/* __ipipe_dispatch_event() -- Low-level event dispatcher. */
int fastcall __ipipe_dispatch_event (unsigned event, void *data)
{
struct ipipe_domain *start_domain, *this_domain, *next_domain;
struct list_head *pos, *npos;
unsigned long flags;
ipipe_declare_cpuid;
int propagate = 1;
ipipe_lock_cpu(flags);
start_domain = this_domain = ipipe_percpu_domain[cpuid];
list_for_each_safe(pos,npos,&__ipipe_pipeline) {
next_domain = list_entry(pos,struct
ipipe_domain,p_link);
/*
* Note: Domain migration may occur while running
* event or interrupt handlers, in which case the
* current register set is going to be recycled for a
* different domain than the initiating one. We do
* care for that, always tracking the current domain
* descriptor upon return from those handlers.
*/
if (next_domain->evhand[event] != NULL) {
ipipe_percpu_domain[cpuid] = next_domain;
ipipe_unlock_cpu(flags);
propagate =
!next_domain->evhand[event](event,start_domain,data);
ipipe_lock_cpu(flags);
if (ipipe_percpu_domain[cpuid] != next_domain)
this_domain =
ipipe_percpu_domain[cpuid];
}
if (next_domain != ipipe_root_domain && /* NEVER sync
the root stage here. */
next_domain->cpudata[cpuid].irq_pending_hi != 0 &&
!test_bit(IPIPE_STALL_FLAG,&next_domain->cpudata[cpuid].status)) {
ipipe_percpu_domain[cpuid] = next_domain;
__ipipe_sync_pipeline(IPIPE_IRQMASK_ANY);
ipipe_load_cpuid();
if (ipipe_percpu_domain[cpuid] != next_domain)
this_domain =
ipipe_percpu_domain[cpuid];
}
ipipe_percpu_domain[cpuid] = this_domain;
if (next_domain == this_domain || !propagate)
break;
}
ipipe_unlock_cpu(flags);
return !propagate;
}
Ksrc/nucleus/shadow.c
static inline int do_losyscall_event (unsigned event, unsigned domid,
void *data) ---------- For ipipe_root_domain()
static inline int do_hisyscall_event (unsigned event, unsigned domid,
void *data) ---------- For Xenomai domain()
No comments:
Post a Comment