| /* |
| * General Purpose functions for the global management of the |
| * Communication Processor Module. |
| * |
| * Copyright (c) 2000 Michael Leslie <mleslie@lineo.com> |
| * Copyright (c) 1997 Dan Malek (dmalek@jlc.net) |
| * |
| * In addition to the individual control of the communication |
| * channels, there are a few functions that globally affect the |
| * communication processor. |
| * |
| * Buffer descriptors must be allocated from the dual ported memory |
| * space. The allocator for that is here. When the communication |
| * process is reset, we reclaim the memory available. There is |
| * currently no deallocator for this memory. |
| * The amount of space available is platform dependent. On the |
| * MBX, the EPPC software loads additional microcode into the |
| * communication processor, and uses some of the DP ram for this |
| * purpose. Current, the first 512 bytes and the last 256 bytes of |
| * memory are used. Right now I am conservative and only use the |
| * memory that can never be used for microcode. If there are |
| * applications that require more DP ram, we can expand the boundaries |
| * but then we have to be careful of any downloaded microcode. |
| * |
| */ |
| |
| /* |
| * Michael Leslie <mleslie@lineo.com> |
| * adapted Dan Malek's ppc8xx drivers to M68360 |
| * |
| */ |
| |
| #include <linux/errno.h> |
| #include <linux/init.h> |
| #include <linux/sched.h> |
| #include <linux/kernel.h> |
| #include <linux/param.h> |
| #include <linux/string.h> |
| #include <linux/mm.h> |
| #include <linux/interrupt.h> |
| #include <asm/irq.h> |
| #include <asm/m68360.h> |
| #include <asm/commproc.h> |
| |
| /* #include <asm/page.h> */ |
| /* #include <asm/pgtable.h> */ |
| extern void *_quicc_base; |
| extern unsigned int system_clock; |
| |
| |
| static uint dp_alloc_base; /* Starting offset in DP ram */ |
| static uint dp_alloc_top; /* Max offset + 1 */ |
| |
| #if 0 |
| static void *host_buffer; /* One page of host buffer */ |
| static void *host_end; /* end + 1 */ |
| #endif |
| |
| /* struct cpm360_t *cpmp; */ /* Pointer to comm processor space */ |
| |
| QUICC *pquicc; |
| /* QUICC *quicc_dpram; */ /* mleslie - temporary; use extern pquicc elsewhere instead */ |
| |
| |
| /* CPM interrupt vector functions. */ |
| struct cpm_action { |
| void (*handler)(void *); |
| void *dev_id; |
| }; |
| static struct cpm_action cpm_vecs[CPMVEC_NR]; |
| static void cpm_interrupt(int irq, void * dev, struct pt_regs * regs); |
| static void cpm_error_interrupt(void *); |
| |
| /* prototypes: */ |
| void cpm_install_handler(int vec, void (*handler)(), void *dev_id); |
| void m360_cpm_reset(void); |
| |
| |
| |
| |
| void __init m360_cpm_reset() |
| { |
| /* pte_t *pte; */ |
| |
| pquicc = (struct quicc *)(_quicc_base); /* initialized in crt0_rXm.S */ |
| |
| /* Perform a CPM reset. */ |
| pquicc->cp_cr = (SOFTWARE_RESET | CMD_FLAG); |
| |
| /* Wait for CPM to become ready (should be 2 clocks). */ |
| while (pquicc->cp_cr & CMD_FLAG); |
| |
| /* On the recommendation of the 68360 manual, p. 7-60 |
| * - Set sdma interrupt service mask to 7 |
| * - Set sdma arbitration ID to 4 |
| */ |
| pquicc->sdma_sdcr = 0x0740; |
| |
| |
| /* Claim the DP memory for our use. |
| */ |
| dp_alloc_base = CPM_DATAONLY_BASE; |
| dp_alloc_top = dp_alloc_base + CPM_DATAONLY_SIZE; |
| |
| |
| /* Set the host page for allocation. |
| */ |
| /* host_buffer = host_page_addr; */ |
| /* host_end = host_page_addr + PAGE_SIZE; */ |
| |
| /* pte = find_pte(&init_mm, host_page_addr); */ |
| /* pte_val(*pte) |= _PAGE_NO_CACHE; */ |
| /* flush_tlb_page(current->mm->mmap, host_buffer); */ |
| |
| /* Tell everyone where the comm processor resides. |
| */ |
| /* cpmp = (cpm360_t *)commproc; */ |
| } |
| |
| |
| /* This is called during init_IRQ. We used to do it above, but this |
| * was too early since init_IRQ was not yet called. |
| */ |
| void |
| cpm_interrupt_init(void) |
| { |
| /* Initialize the CPM interrupt controller. |
| * NOTE THAT pquicc had better have been initialized! |
| * reference: MC68360UM p. 7-377 |
| */ |
| pquicc->intr_cicr = |
| (CICR_SCD_SCC4 | CICR_SCC_SCC3 | CICR_SCB_SCC2 | CICR_SCA_SCC1) | |
| (CPM_INTERRUPT << 13) | |
| CICR_HP_MASK | |
| (CPM_VECTOR_BASE << 5) | |
| CICR_SPS; |
| |
| /* mask all CPM interrupts from reaching the cpu32 core: */ |
| pquicc->intr_cimr = 0; |
| |
| |
| /* mles - If I understand correctly, the 360 just pops over to the CPM |
| * specific vector, obviating the necessity to vector through the IRQ |
| * whose priority the CPM is set to. This needs a closer look, though. |
| */ |
| |
| /* Set our interrupt handler with the core CPU. */ |
| /* if (request_irq(CPM_INTERRUPT, cpm_interrupt, 0, "cpm", NULL) != 0) */ |
| /* panic("Could not allocate CPM IRQ!"); */ |
| |
| /* Install our own error handler. |
| */ |
| /* I think we want to hold off on this one for the moment - mles */ |
| /* cpm_install_handler(CPMVEC_ERROR, cpm_error_interrupt, NULL); */ |
| |
| /* master CPM interrupt enable */ |
| /* pquicc->intr_cicr |= CICR_IEN; */ /* no such animal for 360 */ |
| } |
| |
| |
| |
| /* CPM interrupt controller interrupt. |
| */ |
| static void |
| cpm_interrupt(int irq, void * dev, struct pt_regs * regs) |
| { |
| /* uint vec; */ |
| |
| /* mles: Note that this stuff is currently being performed by |
| * M68360_do_irq(int vec, struct pt_regs *fp), in ../ints.c */ |
| |
| /* figure out the vector */ |
| /* call that vector's handler */ |
| /* clear the irq's bit in the service register */ |
| |
| #if 0 /* old 860 stuff: */ |
| /* Get the vector by setting the ACK bit and then reading |
| * the register. |
| */ |
| ((volatile immap_t *)IMAP_ADDR)->im_cpic.cpic_civr = 1; |
| vec = ((volatile immap_t *)IMAP_ADDR)->im_cpic.cpic_civr; |
| vec >>= 11; |
| |
| |
| if (cpm_vecs[vec].handler != 0) |
| (*cpm_vecs[vec].handler)(cpm_vecs[vec].dev_id); |
| else |
| ((immap_t *)IMAP_ADDR)->im_cpic.cpic_cimr &= ~(1 << vec); |
| |
| /* After servicing the interrupt, we have to remove the status |
| * indicator. |
| */ |
| ((immap_t *)IMAP_ADDR)->im_cpic.cpic_cisr |= (1 << vec); |
| #endif |
| |
| } |
| |
| /* The CPM can generate the error interrupt when there is a race condition |
| * between generating and masking interrupts. All we have to do is ACK it |
| * and return. This is a no-op function so we don't need any special |
| * tests in the interrupt handler. |
| */ |
| static void |
| cpm_error_interrupt(void *dev) |
| { |
| } |
| |
| /* Install a CPM interrupt handler. |
| */ |
| void |
| cpm_install_handler(int vec, void (*handler)(), void *dev_id) |
| { |
| |
| request_irq(vec, handler, 0, "timer", dev_id); |
| |
| /* if (cpm_vecs[vec].handler != 0) */ |
| /* printk(KERN_INFO "CPM interrupt %x replacing %x\n", */ |
| /* (uint)handler, (uint)cpm_vecs[vec].handler); */ |
| /* cpm_vecs[vec].handler = handler; */ |
| /* cpm_vecs[vec].dev_id = dev_id; */ |
| |
| /* ((immap_t *)IMAP_ADDR)->im_cpic.cpic_cimr |= (1 << vec); */ |
| /* pquicc->intr_cimr |= (1 << vec); */ |
| |
| } |
| |
| /* Free a CPM interrupt handler. |
| */ |
| void |
| cpm_free_handler(int vec) |
| { |
| cpm_vecs[vec].handler = NULL; |
| cpm_vecs[vec].dev_id = NULL; |
| /* ((immap_t *)IMAP_ADDR)->im_cpic.cpic_cimr &= ~(1 << vec); */ |
| pquicc->intr_cimr &= ~(1 << vec); |
| } |
| |
| |
| |
| |
| /* Allocate some memory from the dual ported ram. We may want to |
| * enforce alignment restrictions, but right now everyone is a good |
| * citizen. |
| */ |
| uint |
| m360_cpm_dpalloc(uint size) |
| { |
| uint retloc; |
| |
| if ((dp_alloc_base + size) >= dp_alloc_top) |
| return(CPM_DP_NOSPACE); |
| |
| retloc = dp_alloc_base; |
| dp_alloc_base += size; |
| |
| return(retloc); |
| } |
| |
| |
| #if 0 /* mleslie - for now these are simply kmalloc'd */ |
| /* We also own one page of host buffer space for the allocation of |
| * UART "fifos" and the like. |
| */ |
| uint |
| m360_cpm_hostalloc(uint size) |
| { |
| uint retloc; |
| |
| if ((host_buffer + size) >= host_end) |
| return(0); |
| |
| retloc = host_buffer; |
| host_buffer += size; |
| |
| return(retloc); |
| } |
| #endif |
| |
| |
| /* Set a baud rate generator. This needs lots of work. There are |
| * four BRGs, any of which can be wired to any channel. |
| * The internal baud rate clock is the system clock divided by 16. |
| * This assumes the baudrate is 16x oversampled by the uart. |
| */ |
| /* #define BRG_INT_CLK (((bd_t *)__res)->bi_intfreq * 1000000) */ |
| #define BRG_INT_CLK system_clock |
| #define BRG_UART_CLK (BRG_INT_CLK/16) |
| |
| void |
| m360_cpm_setbrg(uint brg, uint rate) |
| { |
| volatile uint *bp; |
| |
| /* This is good enough to get SMCs running..... |
| */ |
| /* bp = (uint *)&cpmp->cp_brgc1; */ |
| bp = (volatile uint *)(&pquicc->brgc[0].l); |
| bp += brg; |
| *bp = ((BRG_UART_CLK / rate - 1) << 1) | CPM_BRG_EN; |
| } |
| |
| |
| /* |
| * Local variables: |
| * c-indent-level: 4 |
| * c-basic-offset: 4 |
| * tab-width: 4 |
| * End: |
| */ |