| |
| /* |
| * |
| Copyright (c) Eicon Networks, 2002. |
| * |
| This source file is supplied for the use with |
| Eicon Networks range of DIVA Server Adapters. |
| * |
| Eicon File Revision : 2.1 |
| * |
| This program is free software; you can redistribute it and/or modify |
| it under the terms of the GNU General Public License as published by |
| the Free Software Foundation; either version 2, or (at your option) |
| any later version. |
| * |
| This program is distributed in the hope that it will be useful, |
| but WITHOUT ANY WARRANTY OF ANY KIND WHATSOEVER INCLUDING ANY |
| implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
| See the GNU General Public License for more details. |
| * |
| You should have received a copy of the GNU General Public License |
| along with this program; if not, write to the Free Software |
| Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. |
| * |
| */ |
| #include "platform.h" |
| #include "di_defs.h" |
| #include "pc.h" |
| #include "pr_pc.h" |
| #include "di.h" |
| #include "mi_pc.h" |
| #include "pc_maint.h" |
| #include "divasync.h" |
| #include "pc_init.h" |
| #include "io.h" |
| #include "helpers.h" |
| #include "dsrv4bri.h" |
| #include "dsp_defs.h" |
| #include "sdp_hdr.h" |
| |
| /*****************************************************************************/ |
| #define MAX_XLOG_SIZE (64 * 1024) |
| |
| /* -------------------------------------------------------------------------- |
| Recovery XLOG from QBRI Card |
| -------------------------------------------------------------------------- */ |
| static void qBri_cpu_trapped (PISDN_ADAPTER IoAdapter) { |
| byte __iomem *base ; |
| word *Xlog ; |
| dword regs[4], TrapID, offset, size ; |
| Xdesc xlogDesc ; |
| int factor = (IoAdapter->tasks == 1) ? 1 : 2; |
| |
| /* |
| * check for trapped MIPS 46xx CPU, dump exception frame |
| */ |
| |
| base = DIVA_OS_MEM_ATTACH_CONTROL(IoAdapter); |
| offset = IoAdapter->ControllerNumber * (IoAdapter->MemorySize >> factor) ; |
| |
| TrapID = READ_DWORD(&base[0x80]) ; |
| |
| if ( (TrapID == 0x99999999) || (TrapID == 0x99999901) ) |
| { |
| dump_trap_frame (IoAdapter, &base[0x90]) ; |
| IoAdapter->trapped = 1 ; |
| } |
| |
| regs[0] = READ_DWORD((base + offset) + 0x70); |
| regs[1] = READ_DWORD((base + offset) + 0x74); |
| regs[2] = READ_DWORD((base + offset) + 0x78); |
| regs[3] = READ_DWORD((base + offset) + 0x7c); |
| regs[0] &= IoAdapter->MemorySize - 1 ; |
| |
| if ( (regs[0] >= offset) |
| && (regs[0] < offset + (IoAdapter->MemorySize >> factor) - 1) ) |
| { |
| if ( !(Xlog = (word *)diva_os_malloc (0, MAX_XLOG_SIZE)) ) { |
| DIVA_OS_MEM_DETACH_CONTROL(IoAdapter, base); |
| return ; |
| } |
| |
| size = offset + (IoAdapter->MemorySize >> factor) - regs[0] ; |
| if ( size > MAX_XLOG_SIZE ) |
| size = MAX_XLOG_SIZE ; |
| memcpy_fromio (Xlog, &base[regs[0]], size) ; |
| xlogDesc.buf = Xlog ; |
| xlogDesc.cnt = READ_WORD(&base[regs[1] & (IoAdapter->MemorySize - 1)]) ; |
| xlogDesc.out = READ_WORD(&base[regs[2] & (IoAdapter->MemorySize - 1)]) ; |
| dump_xlog_buffer (IoAdapter, &xlogDesc) ; |
| diva_os_free (0, Xlog) ; |
| IoAdapter->trapped = 2 ; |
| } |
| DIVA_OS_MEM_DETACH_CONTROL(IoAdapter, base); |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Reset QBRI Hardware |
| -------------------------------------------------------------------------- */ |
| static void reset_qBri_hardware (PISDN_ADAPTER IoAdapter) { |
| word volatile __iomem *qBriReset ; |
| byte volatile __iomem *qBriCntrl ; |
| byte volatile __iomem *p ; |
| |
| qBriReset = (word volatile __iomem *)DIVA_OS_MEM_ATTACH_PROM(IoAdapter); |
| WRITE_WORD(qBriReset, READ_WORD(qBriReset) | PLX9054_SOFT_RESET) ; |
| diva_os_wait (1) ; |
| WRITE_WORD(qBriReset, READ_WORD(qBriReset) & ~PLX9054_SOFT_RESET) ; |
| diva_os_wait (1); |
| WRITE_WORD(qBriReset, READ_WORD(qBriReset) | PLX9054_RELOAD_EEPROM) ; |
| diva_os_wait (1) ; |
| WRITE_WORD(qBriReset, READ_WORD(qBriReset) & ~PLX9054_RELOAD_EEPROM) ; |
| diva_os_wait (1); |
| DIVA_OS_MEM_DETACH_PROM(IoAdapter, qBriReset); |
| |
| qBriCntrl = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| p = &qBriCntrl[DIVA_4BRI_REVISION(IoAdapter) ? (MQ2_BREG_RISC) : (MQ_BREG_RISC)]; |
| WRITE_DWORD(p, 0) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, qBriCntrl); |
| |
| DBG_TRC(("resetted board @ reset addr 0x%08lx", qBriReset)) |
| DBG_TRC(("resetted board @ cntrl addr 0x%08lx", p)) |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Start Card CPU |
| -------------------------------------------------------------------------- */ |
| void start_qBri_hardware (PISDN_ADAPTER IoAdapter) { |
| byte volatile __iomem *qBriReset ; |
| byte volatile __iomem *p ; |
| |
| p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| qBriReset = &p[(DIVA_4BRI_REVISION(IoAdapter)) ? (MQ2_BREG_RISC) : (MQ_BREG_RISC)]; |
| WRITE_DWORD(qBriReset, MQ_RISC_COLD_RESET_MASK) ; |
| diva_os_wait (2) ; |
| WRITE_DWORD(qBriReset, MQ_RISC_WARM_RESET_MASK | MQ_RISC_COLD_RESET_MASK) ; |
| diva_os_wait (10) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p); |
| |
| DBG_TRC(("started processor @ addr 0x%08lx", qBriReset)) |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Stop Card CPU |
| -------------------------------------------------------------------------- */ |
| static void stop_qBri_hardware (PISDN_ADAPTER IoAdapter) { |
| byte volatile __iomem *p ; |
| dword volatile __iomem *qBriReset ; |
| dword volatile __iomem *qBriIrq ; |
| dword volatile __iomem *qBriIsacDspReset ; |
| int rev2 = DIVA_4BRI_REVISION(IoAdapter); |
| int reset_offset = rev2 ? (MQ2_BREG_RISC) : (MQ_BREG_RISC); |
| int irq_offset = rev2 ? (MQ2_BREG_IRQ_TEST) : (MQ_BREG_IRQ_TEST); |
| int hw_offset = rev2 ? (MQ2_ISAC_DSP_RESET) : (MQ_ISAC_DSP_RESET); |
| |
| if ( IoAdapter->ControllerNumber > 0 ) |
| return ; |
| p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| qBriReset = (dword volatile __iomem *)&p[reset_offset]; |
| qBriIsacDspReset = (dword volatile __iomem *)&p[hw_offset]; |
| /* |
| * clear interrupt line (reset Local Interrupt Test Register) |
| */ |
| WRITE_DWORD(qBriReset, 0) ; |
| WRITE_DWORD(qBriIsacDspReset, 0) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p); |
| |
| p = DIVA_OS_MEM_ATTACH_RESET(IoAdapter); |
| WRITE_BYTE(&p[PLX9054_INTCSR], 0x00); /* disable PCI interrupts */ |
| DIVA_OS_MEM_DETACH_RESET(IoAdapter, p); |
| |
| p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| qBriIrq = (dword volatile __iomem *)&p[irq_offset]; |
| WRITE_DWORD(qBriIrq, MQ_IRQ_REQ_OFF) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p); |
| |
| DBG_TRC(("stopped processor @ addr 0x%08lx", qBriReset)) |
| |
| } |
| |
| /* -------------------------------------------------------------------------- |
| FPGA download |
| -------------------------------------------------------------------------- */ |
| #define FPGA_NAME_OFFSET 0x10 |
| |
| static byte * qBri_check_FPGAsrc (PISDN_ADAPTER IoAdapter, char *FileName, |
| dword *Length, dword *code) { |
| byte *File ; |
| char *fpgaFile, *fpgaType, *fpgaDate, *fpgaTime ; |
| dword fpgaFlen, fpgaTlen, fpgaDlen, cnt, year, i ; |
| |
| if (!(File = (byte *)xdiLoadFile (FileName, Length, 0))) { |
| return (NULL) ; |
| } |
| /* |
| * scan file until FF and put id string into buffer |
| */ |
| for ( i = 0 ; File[i] != 0xff ; ) |
| { |
| if ( ++i >= *Length ) |
| { |
| DBG_FTL(("FPGA download: start of data header not found")) |
| xdiFreeFile (File) ; |
| return (NULL) ; |
| } |
| } |
| *code = i++ ; |
| |
| if ( (File[i] & 0xF0) != 0x20 ) |
| { |
| DBG_FTL(("FPGA download: data header corrupted")) |
| xdiFreeFile (File) ; |
| return (NULL) ; |
| } |
| fpgaFlen = (dword) File[FPGA_NAME_OFFSET - 1] ; |
| if ( fpgaFlen == 0 ) |
| fpgaFlen = 12 ; |
| fpgaFile = (char *)&File[FPGA_NAME_OFFSET] ; |
| fpgaTlen = (dword) fpgaFile[fpgaFlen + 2] ; |
| if ( fpgaTlen == 0 ) |
| fpgaTlen = 10 ; |
| fpgaType = (char *)&fpgaFile[fpgaFlen + 3] ; |
| fpgaDlen = (dword) fpgaType[fpgaTlen + 2] ; |
| if ( fpgaDlen == 0 ) |
| fpgaDlen = 11 ; |
| fpgaDate = (char *)&fpgaType[fpgaTlen + 3] ; |
| fpgaTime = (char *)&fpgaDate[fpgaDlen + 3] ; |
| cnt = (dword)(((File[ i ] & 0x0F) << 20) + (File[i + 1] << 12) |
| + (File[i + 2] << 4) + (File[i + 3] >> 4)) ; |
| |
| if ( (dword)(i + (cnt / 8)) > *Length ) |
| { |
| DBG_FTL(("FPGA download: '%s' file too small (%ld < %ld)", |
| FileName, *Length, code + ((cnt + 7) / 8) )) |
| xdiFreeFile (File) ; |
| return (NULL) ; |
| } |
| i = 0 ; |
| do |
| { |
| while ( (fpgaDate[i] != '\0') |
| && ((fpgaDate[i] < '0') || (fpgaDate[i] > '9')) ) |
| { |
| i++; |
| } |
| year = 0 ; |
| while ( (fpgaDate[i] >= '0') && (fpgaDate[i] <= '9') ) |
| year = year * 10 + (fpgaDate[i++] - '0') ; |
| } while ( (year < 2000) && (fpgaDate[i] != '\0') ); |
| |
| switch (IoAdapter->cardType) { |
| case CARDTYPE_DIVASRV_B_2F_PCI: |
| break; |
| |
| default: |
| if ( year >= 2001 ) { |
| IoAdapter->fpga_features |= PCINIT_FPGA_PLX_ACCESS_SUPPORTED ; |
| } |
| } |
| |
| DBG_LOG(("FPGA[%s] file %s (%s %s) len %d", |
| fpgaType, fpgaFile, fpgaDate, fpgaTime, cnt)) |
| return (File) ; |
| } |
| |
| /******************************************************************************/ |
| |
| #define FPGA_PROG 0x0001 /* PROG enable low */ |
| #define FPGA_BUSY 0x0002 /* BUSY high, DONE low */ |
| #define FPGA_CS 0x000C /* Enable I/O pins */ |
| #define FPGA_CCLK 0x0100 |
| #define FPGA_DOUT 0x0400 |
| #define FPGA_DIN FPGA_DOUT /* bidirectional I/O */ |
| |
| int qBri_FPGA_download (PISDN_ADAPTER IoAdapter) { |
| int bit ; |
| byte *File ; |
| dword code, FileLength ; |
| word volatile __iomem *addr = (word volatile __iomem *)DIVA_OS_MEM_ATTACH_PROM(IoAdapter); |
| word val, baseval = FPGA_CS | FPGA_PROG ; |
| |
| |
| |
| if (DIVA_4BRI_REVISION(IoAdapter)) |
| { |
| char* name; |
| |
| switch (IoAdapter->cardType) { |
| case CARDTYPE_DIVASRV_B_2F_PCI: |
| name = "dsbri2f.bit"; |
| break; |
| |
| case CARDTYPE_DIVASRV_B_2M_V2_PCI: |
| case CARDTYPE_DIVASRV_VOICE_B_2M_V2_PCI: |
| name = "dsbri2m.bit"; |
| break; |
| |
| default: |
| name = "ds4bri2.bit"; |
| } |
| |
| File = qBri_check_FPGAsrc (IoAdapter, name, |
| &FileLength, &code); |
| } |
| else |
| { |
| File = qBri_check_FPGAsrc (IoAdapter, "ds4bri.bit", |
| &FileLength, &code) ; |
| } |
| if ( !File ) { |
| DIVA_OS_MEM_DETACH_PROM(IoAdapter, addr); |
| return (0) ; |
| } |
| /* |
| * prepare download, pulse PROGRAM pin down. |
| */ |
| WRITE_WORD(addr, baseval & ~FPGA_PROG) ; /* PROGRAM low pulse */ |
| WRITE_WORD(addr, baseval) ; /* release */ |
| diva_os_wait (50) ; /* wait until FPGA finished internal memory clear */ |
| /* |
| * check done pin, must be low |
| */ |
| if ( READ_WORD(addr) & FPGA_BUSY ) |
| { |
| DBG_FTL(("FPGA download: acknowledge for FPGA memory clear missing")) |
| xdiFreeFile (File) ; |
| DIVA_OS_MEM_DETACH_PROM(IoAdapter, addr); |
| return (0) ; |
| } |
| /* |
| * put data onto the FPGA |
| */ |
| while ( code < FileLength ) |
| { |
| val = ((word)File[code++]) << 3 ; |
| |
| for ( bit = 8 ; bit-- > 0 ; val <<= 1 ) /* put byte onto FPGA */ |
| { |
| baseval &= ~FPGA_DOUT ; /* clr data bit */ |
| baseval |= (val & FPGA_DOUT) ; /* copy data bit */ |
| WRITE_WORD(addr, baseval) ; |
| WRITE_WORD(addr, baseval | FPGA_CCLK) ; /* set CCLK hi */ |
| WRITE_WORD(addr, baseval | FPGA_CCLK) ; /* set CCLK hi */ |
| WRITE_WORD(addr, baseval) ; /* set CCLK lo */ |
| } |
| } |
| xdiFreeFile (File) ; |
| diva_os_wait (100) ; |
| val = READ_WORD(addr) ; |
| |
| DIVA_OS_MEM_DETACH_PROM(IoAdapter, addr); |
| |
| if ( !(val & FPGA_BUSY) ) |
| { |
| DBG_FTL(("FPGA download: chip remains in busy state (0x%04x)", val)) |
| return (0) ; |
| } |
| |
| return (1) ; |
| } |
| |
| static int load_qBri_hardware (PISDN_ADAPTER IoAdapter) { |
| return (0); |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Card ISR |
| -------------------------------------------------------------------------- */ |
| static int qBri_ISR (struct _ISDN_ADAPTER* IoAdapter) { |
| dword volatile __iomem *qBriIrq ; |
| |
| PADAPTER_LIST_ENTRY QuadroList = IoAdapter->QuadroList ; |
| |
| word i ; |
| int serviced = 0 ; |
| byte __iomem *p; |
| |
| p = DIVA_OS_MEM_ATTACH_RESET(IoAdapter); |
| |
| if ( !(READ_BYTE(&p[PLX9054_INTCSR]) & 0x80) ) { |
| DIVA_OS_MEM_DETACH_RESET(IoAdapter, p); |
| return (0) ; |
| } |
| DIVA_OS_MEM_DETACH_RESET(IoAdapter, p); |
| |
| /* |
| * clear interrupt line (reset Local Interrupt Test Register) |
| */ |
| p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| qBriIrq = (dword volatile __iomem *)(&p[DIVA_4BRI_REVISION(IoAdapter) ? (MQ2_BREG_IRQ_TEST) : (MQ_BREG_IRQ_TEST)]); |
| WRITE_DWORD(qBriIrq, MQ_IRQ_REQ_OFF) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p); |
| |
| for ( i = 0 ; i < IoAdapter->tasks; ++i ) |
| { |
| IoAdapter = QuadroList->QuadroAdapter[i] ; |
| |
| if ( IoAdapter && IoAdapter->Initialized |
| && IoAdapter->tst_irq (&IoAdapter->a) ) |
| { |
| IoAdapter->IrqCount++ ; |
| serviced = 1 ; |
| diva_os_schedule_soft_isr (&IoAdapter->isr_soft_isr); |
| } |
| } |
| |
| return (serviced) ; |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Does disable the interrupt on the card |
| -------------------------------------------------------------------------- */ |
| static void disable_qBri_interrupt (PISDN_ADAPTER IoAdapter) { |
| dword volatile __iomem *qBriIrq ; |
| byte __iomem *p; |
| |
| if ( IoAdapter->ControllerNumber > 0 ) |
| return ; |
| /* |
| * clear interrupt line (reset Local Interrupt Test Register) |
| */ |
| p = DIVA_OS_MEM_ATTACH_RESET(IoAdapter); |
| WRITE_BYTE(&p[PLX9054_INTCSR], 0x00); /* disable PCI interrupts */ |
| DIVA_OS_MEM_DETACH_RESET(IoAdapter, p); |
| |
| p = DIVA_OS_MEM_ATTACH_CTLREG(IoAdapter); |
| qBriIrq = (dword volatile __iomem *)(&p[DIVA_4BRI_REVISION(IoAdapter) ? (MQ2_BREG_IRQ_TEST) : (MQ_BREG_IRQ_TEST)]); |
| WRITE_DWORD(qBriIrq, MQ_IRQ_REQ_OFF) ; |
| DIVA_OS_MEM_DETACH_CTLREG(IoAdapter, p); |
| } |
| |
| /* -------------------------------------------------------------------------- |
| Install Adapter Entry Points |
| -------------------------------------------------------------------------- */ |
| static void set_common_qBri_functions (PISDN_ADAPTER IoAdapter) { |
| ADAPTER *a; |
| |
| a = &IoAdapter->a ; |
| |
| a->ram_in = mem_in ; |
| a->ram_inw = mem_inw ; |
| a->ram_in_buffer = mem_in_buffer ; |
| a->ram_look_ahead = mem_look_ahead ; |
| a->ram_out = mem_out ; |
| a->ram_outw = mem_outw ; |
| a->ram_out_buffer = mem_out_buffer ; |
| a->ram_inc = mem_inc ; |
| |
| IoAdapter->out = pr_out ; |
| IoAdapter->dpc = pr_dpc ; |
| IoAdapter->tst_irq = scom_test_int ; |
| IoAdapter->clr_irq = scom_clear_int ; |
| IoAdapter->pcm = (struct pc_maint *)MIPS_MAINT_OFFS ; |
| |
| IoAdapter->load = load_qBri_hardware ; |
| |
| IoAdapter->disIrq = disable_qBri_interrupt ; |
| IoAdapter->rstFnc = reset_qBri_hardware ; |
| IoAdapter->stop = stop_qBri_hardware ; |
| IoAdapter->trapFnc = qBri_cpu_trapped ; |
| |
| IoAdapter->diva_isr_handler = qBri_ISR; |
| |
| IoAdapter->a.io = (void*)IoAdapter ; |
| } |
| |
| static void set_qBri_functions (PISDN_ADAPTER IoAdapter) { |
| if (!IoAdapter->tasks) { |
| IoAdapter->tasks = MQ_INSTANCE_COUNT; |
| } |
| IoAdapter->MemorySize = MQ_MEMORY_SIZE ; |
| set_common_qBri_functions (IoAdapter) ; |
| diva_os_set_qBri_functions (IoAdapter) ; |
| } |
| |
| static void set_qBri2_functions (PISDN_ADAPTER IoAdapter) { |
| if (!IoAdapter->tasks) { |
| IoAdapter->tasks = MQ_INSTANCE_COUNT; |
| } |
| IoAdapter->MemorySize = (IoAdapter->tasks == 1) ? BRI2_MEMORY_SIZE : MQ2_MEMORY_SIZE; |
| set_common_qBri_functions (IoAdapter) ; |
| diva_os_set_qBri2_functions (IoAdapter) ; |
| } |
| |
| /******************************************************************************/ |
| |
| void prepare_qBri_functions (PISDN_ADAPTER IoAdapter) { |
| |
| set_qBri_functions (IoAdapter->QuadroList->QuadroAdapter[0]) ; |
| set_qBri_functions (IoAdapter->QuadroList->QuadroAdapter[1]) ; |
| set_qBri_functions (IoAdapter->QuadroList->QuadroAdapter[2]) ; |
| set_qBri_functions (IoAdapter->QuadroList->QuadroAdapter[3]) ; |
| |
| } |
| |
| void prepare_qBri2_functions (PISDN_ADAPTER IoAdapter) { |
| if (!IoAdapter->tasks) { |
| IoAdapter->tasks = MQ_INSTANCE_COUNT; |
| } |
| |
| set_qBri2_functions (IoAdapter->QuadroList->QuadroAdapter[0]) ; |
| if (IoAdapter->tasks > 1) { |
| set_qBri2_functions (IoAdapter->QuadroList->QuadroAdapter[1]) ; |
| set_qBri2_functions (IoAdapter->QuadroList->QuadroAdapter[2]) ; |
| set_qBri2_functions (IoAdapter->QuadroList->QuadroAdapter[3]) ; |
| } |
| |
| } |
| |
| /* -------------------------------------------------------------------------- */ |