/* * header goes here */ #include #include #include #include #include #include #include #include #include #define NUM_GATES 16 #define MASK_FROM_GATE(gate_num) ((u32)(1 << (NUM_GATES*2 - 1 - idx[gate_num]))) #define THIS_CORE (0) #define LOCK_VALUE (THIS_CORE + 1) #define SEMA4_CP0INE (MVF_SEMA4_BASE_ADDR + 0x40) #define SEMA4_CP0NTF (MVF_SEMA4_BASE_ADDR + 0x80) //#if 0 #include #include #include #include #include #include #include #include #include #include //#endif #include // ************************************ Local Data ************************************************* static MVF_SEMA4* gates[NUM_GATES]; static bool initialized = false; // account for the way the bits are set / returned in CP0INE and CP0NTF static const int idx[16] = {3, 2, 1, 0, 7, 6, 5, 4, 11, 10, 9, 8, 15, 14, 13, 12}; // ************************************ Interrupt handler ************************************************* static irqreturn_t sema4_irq_handler(int irq, void *dev_id) { int gate_num; u32 cp0ntf = readl(MVF_IO_ADDRESS(SEMA4_CP0NTF)); for(gate_num=0; gate_numuse_interrupts) // wake up whoever was aiting wake_up_interruptible(&(gates[gate_num]->wait_queue)); } } } return IRQ_HANDLED; } // ************************************ Utility functions ************************************************* static int find_sema4(MVF_SEMA4 *sema4) { int i; for(i=0; i= NUM_GATES)) return -EINVAL; if(gates[gate_num]) return -EBUSY; *sema4_p = (MVF_SEMA4 *)kmalloc(sizeof(MVF_SEMA4), GFP_KERNEL); if(*sema4_p == NULL) return -ENOMEM; gates[gate_num] = *sema4_p; (*sema4_p)->gate_num = gate_num; (*sema4_p)->use_interrupts = use_interrupts; if(use_interrupts) { init_waitqueue_head(&((*sema4_p)->wait_queue)); local_irq_save(irq_flags); cp0ine = readl(MVF_IO_ADDRESS(SEMA4_CP0INE)); cp0ine |= MASK_FROM_GATE(gate_num); writel(cp0ine, MVF_IO_ADDRESS(SEMA4_CP0INE)); local_irq_restore(irq_flags); } return 0; } int mvf_sema4_deassign(MVF_SEMA4 *sema4) { u32 cp0ine; unsigned long irq_flags; int gate_num = find_sema4(sema4); if(gate_num < 0) return gate_num; if(sema4->use_interrupts) { local_irq_save(irq_flags); cp0ine = readl(MVF_IO_ADDRESS(SEMA4_CP0INE)); cp0ine &= ~MASK_FROM_GATE(gate_num); writel(cp0ine, MVF_IO_ADDRESS(SEMA4_CP0INE)); local_irq_restore(irq_flags); } kfree(sema4); gates[gate_num] = NULL; return 0; } int mvf_sema4_lock(MVF_SEMA4 *sema4, unsigned int timeout_us) { int retval; int gate_num = find_sema4(sema4); if(gate_num < 0) return gate_num; // cant use timeouts if not using interruppts // TODO use spin lock if not using interrupts if((!sema4->use_interrupts) && timeout_us) return -EINVAL; // try to grab it writeb(LOCK_VALUE, MVF_IO_ADDRESS(MVF_SEMA4_BASE_ADDR) + gate_num); if(readb(MVF_IO_ADDRESS(MVF_SEMA4_BASE_ADDR) + gate_num) == LOCK_VALUE) return 0; // no timeout, fail if(!timeout_us) return -EBUSY; // wait forever? if(timeout_us == 0xffffffff) { if(wait_event_interruptible(sema4->wait_queue, (readb(MVF_IO_ADDRESS(MVF_SEMA4_BASE_ADDR) + gate_num) == LOCK_VALUE))) return -ERESTARTSYS; } else { // return: 0 = timeout, >0 = woke up with that many jiffies left, <0 = error retval = wait_event_interruptible_timeout(sema4->wait_queue, (readb(MVF_IO_ADDRESS(MVF_SEMA4_BASE_ADDR) + gate_num) == LOCK_VALUE), usecs_to_jiffies(timeout_us)); if(retval == 0) return -ETIME; else if(retval < 0) return retval; } return 0; } int mvf_sema4_unlock(MVF_SEMA4 *sema4) { int gate_num = find_sema4(sema4); if(gate_num < 0) return gate_num; // unlock it writeb(0, MVF_IO_ADDRESS(MVF_SEMA4_BASE_ADDR) + gate_num); return 0; }