123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443 |
- /*
- * Digium, Inc. Wildcard T100P T1/PRI card Driver
- *
- * Written by Mark Spencer <markster@digium.com>
- * Matthew Fredrickson <creslin@digium.com>
- * William Meadows <wmeadows@digium.com>
- *
- * Copyright (C) 2001-2008, Digium, Inc.
- *
- * All rights reserved.
- *
- */
- /*
- * See http://www.asterisk.org for more information about
- * the Asterisk project. Please do not directly contact
- * any of the maintainers of this project for assistance;
- * the project provides a web site, mailing lists and IRC
- * channels for your use.
- *
- * This program is free software, distributed under the terms of
- * the GNU General Public License Version 2 as published by the
- * Free Software Foundation. See the LICENSE file included with
- * this program for more details.
- */
- #include <linux/kernel.h>
- #include <linux/errno.h>
- #include <linux/module.h>
- #include <linux/init.h>
- #include <linux/usb.h>
- #include <linux/errno.h>
- #include <linux/pci.h>
- #include <linux/spinlock.h>
- #include <linux/moduleparam.h>
- #include <dahdi/kernel.h>
- #define WC_MAX_CARDS 32
- /*
- #define TEST_REGS
- */
- /* Define to get more attention-grabbing but slightly more I/O using
- alarm status */
- #define FANCY_ALARM
- #define DELAY 0x0 /* 30 = 15 cycles, 10 = 8 cycles, 0 = 3 cycles */
- #define WC_CNTL 0x00
- #define WC_OPER 0x01
- #define WC_AUXC 0x02
- #define WC_AUXD 0x03
- #define WC_MASK0 0x04
- #define WC_MASK1 0x05
- #define WC_INTSTAT 0x06
- #define WC_DMAWS 0x08
- #define WC_DMAWI 0x0c
- #define WC_DMAWE 0x10
- #define WC_DMARS 0x18
- #define WC_DMARI 0x1c
- #define WC_DMARE 0x20
- #define WC_CURPOS 0x24
- #define WC_SERC 0x2d
- #define WC_FSCDELAY 0x2f
- #define WC_USERREG 0xc0
- #define WC_CLOCK 0x0
- #define WC_LEDTEST 0x1
- #define WC_VERSION 0x2
- /* Offset between transmit and receive */
- #define WC_OFFSET 4
- #define BIT_CS (1 << 7)
- #define BIT_ADDR (0xf << 3)
- #define BIT_LED0 (1 << 0)
- #define BIT_LED1 (1 << 1)
- #define BIT_TEST (1 << 2)
- static char *chips[] =
- {
- "DS2152",
- "DS21352",
- "DS21552",
- "Unknown Chip (3)",
- "DS2154",
- "DS21354",
- "DS21554",
- "Unknown Chip (7)",
- };
- static int chanmap_t1[] =
- { 2,1,0,
- 6,5,4,
- 10,9,8,
- 14,13,12,
- 18,17,16,
- 22,21,20,
- 26,25,24,
- 30,29,28 };
- static int chanmap_e1[] =
- { 2,1,0,
- 7,6,5,4,
- 11,10,9,8,
- 15,14,13,12,
- 19,18,17,16,
- 23,22,21,20,
- 27,26,25,24,
- 31,30,29,28 };
- #ifdef FANCY_ALARM
- static int altab[] = {
- 0, 0, 0, 1, 2, 3, 4, 6, 8, 9, 11, 13, 16, 18, 20, 22, 24, 25, 27, 28, 29, 30, 31, 31, 32, 31, 31, 30, 29, 28, 27, 25, 23, 22, 20, 18, 16, 13, 11, 9, 8, 6, 4, 3, 2, 1, 0, 0,
- };
- #endif
- struct t1xxp {
- struct pci_dev *dev;
- spinlock_t lock;
- int ise1;
- int num;
- /* Our offset for finding channel 1 */
- int offset;
- char *variety;
- unsigned int intcount;
- int usecount;
- int clocktimeout;
- int sync;
- int dead;
- int blinktimer;
- int alarmtimer;
- int loopupcnt;
- int loopdowncnt;
- int miss;
- int misslast;
- int *chanmap;
- #ifdef FANCY_ALARM
- int alarmpos;
- #endif
- unsigned char ledtestreg;
- unsigned char outbyte;
- unsigned long ioaddr;
- unsigned short canary;
- /* T1 signalling */
- unsigned char txsiga[3];
- unsigned char txsigb[3];
- dma_addr_t readdma;
- dma_addr_t writedma;
- volatile unsigned char *writechunk; /* Double-word aligned write memory */
- volatile unsigned char *readchunk; /* Double-word aligned read memory */
- unsigned char ec_chunk1[31][DAHDI_CHUNKSIZE];
- unsigned char ec_chunk2[31][DAHDI_CHUNKSIZE];
- unsigned char tempo[32];
- struct dahdi_span span; /* Span */
- struct dahdi_chan *chans[31]; /* Channels */
- };
- #define CANARY 0xca1e
- static int debug = 0; /* doesnt do anything */
- static struct t1xxp *cards[WC_MAX_CARDS];
- static inline void start_alarm(struct t1xxp *wc)
- {
- #ifdef FANCY_ALARM
- wc->alarmpos = 0;
- #endif
- wc->blinktimer = 0;
- }
- static inline void stop_alarm(struct t1xxp *wc)
- {
- #ifdef FANCY_ALARM
- wc->alarmpos = 0;
- #endif
- wc->blinktimer = 0;
- }
- static inline void __select_framer(struct t1xxp *wc, int reg)
- {
- /* Top four bits of address from AUX 6-3 */
- wc->outbyte &= ~BIT_CS;
- wc->outbyte &= ~BIT_ADDR;
- wc->outbyte |= (reg & 0xf0) >> 1;
- outb(wc->outbyte, wc->ioaddr + WC_AUXD);
- }
- static inline void __select_control(struct t1xxp *wc)
- {
- if (!(wc->outbyte & BIT_CS)) {
- wc->outbyte |= BIT_CS;
- outb(wc->outbyte, wc->ioaddr + WC_AUXD);
- }
- }
- static int t1xxp_open(struct dahdi_chan *chan)
- {
- struct t1xxp *wc = chan->pvt;
- if (wc->dead)
- return -ENODEV;
- wc->usecount++;
- return 0;
- }
- static int __t1_get_reg(struct t1xxp *wc, int reg)
- {
- unsigned char res;
- __select_framer(wc, reg);
- /* Get value */
- res = inb(wc->ioaddr + WC_USERREG + ((reg & 0xf) << 2));
- return res;
- }
- static int __t1_set_reg(struct t1xxp *wc, int reg, unsigned char val)
- {
- __select_framer(wc, reg);
- /* Send address */
- outb(val, wc->ioaddr + WC_USERREG + ((reg & 0xf) << 2));
- return 0;
- }
- static int __control_set_reg(struct t1xxp *wc, int reg, unsigned char val)
- {
- __select_control(wc);
- outb(val, wc->ioaddr + WC_USERREG + ((reg & 0xf) << 2));
- return 0;
- }
- static int control_set_reg(struct t1xxp *wc, int reg, unsigned char val)
- {
- unsigned long flags;
- int res;
- spin_lock_irqsave(&wc->lock, flags);
- res = __control_set_reg(wc, reg, val);
- spin_unlock_irqrestore(&wc->lock, flags);
- return res;
- }
- static int __control_get_reg(struct t1xxp *wc, int reg)
- {
- unsigned char res;
- /* The following makes UTTERLY no sense, but what was happening
- was that reads in some cases were not actually happening
- on the physical bus. Why, we dunno. But in debugging, we found
- that writing before reading (in this case to an unused position)
- seems to get rid of the problem */
- __control_set_reg(wc,3,0x69); /* do magic here */
- /* now get the read byte from the Xilinx part */
- res = inb(wc->ioaddr + WC_USERREG + ((reg & 0xf) << 2));
- return res;
- }
- static int control_get_reg(struct t1xxp *wc, int reg)
- {
- unsigned long flags;
- int res;
- spin_lock_irqsave(&wc->lock, flags);
- res = __control_get_reg(wc, reg);
- spin_unlock_irqrestore(&wc->lock, flags);
- return res;
- }
- static void t1xxp_release(struct t1xxp *wc)
- {
- unsigned int x;
- dahdi_unregister(&wc->span);
- for (x = 0; x < (wc->ise1 ? 31 : 24); x++) {
- kfree(wc->chans[x]);
- }
- kfree(wc);
- printk(KERN_INFO "Freed a Wildcard\n");
- }
- static int t1xxp_close(struct dahdi_chan *chan)
- {
- struct t1xxp *wc = chan->pvt;
- wc->usecount--;
- /* If we're dead, release us now */
- if (!wc->usecount && wc->dead)
- t1xxp_release(wc);
- return 0;
- }
- static void t1xxp_enable_interrupts(struct t1xxp *wc)
- {
- /* Clear interrupts */
- outb(0xff, wc->ioaddr + WC_INTSTAT);
- /* Enable interrupts (we care about all of them) */
- outb(0x3c /* 0x3f */, wc->ioaddr + WC_MASK0);
- /* No external interrupts */
- outb(0x00, wc->ioaddr + WC_MASK1);
- }
- static void t1xxp_start_dma(struct t1xxp *wc)
- {
- /* Reset Master and TDM */
- outb(DELAY | 0x0f, wc->ioaddr + WC_CNTL);
- set_current_state(TASK_INTERRUPTIBLE);
- schedule_timeout(1);
- outb(DELAY | 0x01, wc->ioaddr + WC_CNTL);
- outb(0x01, wc->ioaddr + WC_OPER);
- if (debug) printk(KERN_DEBUG "Started DMA\n");
- }
- static void __t1xxp_stop_dma(struct t1xxp *wc)
- {
- outb(0x00, wc->ioaddr + WC_OPER);
- }
- static void __t1xxp_disable_interrupts(struct t1xxp *wc)
- {
- outb(0x00, wc->ioaddr + WC_MASK0);
- outb(0x00, wc->ioaddr + WC_MASK1);
- }
- static void __t1xxp_set_clear(struct t1xxp *wc)
- {
- /* Setup registers */
- int x,y;
- unsigned char b;
- /* No such thing under E1 */
- if (wc->ise1) {
- printk(KERN_NOTICE "Can't set clear mode on an E1!\n");
- return;
- }
- for (x=0;x<3;x++) {
- b = 0;
- for (y=0;y<8;y++)
- if (wc->chans[x * 8 + y]->sig & DAHDI_SIG_CLEAR)
- b |= (1 << y);
- __t1_set_reg(wc, 0x39 + x, b);
- }
- }
- static void t1xxp_t1_framer_start(struct t1xxp *wc)
- {
- int i;
- char *coding, *framing;
- unsigned long endjiffies;
- int alreadyrunning = wc->span.flags & DAHDI_FLAG_RUNNING;
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- /* Build up config */
- i = 0x20;
- if (wc->span.lineconfig & DAHDI_CONFIG_ESF) {
- coding = "ESF";
- i = 0x88;
- } else {
- coding = "SF";
- }
- if (wc->span.lineconfig & DAHDI_CONFIG_B8ZS) {
- framing = "B8ZS";
- i |= 0x44;
- } else {
- framing = "AMI";
- }
- __t1_set_reg(wc, 0x38, i);
- if (!(wc->span.lineconfig & DAHDI_CONFIG_ESF)) {
- /* 1c in FDL bit */
- __t1_set_reg(wc, 0x7e, 0x1c);
- } else {
- __t1_set_reg(wc, 0x7e, 0x00);
- }
- /* Set outgoing LBO */
- __t1_set_reg(wc, 0x7c, wc->span.txlevel << 5);
- printk(KERN_DEBUG "Using %s/%s coding/framing\n", coding, framing);
- if (!alreadyrunning) {
- /* Setup the clear channels */
- __t1xxp_set_clear(wc);
- /* Set LIRST bit to 1 */
- __t1_set_reg(wc, 0x0a, 0x80);
- spin_unlock_irqrestore(&wc->lock, flags);
- /* Wait 100ms to give plenty of time for reset */
- endjiffies = jiffies + 10;
- while(endjiffies < jiffies);
- spin_lock_irqsave(&wc->lock, flags);
-
- /* Reset LIRST bit and reset elastic stores */
- __t1_set_reg(wc, 0xa, 0x30);
- wc->span.flags |= DAHDI_FLAG_RUNNING;
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- }
- static void t1xxp_e1_framer_start(struct t1xxp *wc)
- {
- int i;
- char *coding, *framing;
- unsigned long endjiffies;
- int alreadyrunning = wc->span.flags & DAHDI_FLAG_RUNNING;
- unsigned long flags;
- char *crcing = "";
- unsigned char ccr1, tcr1, tcr2;
- spin_lock_irqsave(&wc->lock, flags);
- /* Build up config */
- ccr1 = 0;
- tcr1 = 8;
- tcr2 = 0;
- if (wc->span.lineconfig & DAHDI_CONFIG_CCS) {
- coding = "CCS"; /* Receive CCS */
- ccr1 |= 8;
- } else {
- tcr1 |= 0x20;
- coding = "CAS";
- }
- if (wc->span.lineconfig & DAHDI_CONFIG_HDB3) {
- ccr1 |= 0x44; /* TX/RX HDB3 */
- framing = "HDB3";
- } else {
- framing = "AMI";
- }
- if (wc->span.lineconfig & DAHDI_CONFIG_CRC4) {
- ccr1 |= 0x11;
- tcr2 |= 0x02;
- crcing = " with CRC4";
- }
- __t1_set_reg(wc, 0x12, tcr1);
- __t1_set_reg(wc, 0x13, tcr2);
- __t1_set_reg(wc, 0x14, ccr1);
- __t1_set_reg(wc, 0x18, 0x20); /* 120 Ohm */
-
-
- #if 0 /* XXX Does LBO Matter? XXX */
- /* Set outgoing LBO */
- __t1_set_reg(wc, 0x7c, wc->span.txlevel << 5);
- #endif
- printk(KERN_DEBUG "Using %s/%s coding/framing%s 120 Ohms\n", coding, framing,crcing);
- if (!alreadyrunning) {
-
- __t1_set_reg(wc,0x1b,0x8a); /* CCR3: LIRST & TSCLKM */
- __t1_set_reg(wc,0x20,0x1b); /* TAFR */
- __t1_set_reg(wc,0x21,0x5f); /* TNAFR */
- __t1_set_reg(wc,0x40,0xb); /* TSR1 */
- for(i = 0x41; i <= 0x4f; i++) __t1_set_reg(wc,i,0x55);
- for(i = 0x22; i <= 0x25; i++) __t1_set_reg(wc,i,0xff);
- spin_unlock_irqrestore(&wc->lock, flags);
- /* Wait 100ms to give plenty of time for reset */
- endjiffies = jiffies + 10;
- while(endjiffies < jiffies);
- spin_lock_irqsave(&wc->lock, flags);
-
- __t1_set_reg(wc, 0x1b, 0x9a); /* Set ESR */
- __t1_set_reg(wc, 0x1b, 0x82); /* TSCLKM only now */
- /* Reset LIRST bit and reset elastic stores */
- wc->span.flags |= DAHDI_FLAG_RUNNING;
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- }
- static int t1xxp_framer_sanity_check(struct t1xxp *wc)
- {
- int res;
- int chipid;
- unsigned long flags;
- int x;
- /* Sanity check */
- spin_lock_irqsave(&wc->lock, flags);
- for (x=0x0;x<192;x++)
- __t1_set_reg(wc, x, 0);
- res = __t1_get_reg(wc, 0x0f);
- res = __t1_get_reg(wc, 0x0f);
- chipid = ((res & 0x80) >> 5) | ((res & 0x30) >> 4);
- wc->ise1 = (res & 0x80) ? (1 << 4) : 0;
- spin_unlock_irqrestore(&wc->lock, flags);
- printk(KERN_DEBUG "Framer: %s, Revision: %d (%s)\n", chips[chipid], res & 0xf, wc->ise1 ? "E1" : "T1");
- return 0;
- }
- static int t1xxp_framer_hard_reset(struct t1xxp *wc)
- {
- int x;
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- /* Initialize all registers to 0 */
- for (x=0x0;x<192;x++)
- __t1_set_reg(wc, x, 0);
- if (wc->ise1) {
- /* Set LOTCMC (switch to RCLCK if TCLK fails) */
- __t1_set_reg(wc, 0x1a, 0x04);
-
- /* RSYNC is an input */
- __t1_set_reg(wc, 0x10, 0x20);
- /* Rx elastic store enabled, 2.048 Mhz (in theory) */
- __t1_set_reg(wc, 0x11, 0x06);
- /* TSYNC is an input, Tsis mode */
- __t1_set_reg(wc, 0x12, 0x08);
- /* Tx elastic store enabled, 2.048 Mhz (in theory) */
- __t1_set_reg(wc, 0x1b, 0x82);
-
- } else {
- /* Full-on sync required for T1 */
- __t1_set_reg(wc, 0x2b, 0x08);
- /* RSYNC is an input */
- __t1_set_reg(wc, 0x2c, 0x08);
- /* Enable tx RBS bits */
- __t1_set_reg(wc, 0x35, 0x10);
- /* TSYNC is output */
- __t1_set_reg(wc, 0x36, 0x04);
- /* Tx and Rx elastic store enabled, 2.048 Mhz (in theory) */
- __t1_set_reg(wc, 0x37, 0x9c);
- /* Setup Loopup / Loopdown codes */
- __t1_set_reg(wc, 0x12, 0x22);
- __t1_set_reg(wc, 0x14, 0x80);
- __t1_set_reg(wc, 0x15, 0x80);
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- return 0;
- }
- static int t1xxp_rbsbits(struct dahdi_chan *chan, int bits)
- {
- struct t1xxp *wc = chan->pvt;
- unsigned long flags;
- int b,o;
- unsigned char mask;
-
- /* Byte offset */
- spin_lock_irqsave(&wc->lock, flags);
- if (wc->ise1) {
- if (chan->chanpos < 16) {
- mask = ((bits << 4) | wc->chans[chan->chanpos - 1 + 16]->txsig);
- __t1_set_reg(wc, 0x40 + chan->chanpos, mask);
- }
- else if (chan->chanpos > 16) {
- mask = (bits | (wc->chans[chan->chanpos - 1 - 16]->txsig << 4));
- __t1_set_reg(wc, 0x40 + chan->chanpos - 16, mask);
- }
- wc->chans[chan->chanpos - 1]->txsig = bits;
- } else {
- b = (chan->chanpos - 1) / 8;
- o = (chan->chanpos - 1) % 8;
- mask = (1 << o);
- if (bits & DAHDI_ABIT) {
- /* Set A-bit */
- wc->txsiga[b] |= mask;
- } else {
- /* Clear A-bit */
- wc->txsiga[b] &= ~mask;
- }
- if (bits & DAHDI_BBIT) {
- /* Set B-bit */
- wc->txsigb[b] |= mask;
- } else {
- wc->txsigb[b] &= ~mask;
- }
- /* Output new values */
- __t1_set_reg(wc, 0x70 + b, wc->txsiga[b]);
- __t1_set_reg(wc, 0x73 + b, wc->txsigb[b]);
- __t1_set_reg(wc, 0x76 + b, wc->txsiga[b]);
- __t1_set_reg(wc, 0x79 + b, wc->txsigb[b]);
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- return 0;
- }
- static int t1xxp_ioctl(struct dahdi_chan *chan, unsigned int cmd, unsigned long data)
- {
- switch(cmd) {
- default:
- return -ENOTTY;
- }
- }
- static inline struct t1xxp *t1xxp_from_span(struct dahdi_span *span)
- {
- return container_of(span, struct t1xxp, span);
- }
- static int t1xxp_startup(struct dahdi_span *span)
- {
- struct t1xxp *wc = t1xxp_from_span(span);
- int i,alreadyrunning = span->flags & DAHDI_FLAG_RUNNING;
- /* initialize the start value for the entire chunk of last ec buffer */
- for(i = 0; i < span->channels; i++)
- {
- memset(wc->ec_chunk1[i],
- DAHDI_LIN2X(0,span->chans[i]),DAHDI_CHUNKSIZE);
- memset(wc->ec_chunk2[i],
- DAHDI_LIN2X(0,span->chans[i]),DAHDI_CHUNKSIZE);
- }
- /* Reset framer with proper parameters and start */
- if (wc->ise1)
- t1xxp_e1_framer_start(wc);
- else
- t1xxp_t1_framer_start(wc);
- printk(KERN_INFO "Calling startup (flags is %lu)\n", span->flags);
- if (!alreadyrunning) {
- /* Only if we're not already going */
- t1xxp_enable_interrupts(wc);
- t1xxp_start_dma(wc);
- span->flags |= DAHDI_FLAG_RUNNING;
- }
- return 0;
- }
- static int t1xxp_shutdown(struct dahdi_span *span)
- {
- struct t1xxp *wc = t1xxp_from_span(span);
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- __t1xxp_stop_dma(wc);
- __t1xxp_disable_interrupts(wc);
- span->flags &= ~DAHDI_FLAG_RUNNING;
- spin_unlock_irqrestore(&wc->lock, flags);
- t1xxp_framer_hard_reset(wc);
- return 0;
- }
- static int t1xxp_maint(struct dahdi_span *span, int cmd)
- {
- struct t1xxp *wc = t1xxp_from_span(span);
- int res = 0;
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- if (wc->ise1) {
- switch(cmd) {
- case DAHDI_MAINT_NONE:
- __t1_set_reg(wc,0xa8,0); /* no loops */
- break;
- case DAHDI_MAINT_LOCALLOOP:
- __t1_set_reg(wc,0xa8,0x40); /* local loop */
- break;
- case DAHDI_MAINT_REMOTELOOP:
- __t1_set_reg(wc,0xa8,0x80); /* remote loop */
- break;
- case DAHDI_MAINT_LOOPUP:
- case DAHDI_MAINT_LOOPDOWN:
- case DAHDI_MAINT_LOOPSTOP:
- res = -ENOSYS;
- break;
- default:
- printk(KERN_NOTICE "wct1xxp/E1: Unknown maint command: %d\n", cmd);
- res = -EINVAL;
- break;
- }
- } else {
- switch(cmd) {
- case DAHDI_MAINT_NONE:
- __t1_set_reg(wc,0x19,0); /* no local loop */
- __t1_set_reg(wc,0x0a,0); /* no remote loop */
- break;
- case DAHDI_MAINT_LOCALLOOP:
- __t1_set_reg(wc,0x19,0x40); /* local loop */
- __t1_set_reg(wc,0x0a,0); /* no remote loop */
- break;
- case DAHDI_MAINT_REMOTELOOP:
- __t1_set_reg(wc,0x1e,0); /* no local loop */
- __t1_set_reg(wc,0x0a,0x40); /* remote loop */
- break;
- case DAHDI_MAINT_LOOPUP:
- __t1_set_reg(wc,0x30,2); /* send loopup code */
- __t1_set_reg(wc,0x12,0x22); /* send loopup code */
- __t1_set_reg(wc,0x13,0x80); /* send loopup code */
- break;
- case DAHDI_MAINT_LOOPDOWN:
- __t1_set_reg(wc,0x30,2); /* send loopdown code */
- __t1_set_reg(wc,0x12,0x62); /* send loopdown code */
- __t1_set_reg(wc,0x13,0x90); /* send loopdown code */
- break;
- case DAHDI_MAINT_LOOPSTOP:
- __t1_set_reg(wc,0x30,0); /* stop sending loopup code */
- break;
- default:
- printk(KERN_NOTICE "wct1xxp/T1: Unknown maint command: %d\n", cmd);
- res = -EINVAL;
- }
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- return res;
- }
- static int t1xxp_chanconfig(struct dahdi_chan *chan, int sigtype)
- {
- struct t1xxp *wc = chan->pvt;
- unsigned long flags;
- int alreadyrunning = chan->span->flags & DAHDI_FLAG_RUNNING;
- spin_lock_irqsave(&wc->lock, flags);
- if (alreadyrunning && !wc->ise1)
- __t1xxp_set_clear(wc);
- spin_unlock_irqrestore(&wc->lock, flags);
- return 0;
- }
- static int t1xxp_spanconfig(struct dahdi_span *span, struct dahdi_lineconfig *lc)
- {
- struct t1xxp *wc = t1xxp_from_span(span);
- /* Do we want to SYNC on receive or not */
- wc->sync = (lc->sync) ? 1 : 0;
- /* If already running, apply changes immediately */
- if (span->flags & DAHDI_FLAG_RUNNING)
- return t1xxp_startup(span);
- return 0;
- }
- static const struct dahdi_span_ops t1xxp_span_ops = {
- .owner = THIS_MODULE,
- .spanconfig = t1xxp_spanconfig,
- .chanconfig = t1xxp_chanconfig,
- .startup = t1xxp_startup,
- .shutdown = t1xxp_shutdown,
- .rbsbits = t1xxp_rbsbits,
- .maint = t1xxp_maint,
- .open = t1xxp_open,
- .close = t1xxp_close,
- .ioctl = t1xxp_ioctl,
- };
- static int t1xxp_software_init(struct t1xxp *wc)
- {
- int x;
- /* Find position */
- for (x=0;x<WC_MAX_CARDS;x++) {
- if (!cards[x]) {
- cards[x] = wc;
- break;
- }
- }
- if (x >= WC_MAX_CARDS)
- return -1;
- wc->num = x;
- sprintf(wc->span.name, "WCT1/%d", wc->num);
- snprintf(wc->span.desc, sizeof(wc->span.desc) - 1, "%s Card %d", wc->variety, wc->num);
- wc->span.manufacturer = "Digium";
- dahdi_copy_string(wc->span.devicetype, wc->variety, sizeof(wc->span.devicetype));
- snprintf(wc->span.location, sizeof(wc->span.location) - 1,
- "PCI Bus %02d Slot %02d", wc->dev->bus->number, PCI_SLOT(wc->dev->devfn) + 1);
- wc->span.irq = wc->dev->irq;
- wc->span.chans = wc->chans;
- wc->span.flags = DAHDI_FLAG_RBS;
- if (wc->ise1) {
- wc->span.channels = 31;
- wc->span.deflaw = DAHDI_LAW_ALAW;
- wc->span.linecompat = DAHDI_CONFIG_HDB3 | DAHDI_CONFIG_CCS | DAHDI_CONFIG_CRC4;
- wc->span.spantype = "E1";
- } else {
- wc->span.channels = 24;
- wc->span.deflaw = DAHDI_LAW_MULAW;
- wc->span.linecompat = DAHDI_CONFIG_AMI | DAHDI_CONFIG_B8ZS | DAHDI_CONFIG_D4 | DAHDI_CONFIG_ESF;
- wc->span.spantype = "T1";
- }
- init_waitqueue_head(&wc->span.maintq);
- for (x=0;x<wc->span.channels;x++) {
- sprintf(wc->chans[x]->name, "WCT1/%d/%d", wc->num, x + 1);
- wc->chans[x]->sigcap = DAHDI_SIG_EM | DAHDI_SIG_CLEAR | DAHDI_SIG_EM_E1 |
- DAHDI_SIG_FXSLS | DAHDI_SIG_FXSGS |
- DAHDI_SIG_FXSKS | DAHDI_SIG_FXOLS | DAHDI_SIG_DACS_RBS |
- DAHDI_SIG_FXOGS | DAHDI_SIG_FXOKS | DAHDI_SIG_CAS | DAHDI_SIG_SF;
- wc->chans[x]->pvt = wc;
- wc->chans[x]->chanpos = x + 1;
- }
- wc->span.ops = &t1xxp_span_ops;
- if (dahdi_register(&wc->span, 0)) {
- printk(KERN_NOTICE "Unable to register span with DAHDI\n");
- return -1;
- }
- return 0;
- }
- static inline void __handle_leds(struct t1xxp *wc)
- {
- int oldreg;
- wc->blinktimer++;
- if (wc->span.alarms & (DAHDI_ALARM_RED | DAHDI_ALARM_BLUE)) {
- /* Red/Blue alarm */
- #ifdef FANCY_ALARM
- if (wc->blinktimer == (altab[wc->alarmpos] >> 1)) {
- wc->ledtestreg = (wc->ledtestreg | BIT_LED1) & ~BIT_LED0;
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- }
- if (wc->blinktimer == 0xf) {
- wc->ledtestreg = wc->ledtestreg & ~(BIT_LED0 | BIT_LED1);
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- wc->blinktimer = -1;
- wc->alarmpos++;
- if (wc->alarmpos >= (sizeof(altab) / sizeof(altab[0])))
- wc->alarmpos = 0;
- }
- #else
- if (wc->blinktimer == 160) {
- wc->ledtestreg = (wc->ledtestreg | BIT_LED1) & ~BIT_LED0;
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- } else if (wc->blinktimer == 480) {
- wc->ledtestreg = wc->ledtestreg & ~(BIT_LED0 | BIT_LED1);
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- wc->blinktimer = 0;
- }
- #endif
- } else if (wc->span.alarms & DAHDI_ALARM_YELLOW) {
- /* Yellow Alarm */
- if (!(wc->blinktimer % 2))
- wc->ledtestreg = (wc->ledtestreg | BIT_LED1) & ~BIT_LED0;
- else
- wc->ledtestreg = (wc->ledtestreg | BIT_LED0) & ~BIT_LED1;
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- } else {
- /* No Alarm */
- oldreg = wc->ledtestreg;
- if (wc->span.maintstat != DAHDI_MAINT_NONE)
- wc->ledtestreg |= BIT_TEST;
- else
- wc->ledtestreg &= ~BIT_TEST;
- if (wc->span.flags & DAHDI_FLAG_RUNNING)
- wc->ledtestreg = (wc->ledtestreg | BIT_LED0) & ~BIT_LED1;
- else
- wc->ledtestreg = wc->ledtestreg & ~(BIT_LED0 | BIT_LED1);
- if (oldreg != wc->ledtestreg)
- __control_set_reg(wc, WC_LEDTEST, wc->ledtestreg);
- }
- }
- static void t1xxp_transmitprep(struct t1xxp *wc, int ints)
- {
- volatile unsigned char *txbuf;
- int x,y;
- int pos;
- if (ints & 0x04 /* 0x01 */) {
- /* We just finished sending the first buffer, start filling it
- now */
- txbuf = wc->writechunk;
- } else {
- /* Just finished sending second buffer, fill it now */
- txbuf = wc->writechunk + 32 * DAHDI_CHUNKSIZE;
- }
- dahdi_transmit(&wc->span);
- for (x=0;x<wc->offset;x++)
- txbuf[x] = wc->tempo[x];
- for (y=0;y<DAHDI_CHUNKSIZE;y++) {
- for (x=0;x<wc->span.channels;x++) {
- pos = y * 32 + wc->chanmap[x] + wc->offset;
- /* Put channel number as outgoing data */
- if (pos < 32 * DAHDI_CHUNKSIZE)
- txbuf[pos] = wc->chans[x]->writechunk[y];
- else
- wc->tempo[pos - 32 * DAHDI_CHUNKSIZE] = wc->chans[x]->writechunk[y];
- }
- }
- }
- static void t1xxp_receiveprep(struct t1xxp *wc, int ints)
- {
- volatile unsigned char *rxbuf;
- volatile unsigned int *canary;
- int x;
- int y;
- unsigned int oldcan;
- if (ints & 0x04) {
- /* Just received first buffer */
- rxbuf = wc->readchunk;
- canary = (unsigned int *)(wc->readchunk + DAHDI_CHUNKSIZE * 64 - 4);
- } else {
- rxbuf = wc->readchunk + DAHDI_CHUNKSIZE * 32;
- canary = (unsigned int *)(wc->readchunk + DAHDI_CHUNKSIZE * 32 - 4);
- }
- oldcan = *canary;
- if (((oldcan & 0xffff0000) >> 16) != CANARY) {
- /* Check top part */
- if (debug) printk(KERN_DEBUG "Expecting top %04x, got %04x\n", CANARY, (oldcan & 0xffff0000) >> 16);
- wc->span.irqmisses++;
- } else if ((oldcan & 0xffff) != ((wc->canary - 1) & 0xffff)) {
- if (debug) printk(KERN_DEBUG "Expecting bottom %d, got %d\n", wc->canary - 1, oldcan & 0xffff);
- wc->span.irqmisses++;
- }
- for (y=0;y<DAHDI_CHUNKSIZE;y++) {
- for (x=0;x<wc->span.channels;x++) {
- /* XXX Optimize, remove * and + XXX */
- /* Must map received channels into appropriate data */
- wc->chans[x]->readchunk[y] =
- rxbuf[32 * y + ((wc->chanmap[x] + WC_OFFSET + wc->offset) & 0x1f)];
- }
- if (!wc->ise1) {
- for (x=3;x<32;x+=4) {
- if (rxbuf[32 * y + ((x + WC_OFFSET) & 0x1f)] == 0x7f) {
- if (wc->offset != (x-3)) {
- /* Resync */
- control_set_reg(wc, WC_CLOCK, 0x02 | wc->sync | wc->ise1);
- wc->clocktimeout = 100;
- #if 1
- if (debug) printk(KERN_DEBUG "T1: Lost our place, resyncing\n");
- #endif
- }
- }
- }
- } else {
- if (!wc->clocktimeout && !wc->span.alarms) {
- if ((rxbuf[32 * y + ((3 + WC_OFFSET + wc->offset) & 0x1f)] & 0x7f) != 0x1b) {
- if (wc->miss) {
- if (debug) printk(KERN_DEBUG "Double miss (%d, %d)...\n", wc->misslast, rxbuf[32 * y + ((3 + WC_OFFSET + wc->offset) & 0x1f)]);
- control_set_reg(wc, WC_CLOCK, 0x02 | wc->sync | wc->ise1);
- wc->clocktimeout = 100;
- } else {
- wc->miss = 1;
- wc->misslast = rxbuf[32 * y + ((3 + WC_OFFSET + wc->offset) & 0x1f)];
- }
- } else {
- wc->miss = 0;
- }
- } else {
- wc->miss = 0;
- }
- }
- }
- /* Store the next canary */
- canary = (unsigned int *)(rxbuf + DAHDI_CHUNKSIZE * 32 - 4);
- *canary = (wc->canary++) | (CANARY << 16);
- for (x=0;x<wc->span.channels;x++) {
- dahdi_ec_chunk(wc->chans[x], wc->chans[x]->readchunk, wc->ec_chunk2[x]);
- memcpy(wc->ec_chunk2[x],wc->ec_chunk1[x],DAHDI_CHUNKSIZE);
- memcpy(wc->ec_chunk1[x],wc->chans[x]->writechunk,DAHDI_CHUNKSIZE);
- }
- dahdi_receive(&wc->span);
- }
- static void t1xxp_check_sigbits(struct t1xxp *wc, int x)
- {
- int a,b,i,y,rxs;
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- if (wc->ise1) {
- /* Read 5 registers at a time, loading 10 channels at a time */
- for (i = (x * 5); i < (x * 5) + 5; i++) {
- a = __t1_get_reg(wc, 0x31 + i);
- /* Get high channel in low bits */
- rxs = (a & 0xf);
- if (!(wc->chans[i+16]->sig & DAHDI_SIG_CLEAR)) {
- if (wc->chans[i+16]->rxsig != rxs) {
- spin_unlock_irqrestore(&wc->lock, flags);
- dahdi_rbsbits(wc->chans[i+16], rxs);
- spin_lock_irqsave(&wc->lock, flags);
- }
- }
- rxs = (a >> 4) & 0xf;
- if (!(wc->chans[i]->sig & DAHDI_SIG_CLEAR)) {
- if (wc->chans[i]->rxsig != rxs) {
- spin_unlock_irqrestore(&wc->lock, flags);
- dahdi_rbsbits(wc->chans[i], rxs);
- spin_lock_irqsave(&wc->lock, flags);
- }
- }
- }
- } else {
- a = __t1_get_reg(wc, 0x60 + x);
- b = __t1_get_reg(wc, 0x63 + x);
- for (y=0;y<8;y++) {
- i = x * 8 + y;
- rxs = 0;
- if (a & (1 << y))
- rxs |= DAHDI_ABIT;
- if (b & (1 << y))
- rxs |= DAHDI_BBIT;
- if (!(wc->chans[i]->sig & DAHDI_SIG_CLEAR)) {
- if (wc->chans[i]->rxsig != rxs) {
- spin_unlock_irqrestore(&wc->lock, flags);
- dahdi_rbsbits(wc->chans[i], rxs);
- spin_lock_irqsave(&wc->lock, flags);
- }
- }
- }
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- }
- static void t1xxp_check_alarms(struct t1xxp *wc)
- {
- unsigned char c,d;
- int alarms;
- int x,j;
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- if (wc->ise1) {
- __t1_set_reg(wc, 0x06, 0xff);
- c = __t1_get_reg(wc, 0x6);
- } else {
- /* Get RIR2 */
- c = __t1_get_reg(wc, 0x31);
- wc->span.rxlevel = c >> 6;
-
- /* Get status register s*/
- __t1_set_reg(wc, 0x20, 0xff);
- c = __t1_get_reg(wc, 0x20);
- }
- /* Assume no alarms */
- alarms = 0;
- /* And consider only carrier alarms */
- wc->span.alarms &= (DAHDI_ALARM_RED | DAHDI_ALARM_BLUE | DAHDI_ALARM_NOTOPEN);
- if (wc->ise1) {
- /* XXX Implement me XXX */
- } else {
- /* Detect loopup code if we're not sending one */
- if ((!wc->span.mainttimer) && (c & 0x80)) {
- /* Loop-up code detected */
- if ((wc->loopupcnt++ > 80) && (wc->span.maintstat != DAHDI_MAINT_REMOTELOOP)) {
- __t1_set_reg(wc, 0x1e, 0); /* No local loop */
- __t1_set_reg(wc, 0x0a, 0x40); /* Remote Loop */
- wc->span.maintstat = DAHDI_MAINT_REMOTELOOP;
- }
- } else {
- wc->loopupcnt = 0;
- }
- /* Same for loopdown code */
- if ((!wc->span.mainttimer) && (c & 0x40)) {
- /* Loop-down code detected */
- if ((wc->loopdowncnt++ > 80) && (wc->span.maintstat == DAHDI_MAINT_REMOTELOOP)) {
- __t1_set_reg(wc, 0x1e, 0); /* No local loop */
- __t1_set_reg(wc, 0x0a, 0x0); /* No remote Loop */
- wc->span.maintstat = DAHDI_MAINT_NONE;
- }
- } else
- wc->loopdowncnt = 0;
- }
- if (wc->span.lineconfig & DAHDI_CONFIG_NOTOPEN) {
- for (x=0,j=0;x < wc->span.channels;x++)
- if ((wc->chans[x]->flags & DAHDI_FLAG_OPEN) ||
- (wc->chans[x]->flags & DAHDI_FLAG_NETDEV))
- j++;
- if (!j)
- alarms |= DAHDI_ALARM_NOTOPEN;
- }
- if (wc->ise1) {
- if (c & 0x9)
- alarms |= DAHDI_ALARM_RED;
- if (c & 0x2)
- alarms |= DAHDI_ALARM_BLUE;
- } else {
- /* Check actual alarm status */
- if (c & 0x3)
- alarms |= DAHDI_ALARM_RED;
- if (c & 0x8)
- alarms |= DAHDI_ALARM_BLUE;
- }
- /* Keep track of recovering */
- if ((!alarms) && wc->span.alarms)
- wc->alarmtimer = DAHDI_ALARMSETTLE_TIME;
- /* If receiving alarms, go into Yellow alarm state */
- if (alarms && (!wc->span.alarms)) {
- #if 0
- printk(KERN_DEBUG "Going into yellow alarm\n");
- #endif
- if (wc->ise1)
- __t1_set_reg(wc, 0x21, 0x7f);
- else
- __t1_set_reg(wc, 0x35, 0x11);
- }
- if (wc->span.alarms != alarms) {
- d = __control_get_reg(wc, WC_CLOCK);
- start_alarm(wc);
- if (!(alarms & (DAHDI_ALARM_RED | DAHDI_ALARM_BLUE | DAHDI_ALARM_LOOPBACK)) &&
- wc->sync) {
- /* Use the receive signalling */
- wc->span.syncsrc = wc->span.spanno;
- d |= 1;
- } else {
- wc->span.syncsrc = 0;
- d &= ~1;
- }
- __control_set_reg(wc, WC_CLOCK, d);
- }
- if (wc->alarmtimer)
- alarms |= DAHDI_ALARM_RECOVER;
- if (c & 0x4)
- alarms |= DAHDI_ALARM_YELLOW;
- wc->span.alarms = alarms;
- spin_unlock_irqrestore(&wc->lock, flags);
- dahdi_alarm_notify(&wc->span);
- }
- static void t1xxp_do_counters(struct t1xxp *wc)
- {
- unsigned long flags;
- spin_lock_irqsave(&wc->lock, flags);
- if (wc->alarmtimer) {
- if (!--wc->alarmtimer) {
- wc->span.alarms &= ~(DAHDI_ALARM_RECOVER);
- /* Clear yellow alarm */
- #if 0
- printk(KERN_DEBUG "Coming out of alarm\n");
- #endif
- if (wc->ise1)
- __t1_set_reg(wc, 0x21, 0x5f);
- else
- __t1_set_reg(wc, 0x35, 0x10);
- spin_unlock_irqrestore(&wc->lock, flags);
- dahdi_alarm_notify(&wc->span);
- spin_lock_irqsave(&wc->lock, flags);
- }
- }
- spin_unlock_irqrestore(&wc->lock, flags);
- }
- DAHDI_IRQ_HANDLER(t1xxp_interrupt)
- {
- struct t1xxp *wc = dev_id;
- unsigned char ints;
- unsigned long flags;
- int x;
- ints = inb(wc->ioaddr + WC_INTSTAT);
- if (!ints)
- return IRQ_NONE;
- outb(ints, wc->ioaddr + WC_INTSTAT);
- if (!wc->intcount) {
- if (debug) printk(KERN_DEBUG "Got interrupt: 0x%04x\n", ints);
- }
- wc->intcount++;
- if (wc->clocktimeout && !--wc->clocktimeout)
- control_set_reg(wc, WC_CLOCK, 0x00 | wc->sync | wc->ise1);
- if (ints & 0x0f) {
- t1xxp_receiveprep(wc, ints);
- t1xxp_transmitprep(wc, ints);
- }
- spin_lock_irqsave(&wc->lock, flags);
- #if 1
- __handle_leds(wc);
- #endif
- spin_unlock_irqrestore(&wc->lock, flags);
- /* Count down timers */
- t1xxp_do_counters(wc);
- /* Do some things that we don't have to do very often */
- x = wc->intcount & 15 /* 63 */;
- switch(x) {
- case 0:
- case 1:
- case 2:
- t1xxp_check_sigbits(wc, x);
- break;
- case 4:
- /* Check alarms 1/4 as frequently */
- if (!(wc->intcount & 0x30))
- t1xxp_check_alarms(wc);
- break;
- }
-
- if (ints & 0x10)
- printk(KERN_INFO "PCI Master abort\n");
- if (ints & 0x20)
- printk(KERN_INFO "PCI Target abort\n");
- return IRQ_RETVAL(1);
- }
- static int t1xxp_hardware_init(struct t1xxp *wc)
- {
- /* Hardware PCI stuff */
- /* Reset chip and registers */
- outb(DELAY | 0x0e, wc->ioaddr + WC_CNTL);
- /* Set all outputs to 0 */
- outb(0x00, wc->ioaddr + WC_AUXD);
- /* Set all to outputs except AUX1 (TDO). */
- outb(0xfd, wc->ioaddr + WC_AUXC);
- /* Configure the serial port: double clock, 20ns width, no inversion,
- MSB first */
- outb(0xc8, wc->ioaddr + WC_SERC);
- /* Internally delay FSC by one */
- outb(0x01, wc->ioaddr + WC_FSCDELAY);
- /* Back to normal, with automatic DMA wrap around */
- outb(DELAY | 0x01, wc->ioaddr + WC_CNTL);
-
- /* Make sure serial port and DMA are out of reset */
- outb(inb(wc->ioaddr + WC_CNTL) & 0xf9, WC_CNTL);
-
- /* Setup DMA Addresses */
- /* Start at writedma */
- outl(wc->writedma, wc->ioaddr + WC_DMAWS); /* Write start */
- /* First frame */
- outl(wc->writedma + DAHDI_CHUNKSIZE * 32 - 4, wc->ioaddr + WC_DMAWI); /* Middle (interrupt) */
- /* Second frame */
- outl(wc->writedma + DAHDI_CHUNKSIZE * 32 * 2 - 4, wc->ioaddr + WC_DMAWE); /* End */
-
- outl(wc->readdma, wc->ioaddr + WC_DMARS); /* Read start */
- /* First frame */
- outl(wc->readdma + DAHDI_CHUNKSIZE * 32 - 4, wc->ioaddr + WC_DMARI); /* Middle (interrupt) */
- /* Second frame */
- outl(wc->readdma + DAHDI_CHUNKSIZE * 32 * 2 - 4, wc->ioaddr + WC_DMARE); /* End */
-
- if (debug) printk(KERN_DEBUG "Setting up DMA (write/read = %08lx/%08lx)\n", (long)wc->writedma, (long)wc->readdma);
- /* Check out the controller */
- if (debug) printk(KERN_DEBUG "Controller version: %02x\n", control_get_reg(wc, WC_VERSION));
- control_set_reg(wc, WC_LEDTEST, 0x00);
- /* Sanity check also determines e1 or t1 */
- if (t1xxp_framer_sanity_check(wc))
- return -1;
- if (wc->ise1)
- wc->chanmap = chanmap_e1;
- else
- wc->chanmap = chanmap_t1;
- /* Setup clock appropriately */
- control_set_reg(wc, WC_CLOCK, 0x02 | wc->sync | wc->ise1);
- wc->clocktimeout = 100;
-
- /* Reset the T1 and report */
- t1xxp_framer_hard_reset(wc);
- start_alarm(wc);
- return 0;
- }
- static int __devinit t1xxp_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
- {
- struct t1xxp *wc;
- unsigned int *canary;
- unsigned int x;
-
- if (pci_enable_device(pdev)) {
- printk(KERN_ERR "%s: pci_enable_device failed\n", __FUNCTION__);
- return -EIO;
- }
- if (!(wc = kmalloc(sizeof(*wc), GFP_KERNEL))) {
- printk(KERN_ERR "%s: Failed allocation a wc\n", __FUNCTION__);
- return -ENOMEM;
- }
- memset(wc, 0x0, sizeof(*wc));
- spin_lock_init(&wc->lock);
- wc->ioaddr = pci_resource_start(pdev, 0);
- wc->dev = pdev;
- wc->offset = 28; /* And you thought 42 was the answer */
-
- wc->writechunk =
- /* 32 channels, Double-buffer, Read/Write */
- (unsigned char *)pci_alloc_consistent(pdev, DAHDI_MAX_CHUNKSIZE * 32 * 2 * 2, &wc->writedma);
- if (!wc->writechunk) {
- printk(KERN_NOTICE "wct1xxp: Unable to allocate DMA-able memory\n");
- return -ENOMEM;
- }
-
- /* Read is after the whole write piece (in bytes) */
- wc->readchunk = wc->writechunk + DAHDI_CHUNKSIZE * 32 * 2;
-
- /* Same thing... */
- wc->readdma = wc->writedma + DAHDI_CHUNKSIZE * 32 * 2;
-
- /* Initialize Write/Buffers to all blank data */
- memset((void *)wc->writechunk,0x00,DAHDI_MAX_CHUNKSIZE * 2 * 2 * 32);
- /* Initialize canary */
- canary = (unsigned int *)(wc->readchunk + DAHDI_CHUNKSIZE * 64 - 4);
- *canary = (CANARY << 16) | (0xffff);
-
- /* Enable bus mastering */
- pci_set_master(pdev);
-
- /* Keep track of which device we are */
- pci_set_drvdata(pdev, wc);
-
- if (request_irq(pdev->irq, t1xxp_interrupt, DAHDI_IRQ_SHARED_DISABLED, "t1xxp", wc)) {
- printk(KERN_NOTICE "t1xxp: Unable to request IRQ %d\n", pdev->irq);
- kfree(wc);
- return -EIO;
- }
- /* Initialize hardware */
- t1xxp_hardware_init(wc);
-
- /* We now know which version of card we have */
- if (wc->ise1) {
- wc->variety = "Digium Wildcard E100P E1/PRA";
- } else {
- wc->variety = "Digium Wildcard T100P T1/PRI";
- }
- for (x = 0; x < (wc->ise1 ? 31 : 24); x++) {
- if (!(wc->chans[x] = kmalloc(sizeof(*wc->chans[x]), GFP_KERNEL))) {
- while (x) {
- kfree(wc->chans[--x]);
- }
- kfree(wc);
- return -ENOMEM;
- }
- memset(wc->chans[x], 0, sizeof(*wc->chans[x]));
- }
- /* Misc. software stuff */
- t1xxp_software_init(wc);
-
- printk(KERN_INFO "Found a Wildcard: %s\n", wc->variety);
- return 0;
- }
- static void t1xxp_stop_stuff(struct t1xxp *wc)
- {
- /* Kill clock */
- control_set_reg(wc, WC_CLOCK, 0);
- /* Turn off LED's */
- control_set_reg(wc, WC_LEDTEST, 0);
- /* Reset the T1 */
- t1xxp_framer_hard_reset(wc);
- }
- static void __devexit t1xxp_remove_one(struct pci_dev *pdev)
- {
- struct t1xxp *wc = pci_get_drvdata(pdev);
- if (wc) {
- /* Stop any DMA */
- __t1xxp_stop_dma(wc);
- /* In case hardware is still there */
- __t1xxp_disable_interrupts(wc);
-
- t1xxp_stop_stuff(wc);
- /* Immediately free resources */
- pci_free_consistent(pdev, DAHDI_MAX_CHUNKSIZE * 2 * 2 * 32 * 4, (void *)wc->writechunk, wc->writedma);
- free_irq(pdev->irq, wc);
- /* Reset PCI chip and registers */
- outb(DELAY | 0x0e, wc->ioaddr + WC_CNTL);
- /* Release span, possibly delayed */
- if (!wc->usecount)
- t1xxp_release(wc);
- else
- wc->dead = 1;
- }
- }
- static DEFINE_PCI_DEVICE_TABLE(t1xxp_pci_tbl) = {
- { 0xe159, 0x0001, 0x6159, PCI_ANY_ID, 0, 0, (unsigned long) "Digium Wildcard T100P T1/PRI or E100P E1/PRA Board" },
- { 0 }
- };
- MODULE_DEVICE_TABLE(pci,t1xxp_pci_tbl);
- static struct pci_driver t1xxp_driver = {
- .name = "t1xxp",
- .probe = t1xxp_init_one,
- .remove = __devexit_p(t1xxp_remove_one),
- .suspend = NULL,
- .resume = NULL,
- . id_table = t1xxp_pci_tbl,
- };
- static int __init t1xxp_init(void)
- {
- int res;
- res = dahdi_pci_module(&t1xxp_driver);
- if (res)
- return -ENODEV;
- return 0;
- }
- static void __exit t1xxp_cleanup(void)
- {
- pci_unregister_driver(&t1xxp_driver);
- }
- module_param(debug, int, 0600);
- MODULE_DESCRIPTION("Wildcard T100P/E100P Driver");
- MODULE_AUTHOR("Mark Spencer <markster@digium.com>");
- MODULE_LICENSE("GPL v2");
- module_init(t1xxp_init);
- module_exit(t1xxp_cleanup);
|