123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545 |
- /*
- * A4A,A4B,A8A,A8B TDM FXS/FXO Interface Driver for DAHDI Telephony interface
- *
- * Copyright (C) 2013 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.
- */
- #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- #include <linux/version.h>
- #include <linux/kernel.h>
- #include <linux/errno.h>
- #include <linux/module.h>
- #include <linux/init.h>
- #include <linux/pci.h>
- #include <linux/sched.h>
- #include <linux/workqueue.h>
- #include <linux/delay.h>
- #include <linux/moduleparam.h>
- #include <linux/firmware.h>
- #include <linux/crc32.h>
- #include <stdbool.h>
- #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 30)
- /* Define this if you would like to load the modules in parallel. While this
- * can speed up loads when multiple cards handled by this driver are installed,
- * it also makes it impossible to abort module loads with ctrl-c */
- #undef USE_ASYNC_INIT
- #include <linux/async.h>
- #else
- #undef USE_ASYNC_INIT
- #endif
- #include <dahdi/kernel.h>
- #include <dahdi/wctdm_user.h>
- #include "proslic.h"
- #include <dahdi/kernel.h>
- #include <linux/version.h>
- #include <linux/mutex.h>
- #include <oct612x.h>
- #include "wcxb.h"
- #include "wcxb_spi.h"
- #include "wcxb_flash.h"
- /*!
- * \brief Default ringer debounce (in ms)
- */
- #define DEFAULT_RING_DEBOUNCE 1024
- #define POLARITY_DEBOUNCE 64 /* Polarity debounce (in ms) */
- #define OHT_TIMER 6000 /* How long after RING to retain OHT */
- #define FLAG_EXPRESS (1 << 0)
- #define NUM_MODULES 8
- #define CMD_WR(addr, val) (((addr<<8)&0xff00) | (val&0xff))
- enum battery_state {
- BATTERY_UNKNOWN = 0,
- BATTERY_DEBOUNCING_PRESENT,
- BATTERY_DEBOUNCING_PRESENT_FROM_LOST_ALARM,
- BATTERY_DEBOUNCING_PRESENT_ALARM,
- BATTERY_PRESENT,
- BATTERY_DEBOUNCING_LOST,
- BATTERY_DEBOUNCING_LOST_FROM_PRESENT_ALARM,
- BATTERY_DEBOUNCING_LOST_ALARM,
- BATTERY_LOST,
- };
- enum ring_detector_state {
- RINGOFF = 0,
- DEBOUNCING_RINGING_POSITIVE,
- DEBOUNCING_RINGING_NEGATIVE,
- RINGING,
- DEBOUNCING_RINGOFF,
- };
- enum polarity_state {
- UNKNOWN_POLARITY = 0,
- POLARITY_DEBOUNCE_POSITIVE,
- POLARITY_POSITIVE,
- POLARITY_DEBOUNCE_NEGATIVE,
- POLARITY_NEGATIVE,
- };
- struct wcaxx_chan {
- struct dahdi_chan chan;
- struct dahdi_echocan_state ec;
- int timeslot;
- unsigned int hwpreec_enabled:1;
- };
- struct fxo {
- enum ring_detector_state ring_state:4;
- enum battery_state battery_state:4;
- enum polarity_state polarity_state:4;
- u8 ring_polarity_change_count:4;
- u8 hook_ring_shadow;
- s8 line_voltage_status;
- int offhook;
- int neonmwi_state;
- int neonmwi_last_voltage;
- unsigned int neonmwi_debounce;
- unsigned int neonmwi_offcounter;
- unsigned long display_fxovoltage;
- unsigned long ringdebounce_timer;
- unsigned long battdebounce_timer;
- unsigned long poldebounce_timer;
- };
- struct fxs {
- int idletxhookstate; /* IDLE changing hook state */
- /* lasttxhook reflects the last value written to the proslic's reg 64
- * (LINEFEED_CONTROL) in bits 0-2. Bit 4 indicates if the last write is
- * pending i.e. it is in process of being written to the register NOTE:
- * in order for this value to actually be written to the proslic, the
- * appropriate matching value must be written into the sethook variable
- * so that it gets queued and handled by the voicebus ISR.
- */
- u8 lasttxhook;
- u8 linefeed_control_shadow;
- u8 hook_state_shadow;
- u8 oht_active:1;
- u8 off_hook:1;
- int palarms;
- struct dahdi_vmwi_info vmwisetting;
- int vmwi_active_messages;
- int vmwi_linereverse;
- int reversepolarity; /* polarity reversal */
- struct {
- u8 vals[12];
- } calregs;
- unsigned long check_alarm;
- unsigned long check_proslic;
- unsigned long oppending_timeout;
- unsigned long ohttimer;
- };
- #define fxs_lf(fxs, value) _fxs_lf((fxs), SLIC_LF_##value)
- static inline bool _fxs_lf(const struct fxs *fxs, const unsigned value)
- {
- return (fxs->lasttxhook & SLIC_LF_SETMASK) == value;
- }
- enum module_type {
- NONE = 0,
- FXS,
- FXO,
- };
- #define MODULE_POLL_TIME_MS 10
- struct wcaxx_mod_poll {
- struct wcxb_spi_message m;
- struct wcxb_spi_transfer t;
- struct wcaxx_module *mod;
- struct wcaxx *wc;
- u8 buffer[6];
- u8 master_buffer[6];
- };
- struct wcaxx_module {
- union modtypes {
- struct fxo fxo;
- struct fxs fxs;
- } mod;
- u8 card;
- u8 subaddr;
- enum module_type type;
- int sethook; /* pending hook state command */
- int dacssrc;
- struct wcxb_spi_device *spi;
- struct wcaxx_mod_poll *mod_poll;
- };
- struct _device_desc {
- const char *name;
- unsigned int ports;
- };
- static const struct _device_desc device_a8a = { "Wildcard A8A", 8};
- static const struct _device_desc device_a8b = { "Wildcard A8B", 8};
- static const struct _device_desc device_a4a = { "Wildcard A4A", 4};
- static const struct _device_desc device_a4b = { "Wildcard A4B", 4};
- struct wcaxx {
- const struct _device_desc *desc;
- const char *board_name;
- unsigned long framecount;
- unsigned long module_poll_time;
- int mods_per_board;
- spinlock_t reglock;
- struct wcaxx_module mods[NUM_MODULES];
- struct wcxb xb;
- struct dahdi_span span;
- struct wcaxx_chan *chans[NUM_MODULES];
- struct dahdi_echocan_state *ec[NUM_MODULES];
- int companding;
- struct dahdi_device *ddev;
- struct wcxb_spi_master *master;
- #define INITIALIZED 0
- unsigned long bit_flags;
- /* 4 SPI devices that are matched to the chip selects. The 4 port
- * modules will share a single SPI device since they use the same chip
- * select. */
- struct wcxb_spi_device *spi_devices[4];
- struct vpm450m *vpm;
- struct list_head card_node;
- u16 num;
- };
- static inline bool is_pcie(const struct wcaxx *wc)
- {
- return (wc->desc == &device_a8b) || (wc->desc == &device_a4b);
- }
- static inline bool is_four_port(const struct wcaxx *wc)
- {
- return (4 == wc->desc->ports);
- }
- #include <linux/slab.h>
- #include <linux/vmalloc.h>
- #include <linux/string.h>
- #include <linux/time.h>
- #include <linux/version.h>
- #include <linux/firmware.h>
- #include "oct6100api/oct6100_api.h"
- #define ECHOCAN_NUM_CHANS 8
- #define FLAG_DTMF (1 << 0)
- #define FLAG_MUTE (1 << 1)
- #define FLAG_ECHO (1 << 2)
- #define OCT_CHIP_ID 0
- #define OCT_MAX_TDM_STREAMS 4
- #define OCT_TONEEVENT_BUFFER_SIZE 128
- #define SOUT_STREAM 1
- #define RIN_STREAM 0
- #define SIN_STREAM 2
- static int vpmsupport = 1;
- static int wcaxx_vpm_init(struct wcaxx *wc);
- static void echocan_free(struct dahdi_chan *chan,
- struct dahdi_echocan_state *ec);
- static const struct dahdi_echocan_features vpm_ec_features = {
- .NLP_automatic = 1,
- .CED_tx_detect = 1,
- .CED_rx_detect = 1,
- };
- static const struct dahdi_echocan_ops vpm_ec_ops = {
- .echocan_free = echocan_free,
- };
- struct vpm450m {
- tPOCT6100_INSTANCE_API pApiInstance;
- struct oct612x_context context;
- UINT32 aulEchoChanHndl[32];
- int chanflags[32];
- int ecmode[32];
- int numchans;
- };
- static int wcaxx_oct612x_write(struct oct612x_context *context,
- u32 address, u16 value)
- {
- struct wcaxx *wc = dev_get_drvdata(context->dev);
- wcxb_set_echocan_reg(&wc->xb, address, value);
- return 0;
- }
- static int wcaxx_oct612x_read(struct oct612x_context *context, u32 address,
- u16 *value)
- {
- struct wcaxx *wc = dev_get_drvdata(context->dev);
- *value = wcxb_get_echocan_reg(&wc->xb, address);
- return 0;
- }
- static int wcaxx_oct612x_write_smear(struct oct612x_context *context,
- u32 address, u16 value, size_t count)
- {
- unsigned int i;
- struct wcaxx *wc = dev_get_drvdata(context->dev);
- for (i = 0; i < count; ++i)
- wcxb_set_echocan_reg(&wc->xb, address + (i << 1), value);
- return 0;
- }
- static int wcaxx_oct612x_write_burst(struct oct612x_context *context,
- u32 address, const u16 *buffer,
- size_t count)
- {
- unsigned int i;
- struct wcaxx *wc = dev_get_drvdata(context->dev);
- for (i = 0; i < count; ++i)
- wcxb_set_echocan_reg(&wc->xb, address + (i << 1), buffer[i]);
- return 0;
- }
- static int wcaxx_oct612x_read_burst(struct oct612x_context *context,
- u32 address, u16 *buffer, size_t count)
- {
- unsigned int i;
- struct wcaxx *wc = dev_get_drvdata(context->dev);
- for (i = 0; i < count; ++i)
- buffer[i] = wcxb_get_echocan_reg(&wc->xb, address + (i << 1));
- return 0;
- }
- static const struct oct612x_ops wcaxx_oct612x_ops = {
- .write = wcaxx_oct612x_write,
- .read = wcaxx_oct612x_read,
- .write_smear = wcaxx_oct612x_write_smear,
- .write_burst = wcaxx_oct612x_write_burst,
- .read_burst = wcaxx_oct612x_read_burst,
- };
- static void vpm450m_setecmode(struct vpm450m *vpm450m, int channel, int mode)
- {
- tOCT6100_CHANNEL_MODIFY *modify;
- UINT32 ulResult;
- if (vpm450m->ecmode[channel] == mode)
- return;
- modify = kzalloc(sizeof(tOCT6100_CHANNEL_MODIFY), GFP_ATOMIC);
- if (!modify) {
- pr_notice("Unable to allocate memory for setec!\n");
- return;
- }
- Oct6100ChannelModifyDef(modify);
- modify->ulEchoOperationMode = mode;
- modify->ulChannelHndl = vpm450m->aulEchoChanHndl[channel];
- ulResult = Oct6100ChannelModify(vpm450m->pApiInstance, modify);
- if (ulResult != GENERIC_OK) {
- pr_notice("Failed to apply echo can changes on channel %d %d %08x!\n",
- vpm450m->aulEchoChanHndl[channel], channel, ulResult);
- } else {
- #ifdef OCTASIC_DEBUG
- pr_debug("Echo can on channel %d set to %d\n", channel, mode);
- #endif
- vpm450m->ecmode[channel] = mode;
- }
- kfree(modify);
- }
- static void vpm450m_setec(struct vpm450m *vpm450m, int channel, int eclen)
- {
- if (eclen) {
- vpm450m->chanflags[channel] |= FLAG_ECHO;
- vpm450m_setecmode(vpm450m, channel,
- cOCT6100_ECHO_OP_MODE_NORMAL);
- } else {
- vpm450m->chanflags[channel] &= ~FLAG_ECHO;
- if (vpm450m->chanflags[channel] & (FLAG_DTMF | FLAG_MUTE)) {
- vpm450m_setecmode(vpm450m, channel,
- cOCT6100_ECHO_OP_MODE_HT_RESET);
- } else {
- vpm450m_setecmode(vpm450m, channel,
- cOCT6100_ECHO_OP_MODE_POWER_DOWN);
- }
- }
- }
- static UINT32 tdmmode_chan_to_slot_map(int mode, int channel)
- {
- /* Four phases on the tdm bus, skip three of them per channel */
- /* Due to a bug in the octasic, we had to move the data onto phase 2 */
- return 1+(channel*4);
- }
- static int echocan_initialize_channel(struct vpm450m *vpm,
- int channel, int mode)
- {
- tOCT6100_CHANNEL_OPEN ChannelOpen;
- UINT32 law_to_use = cOCT6100_PCM_U_LAW;
- UINT32 tdmslot_setting;
- UINT32 ulResult;
- if (0 > channel || ECHOCAN_NUM_CHANS <= channel)
- return -1;
- tdmslot_setting = tdmmode_chan_to_slot_map(mode, channel);
- /* Fill Open channel structure with defaults */
- Oct6100ChannelOpenDef(&ChannelOpen);
- /* Assign the handle memory.*/
- ChannelOpen.pulChannelHndl = &vpm->aulEchoChanHndl[channel];
- ChannelOpen.ulUserChanId = channel;
- /* Enable Tone disabling for Fax and Modems */
- ChannelOpen.fEnableToneDisabler = TRUE;
- /* Passthrough TDM data by default, no echocan */
- ChannelOpen.ulEchoOperationMode = cOCT6100_ECHO_OP_MODE_POWER_DOWN;
- /* Configure the TDM settings.*/
- /* Input from the framer */
- ChannelOpen.TdmConfig.ulSinStream = SIN_STREAM;
- ChannelOpen.TdmConfig.ulSinTimeslot = tdmslot_setting;
- ChannelOpen.TdmConfig.ulSinPcmLaw = law_to_use;
- /* Input from the Host (pre-framer) */
- ChannelOpen.TdmConfig.ulRinStream = RIN_STREAM;
- ChannelOpen.TdmConfig.ulRinTimeslot = tdmslot_setting;
- ChannelOpen.TdmConfig.ulRinPcmLaw = law_to_use;
- /* Output to the Host */
- ChannelOpen.TdmConfig.ulSoutStream = SOUT_STREAM;
- ChannelOpen.TdmConfig.ulSoutTimeslot = tdmslot_setting;
- ChannelOpen.TdmConfig.ulSoutPcmLaw = law_to_use;
- /* From asterisk after echo-cancellation - goes nowhere */
- ChannelOpen.TdmConfig.ulRoutStream = cOCT6100_UNASSIGNED;
- ChannelOpen.TdmConfig.ulRoutTimeslot = cOCT6100_UNASSIGNED;
- ChannelOpen.TdmConfig.ulRoutPcmLaw = law_to_use;
- /* Set the desired VQE features.*/
- ChannelOpen.VqeConfig.fEnableNlp = TRUE;
- ChannelOpen.VqeConfig.fRinDcOffsetRemoval = TRUE;
- ChannelOpen.VqeConfig.fSinDcOffsetRemoval = TRUE;
- ChannelOpen.VqeConfig.ulComfortNoiseMode =
- cOCT6100_COMFORT_NOISE_NORMAL;
- /* Open the channel.*/
- ulResult = Oct6100ChannelOpen(vpm->pApiInstance, &ChannelOpen);
- return ulResult;
- }
- static struct vpm450m *init_vpm450m(struct wcaxx *wc, int isalaw,
- const struct firmware *firmware)
- {
- tOCT6100_CHIP_OPEN *ChipOpen;
- tOCT6100_GET_INSTANCE_SIZE InstanceSize;
- tOCT6100_CHANNEL_OPEN *ChannelOpen;
- UINT32 ulResult;
- struct vpm450m *vpm450m;
- int x, i;
- vpm450m = kzalloc(sizeof(struct vpm450m), GFP_KERNEL);
- if (!vpm450m) {
- dev_info(&wc->xb.pdev->dev, "Unable to allocate vpm450m struct\n");
- return NULL;
- }
- vpm450m->context.dev = &wc->xb.pdev->dev;
- vpm450m->context.ops = &wcaxx_oct612x_ops;
- ChipOpen = kzalloc(sizeof(tOCT6100_CHIP_OPEN), GFP_KERNEL);
- if (!ChipOpen) {
- dev_info(&wc->xb.pdev->dev, "Unable to allocate ChipOpen\n");
- kfree(vpm450m);
- return NULL;
- }
- ChannelOpen = kzalloc(sizeof(tOCT6100_CHANNEL_OPEN), GFP_KERNEL);
- if (!ChannelOpen) {
- dev_info(&wc->xb.pdev->dev, "Unable to allocate ChannelOpen\n");
- kfree(vpm450m);
- kfree(ChipOpen);
- return NULL;
- }
- for (x = 0; x < ARRAY_SIZE(vpm450m->ecmode); x++)
- vpm450m->ecmode[x] = -1;
- vpm450m->numchans = ECHOCAN_NUM_CHANS;
- dev_info(&wc->xb.pdev->dev, "Echo cancellation for %d channels\n",
- wc->desc->ports);
- Oct6100ChipOpenDef(ChipOpen);
- ChipOpen->pProcessContext = &vpm450m->context;
- /* Change default parameters as needed */
- /* upclk oscillator is at 33.33 Mhz */
- ChipOpen->ulUpclkFreq = cOCT6100_UPCLK_FREQ_33_33_MHZ;
- /* mclk will be generated by internal PLL at 133 Mhz */
- ChipOpen->fEnableMemClkOut = TRUE;
- ChipOpen->ulMemClkFreq = cOCT6100_MCLK_FREQ_133_MHZ;
- /* User defined Chip ID.*/
- ChipOpen->ulUserChipId = OCT_CHIP_ID;
- /* Set the maximums that the chip needs to support */
- ChipOpen->ulMaxChannels = vpm450m->numchans;
- ChipOpen->ulMaxTdmStreams = OCT_MAX_TDM_STREAMS;
- /* External Memory Settings */
- /* Use DDR memory.*/
- ChipOpen->ulMemoryType = cOCT6100_MEM_TYPE_DDR;
- ChipOpen->ulNumMemoryChips = 1;
- ChipOpen->ulMemoryChipSize = cOCT6100_MEMORY_CHIP_SIZE_32MB;
- ChipOpen->pbyImageFile = (PUINT8) firmware->data;
- ChipOpen->ulImageSize = firmware->size;
- /* Set TDM data stream frequency */
- for (i = 0; i < ChipOpen->ulMaxTdmStreams; i++)
- ChipOpen->aulTdmStreamFreqs[i] = cOCT6100_TDM_STREAM_FREQ_8MHZ;
- /* Configure TDM sampling */
- ChipOpen->ulTdmSampling = cOCT6100_TDM_SAMPLE_AT_FALLING_EDGE;
- /* Disable to save RAM footprint space */
- ChipOpen->fEnableChannelRecording = FALSE;
- /* In this example we will maintain the API using polling so interrupts
- * must be disabled */
- ChipOpen->InterruptConfig.ulErrorH100Config =
- cOCT6100_INTERRUPT_DISABLE;
- ChipOpen->InterruptConfig.ulErrorMemoryConfig =
- cOCT6100_INTERRUPT_DISABLE;
- ChipOpen->InterruptConfig.ulFatalGeneralConfig =
- cOCT6100_INTERRUPT_DISABLE;
- ChipOpen->InterruptConfig.ulFatalMemoryConfig =
- cOCT6100_INTERRUPT_DISABLE;
- ChipOpen->ulSoftToneEventsBufSize = OCT_TONEEVENT_BUFFER_SIZE;
- /* Inserting default values into tOCT6100_GET_INSTANCE_SIZE structure
- * parameters. */
- Oct6100GetInstanceSizeDef(&InstanceSize);
- /* Get the size of the OCT6100 instance structure. */
- ulResult = Oct6100GetInstanceSize(ChipOpen, &InstanceSize);
- if (ulResult != cOCT6100_ERR_OK) {
- dev_info(&wc->xb.pdev->dev, "Unable to get instance size: %x\n",
- ulResult);
- return NULL;
- }
- vpm450m->pApiInstance = vmalloc(InstanceSize.ulApiInstanceSize);
- if (!vpm450m->pApiInstance) {
- dev_info(&wc->xb.pdev->dev,
- "Out of memory (can't allocate %d bytes)!\n",
- InstanceSize.ulApiInstanceSize);
- return NULL;
- }
- /* Perform the actual configuration of the chip. */
- wcxb_enable_echocan_dram(&wc->xb);
- ulResult = Oct6100ChipOpen(vpm450m->pApiInstance, ChipOpen);
- if (ulResult != cOCT6100_ERR_OK) {
- dev_info(&wc->xb.pdev->dev, "Unable to Oct6100ChipOpen: %x\n",
- ulResult);
- return NULL;
- }
- /* OCT6100 is now booted and channels can be opened */
- /* Open all channels */
- for (i = 0; i < ECHOCAN_NUM_CHANS; i++) {
- ulResult = echocan_initialize_channel(vpm450m, i, isalaw);
- if (0 != ulResult) {
- dev_info(&wc->xb.pdev->dev,
- "Unable to echocan_initialize_channel: %x\n",
- ulResult);
- return NULL;
- }
- }
- if (vpmsupport)
- wcxb_enable_echocan(&wc->xb);
- else
- wcxb_disable_echocan(&wc->xb);
- kfree(ChipOpen);
- kfree(ChannelOpen);
- return vpm450m;
- }
- static void release_vpm450m(struct vpm450m *vpm450m)
- {
- UINT32 ulResult;
- tOCT6100_CHIP_CLOSE ChipClose;
- Oct6100ChipCloseDef(&ChipClose);
- ulResult = Oct6100ChipClose(vpm450m->pApiInstance, &ChipClose);
- if (ulResult != cOCT6100_ERR_OK)
- pr_notice("Failed to close chip, code %08x!\n", ulResult);
- vfree(vpm450m->pApiInstance);
- kfree(vpm450m);
- }
- static const char *wcaxx_echocan_name(const struct dahdi_chan *chan)
- {
- struct wcaxx *wc = chan->pvt;
- if (wc->vpm)
- return "VPMOCT032";
- else
- return NULL;
- }
- static int wcaxx_echocan_create(struct dahdi_chan *chan,
- struct dahdi_echocanparams *ecp,
- struct dahdi_echocanparam *p,
- struct dahdi_echocan_state **ec)
- {
- struct wcaxx *wc = chan->pvt;
- int channel;
- const struct dahdi_echocan_ops *ops;
- const struct dahdi_echocan_features *features;
- if (!vpmsupport || !wc->vpm)
- return -ENODEV;
- ops = &vpm_ec_ops;
- features = &vpm_ec_features;
- if (ecp->param_count > 0) {
- dev_warn(&wc->xb.pdev->dev,
- "%s echo canceller does not support parameters; failing request\n",
- chan->ec_factory->get_name(chan));
- return -EINVAL;
- }
- *ec = wc->ec[chan->chanpos - 1];
- (*ec)->ops = ops;
- (*ec)->features = *features;
- channel = chan->chanpos-1;
- if (wc->vpm)
- vpm450m_setec(wc->vpm, channel, ecp->tap_length);
- return 0;
- }
- static void echocan_free(struct dahdi_chan *chan,
- struct dahdi_echocan_state *ec)
- {
- struct wcaxx *wc = chan->pvt;
- int channel;
- memset(ec, 0, sizeof(*ec));
- channel = chan->chanpos - 1;
- if (wc->vpm)
- vpm450m_setec(wc->vpm, channel, 0);
- }
- static int wcaxx_vpm_init(struct wcaxx *wc)
- {
- int companding = 0;
- struct firmware embedded_firmware;
- const struct firmware *firmware = &embedded_firmware;
- #if !defined(HOTPLUG_FIRMWARE)
- extern void _binary_dahdi_fw_oct6114_032_bin_size;
- extern u8 _binary_dahdi_fw_oct6114_032_bin_start[];
- #else
- static const char oct032_firmware[] = "dahdi-fw-oct6114-032.bin";
- #endif
- int res;
- if (!vpmsupport) {
- dev_info(&wc->xb.pdev->dev, "VPM: Support Disabled\n");
- return -1;
- }
- wcxb_reset_echocan(&wc->xb);
- if (!wcxb_is_echocan_present(&wc->xb)) {
- dev_info(&wc->xb.pdev->dev, "VPM not present.\n");
- return -1;
- }
- #if defined(HOTPLUG_FIRMWARE)
- res = request_firmware(&firmware, oct032_firmware, &wc->xb.pdev->dev);
- if ((0 != res) || !firmware) {
- dev_notice(&wc->xb.pdev->dev,
- "VPM450: firmware %s not available from userspace\n",
- oct032_firmware);
- return -1;
- }
- #else
- embedded_firmware.data = _binary_dahdi_fw_oct6114_032_bin_start;
- /* Yes... this is weird. objcopy gives us a symbol containing
- the size of the firmware, not a pointer a variable containing
- the size. The only way we can get the value of the symbol
- is to take its address, so we define it as a pointer and
- then cast that value to the proper type.
- */
- embedded_firmware.size = (size_t)&_binary_dahdi_fw_oct6114_032_bin_size;
- #endif
- wc->vpm = init_vpm450m(wc, companding, firmware);
- if (!wc->vpm) {
- dev_notice(&wc->xb.pdev->dev, "VPM450: Failed to initialize\n");
- if (firmware != &embedded_firmware)
- release_firmware(firmware);
- return -EIO;
- }
- if (firmware != &embedded_firmware)
- release_firmware(firmware);
- dev_info(&wc->xb.pdev->dev,
- "VPM450: Present and operational servicing %d span\n", 1);
- return 0;
- }
- static inline bool is_initialized(struct wcaxx *wc)
- {
- return (test_bit(INITIALIZED, &wc->bit_flags) > 0);
- }
- static inline struct wcxb_spi_device *
- _get_spi_device_for_8_port(struct wcaxx *wc, unsigned int port, bool altcs)
- {
- switch (port) {
- case 0:
- return wc->spi_devices[0];
- case 1:
- return (altcs) ? wc->spi_devices[0] : wc->spi_devices[1];
- case 2:
- WARN_ON(!altcs);
- return wc->spi_devices[0];
- case 3:
- WARN_ON(!altcs);
- return wc->spi_devices[0];
- case 4:
- return wc->spi_devices[2];
- case 5:
- return (altcs) ? wc->spi_devices[2] : wc->spi_devices[3];
- case 6:
- WARN_ON(!altcs);
- return wc->spi_devices[2];
- case 7:
- WARN_ON(!altcs);
- return wc->spi_devices[2];
- default:
- WARN_ON(1);
- return wc->spi_devices[0];
- }
- }
- static inline struct wcxb_spi_device *
- _get_spi_device_for_4_port(struct wcaxx *wc, unsigned int port)
- {
- if (port > 3) {
- WARN_ON(1);
- return wc->spi_devices[0];
- } else {
- return wc->spi_devices[port];
- }
- }
- static inline struct wcxb_spi_device *
- get_spi_device_for_port(struct wcaxx *wc, unsigned int port, bool altcs)
- {
- if (is_four_port(wc))
- return _get_spi_device_for_4_port(wc, port);
- else
- return _get_spi_device_for_8_port(wc, port, altcs);
- }
- static u8 wcaxx_getreg(struct wcaxx *wc,
- struct wcaxx_module *const mod, int addr);
- static void wcaxx_setreg(struct wcaxx *wc, struct wcaxx_module *const mod,
- int addr, int val);
- static DEFINE_MUTEX(card_list_lock);
- static LIST_HEAD(card_list);
- #include "adt_lec.h"
- /*
- Experimental max loop current limit for the proslic
- Loop current limit is from 20 mA to 41 mA in steps of 3
- (according to datasheet)
- So set the value below to:
- 0x00 : 20mA (default)
- 0x01 : 23mA
- 0x02 : 26mA
- 0x03 : 29mA
- 0x04 : 32mA
- 0x05 : 35mA
- 0x06 : 37mA
- 0x07 : 41mA
- */
- static int loopcurrent = 20;
- /* Following define is a logical exclusive OR to determine if the polarity of
- * an fxs line is to be reversed. The items taken into account are:
- * overall polarity reversal for the module,
- * polarity reversal for the port,
- * and the state of the line reversal MWI indicator
- */
- #define POLARITY_XOR(fxs) \
- ((reversepolarity != 0) ^ ((fxs)->reversepolarity != 0) ^ \
- ((fxs)->vmwi_linereverse != 0))
- static int reversepolarity;
- static alpha indirect_regs[] = {
- {0, 255, "DTMF_ROW_0_PEAK", 0x55C2},
- {1, 255, "DTMF_ROW_1_PEAK", 0x51E6},
- {2, 255, "DTMF_ROW2_PEAK", 0x4B85},
- {3, 255, "DTMF_ROW3_PEAK", 0x4937},
- {4, 255, "DTMF_COL1_PEAK", 0x3333},
- {5, 255, "DTMF_FWD_TWIST", 0x0202},
- {6, 255, "DTMF_RVS_TWIST", 0x0202},
- {7, 255, "DTMF_ROW_RATIO_TRES", 0x0198},
- {8, 255, "DTMF_COL_RATIO_TRES", 0x0198},
- {9, 255, "DTMF_ROW_2ND_ARM", 0x0611},
- {10, 255, "DTMF_COL_2ND_ARM", 0x0202},
- {11, 255, "DTMF_PWR_MIN_TRES", 0x00E5},
- {12, 255, "DTMF_OT_LIM_TRES", 0x0A1C},
- {13, 0, "OSC1_COEF", 0x7B30},
- {14, 1, "OSC1X", 0x0063},
- {15, 2, "OSC1Y", 0x0000},
- {16, 3, "OSC2_COEF", 0x7870},
- {17, 4, "OSC2X", 0x007D},
- {18, 5, "OSC2Y", 0x0000},
- {19, 6, "RING_V_OFF", 0x0000},
- {20, 7, "RING_OSC", 0x7EF0},
- {21, 8, "RING_X", 0x0160},
- {22, 9, "RING_Y", 0x0000},
- {23, 255, "PULSE_ENVEL", 0x2000},
- {24, 255, "PULSE_X", 0x2000},
- {25, 255, "PULSE_Y", 0x0000},
- {26, 13, "RECV_DIGITAL_GAIN", 0x2000}, /* playback volume set lower */
- {27, 14, "XMIT_DIGITAL_GAIN", 0x4000},
- {28, 15, "LOOP_CLOSE_TRES", 0x1000},
- {29, 16, "RING_TRIP_TRES", 0x3600},
- {30, 17, "COMMON_MIN_TRES", 0x1000},
- {31, 18, "COMMON_MAX_TRES", 0x0200},
- {32, 19, "PWR_ALARM_Q1Q2", 0x07C0},
- {33, 20, "PWR_ALARM_Q3Q4", 0x4C00 /* 0x2600 */},
- {34, 21, "PWR_ALARM_Q5Q6", 0x1B80},
- {35, 22, "LOOP_CLOSURE_FILTER", 0x8000},
- {36, 23, "RING_TRIP_FILTER", 0x0320},
- {37, 24, "TERM_LP_POLE_Q1Q2", 0x008C},
- {38, 25, "TERM_LP_POLE_Q3Q4", 0x0100},
- {39, 26, "TERM_LP_POLE_Q5Q6", 0x0010},
- {40, 27, "CM_BIAS_RINGING", 0x0C00},
- {41, 64, "DCDC_MIN_V", 0x0C00},
- {42, 255, "DCDC_XTRA", 0x1000},
- {43, 66, "LOOP_CLOSE_TRES_LOW", 0x1000},
- };
- /* Undefine to enable Power alarm / Transistor debug -- note: do not
- enable for normal operation! */
- /* #define PAQ_DEBUG */
- #define DEBUG_CARD (1 << 0)
- #define DEBUG_ECHOCAN (1 << 1)
- #include "fxo_modes.h"
- static inline struct dahdi_chan *
- get_dahdi_chan(const struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- return wc->span.chans[mod->card];
- }
- static inline void mod_hooksig(struct wcaxx *wc,
- struct wcaxx_module *mod,
- enum dahdi_rxsig rxsig)
- {
- dahdi_hooksig(get_dahdi_chan(wc, mod), rxsig);
- }
- static void wcaxx_release(struct wcaxx *wc);
- static int fxovoltage;
- static unsigned int battdebounce;
- static unsigned int battalarm;
- static unsigned int battthresh;
- static int debug;
- static int int_mode;
- #ifdef DEBUG
- static int robust;
- static int digitalloopback;
- #endif
- static int lowpower;
- static int boostringer;
- static int fastringer;
- static int _opermode;
- static char *opermode = "FCC";
- static int fxshonormode;
- static int alawoverride;
- static char *companding = "auto";
- static int fastpickup = -1; /* -1 auto, 0 no, 1 yes */
- static int fxotxgain;
- static int fxorxgain;
- static int fxstxgain;
- static int fxsrxgain;
- static int nativebridge;
- static int ringdebounce = DEFAULT_RING_DEBOUNCE;
- static int latency = WCXB_DEFAULT_LATENCY;
- static unsigned int max_latency = WCXB_DEFAULT_MAXLATENCY;
- static int forceload;
- #define MS_PER_HOOKCHECK (1)
- #define NEONMWI_ON_DEBOUNCE (100/MS_PER_HOOKCHECK)
- static int neonmwi_monitor;
- static int neonmwi_level = 75; /* neon mwi trip voltage */
- static int neonmwi_envelope = 10;
- /* Time in milliseconds the monitor is checked before saying no message is
- * waiting */
- static int neonmwi_offlimit = 16000;
- static int neonmwi_offlimit_cycles;
- static int wcaxx_init_proslic(struct wcaxx *wc,
- struct wcaxx_module *const mod, int fast,
- int manual, int sane);
- struct wcaxx_setreg_memory {
- struct wcxb_spi_message m;
- struct wcxb_spi_transfer t;
- u8 buffer[3];
- };
- /**
- * wcxb_spi_complete_setreg - Cleanup after a SPI write.
- *
- * We don't care about the results of setreg. Just go ahead and free up the
- * messages.
- *
- */
- static void wcaxx_complete_setreg(void *arg)
- {
- struct wcaxx_setreg_memory *setreg = arg;
- kfree(setreg);
- }
- static void wcaxx_setreg(struct wcaxx *wc, struct wcaxx_module *mod,
- int addr, int val)
- {
- struct wcaxx_setreg_memory *setreg = kzalloc(sizeof(*setreg),
- GFP_ATOMIC);
- struct wcxb_spi_message *const m = &setreg->m;
- struct wcxb_spi_transfer *const t = &setreg->t;
- if (!setreg) {
- WARN_ON_ONCE(!setreg);
- return;
- }
- wcxb_spi_message_init(m);
- t->tx_buf = setreg->buffer;
- wcxb_spi_message_add_tail(t, m);
- if (FXO == mod->type) {
- static const int ADDRS[4] = {0x00, 0x08, 0x04, 0x0c};
- setreg->buffer[0] = 0x20 | ADDRS[mod->subaddr];
- } else {
- setreg->buffer[0] = 1 << mod->subaddr;
- }
- setreg->buffer[1] = (addr) & 0x7f;
- setreg->buffer[2] = val;
- t->len = 3;
- m->complete = &wcaxx_complete_setreg;
- m->arg = setreg;
- wcxb_spi_async(mod->spi, m);
- }
- /**
- * wcaxx_fsxinit - Initilize all SPI devices to 3 byte mode.
- *
- * All the modules on the card need to be initialized to 3 byte mode in order to
- * talk to the daisy-chained SLIC / DAA on the quad modules.
- *
- */
- static void wcaxx_fxsinit(struct wcxb_spi_device *const spi)
- {
- int res;
- u8 data_byte[2] = {0, 0x80};
- struct wcxb_spi_transfer t;
- struct wcxb_spi_message m;
- memset(&t, 0, sizeof(t));
- wcxb_spi_message_init(&m);
- t.tx_buf = data_byte;
- t.len = sizeof(data_byte);
- wcxb_spi_message_add_tail(&t, &m);
- res = wcxb_spi_sync(spi, &m);
- WARN_ON_ONCE(0 != res);
- return;
- }
- static u8
- wcaxx_getreg(struct wcaxx *wc, struct wcaxx_module *const mod, int addr)
- {
- int res;
- u8 buffer[3];
- struct wcxb_spi_message m;
- struct wcxb_spi_transfer t;
- memset(&t, 0, sizeof(t));
- wcxb_spi_message_init(&m);
- t.tx_buf = t.rx_buf = buffer;
- t.len = sizeof(buffer);
- wcxb_spi_message_add_tail(&t, &m);
- if (FXO == mod->type) {
- static const int ADDRS[4] = {0x00, 0x08, 0x04, 0x0c};
- buffer[0] = 0x60 | ADDRS[mod->subaddr];
- buffer[1] = addr & 0x7f;
- buffer[2] = 0;
- } else {
- buffer[0] = 1 << mod->subaddr;
- buffer[1] = (addr | 0x80) & 0xff;
- buffer[2] = 0;
- }
- res = wcxb_spi_sync(mod->spi, &m);
- WARN_ON_ONCE(0 != res);
- return buffer[2];
- }
- static int wcaxx_getregs(struct wcaxx *wc, struct wcaxx_module *const mod,
- int *const addresses, const size_t count)
- {
- int x;
- for (x = 0; x < count; ++x)
- addresses[x] = wcaxx_getreg(wc, mod, addresses[x]);
- return 0;
- }
- static int wait_access(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- unsigned char data = 0;
- int count = 0;
- #define MAX 10 /* attempts */
- /* Wait for indirect access */
- while (count++ < MAX) {
- data = wcaxx_getreg(wc, mod, I_STATUS);
- if (!data)
- return 0;
- }
- if (count > (MAX-1)) {
- dev_notice(&wc->xb.pdev->dev,
- " ##### Loop error (%02x) #####\n", data);
- }
- return 0;
- }
- static unsigned char translate_3215(unsigned char address)
- {
- int x;
- for (x = 0; x < ARRAY_SIZE(indirect_regs); x++) {
- if (indirect_regs[x].address == address) {
- address = indirect_regs[x].altaddr;
- break;
- }
- }
- return address;
- }
- static int wcaxx_proslic_setreg_indirect(struct wcaxx *wc,
- struct wcaxx_module *const mod,
- unsigned char address,
- unsigned short data)
- {
- int res = -1;
- address = translate_3215(address);
- if (address == 255)
- return 0;
- if (!wait_access(wc, mod)) {
- wcaxx_setreg(wc, mod, IDA_LO, (u8)(data & 0xFF));
- wcaxx_setreg(wc, mod, IDA_HI, (u8)((data & 0xFF00)>>8));
- wcaxx_setreg(wc, mod, IAA, address);
- res = 0;
- };
- return res;
- }
- static int wcaxx_proslic_getreg_indirect(struct wcaxx *wc,
- struct wcaxx_module *const mod,
- unsigned char address)
- {
- int res = -1;
- char *p = NULL;
- address = translate_3215(address);
- if (address == 255)
- return 0;
- if (!wait_access(wc, mod)) {
- wcaxx_setreg(wc, mod, IAA, address);
- if (!wait_access(wc, mod)) {
- int addresses[2] = {IDA_LO, IDA_HI};
- wcaxx_getregs(wc, mod, addresses,
- ARRAY_SIZE(addresses));
- res = addresses[0] | (addresses[1] << 8);
- } else
- p = "Failed to wait inside\n";
- } else
- p = "failed to wait\n";
- if (p)
- dev_notice(&wc->xb.pdev->dev, "%s", p);
- return res;
- }
- static int
- wcaxx_proslic_init_indirect_regs(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- unsigned char i;
- for (i = 0; i < ARRAY_SIZE(indirect_regs); i++) {
- if (wcaxx_proslic_setreg_indirect(wc, mod,
- indirect_regs[i].address,
- indirect_regs[i].initial))
- return -1;
- }
- return 0;
- }
- static int wcaxx_proslic_verify_indirect_regs(struct wcaxx *wc,
- struct wcaxx_module *mod)
- {
- int passed = 1;
- unsigned short i, initial;
- int j;
- for (i = 0; i < ARRAY_SIZE(indirect_regs); i++) {
- j = wcaxx_proslic_getreg_indirect(wc, mod,
- (u8)indirect_regs[i].address);
- if (j < 0) {
- dev_notice(&wc->xb.pdev->dev,
- "Failed to read indirect register %d\n", i);
- return -1;
- }
- initial = indirect_regs[i].initial;
- if ((j != initial) && (indirect_regs[i].altaddr != 255)) {
- dev_notice(&wc->xb.pdev->dev,
- "!!!!!!! %s iREG %X = %X should be %X\n",
- indirect_regs[i].name,
- indirect_regs[i].address, j, initial);
- passed = 0;
- }
- }
- if (passed) {
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Init Indirect Registers completed successfully.\n");
- }
- } else {
- dev_notice(&wc->xb.pdev->dev,
- " !!!!! Init Indirect Registers UNSUCCESSFULLY.\n");
- return -1;
- }
- return 0;
- }
- /**
- * wcaxx_proslic_check_oppending -
- *
- * Ensures that a write to the line feed register on the SLIC has been
- * processed. If it hasn't after the timeout value, then it will resend the
- * command and wait for another timeout period.
- *
- */
- static void wcaxx_proslic_check_oppending(struct wcaxx *wc,
- struct wcaxx_module *const mod)
- {
- struct fxs *const fxs = &mod->mod.fxs;
- unsigned long flags;
- if (!(fxs->lasttxhook & SLIC_LF_OPPENDING))
- return;
- /* Monitor the Pending LF state change, for the next 100ms */
- spin_lock_irqsave(&wc->reglock, flags);
- if (!(fxs->lasttxhook & SLIC_LF_OPPENDING)) {
- spin_unlock_irqrestore(&wc->reglock, flags);
- return;
- }
- if ((fxs->linefeed_control_shadow & SLIC_LF_SETMASK) ==
- (fxs->lasttxhook & SLIC_LF_SETMASK)) {
- fxs->lasttxhook &= SLIC_LF_SETMASK;
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "SLIC_LF OK: card=%d shadow=%02x "
- "lasttxhook=%02x framecount=%ld\n", mod->card,
- fxs->linefeed_control_shadow,
- fxs->lasttxhook, wc->framecount);
- }
- } else if (time_after(wc->framecount, fxs->oppending_timeout)) {
- /* Check again in 100 ms */
- fxs->oppending_timeout = wc->framecount + 100;
- wcaxx_setreg(wc, mod, LINE_STATE, fxs->lasttxhook);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "SLIC_LF RETRY: card=%d shadow=%02x "
- "lasttxhook=%02x framecount=%ld\n", mod->card,
- fxs->linefeed_control_shadow,
- fxs->lasttxhook, wc->framecount);
- }
- }
- spin_unlock_irqrestore(&wc->reglock, flags);
- }
- /* 256ms interrupt */
- static void wcaxx_proslic_recheck_sanity(struct wcaxx *wc,
- struct wcaxx_module *const mod)
- {
- struct fxs *const fxs = &mod->mod.fxs;
- int res;
- unsigned long flags;
- const unsigned int MAX_ALARMS = 10;
- #ifdef PAQ_DEBUG
- res = mod->isrshadow[1];
- res &= ~0x3;
- if (res) {
- mod->isrshadow[1] = 0;
- fxs->palarms++;
- if (fxs->palarms < MAX_ALARMS) {
- dev_notice(&wc->xb.pdev->dev,
- "Power alarm (%02x) on module %d, resetting!\n",
- res, card + 1);
- mod->sethook = CMD_WR(19, res);
- /* Update shadow register to avoid extra power alarms
- * until next read */
- mod->isrshadow[1] = 0;
- } else {
- if (fxs->palarms == MAX_ALARMS) {
- dev_notice(&wc->xb.pdev->dev,
- "Too many power alarms on card %d, NOT resetting!\n",
- card + 1);
- }
- }
- }
- #else
- spin_lock_irqsave(&wc->reglock, flags);
- /* reg 64 has to be zero at last isr read */
- res = !fxs->linefeed_control_shadow &&
- !(fxs->lasttxhook & SLIC_LF_OPPENDING) && /* not a transition */
- fxs->lasttxhook; /* not an intended zero */
- if (res) {
- fxs->palarms++;
- if (fxs->palarms < MAX_ALARMS) {
- dev_notice(&wc->xb.pdev->dev,
- "Power alarm on module %d, resetting!\n",
- mod->card + 1);
- if (fxs->lasttxhook == SLIC_LF_RINGING) {
- fxs->lasttxhook = POLARITY_XOR(fxs) ?
- SLIC_LF_ACTIVE_REV :
- SLIC_LF_ACTIVE_FWD;
- }
- fxs->lasttxhook |= SLIC_LF_OPPENDING;
- mod->sethook = CMD_WR(LINE_STATE, fxs->lasttxhook);
- fxs->oppending_timeout = wc->framecount + 100;
- /* Update shadow register to avoid extra power alarms
- * until next read */
- fxs->linefeed_control_shadow = fxs->lasttxhook;
- } else {
- if (fxs->palarms == MAX_ALARMS) {
- dev_notice(&wc->xb.pdev->dev,
- "Too many power alarms on card %d, "
- "NOT resetting!\n", mod->card + 1);
- }
- }
- }
- spin_unlock_irqrestore(&wc->reglock, flags);
- #endif
- }
- static inline bool is_fxo_ringing(const struct fxo *const fxo)
- {
- return ((fxo->hook_ring_shadow & 0x60) &&
- ((fxo->battery_state == BATTERY_PRESENT) ||
- (fxo->battery_state == BATTERY_DEBOUNCING_LOST)));
- }
- static inline bool is_fxo_ringing_positive(const struct fxo *const fxo)
- {
- return (((fxo->hook_ring_shadow & 0x60) == 0x20) &&
- ((fxo->battery_state == BATTERY_PRESENT) ||
- (fxo->battery_state == BATTERY_DEBOUNCING_LOST)));
- }
- static inline bool is_fxo_ringing_negative(const struct fxo *const fxo)
- {
- return (((fxo->hook_ring_shadow & 0x60) == 0x40) &&
- ((fxo->battery_state == BATTERY_PRESENT) ||
- (fxo->battery_state == BATTERY_DEBOUNCING_LOST)));
- }
- static inline void set_ring(struct fxo *fxo, enum ring_detector_state new)
- {
- fxo->ring_state = new;
- }
- static void wcaxx_fxo_ring_detect(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- static const unsigned int POLARITY_CHANGES_NEEDED = 2;
- /* Look for ring status bits (Ring Detect Signal Negative and Ring
- * Detect Signal Positive) to transition back and forth
- * POLARITY_CHANGES_NEEDED times to indicate that a ring is occurring.
- * Provide some number of samples to allow for the transitions to occur
- * before giving up. NOTE: neon mwi voltages will trigger one of these
- * bits to go active but not to have transitions between the two bits
- * (i.e. no negative to positive or positive to negative traversals) */
- switch (fxo->ring_state) {
- case DEBOUNCING_RINGING_POSITIVE:
- if (is_fxo_ringing_negative(fxo)) {
- if (++fxo->ring_polarity_change_count >
- POLARITY_CHANGES_NEEDED) {
- mod_hooksig(wc, mod, DAHDI_RXSIG_RING);
- set_ring(fxo, RINGING);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "RING on %s!\n",
- get_dahdi_chan(wc, mod)->name);
- }
- } else {
- set_ring(fxo, DEBOUNCING_RINGING_NEGATIVE);
- }
- } else if (time_after(wc->framecount,
- fxo->ringdebounce_timer)) {
- set_ring(fxo, RINGOFF);
- }
- break;
- case DEBOUNCING_RINGING_NEGATIVE:
- if (is_fxo_ringing_positive(fxo)) {
- if (++fxo->ring_polarity_change_count >
- POLARITY_CHANGES_NEEDED) {
- mod_hooksig(wc, mod, DAHDI_RXSIG_RING);
- set_ring(fxo, RINGING);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "RING on %s!\n",
- get_dahdi_chan(wc, mod)->name);
- }
- } else {
- set_ring(fxo, DEBOUNCING_RINGING_POSITIVE);
- }
- } else if (time_after(wc->framecount,
- fxo->ringdebounce_timer)) {
- set_ring(fxo, RINGOFF);
- }
- break;
- case RINGING:
- if (!is_fxo_ringing(fxo)) {
- set_ring(fxo, DEBOUNCING_RINGOFF);
- fxo->ringdebounce_timer =
- wc->framecount + ringdebounce / 8;
- }
- break;
- case DEBOUNCING_RINGOFF:
- if (!is_fxo_ringing(fxo)) {
- if (time_after(wc->framecount,
- fxo->ringdebounce_timer)) {
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "NO RING on %s!\n",
- get_dahdi_chan(wc, mod)->name);
- }
- mod_hooksig(wc, mod, DAHDI_RXSIG_OFFHOOK);
- set_ring(fxo, RINGOFF);
- }
- } else {
- set_ring(fxo, RINGING);
- }
- break;
- case RINGOFF:
- if (is_fxo_ringing(fxo)) {
- /* Look for positive/negative crossings in ring status
- * reg */
- if (is_fxo_ringing_positive(fxo))
- set_ring(fxo, DEBOUNCING_RINGING_POSITIVE);
- else
- set_ring(fxo, DEBOUNCING_RINGING_NEGATIVE);
- fxo->ringdebounce_timer =
- wc->framecount + ringdebounce / 8;
- fxo->ring_polarity_change_count = 0;
- }
- break;
- }
- }
- #define MS_PER_CHECK_HOOK 1
- static void
- wcaxx_check_battery_lost(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- /* possible existing states:
- battery lost, no debounce timer
- battery lost, debounce timer (going to battery present)
- battery present or unknown, no debounce timer
- battery present or unknown, debounce timer (going to battery lost)
- */
- switch (fxo->battery_state) {
- case BATTERY_DEBOUNCING_PRESENT_ALARM:
- fxo->battery_state = BATTERY_DEBOUNCING_LOST_FROM_PRESENT_ALARM;
- fxo->battdebounce_timer = wc->framecount + battdebounce;
- break;
- case BATTERY_DEBOUNCING_PRESENT:
- fxo->battery_state = BATTERY_LOST;
- break;
- case BATTERY_DEBOUNCING_PRESENT_FROM_LOST_ALARM:
- fxo->battery_state = BATTERY_DEBOUNCING_LOST_ALARM;
- fxo->battdebounce_timer = wc->framecount +
- battalarm - battdebounce;
- break;
- case BATTERY_UNKNOWN:
- mod_hooksig(wc, mod, DAHDI_RXSIG_ONHOOK);
- case BATTERY_PRESENT:
- fxo->battery_state = BATTERY_DEBOUNCING_LOST;
- fxo->battdebounce_timer = wc->framecount + battdebounce;
- break;
- case BATTERY_DEBOUNCING_LOST_FROM_PRESENT_ALARM:
- case BATTERY_DEBOUNCING_LOST: /* Intentional drop through */
- if (time_after(wc->framecount, fxo->battdebounce_timer)) {
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "NO BATTERY on %d/%d!\n",
- wc->span.spanno,
- mod->card + 1);
- }
- #ifdef JAPAN
- if (!wc->ohdebounce && wc->offhook) {
- dahdi_hooksig(wc->aspan->chans[card],
- DAHDI_RXSIG_ONHOOK);
- if (debug) {
- dev_info(&wc->vb.pdev->dev,
- "Signalled On Hook\n");
- }
- #ifdef ZERO_BATT_RING
- wc->onhook++;
- #endif
- }
- #else
- mod_hooksig(wc, mod, DAHDI_RXSIG_ONHOOK);
- #endif
- /* set the alarm timer, taking into account that part
- * of its time period has already passed while
- * debouncing occurred */
- fxo->battery_state = BATTERY_DEBOUNCING_LOST_ALARM;
- fxo->battdebounce_timer = wc->framecount +
- battalarm - battdebounce;
- }
- break;
- case BATTERY_DEBOUNCING_LOST_ALARM:
- if (time_after(wc->framecount, fxo->battdebounce_timer)) {
- fxo->battery_state = BATTERY_LOST;
- dahdi_alarm_channel(get_dahdi_chan(wc, mod),
- DAHDI_ALARM_RED);
- }
- break;
- case BATTERY_LOST:
- break;
- }
- }
- static void
- wcaxx_check_battery_present(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- switch (fxo->battery_state) {
- case BATTERY_DEBOUNCING_PRESENT_FROM_LOST_ALARM:
- case BATTERY_DEBOUNCING_PRESENT: /* intentional drop through */
- if (time_after(wc->framecount, fxo->battdebounce_timer)) {
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "BATTERY on %d/%d (%s)!\n",
- wc->span.spanno, mod->card + 1,
- (fxo->line_voltage_status < 0) ?
- "-" : "+");
- }
- #ifdef ZERO_BATT_RING
- if (wc->onhook) {
- wc->onhook = 0;
- dahdi_hooksig(wc->aspan->chans[card],
- DAHDI_RXSIG_OFFHOOK);
- if (debug) {
- dev_info(&wc->vb.pdev->dev,
- "Signalled Off Hook\n");
- }
- }
- #else
- mod_hooksig(wc, mod, DAHDI_RXSIG_OFFHOOK);
- #endif
- /* set the alarm timer, taking into account that part
- * of its time period has already passed while
- * debouncing occurred */
- fxo->battery_state = BATTERY_DEBOUNCING_PRESENT_ALARM;
- fxo->battdebounce_timer = wc->framecount +
- battalarm - battdebounce;
- }
- break;
- case BATTERY_DEBOUNCING_PRESENT_ALARM:
- if (time_after(wc->framecount, fxo->battdebounce_timer)) {
- fxo->battery_state = BATTERY_PRESENT;
- dahdi_alarm_channel(get_dahdi_chan(wc, mod),
- DAHDI_ALARM_NONE);
- }
- break;
- case BATTERY_PRESENT:
- break;
- case BATTERY_DEBOUNCING_LOST_ALARM:
- fxo->battery_state = BATTERY_DEBOUNCING_PRESENT_FROM_LOST_ALARM;
- fxo->battdebounce_timer = wc->framecount + battdebounce;
- break;
- case BATTERY_DEBOUNCING_LOST_FROM_PRESENT_ALARM:
- fxo->battery_state = BATTERY_DEBOUNCING_PRESENT_ALARM;
- fxo->battdebounce_timer = wc->framecount +
- battalarm - battdebounce;
- break;
- case BATTERY_DEBOUNCING_LOST:
- fxo->battery_state = BATTERY_PRESENT;
- break;
- case BATTERY_UNKNOWN:
- mod_hooksig(wc, mod, DAHDI_RXSIG_OFFHOOK);
- case BATTERY_LOST: /* intentional drop through */
- fxo->battery_state = BATTERY_DEBOUNCING_PRESENT;
- fxo->battdebounce_timer = wc->framecount + battdebounce;
- break;
- }
- }
- static void
- wcaxx_fxo_stop_debouncing_polarity(struct wcaxx *wc,
- struct wcaxx_module *const mod)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- switch (fxo->polarity_state) {
- case UNKNOWN_POLARITY:
- break;
- case POLARITY_DEBOUNCE_POSITIVE:
- fxo->polarity_state = POLARITY_NEGATIVE;
- break;
- case POLARITY_POSITIVE:
- break;
- case POLARITY_DEBOUNCE_NEGATIVE:
- fxo->polarity_state = POLARITY_POSITIVE;
- break;
- case POLARITY_NEGATIVE:
- break;
- };
- }
- static void
- wcaxx_fxo_check_polarity(struct wcaxx *wc, struct wcaxx_module *const mod,
- const bool positive_polarity)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- switch (fxo->polarity_state) {
- case UNKNOWN_POLARITY:
- fxo->polarity_state = (positive_polarity) ? POLARITY_POSITIVE :
- POLARITY_NEGATIVE;
- break;
- case POLARITY_DEBOUNCE_POSITIVE:
- if (!positive_polarity) {
- fxo->polarity_state = POLARITY_NEGATIVE;
- } else if (time_after(wc->framecount, fxo->poldebounce_timer)) {
- fxo->polarity_state = POLARITY_POSITIVE;
- dahdi_qevent_lock(get_dahdi_chan(wc, mod),
- DAHDI_EVENT_POLARITY);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "%s: Polarity NEGATIVE -> POSITIVE\n",
- get_dahdi_chan(wc, mod)->name);
- }
- }
- break;
- case POLARITY_POSITIVE:
- if (!positive_polarity) {
- fxo->polarity_state = POLARITY_DEBOUNCE_NEGATIVE;
- fxo->poldebounce_timer = wc->framecount +
- POLARITY_DEBOUNCE;
- }
- break;
- case POLARITY_DEBOUNCE_NEGATIVE:
- if (positive_polarity) {
- fxo->polarity_state = POLARITY_POSITIVE;
- } else if (time_after(wc->framecount, fxo->poldebounce_timer)) {
- dahdi_qevent_lock(get_dahdi_chan(wc, mod),
- DAHDI_EVENT_POLARITY);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "%s: Polarity POSITIVE -> NEGATIVE\n",
- get_dahdi_chan(wc, mod)->name);
- }
- fxo->polarity_state = POLARITY_NEGATIVE;
- }
- break;
- case POLARITY_NEGATIVE:
- if (positive_polarity) {
- fxo->polarity_state = POLARITY_DEBOUNCE_POSITIVE;
- fxo->poldebounce_timer = wc->framecount +
- POLARITY_DEBOUNCE;
- }
- break;
- };
- }
- static bool is_neon_voltage_present(const struct fxo *fxo, u8 abs_voltage)
- {
- return (fxo->battery_state == BATTERY_PRESENT &&
- abs_voltage > neonmwi_level &&
- (0 == fxo->neonmwi_last_voltage ||
- ((fxo->line_voltage_status >= fxo->neonmwi_last_voltage -
- neonmwi_envelope) &&
- (fxo->line_voltage_status <= fxo->neonmwi_last_voltage +
- neonmwi_envelope)
- )
- )
- );
- }
- static void do_neon_monitor(struct wcaxx *wc,
- struct wcaxx_module *mod, u8 abs_voltage)
- {
- struct fxo *const fxo = &mod->mod.fxo;
- struct dahdi_chan *const chan = get_dahdi_chan(wc, mod);
- /* Look for 4 consecutive voltage readings where the voltage is over the
- * neon limit but does not vary greatly from the last reading */
- if (is_neon_voltage_present(fxo, abs_voltage)) {
- fxo->neonmwi_last_voltage = fxo->line_voltage_status;
- if (NEONMWI_ON_DEBOUNCE == fxo->neonmwi_debounce) {
- fxo->neonmwi_offcounter = neonmwi_offlimit_cycles;
- if (0 == fxo->neonmwi_state) {
- dahdi_qevent_lock(chan,
- DAHDI_EVENT_NEONMWI_ACTIVE);
- fxo->neonmwi_state = 1;
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "NEON MWI active for card %d\n",
- mod->card+1);
- }
- }
- fxo->neonmwi_debounce++;
- } else if (NEONMWI_ON_DEBOUNCE > fxo->neonmwi_debounce) {
- fxo->neonmwi_debounce++;
- } else {
- fxo->neonmwi_offcounter = neonmwi_offlimit_cycles;
- }
- } else {
- fxo->neonmwi_debounce = 0;
- fxo->neonmwi_last_voltage = 0;
- }
- /* If no neon mwi pulse for given period of time, indicte no neon mwi
- * state */
- if (fxo->neonmwi_state && 0 < fxo->neonmwi_offcounter) {
- fxo->neonmwi_offcounter--;
- if (0 == fxo->neonmwi_offcounter) {
- dahdi_qevent_lock(get_dahdi_chan(wc, mod),
- DAHDI_EVENT_NEONMWI_INACTIVE);
- fxo->neonmwi_state = 0;
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "NEON MWI cleared for card %d\n",
- mod->card+1);
- }
- }
- }
- }
- static void
- wcaxx_voicedaa_check_hook(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- signed char b;
- u8 abs_voltage;
- struct fxo *const fxo = &mod->mod.fxo;
- /* Try to track issues that plague slot one FXO's */
- b = fxo->hook_ring_shadow & 0x9b;
- if (fxo->offhook) {
- if (b != 0x9)
- wcaxx_setreg(wc, mod, 5, 0x9);
- } else {
- if (b != 0x8)
- wcaxx_setreg(wc, mod, 5, 0x8);
- wcaxx_fxo_ring_detect(wc, mod);
- }
- abs_voltage = abs(fxo->line_voltage_status);
- if (fxovoltage && time_after(wc->framecount, fxo->display_fxovoltage)) {
- /* Every 100 ms */
- fxo->display_fxovoltage = wc->framecount + 100;
- dev_info(&wc->xb.pdev->dev,
- "Port %d: Voltage: %d\n",
- mod->card + 1, fxo->line_voltage_status);
- }
- if (unlikely(DAHDI_RXSIG_INITIAL ==
- get_dahdi_chan(wc, mod)->rxhooksig)) {
- /*
- * dahdi-base will set DAHDI_RXSIG_INITIAL after a
- * DAHDI_STARTUP or DAHDI_CHANCONFIG ioctl so that new events
- * will be queued on the channel with the current received
- * hook state. Channels that use robbed-bit signalling always
- * report the current received state via the dahdi_rbsbits
- * call. Since we only call dahdi_hooksig when we've detected
- * a change to report, let's forget our current state in order
- * to force us to report it again via dahdi_hooksig.
- *
- */
- fxo->battery_state = BATTERY_UNKNOWN;
- }
- if (abs_voltage < battthresh) {
- wcaxx_fxo_stop_debouncing_polarity(wc, mod);
- wcaxx_check_battery_lost(wc, mod);
- } else {
- wcaxx_check_battery_present(wc, mod);
- wcaxx_fxo_check_polarity(wc, mod,
- (fxo->line_voltage_status > 0));
- }
- /* Look for neon mwi pulse */
- if (neonmwi_monitor && !fxo->offhook)
- do_neon_monitor(wc, mod, abs_voltage);
- #undef MS_PER_CHECK_HOOK
- }
- static void
- wcaxx_fxs_hooksig(struct wcaxx *wc, struct wcaxx_module *const mod,
- enum dahdi_txsig txsig)
- {
- int x = 0;
- unsigned long flags;
- struct fxs *const fxs = &mod->mod.fxs;
- spin_lock_irqsave(&wc->reglock, flags);
- switch (txsig) {
- case DAHDI_TXSIG_ONHOOK:
- switch (get_dahdi_chan(wc, mod)->sig) {
- case DAHDI_SIG_FXOGS:
- x = (POLARITY_XOR(fxs)) ?
- SLIC_LF_RING_OPEN :
- SLIC_LF_TIP_OPEN;
- break;
- case DAHDI_SIG_EM:
- case DAHDI_SIG_FXOKS:
- case DAHDI_SIG_FXOLS:
- default:
- x = fxs->idletxhookstate;
- break;
- }
- break;
- case DAHDI_TXSIG_OFFHOOK:
- switch (get_dahdi_chan(wc, mod)->sig) {
- case DAHDI_SIG_EM:
- x = (POLARITY_XOR(fxs)) ?
- SLIC_LF_ACTIVE_FWD :
- SLIC_LF_ACTIVE_REV;
- break;
- default:
- x = fxs->idletxhookstate;
- break;
- }
- break;
- case DAHDI_TXSIG_START:
- x = SLIC_LF_RINGING;
- break;
- case DAHDI_TXSIG_KEWL:
- x = SLIC_LF_OPEN;
- break;
- default:
- spin_unlock_irqrestore(&wc->reglock, flags);
- dev_notice(&wc->xb.pdev->dev,
- "Can't set tx state to %d\n", txsig);
- return;
- }
- if (x != fxs->lasttxhook) {
- fxs->lasttxhook = x | SLIC_LF_OPPENDING;
- mod->sethook = CMD_WR(LINE_STATE, fxs->lasttxhook);
- fxs->oppending_timeout = wc->framecount + 100;
- spin_unlock_irqrestore(&wc->reglock, flags);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Setting FXS hook state to %d (%02x) framecount=%ld\n",
- txsig, x, wc->framecount);
- }
- } else {
- spin_unlock_irqrestore(&wc->reglock, flags);
- }
- }
- static void
- wcaxx_fxs_off_hook(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- struct fxs *const fxs = &mod->mod.fxs;
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "fxs_off_hook: Card %d Going off hook\n", mod->card);
- }
- switch (fxs->lasttxhook) {
- case SLIC_LF_RINGING: /* Ringing */
- case SLIC_LF_OHTRAN_FWD: /* Forward On Hook Transfer */
- case SLIC_LF_OHTRAN_REV: /* Reverse On Hook Transfer */
- /* just detected OffHook, during Ringing or OnHookTransfer */
- fxs->idletxhookstate = POLARITY_XOR(fxs) ?
- SLIC_LF_ACTIVE_REV :
- SLIC_LF_ACTIVE_FWD;
- break;
- }
- if ((fxs->lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_OPEN)
- wcaxx_fxs_hooksig(wc, mod, DAHDI_TXSIG_OFFHOOK);
- dahdi_hooksig(get_dahdi_chan(wc, mod), DAHDI_RXSIG_OFFHOOK);
- #ifdef DEBUG
- if (robust)
- wcaxx_init_proslic(wc, mod, 1, 0, 1);
- #endif
- }
- /**
- * wcaxx_fxs_on_hook - Report on hook to DAHDI.
- * @wc: Board hosting the module.
- * @card: Index of the module / port to place on hook.
- *
- * If we are intentionally dropping battery to signal a forward
- * disconnect we do not want to place the line "On-Hook". In this
- * case, the core of DAHDI will place us on hook when one of the RBS
- * timers expires.
- *
- */
- static void
- wcaxx_fxs_on_hook(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "fxs_on_hook: Card %d Going on hook\n", mod->card);
- }
- if ((mod->mod.fxs.lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_OPEN)
- wcaxx_fxs_hooksig(wc, mod, DAHDI_TXSIG_ONHOOK);
- dahdi_hooksig(get_dahdi_chan(wc, mod), DAHDI_RXSIG_ONHOOK);
- }
- static void
- wcaxx_isr_misc_fxs(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- struct fxs *const fxs = &mod->mod.fxs;
- unsigned long flags;
- if (time_after(wc->framecount, fxs->check_alarm)) {
- /* Accept an alarm once per 10 seconds */
- fxs->check_alarm = wc->framecount + (1000*10);
- if (fxs->palarms)
- fxs->palarms--;
- }
- if (fxs->off_hook && !(fxs->hook_state_shadow & 1)) {
- wcaxx_fxs_on_hook(wc, mod);
- fxs->off_hook = 0;
- } else if (!fxs->off_hook && (fxs->hook_state_shadow & 1)) {
- wcaxx_fxs_off_hook(wc, mod);
- fxs->off_hook = 1;
- }
- wcaxx_proslic_check_oppending(wc, mod);
- if (time_after(wc->framecount, fxs->check_proslic)) {
- fxs->check_proslic = wc->framecount + 250; /* every 250ms */
- wcaxx_proslic_recheck_sanity(wc, mod);
- }
- if (SLIC_LF_RINGING == fxs->lasttxhook) {
- /* RINGing, prepare for OHT */
- fxs->ohttimer = wc->framecount + OHT_TIMER;
- /* OHT mode when idle */
- fxs->idletxhookstate = POLARITY_XOR(fxs) ? SLIC_LF_OHTRAN_REV :
- SLIC_LF_OHTRAN_FWD;
- } else if (fxs->oht_active) {
- /* check if still OnHook */
- if (!fxs->off_hook) {
- if (time_before(wc->framecount, fxs->ohttimer))
- return;
- /* Switch to active */
- fxs->idletxhookstate = POLARITY_XOR(fxs) ?
- SLIC_LF_ACTIVE_REV : SLIC_LF_ACTIVE_FWD;
- spin_lock_irqsave(&wc->reglock, flags);
- if (SLIC_LF_OHTRAN_FWD == fxs->lasttxhook) {
- /* Apply the change if appropriate */
- fxs->lasttxhook = SLIC_LF_OPPENDING |
- SLIC_LF_ACTIVE_FWD;
- /* Data enqueued here */
- mod->sethook = CMD_WR(LINE_STATE,
- fxs->lasttxhook);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d OnHookTransfer stop\n",
- mod->card);
- }
- } else if (SLIC_LF_OHTRAN_REV == fxs->lasttxhook) {
- /* Apply the change if appropriate */
- fxs->lasttxhook = SLIC_LF_OPPENDING |
- SLIC_LF_ACTIVE_REV;
- /* Data enqueued here */
- mod->sethook = CMD_WR(LINE_STATE,
- fxs->lasttxhook);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d OnHookTransfer stop\n",
- mod->card);
- }
- }
- spin_unlock_irqrestore(&wc->reglock, flags);
- } else {
- fxs->oht_active = 0;
- /* Switch to active */
- fxs->idletxhookstate = POLARITY_XOR(fxs) ?
- SLIC_LF_ACTIVE_REV : SLIC_LF_ACTIVE_FWD;
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d OnHookTransfer abort\n",
- mod->card);
- }
- }
- }
- }
- static void wcaxx_handle_receive(struct wcxb *xb, void *_frame)
- {
- int i, j;
- struct wcaxx *wc = container_of(xb, struct wcaxx, xb);
- u8 *const frame = _frame;
- wc->framecount++;
- if (time_after(wc->framecount, wc->module_poll_time)) {
- for (i = 0; i < wc->mods_per_board; i++) {
- struct wcaxx_module *const mod = &wc->mods[i];
- if (mod->mod_poll) {
- wcxb_spi_async(mod->spi, &mod->mod_poll->m);
- mod->mod_poll = NULL;
- }
- }
- wc->module_poll_time = wc->framecount + MODULE_POLL_TIME_MS;
- }
- /* TODO: This protection needs to be thought about. */
- if (!test_bit(DAHDI_FLAGBIT_REGISTERED, &wc->span.flags))
- return;
- for (j = 0; j < DAHDI_CHUNKSIZE; j++) {
- for (i = 0; i < wc->span.channels; i++) {
- wc->chans[i]->chan.readchunk[j] =
- frame[j*WCXB_DMA_CHAN_SIZE+(1+i*4)];
- }
- }
- for (i = 0; i < wc->span.channels; i++) {
- struct dahdi_chan *const c = wc->span.chans[i];
- __dahdi_ec_chunk(c, c->readchunk, c->readchunk, c->writechunk);
- }
- _dahdi_receive(&wc->span);
- return;
- }
- static void wcaxx_handle_transmit(struct wcxb *xb, void *_frame)
- {
- int i, j;
- struct wcaxx *wc = container_of(xb, struct wcaxx, xb);
- u8 *const frame = _frame;
- wcxb_spi_handle_interrupt(wc->master);
- /* TODO: This protection needs to be thought about. */
- if (!test_bit(DAHDI_FLAGBIT_REGISTERED, &wc->span.flags))
- return;
- _dahdi_transmit(&wc->span);
- for (j = 0; j < DAHDI_CHUNKSIZE; j++) {
- for (i = 0; i < wc->span.channels; i++) {
- struct dahdi_chan *c = &wc->chans[i]->chan;
- frame[j*WCXB_DMA_CHAN_SIZE+(1+i*4)] = c->writechunk[j];
- }
- }
- return;
- }
- static int wcaxx_voicedaa_insane(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- int blah;
- blah = wcaxx_getreg(wc, mod, 2);
- if (blah != 0x3)
- return -2;
- blah = wcaxx_getreg(wc, mod, 11);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "VoiceDAA System: %02x\n", blah & 0xf);
- }
- return 0;
- }
- static int
- wcaxx_proslic_insane(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- int blah, reg1, insane_report;
- insane_report = 0;
- blah = wcaxx_getreg(wc, mod, 0);
- if (blah != 0xff && (debug & DEBUG_CARD)) {
- dev_info(&wc->xb.pdev->dev,
- "ProSLIC on module %d, product %d, "
- "version %d\n", mod->card, (blah & 0x30) >> 4,
- (blah & 0xf));
- }
- #if 0
- if ((blah & 0x30) >> 4) {
- dev_info(&wc->xb.pdev->dev,
- "ProSLIC on module %d is not a 3210.\n", mod->card);
- return -1;
- }
- #endif
- if (((blah & 0xf) == 0) || ((blah & 0xf) == 0xf)) {
- /* SLIC not loaded */
- return -1;
- }
- /* let's be really sure this is an FXS before we continue */
- reg1 = wcaxx_getreg(wc, mod, 1);
- if ((0x80 != (blah & 0xf0)) || (0x88 != reg1)) {
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "DEBUG: not FXS b/c reg0=%x or "
- "reg1 != 0x88 (%x).\n", blah, reg1);
- }
- return -1;
- }
- blah = wcaxx_getreg(wc, mod, 8);
- if (blah != 0x2) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d insane (1) %d should be 2\n",
- mod->card, blah);
- return -1;
- } else if (insane_report) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d Reg 8 Reads %d Expected "
- "is 0x2\n", mod->card, blah);
- }
- blah = wcaxx_getreg(wc, mod, 64);
- if (blah != 0x0) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d insane (2)\n",
- mod->card);
- return -1;
- } else if (insane_report) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d Reg 64 Reads %d Expected "
- "is 0x0\n", mod->card, blah);
- }
- blah = wcaxx_getreg(wc, mod, 11);
- if (blah != 0x33) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d insane (3)\n", mod->card);
- return -1;
- } else if (insane_report) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC on module %d Reg 11 Reads %d "
- "Expected is 0x33\n", mod->card, blah);
- }
- /* Just be sure it's setup right. */
- wcaxx_setreg(wc, mod, 30, 0);
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "ProSLIC on module %d seems sane.\n", mod->card);
- }
- return 0;
- }
- static int
- wcaxx_proslic_powerleak_test(struct wcaxx *wc,
- struct wcaxx_module *const mod)
- {
- unsigned long start;
- unsigned char vbat;
- /* Turn off linefeed */
- wcaxx_setreg(wc, mod, LINE_STATE, 0);
- /* Power down */
- wcaxx_setreg(wc, mod, 14, 0x10);
- start = jiffies;
- /* TODO: Why is this sleep necessary. Without it, the first read
- * comes back with a 0 value. */
- msleep(20);
- while ((vbat = wcaxx_getreg(wc, mod, 82)) > 0x6) {
- if (time_after(jiffies, start + HZ/4))
- break;
- }
- if (vbat < 0x06) {
- dev_notice(&wc->xb.pdev->dev,
- "Excessive leakage detected on module %d: %d "
- "volts (%02x) after %d ms\n", mod->card,
- 376 * vbat / 1000, vbat,
- (int)((jiffies - start) * 1000 / HZ));
- return -1;
- } else if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Post-leakage voltage: %d volts\n", 376 * vbat / 1000);
- }
- return 0;
- }
- static int wcaxx_powerup_proslic(struct wcaxx *wc,
- struct wcaxx_module *mod, int fast)
- {
- unsigned char vbat;
- unsigned long origjiffies;
- int lim;
- /* Set period of DC-DC converter to 1/64 khz */
- wcaxx_setreg(wc, mod, 92, 0xc0 /* was 0xff */);
- /* Wait for VBat to powerup */
- origjiffies = jiffies;
- /* Disable powerdown */
- wcaxx_setreg(wc, mod, 14, 0);
- /* If fast, don't bother checking anymore */
- if (fast)
- return 0;
- while ((vbat = wcaxx_getreg(wc, mod, 82)) < 0xc0) {
- /* Wait no more than 500ms */
- if ((jiffies - origjiffies) > HZ/2)
- break;
- }
- if (vbat < 0xc0) {
- dev_notice(&wc->xb.pdev->dev, "ProSLIC on module %d failed to powerup within %d ms (%d mV only)\n\n -- DID YOU REMEMBER TO PLUG IN THE HD POWER CABLE TO THE TDM CARD??\n",
- mod->card, (int)(((jiffies - origjiffies) * 1000 / HZ)),
- vbat * 375);
- return -1;
- } else if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "ProSLIC on module %d powered up to -%d volts (%02x) "
- "in %d ms\n", mod->card, vbat * 376 / 1000, vbat,
- (int)(((jiffies - origjiffies) * 1000 / HZ)));
- }
- /* Proslic max allowed loop current, reg 71 LOOP_I_LIMIT */
- /* If out of range, just set it to the default value */
- lim = (loopcurrent - 20) / 3;
- if (loopcurrent > 41) {
- lim = 0;
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Loop current out of range! Setting to default 20mA!\n");
- }
- } else if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Loop current set to %dmA!\n", (lim*3)+20);
- }
- wcaxx_setreg(wc, mod, LOOP_I_LIMIT, lim);
- /* Engage DC-DC converter */
- wcaxx_setreg(wc, mod, 93, 0x19 /* was 0x19 */);
- return 0;
- }
- static int
- wcaxx_proslic_manual_calibrate(struct wcaxx *wc,
- struct wcaxx_module *const mod)
- {
- unsigned long origjiffies;
- unsigned char i;
- /* Disable all interupts in DR21-23 */
- wcaxx_setreg(wc, mod, 21, 0);
- wcaxx_setreg(wc, mod, 22, 0);
- wcaxx_setreg(wc, mod, 23, 0);
- wcaxx_setreg(wc, mod, 64, 0);
- /* (0x18) Calibrations without the ADC and DAC offset and without
- * common mode calibration. */
- wcaxx_setreg(wc, mod, 97, 0x18);
- /* (0x47) Calibrate common mode and differential DAC mode DAC + ILIM */
- wcaxx_setreg(wc, mod, 96, 0x47);
- origjiffies = jiffies;
- while (wcaxx_getreg(wc, mod, 96) != 0) {
- if ((jiffies-origjiffies) > 80)
- return -1;
- }
- /* Initialized DR 98 and 99 to get consistant results. 98 and 99 are
- * the results registers and the search should have same intial
- * conditions.
- */
- /******* The following is the manual gain mismatch calibration ********/
- /******* This is also available as a function *************************/
- msleep(20);
- wcaxx_proslic_setreg_indirect(wc, mod, 88, 0);
- wcaxx_proslic_setreg_indirect(wc, mod, 89, 0);
- wcaxx_proslic_setreg_indirect(wc, mod, 90, 0);
- wcaxx_proslic_setreg_indirect(wc, mod, 91, 0);
- wcaxx_proslic_setreg_indirect(wc, mod, 92, 0);
- wcaxx_proslic_setreg_indirect(wc, mod, 93, 0);
- /* This is necessary if the calibration occurs other than at reset */
- wcaxx_setreg(wc, mod, 98, 0x10);
- wcaxx_setreg(wc, mod, 99, 0x10);
- for (i = 0x1f; i > 0; i--) {
- wcaxx_setreg(wc, mod, 98, i);
- msleep(40);
- if ((wcaxx_getreg(wc, mod, 88)) == 0)
- break;
- }
- for (i = 0x1f; i > 0; i--) {
- wcaxx_setreg(wc, mod, 99, i);
- msleep(40);
- if ((wcaxx_getreg(wc, mod, 89)) == 0)
- break;
- }
- /******** The preceding is the manual gain mismatch calibration *******/
- /******** The following is the longitudinal Balance Cal ***************/
- wcaxx_setreg(wc, mod, 64, 1);
- msleep(100);
- wcaxx_setreg(wc, mod, 64, 0);
- /* enable interrupt for the balance Cal */
- wcaxx_setreg(wc, mod, 23, 0x4);
- /* this is a singular calibration bit for longitudinal calibration */
- wcaxx_setreg(wc, mod, 97, 0x1);
- wcaxx_setreg(wc, mod, 96, 0x40);
- wcaxx_getreg(wc, mod, 96); /* Read Reg 96 just cause */
- wcaxx_setreg(wc, mod, 21, 0xFF);
- wcaxx_setreg(wc, mod, 22, 0xFF);
- wcaxx_setreg(wc, mod, 23, 0xFF);
- /**The preceding is the longitudinal Balance Cal***/
- return 0;
- }
- static int
- wcaxx_proslic_calibrate(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- unsigned long origjiffies;
- int x;
- /* Perform all calibrations */
- wcaxx_setreg(wc, mod, 97, 0x1f);
- /* Begin, no speedup */
- wcaxx_setreg(wc, mod, 96, 0x5f);
- /* Wait for it to finish */
- origjiffies = jiffies;
- while (wcaxx_getreg(wc, mod, 96)) {
- if (time_after(jiffies, (origjiffies + (2*HZ)))) {
- dev_notice(&wc->xb.pdev->dev,
- "Timeout waiting for calibration of "
- "module %d\n", mod->card);
- return -1;
- }
- }
- if (debug & DEBUG_CARD) {
- /* Print calibration parameters */
- dev_info(&wc->xb.pdev->dev,
- "Calibration Vector Regs 98 - 107:\n");
- for (x = 98; x < 108; x++) {
- dev_info(&wc->xb.pdev->dev,
- "%d: %02x\n", x, wcaxx_getreg(wc, mod, x));
- }
- }
- return 0;
- }
- /*********************************************************************
- * Set the hwgain on the analog modules
- *
- * card = the card position for this module (0-23)
- * gain = gain in dB x10 (e.g. -3.5dB would be gain=-35)
- * tx = (0 for rx; 1 for tx)
- *
- *******************************************************************/
- static int
- wcaxx_set_hwgain(struct wcaxx *wc, struct wcaxx_module *mod,
- __s32 gain, __u32 tx)
- {
- if (mod->type != FXO) {
- dev_notice(&wc->xb.pdev->dev,
- "Cannot adjust gain. Unsupported module type!\n");
- return -1;
- }
- if (tx) {
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "setting FXO tx gain for card=%d to %d\n",
- mod->card, gain);
- }
- if (gain >= -150 && gain <= 0) {
- wcaxx_setreg(wc, mod, 38, 16 + (gain / -10));
- wcaxx_setreg(wc, mod, 40, 16 + (-gain % 10));
- } else if (gain <= 120 && gain > 0) {
- wcaxx_setreg(wc, mod, 38, gain/10);
- wcaxx_setreg(wc, mod, 40, (gain%10));
- } else {
- dev_notice(&wc->xb.pdev->dev,
- "FXO tx gain is out of range (%d)\n", gain);
- return -1;
- }
- } else { /* rx */
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "setting FXO rx gain for card=%d to %d\n",
- mod->card, gain);
- }
- if (gain >= -150 && gain <= 0) {
- wcaxx_setreg(wc, mod, 39, 16 + (gain / -10));
- wcaxx_setreg(wc, mod, 41, 16 + (-gain % 10));
- } else if (gain <= 120 && gain > 0) {
- wcaxx_setreg(wc, mod, 39, gain/10);
- wcaxx_setreg(wc, mod, 41, (gain%10));
- } else {
- dev_notice(&wc->xb.pdev->dev,
- "FXO rx gain is out of range (%d)\n", gain);
- return -1;
- }
- }
- return 0;
- }
- static int set_lasttxhook_interruptible(struct wcaxx *wc, struct fxs *fxs,
- unsigned newval, int *psethook)
- {
- int res = 0;
- unsigned long flags;
- int timeout = 0;
- do {
- spin_lock_irqsave(&wc->reglock, flags);
- if (SLIC_LF_OPPENDING & fxs->lasttxhook) {
- spin_unlock_irqrestore(&wc->reglock, flags);
- if (timeout++ > 100)
- return -1;
- msleep(100);
- } else {
- fxs->lasttxhook = (newval & SLIC_LF_SETMASK) |
- SLIC_LF_OPPENDING;
- *psethook = CMD_WR(LINE_STATE, fxs->lasttxhook);
- spin_unlock_irqrestore(&wc->reglock, flags);
- break;
- }
- } while (1);
- return res;
- }
- /* Must be called from within an interruptible context */
- static int set_vmwi(struct wcaxx *wc, struct wcaxx_module *const mod)
- {
- int x;
- struct fxs *const fxs = &mod->mod.fxs;
- /* Presently only supports line reversal MWI */
- if ((fxs->vmwi_active_messages) &&
- (fxs->vmwisetting.vmwi_type & DAHDI_VMWI_LREV))
- fxs->vmwi_linereverse = 1;
- else
- fxs->vmwi_linereverse = 0;
- /* Set line polarity for new VMWI state */
- if (POLARITY_XOR(fxs)) {
- fxs->idletxhookstate |= SLIC_LF_REVMASK;
- /* Do not set while currently ringing or open */
- if (((fxs->lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_RINGING) &&
- ((fxs->lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_OPEN)) {
- x = fxs->lasttxhook;
- x |= SLIC_LF_REVMASK;
- set_lasttxhook_interruptible(wc, fxs, x, &mod->sethook);
- }
- } else {
- fxs->idletxhookstate &= ~SLIC_LF_REVMASK;
- /* Do not set while currently ringing or open */
- if (((fxs->lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_RINGING) &&
- ((fxs->lasttxhook & SLIC_LF_SETMASK) != SLIC_LF_OPEN)) {
- x = fxs->lasttxhook;
- x &= ~SLIC_LF_REVMASK;
- set_lasttxhook_interruptible(wc, fxs, x, &mod->sethook);
- }
- }
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "Setting VMWI on channel %d, messages=%d, lrev=%d\n",
- mod->card, fxs->vmwi_active_messages,
- fxs->vmwi_linereverse);
- }
- return 0;
- }
- static void
- wcaxx_voicedaa_set_ts(struct wcaxx *wc, struct wcaxx_module *mod, int ts)
- {
- /* 34 bits from framesysc to the first channel, 8 bits in each ts * (th
- * e timeslot we're assigning + 1 to skip for VPMOCT issue on first
- * timeslot + 3 in that there are 4 bytes assigned for each timeslot on
- * framer which was copied to this card */
- /* 34 + 8 * (ts + 1 + 3) */
- wcaxx_setreg(wc, mod, 34, (ts * 8 + 42 + (ts * 3 * 8)) & 0xff);
- wcaxx_setreg(wc, mod, 35, (ts * 8 + 42 + (ts * 3 * 8)) >> 8);
- wcaxx_setreg(wc, mod, 36, (ts * 8 + 42 + (ts * 3 * 8)) & 0xff);
- wcaxx_setreg(wc, mod, 37, (ts * 8 + 42 + (ts * 3 * 8)) >> 8);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "voicedaa: card %d new timeslot: %d\n",
- mod->card + 1, ts);
- }
- }
- static int
- wcaxx_init_voicedaa(struct wcaxx *wc, struct wcaxx_module *mod,
- int fast, int manual, int sane)
- {
- unsigned char reg16 = 0, reg26 = 0, reg30 = 0, reg31 = 0;
- unsigned long flags;
- unsigned long newjiffies;
- /* Send a short write to the device in order to reset the SPI state
- * machine. It may be out of sync since the driver was probing for an
- * FXS device on that chip select. */
- /* wcxb_spi_short_write(mod->spi); */
- spin_lock_irqsave(&wc->reglock, flags);
- mod->type = FXO;
- spin_unlock_irqrestore(&wc->reglock, flags);
- if (!sane && wcaxx_voicedaa_insane(wc, mod))
- return -2;
- /* Software reset */
- wcaxx_setreg(wc, mod, 1, 0x80);
- msleep(100);
- /* Set On-hook speed, Ringer impedence, and ringer threshold */
- reg16 |= (fxo_modes[_opermode].ohs << 6);
- reg16 |= (fxo_modes[_opermode].rz << 1);
- reg16 |= (fxo_modes[_opermode].rt);
- wcaxx_setreg(wc, mod, 16, reg16);
- /* Enable ring detector full-wave rectifier mode */
- wcaxx_setreg(wc, mod, 18, 2);
- wcaxx_setreg(wc, mod, 24, 0);
- /* Set DC Termination:
- Tip/Ring voltage adjust, minimum operational current, current
- limitation */
- reg26 |= (fxo_modes[_opermode].dcv << 6);
- reg26 |= (fxo_modes[_opermode].mini << 4);
- reg26 |= (fxo_modes[_opermode].ilim << 1);
- wcaxx_setreg(wc, mod, 26, reg26);
- /* Set AC Impedence */
- reg30 = (fxo_modes[_opermode].acim);
- wcaxx_setreg(wc, mod, 30, reg30);
- /* Misc. DAA parameters */
- /* If fast pickup is set, then the off hook counter will be set to 8
- * ms, otherwise 128 ms. */
- reg31 = (fastpickup) ? 0xe3 : 0xa3;
- reg31 |= (fxo_modes[_opermode].ohs2 << 3);
- wcaxx_setreg(wc, mod, 31, reg31);
- wcaxx_voicedaa_set_ts(wc, mod, mod->card);
- /* Enable ISO-Cap */
- wcaxx_setreg(wc, mod, 6, 0x00);
- /* Turn off the calibration delay when fastpickup is enabled. */
- if (fastpickup)
- wcaxx_setreg(wc, mod, 17, wcaxx_getreg(wc, mod, 17) | 0x20);
- /* Wait 2000ms for ISO-cap to come up */
- newjiffies = jiffies + msecs_to_jiffies(2000);
- while (time_before(jiffies, newjiffies) &&
- !(wcaxx_getreg(wc, mod, 11) & 0xf0))
- msleep(100);
- if (!(wcaxx_getreg(wc, mod, 11) & 0xf0)) {
- dev_notice(&wc->xb.pdev->dev, "VoiceDAA did not bring up ISO link properly!\n");
- return -1;
- }
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev, "ISO-Cap is now up, line side: %02x rev %02x\n",
- wcaxx_getreg(wc, mod, 11) >> 4,
- (wcaxx_getreg(wc, mod, 13) >> 2) & 0xf);
- }
- /* Enable on-hook line monitor */
- wcaxx_setreg(wc, mod, 5, 0x08);
- /* Take values for fxotxgain and fxorxgain and apply them to module */
- wcaxx_set_hwgain(wc, mod, fxotxgain, 1);
- wcaxx_set_hwgain(wc, mod, fxorxgain, 0);
- #ifdef DEBUG
- if (digitalloopback) {
- dev_info(&wc->xb.pdev->dev,
- "Turning on digital loopback for port %d.\n",
- mod->card + 1);
- wcaxx_setreg(wc, mod, 10, 0x01);
- }
- #endif
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "DEBUG fxotxgain:%i.%i fxorxgain:%i.%i\n",
- (wcaxx_getreg(wc, mod, 38)/16) ?
- -(wcaxx_getreg(wc, mod, 38) - 16) :
- wcaxx_getreg(wc, mod, 38),
- (wcaxx_getreg(wc, mod, 40)/16) ?
- -(wcaxx_getreg(wc, mod, 40) - 16) :
- wcaxx_getreg(wc, mod, 40),
- (wcaxx_getreg(wc, mod, 39)/16) ?
- -(wcaxx_getreg(wc, mod, 39) - 16) :
- wcaxx_getreg(wc, mod, 39),
- (wcaxx_getreg(wc, mod, 41)/16) ?
- -(wcaxx_getreg(wc, mod, 41) - 16) :
- wcaxx_getreg(wc, mod, 41));
- }
- return 0;
- }
- static void
- wcaxx_proslic_set_ts(struct wcaxx *wc, struct wcaxx_module *mod, int ts)
- {
- /* Tx Start low byte 0 */
- wcaxx_setreg(wc, mod, 2, (ts * 8 + 42 + (ts * 3 * 8)) & 0xff);
- /* Tx Start high byte 0 */
- wcaxx_setreg(wc, mod, 3, (ts * 8 + 42 + (ts * 3 * 8)) >> 8);
- /* Rx Start low byte 0 */
- wcaxx_setreg(wc, mod, 4, (ts * 8 + 42 + (ts * 3 * 8)) & 0xff);
- /* Rx Start high byte 0 */
- wcaxx_setreg(wc, mod, 5, (ts * 8 + 42 + (ts * 3 * 8)) >> 8);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "proslic: card %d new timeslot: %d\n",
- mod->card + 1, ts);
- }
- }
- static int
- wcaxx_init_proslic(struct wcaxx *wc, struct wcaxx_module *const mod,
- int fast, int manual, int sane)
- {
- struct fxs *const fxs = &mod->mod.fxs;
- unsigned short tmp[5];
- unsigned long flags;
- unsigned char r19, r9;
- int x;
- int fxsmode = 0;
- int addresses[ARRAY_SIZE(fxs->calregs.vals)];
- #if 0 /* TODO */
- if (wc->mods[mod->card & 0xfc].type == QRV)
- return -2;
- #endif
- spin_lock_irqsave(&wc->reglock, flags);
- mod->type = FXS;
- spin_unlock_irqrestore(&wc->reglock, flags);
- /* msleep(100); */
- /* Sanity check the ProSLIC */
- if (!sane && wcaxx_proslic_insane(wc, mod))
- return -2;
- /* Initialize VMWI settings */
- memset(&(fxs->vmwisetting), 0, sizeof(fxs->vmwisetting));
- fxs->vmwi_linereverse = 0;
- /* By default, don't send on hook */
- if (!reversepolarity != !fxs->reversepolarity)
- fxs->idletxhookstate = SLIC_LF_ACTIVE_REV;
- else
- fxs->idletxhookstate = SLIC_LF_ACTIVE_FWD;
- if (sane) {
- /* Make sure we turn off the DC->DC converter to prevent
- * anything from blowing up */
- wcaxx_setreg(wc, mod, 14, 0x10);
- }
- if (wcaxx_proslic_init_indirect_regs(wc, mod)) {
- dev_info(&wc->xb.pdev->dev,
- "Indirect Registers failed to initialize on "
- "module %d.\n", mod->card);
- return -1;
- }
- /* Clear scratch pad area */
- wcaxx_proslic_setreg_indirect(wc, mod, 97, 0);
- /* Clear digital loopback */
- wcaxx_setreg(wc, mod, 8, 0);
- /* Revision C optimization */
- wcaxx_setreg(wc, mod, 108, 0xeb);
- /* Disable automatic VBat switching for safety to prevent
- * Q7 from accidently turning on and burning out.
- * If pulse dialing has trouble at high REN loads change this to 0x17 */
- wcaxx_setreg(wc, mod, 67, 0x07);
- /* Turn off Q7 */
- wcaxx_setreg(wc, mod, 66, 1);
- /* Flush ProSLIC digital filters by setting to clear, while
- saving old values */
- for (x = 0; x < 5; x++) {
- tmp[x] = wcaxx_proslic_getreg_indirect(wc, mod, x + 35);
- wcaxx_proslic_setreg_indirect(wc, mod, x + 35, 0x8000);
- }
- /* Power up the DC-DC converter */
- if (wcaxx_powerup_proslic(wc, mod, fast)) {
- dev_notice(&wc->xb.pdev->dev,
- "Unable to do INITIAL ProSLIC powerup on "
- "module %d\n", mod->card);
- return -1;
- }
- if (!fast) {
- /* Check for power leaks */
- if (wcaxx_proslic_powerleak_test(wc, mod)) {
- dev_notice(&wc->xb.pdev->dev,
- "ProSLIC module %d failed leakage test. "
- "Check for short circuit\n", mod->card);
- }
- /* Power up again */
- if (wcaxx_powerup_proslic(wc, mod, fast)) {
- dev_notice(&wc->xb.pdev->dev,
- "Unable to do FINAL ProSLIC powerup on "
- "module %d\n", mod->card);
- return -1;
- }
- #ifndef NO_CALIBRATION
- /* Perform calibration */
- if (manual) {
- if (wcaxx_proslic_manual_calibrate(wc, mod)) {
- dev_dbg(&wc->xb.pdev->dev,
- "Proslic failed on Manual Calibration\n");
- if (wcaxx_proslic_manual_calibrate(wc, mod)) {
- dev_notice(&wc->xb.pdev->dev,
- "Proslic Failed on Second Attempt to Calibrate Manually. (Try -DNO_CALIBRATION in Makefile)\n");
- return -1;
- }
- dev_info(&wc->xb.pdev->dev,
- "Proslic Passed Manual Calibration on Second Attempt\n");
- }
- } else {
- if (wcaxx_proslic_calibrate(wc, mod)) {
- dev_dbg(&wc->xb.pdev->dev,
- "ProSlic died on Auto Calibration.\n");
- if (wcaxx_proslic_calibrate(wc, mod)) {
- dev_notice(&wc->xb.pdev->dev,
- "Proslic Failed on Second Attempt to Auto Calibrate\n");
- return -1;
- }
- dev_info(&wc->xb.pdev->dev,
- "Proslic Passed Auto Calibration on Second Attempt\n");
- }
- }
- /* Perform DC-DC calibration */
- wcaxx_setreg(wc, mod, 93, 0x99);
- r19 = wcaxx_getreg(wc, mod, 107);
- if ((r19 < 0x2) || (r19 > 0xd)) {
- dev_notice(&wc->xb.pdev->dev,
- "DC-DC cal has a surprising direct 107 of 0x%02x!\n",
- r19);
- wcaxx_setreg(wc, mod, 107, 0x8);
- }
- /* Save calibration vectors */
- for (x = 0; x < ARRAY_SIZE(addresses); x++)
- addresses[x] = 96 + x;
- wcaxx_getregs(wc, mod, addresses, ARRAY_SIZE(addresses));
- for (x = 0; x < ARRAY_SIZE(fxs->calregs.vals); x++)
- fxs->calregs.vals[x] = addresses[x];
- #endif
- } else {
- /* Restore calibration registers */
- for (x = 0; x < ARRAY_SIZE(fxs->calregs.vals); x++)
- wcaxx_setreg(wc, mod, 96 + x, fxs->calregs.vals[x]);
- }
- /* Calibration complete, restore original values */
- for (x = 0; x < 5; x++)
- wcaxx_proslic_setreg_indirect(wc, mod, x + 35, tmp[x]);
- if (wcaxx_proslic_verify_indirect_regs(wc, mod)) {
- dev_info(&wc->xb.pdev->dev, "Indirect Registers failed verification.\n");
- return -1;
- }
- /* U-Law 8-bit interface */
- wcaxx_proslic_set_ts(wc, mod, mod->card);
- wcaxx_setreg(wc, mod, 18, 0xff); /* clear all interrupt */
- wcaxx_setreg(wc, mod, 19, 0xff);
- wcaxx_setreg(wc, mod, 20, 0xff);
- wcaxx_setreg(wc, mod, 22, 0xff);
- wcaxx_setreg(wc, mod, 73, 0x04);
- wcaxx_setreg(wc, mod, 69, 0x4);
- if (fxshonormode) {
- static const int ACIM2TISS[16] = { 0x0, 0x1, 0x4, 0x5, 0x7,
- 0x0, 0x0, 0x6, 0x0, 0x0,
- 0x0, 0x2, 0x0, 0x3 };
- fxsmode = ACIM2TISS[fxo_modes[_opermode].acim];
- wcaxx_setreg(wc, mod, 10, 0x08 | fxsmode);
- if (fxo_modes[_opermode].ring_osc) {
- wcaxx_proslic_setreg_indirect(wc, mod, 20,
- fxo_modes[_opermode].ring_osc);
- }
- if (fxo_modes[_opermode].ring_x) {
- wcaxx_proslic_setreg_indirect(wc, mod, 21,
- fxo_modes[_opermode].ring_x);
- }
- }
- if (lowpower)
- wcaxx_setreg(wc, mod, 72, 0x10);
- if (fastringer) {
- /* Speed up Ringer */
- wcaxx_proslic_setreg_indirect(wc, mod, 20, 0x7e6d);
- wcaxx_proslic_setreg_indirect(wc, mod, 21, 0x01b9);
- /* Beef up Ringing voltage to 89V */
- if (boostringer) {
- wcaxx_setreg(wc, mod, 74, 0x3f);
- if (wcaxx_proslic_setreg_indirect(wc, mod, 21, 0x247))
- return -1;
- dev_info(&wc->xb.pdev->dev,
- "Boosting fast ringer on slot %d (89V peak)\n",
- mod->card + 1);
- } else if (lowpower) {
- if (wcaxx_proslic_setreg_indirect(wc, mod, 21, 0x14b))
- return -1;
- dev_info(&wc->xb.pdev->dev,
- "Reducing fast ring power on slot %d "
- "(50V peak)\n", mod->card + 1);
- } else
- dev_info(&wc->xb.pdev->dev,
- "Speeding up ringer on slot %d (25Hz)\n",
- mod->card + 1);
- } else {
- /* Beef up Ringing voltage to 89V */
- if (boostringer) {
- wcaxx_setreg(wc, mod, 74, 0x3f);
- if (wcaxx_proslic_setreg_indirect(wc, mod, 21, 0x1d1))
- return -1;
- dev_info(&wc->xb.pdev->dev,
- "Boosting ringer on slot %d (89V peak)\n",
- mod->card + 1);
- } else if (lowpower) {
- if (wcaxx_proslic_setreg_indirect(wc, mod, 21, 0x108))
- return -1;
- dev_info(&wc->xb.pdev->dev,
- "Reducing ring power on slot %d "
- "(50V peak)\n", mod->card + 1);
- }
- }
- if (fxstxgain || fxsrxgain) {
- r9 = wcaxx_getreg(wc, mod, 9);
- switch (fxstxgain) {
- case 35:
- r9 += 8;
- break;
- case -35:
- r9 += 4;
- break;
- case 0:
- break;
- }
- switch (fxsrxgain) {
- case 35:
- r9 += 2;
- break;
- case -35:
- r9 += 1;
- break;
- case 0:
- break;
- }
- wcaxx_setreg(wc, mod, 9, r9);
- }
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "DEBUG: fxstxgain:%s fxsrxgain:%s\n",
- ((wcaxx_getreg(wc, mod, 9) / 8) == 1) ?
- "3.5" : ((wcaxx_getreg(wc, mod, 9) / 4) == 1) ?
- "-3.5" : "0.0",
- ((wcaxx_getreg(wc, mod, 9) / 2) == 1) ?
- "3.5" : ((wcaxx_getreg(wc, mod, 9) % 2) ?
- "-3.5" : "0.0"));
- }
- fxs->lasttxhook = fxs->idletxhookstate;
- wcaxx_setreg(wc, mod, LINE_STATE, fxs->lasttxhook);
- /* Preset the shadow register so that we won't get a power alarm when
- * we finish initialization, otherwise the line state register may not
- * have been read yet. */
- fxs->linefeed_control_shadow = fxs->lasttxhook;
- return 0;
- }
- static void wcaxx_get_fxs_regs(struct wcaxx *wc, struct wcaxx_module *mod,
- struct wctdm_regs *regs)
- {
- int x;
- for (x = 0; x < NUM_INDIRECT_REGS; x++)
- regs->indirect[x] = wcaxx_proslic_getreg_indirect(wc, mod, x);
- for (x = 0; x < NUM_REGS; x++)
- regs->direct[x] = wcaxx_getreg(wc, mod, x);
- }
- static void wcaxx_get_fxo_regs(struct wcaxx *wc, struct wcaxx_module *mod,
- struct wctdm_regs *regs)
- {
- const unsigned int NUM_FXO_REGS = 60;
- int x;
- for (x = 0; x < NUM_FXO_REGS; x++)
- regs->direct[x] = wcaxx_getreg(wc, mod, x);
- }
- static int
- wcaxx_ioctl(struct dahdi_chan *chan, unsigned int cmd, unsigned long data)
- {
- struct wctdm_stats stats;
- struct wctdm_regop regop;
- struct wctdm_echo_coefs echoregs;
- struct dahdi_hwgain hwgain;
- struct wcaxx *wc = chan->pvt;
- int x;
- struct wcaxx_module *const mod = &wc->mods[chan->chanpos - 1];
- struct fxs *const fxs = &mod->mod.fxs;
- switch (cmd) {
- case DAHDI_ONHOOKTRANSFER:
- if (mod->type != FXS)
- return -EINVAL;
- if (get_user(x, (__user int *) data))
- return -EFAULT;
- /* Active mode when idle */
- fxs->idletxhookstate = POLARITY_XOR(fxs) ?
- SLIC_LF_ACTIVE_REV :
- SLIC_LF_ACTIVE_FWD;
- if (fxs_lf(fxs, ACTIVE_FWD) || fxs_lf(fxs, ACTIVE_REV)) {
- int res;
- res = set_lasttxhook_interruptible(wc, fxs,
- (POLARITY_XOR(fxs) ?
- SLIC_LF_OHTRAN_REV : SLIC_LF_OHTRAN_FWD),
- &mod->sethook);
- if (debug & DEBUG_CARD) {
- if (res) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d TIMEOUT: "
- "OnHookTransfer start\n",
- chan->chanpos - 1);
- } else {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d OnHookTransfer "
- "start\n", chan->chanpos - 1);
- }
- }
- }
- fxs->ohttimer = wc->framecount + x;
- fxs->oht_active = 1;
- break;
- case DAHDI_VMWI_CONFIG:
- if (mod->type != FXS)
- return -EINVAL;
- if (copy_from_user(&(fxs->vmwisetting),
- (__user void *)data,
- sizeof(fxs->vmwisetting)))
- return -EFAULT;
- set_vmwi(wc, mod);
- break;
- case DAHDI_VMWI:
- if (mod->type != FXS)
- return -EINVAL;
- if (get_user(x, (__user int *) data))
- return -EFAULT;
- if (0 > x)
- return -EFAULT;
- fxs->vmwi_active_messages = x;
- set_vmwi(wc, mod);
- break;
- case WCTDM_GET_STATS:
- if (mod->type == FXS) {
- stats.tipvolt = wcaxx_getreg(wc, mod, 80) * -376;
- stats.ringvolt = wcaxx_getreg(wc, mod, 81) * -376;
- stats.batvolt = wcaxx_getreg(wc, mod, 82) * -376;
- } else if (mod->type == FXO) {
- stats.tipvolt = (s8)wcaxx_getreg(wc, mod, 29) * 1000;
- stats.ringvolt = (s8)wcaxx_getreg(wc, mod, 29) * 1000;
- stats.batvolt = (s8)wcaxx_getreg(wc, mod, 29) * 1000;
- } else
- return -EINVAL;
- if (copy_to_user((__user void *) data, &stats, sizeof(stats)))
- return -EFAULT;
- break;
- case WCTDM_GET_REGS:
- {
- struct wctdm_regs *regs = kzalloc(sizeof(*regs), GFP_KERNEL);
- if (!regs)
- return -ENOMEM;
- if (mod->type == FXS)
- wcaxx_get_fxs_regs(wc, mod, regs);
- else
- wcaxx_get_fxo_regs(wc, mod, regs);
- if (copy_to_user((__user void *)data, regs, sizeof(*regs))) {
- kfree(regs);
- return -EFAULT;
- }
- kfree(regs);
- break;
- }
- case WCTDM_SET_REG:
- if (copy_from_user(®op, (__user void *) data, sizeof(regop)))
- return -EFAULT;
- if (regop.indirect) {
- if (mod->type != FXS)
- return -EINVAL;
- dev_info(&wc->xb.pdev->dev,
- "Setting indirect %d to 0x%04x on %d\n",
- regop.reg, regop.val, chan->chanpos);
- wcaxx_proslic_setreg_indirect(wc, mod, regop.reg,
- regop.val);
- } else {
- regop.val &= 0xff;
- if (regop.reg == LINE_STATE) {
- /* Set feedback register to indicate the new
- * state that is being set */
- fxs->lasttxhook = (regop.val & 0x0f) |
- SLIC_LF_OPPENDING;
- }
- dev_info(&wc->xb.pdev->dev,
- "Setting direct %d to %04x on %d\n",
- regop.reg, regop.val, chan->chanpos);
- wcaxx_setreg(wc, mod, regop.reg, regop.val);
- }
- break;
- case WCTDM_SET_ECHOTUNE:
- dev_info(&wc->xb.pdev->dev, "-- Setting echo registers:\n");
- if (copy_from_user(&echoregs, (__user void *) data,
- sizeof(echoregs)))
- return -EFAULT;
- if (mod->type == FXO) {
- /* Set the ACIM register */
- wcaxx_setreg(wc, mod, 30, echoregs.acim);
- /* Set the digital echo canceller registers */
- wcaxx_setreg(wc, mod, 45, echoregs.coef1);
- wcaxx_setreg(wc, mod, 46, echoregs.coef2);
- wcaxx_setreg(wc, mod, 47, echoregs.coef3);
- wcaxx_setreg(wc, mod, 48, echoregs.coef4);
- wcaxx_setreg(wc, mod, 49, echoregs.coef5);
- wcaxx_setreg(wc, mod, 50, echoregs.coef6);
- wcaxx_setreg(wc, mod, 51, echoregs.coef7);
- wcaxx_setreg(wc, mod, 52, echoregs.coef8);
- dev_info(&wc->xb.pdev->dev, "-- Set echo registers successfully\n");
- break;
- } else {
- return -EINVAL;
- }
- break;
- case DAHDI_SET_HWGAIN:
- if (copy_from_user(&hwgain, (__user void *) data,
- sizeof(hwgain)))
- return -EFAULT;
- wcaxx_set_hwgain(wc, mod, hwgain.newgain, hwgain.tx);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "Setting hwgain on channel %d to %d for %s direction\n",
- chan->chanpos-1, hwgain.newgain,
- ((hwgain.tx) ? "tx" : "rx"));
- }
- break;
- case DAHDI_TONEDETECT:
- /* Hardware DTMF detection is not supported. */
- return -ENOSYS;
- case DAHDI_SETPOLARITY:
- if (get_user(x, (__user int *) data))
- return -EFAULT;
- if (mod->type != FXS)
- return -EINVAL;
- /* Can't change polarity while ringing or when open */
- if (((fxs->lasttxhook & SLIC_LF_SETMASK) == SLIC_LF_RINGING) ||
- ((fxs->lasttxhook & SLIC_LF_SETMASK) == SLIC_LF_OPEN)) {
- if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d Unable to Set Polarity\n",
- chan->chanpos - 1);
- }
- return -EINVAL;
- }
- fxs->reversepolarity = (x) ? 1 : 0;
- if (POLARITY_XOR(fxs)) {
- fxs->idletxhookstate |= SLIC_LF_REVMASK;
- x = fxs->lasttxhook & SLIC_LF_SETMASK;
- x |= SLIC_LF_REVMASK;
- if (x != fxs->lasttxhook) {
- x = set_lasttxhook_interruptible(wc, fxs, x,
- &mod->sethook);
- if ((debug & DEBUG_CARD) && x) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d TIMEOUT: Set Reverse Polarity\n",
- chan->chanpos - 1);
- } else if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d Set Reverse Polarity\n",
- chan->chanpos - 1);
- }
- }
- } else {
- fxs->idletxhookstate &= ~SLIC_LF_REVMASK;
- x = fxs->lasttxhook & SLIC_LF_SETMASK;
- x &= ~SLIC_LF_REVMASK;
- if (x != fxs->lasttxhook) {
- x = set_lasttxhook_interruptible(wc, fxs, x,
- &mod->sethook);
- if ((debug & DEBUG_CARD) & x) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d TIMEOUT: Set Normal Polarity\n",
- chan->chanpos - 1);
- } else if (debug & DEBUG_CARD) {
- dev_info(&wc->xb.pdev->dev,
- "Channel %d Set Normal Polarity\n",
- chan->chanpos - 1);
- }
- }
- }
- break;
- default:
- return -ENOTTY;
- }
- return 0;
- }
- static int wcaxx_open(struct dahdi_chan *chan)
- {
- struct wcaxx *const wc = chan->pvt;
- unsigned long flags;
- struct wcaxx_module *const mod = &wc->mods[chan->chanpos - 1];
- #if 0
- if (wc->dead)
- return -ENODEV;
- #endif
- if (mod->type == FXO) {
- /* Reset the mwi indicators */
- spin_lock_irqsave(&wc->reglock, flags);
- mod->mod.fxo.neonmwi_debounce = 0;
- mod->mod.fxo.neonmwi_offcounter = 0;
- mod->mod.fxo.neonmwi_state = 0;
- spin_unlock_irqrestore(&wc->reglock, flags);
- }
- return 0;
- }
- static inline struct wcaxx *span_to_wcaxx(struct dahdi_span *span)
- {
- struct wcaxx *wc = container_of(span, struct wcaxx, span);
- return wc;
- }
- static int wcaxx_watchdog(struct dahdi_span *span, int event)
- {
- struct wcaxx *wc = span_to_wcaxx(span);
- dev_info(&wc->xb.pdev->dev, "TDM: Called watchdog\n");
- return 0;
- }
- static int wcaxx_close(struct dahdi_chan *chan)
- {
- struct wcaxx *wc;
- int x;
- wc = chan->pvt;
- for (x = 0; x < wc->mods_per_board; x++) {
- struct wcaxx_module *const mod = &wc->mods[x];
- if (FXS == mod->type) {
- mod->mod.fxs.idletxhookstate =
- POLARITY_XOR(&mod->mod.fxs) ? SLIC_LF_ACTIVE_REV :
- SLIC_LF_ACTIVE_FWD;
- }
- }
- return 0;
- }
- static int wcaxx_hooksig(struct dahdi_chan *chan, enum dahdi_txsig txsig)
- {
- struct wcaxx *wc = chan->pvt;
- struct wcaxx_module *const mod = &wc->mods[chan->chanpos - 1];
- if (mod->type == FXO) {
- switch (txsig) {
- case DAHDI_TXSIG_START:
- case DAHDI_TXSIG_OFFHOOK:
- mod->mod.fxo.offhook = 1;
- mod->sethook = CMD_WR(5, 0x9);
- /* wcaxx_setreg(wc, chan->chanpos - 1, 5, 0x9); */
- break;
- case DAHDI_TXSIG_ONHOOK:
- mod->mod.fxo.offhook = 0;
- mod->sethook = CMD_WR(5, 0x8);
- /* wcaxx_setreg(wc, chan->chanpos - 1, 5, 0x8); */
- break;
- default:
- dev_notice(&wc->xb.pdev->dev,
- "Can't set tx state to %d\n", txsig);
- break;
- }
- } else if (mod->type == FXS) {
- wcaxx_fxs_hooksig(wc, mod, txsig);
- }
- return 0;
- }
- static void wcaxx_dacs_connect(struct wcaxx *wc, int srccard, int dstcard)
- {
- struct wcaxx_module *const srcmod = &wc->mods[srccard];
- struct wcaxx_module *const dstmod = &wc->mods[dstcard];
- unsigned int type;
- if (wc->mods[dstcard].dacssrc > -1) {
- dev_notice(&wc->xb.pdev->dev, "wcaxx_dacs_connect: Can't have double sourcing yet!\n");
- return;
- }
- type = wc->mods[srccard].type;
- if ((type == FXS) || (type == FXO)) {
- dev_notice(&wc->xb.pdev->dev,
- "wcaxx_dacs_connect: Unsupported modtype for "
- "card %d\n", srccard);
- return;
- }
- type = wc->mods[dstcard].type;
- if ((type != FXS) && (type != FXO)) {
- dev_notice(&wc->xb.pdev->dev,
- "wcaxx_dacs_connect: Unsupported modtype "
- "for card %d\n", dstcard);
- return;
- }
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "connect %d => %d\n", srccard, dstcard);
- }
- dstmod->dacssrc = srccard;
- /* make srccard transmit to srccard+24 on the TDM bus */
- if (srcmod->type == FXS) {
- /* proslic */
- wcaxx_setreg(wc, srcmod, PCM_XMIT_START_COUNT_LSB,
- ((srccard+24) * 8) & 0xff);
- wcaxx_setreg(wc, srcmod, PCM_XMIT_START_COUNT_MSB,
- ((srccard+24) * 8) >> 8);
- } else if (srcmod->type == FXO) {
- /* daa TX */
- wcaxx_setreg(wc, srcmod, 34, ((srccard+24) * 8) & 0xff);
- wcaxx_setreg(wc, srcmod, 35, ((srccard+24) * 8) >> 8);
- }
- /* have dstcard receive from srccard+24 on the TDM bus */
- if (dstmod->type == FXS) {
- /* proslic */
- wcaxx_setreg(wc, dstmod, PCM_RCV_START_COUNT_LSB,
- ((srccard+24) * 8) & 0xff);
- wcaxx_setreg(wc, dstmod, PCM_RCV_START_COUNT_MSB,
- ((srccard+24) * 8) >> 8);
- } else if (dstmod->type == FXO) {
- /* daa RX */
- wcaxx_setreg(wc, dstmod, 36, ((srccard+24) * 8) & 0xff);
- wcaxx_setreg(wc, dstmod, 37, ((srccard+24) * 8) >> 8);
- }
- }
- static void wcaxx_dacs_disconnect(struct wcaxx *wc, int card)
- {
- struct wcaxx_module *const mod = &wc->mods[card];
- struct wcaxx_module *dacssrc;
- if (mod->dacssrc <= -1)
- return;
- dacssrc = &wc->mods[mod->dacssrc];
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "wcaxx_dacs_disconnect: restoring TX for %d and RX for %d\n",
- mod->dacssrc, card);
- }
- /* restore TX (source card) */
- if (dacssrc->type == FXS) {
- wcaxx_setreg(wc, dacssrc, PCM_XMIT_START_COUNT_LSB,
- (mod->dacssrc * 8) & 0xff);
- wcaxx_setreg(wc, dacssrc, PCM_XMIT_START_COUNT_MSB,
- (mod->dacssrc * 8) >> 8);
- } else if (dacssrc->type == FXO) {
- wcaxx_setreg(wc, mod, 34, (card * 8) & 0xff);
- wcaxx_setreg(wc, mod, 35, (card * 8) >> 8);
- } else {
- dev_warn(&wc->xb.pdev->dev,
- "WARNING: wcaxx_dacs_disconnect() called "
- "on unsupported modtype\n");
- }
- /* restore RX (this card) */
- if (FXS == mod->type) {
- wcaxx_setreg(wc, mod, PCM_RCV_START_COUNT_LSB,
- (card * 8) & 0xff);
- wcaxx_setreg(wc, mod, PCM_RCV_START_COUNT_MSB,
- (card * 8) >> 8);
- } else if (FXO == mod->type) {
- wcaxx_setreg(wc, mod, 36, (card * 8) & 0xff);
- wcaxx_setreg(wc, mod, 37, (card * 8) >> 8);
- } else {
- dev_warn(&wc->xb.pdev->dev,
- "WARNING: wcaxx_dacs_disconnect() called "
- "on unsupported modtype\n");
- }
- mod->dacssrc = -1;
- }
- static int wcaxx_dacs(struct dahdi_chan *dst, struct dahdi_chan *src)
- {
- struct wcaxx *wc;
- if (!nativebridge)
- return 0; /* should this return -1 since unsuccessful? */
- wc = dst->pvt;
- if (src) {
- wcaxx_dacs_connect(wc, src->chanpos - 1, dst->chanpos - 1);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "dacs connecct: %d -> %d!\n\n",
- src->chanpos, dst->chanpos);
- }
- } else {
- wcaxx_dacs_disconnect(wc, dst->chanpos - 1);
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "dacs disconnect: %d!\n", dst->chanpos);
- }
- }
- return 0;
- }
- /**
- * wcaxx_chanconfig - Called when the channels are being configured.
- *
- * Ensure that the card is completely ready to go before we allow the channels
- * to be completely configured. This is to allow lengthy initialization
- * actions to take place in background on driver load and ensure we're synced
- * up by the time dahdi_cfg is run.
- *
- */
- static int
- wcaxx_chanconfig(struct file *file, struct dahdi_chan *chan, int sigtype)
- {
- struct wcaxx *wc = chan->pvt;
- if ((file->f_flags & O_NONBLOCK) && !is_initialized(wc))
- return -EAGAIN;
- return 0;
- }
- /*
- * wcaxx_assigned - Called when span is assigned.
- * @span: The span that is now assigned.
- *
- * This function is called by the core of DAHDI after the span number and
- * channel numbers have been assigned.
- *
- */
- static void wcaxx_assigned(struct dahdi_span *span)
- {
- struct dahdi_span *s;
- struct dahdi_device *ddev = span->parent;
- struct wcaxx *wc = NULL;
- list_for_each_entry(s, &ddev->spans, device_node) {
- wc = container_of(s, struct wcaxx, span);
- if (!test_bit(DAHDI_FLAGBIT_REGISTERED, &s->flags))
- return;
- }
- }
- static const struct dahdi_span_ops wcaxx_span_ops = {
- .owner = THIS_MODULE,
- .hooksig = wcaxx_hooksig,
- .open = wcaxx_open,
- .close = wcaxx_close,
- .ioctl = wcaxx_ioctl,
- .watchdog = wcaxx_watchdog,
- .chanconfig = wcaxx_chanconfig,
- .dacs = wcaxx_dacs,
- .assigned = wcaxx_assigned,
- .echocan_create = wcaxx_echocan_create,
- .echocan_name = wcaxx_echocan_name,
- };
- static struct wcaxx_chan *
- wcaxx_init_chan(struct wcaxx *wc, struct dahdi_span *s, int channo)
- {
- struct wcaxx_chan *c;
- c = kzalloc(sizeof(*c), GFP_KERNEL);
- if (!c)
- return NULL;
- snprintf(c->chan.name, sizeof(c->chan.name), "WCTDM/%d/%d",
- wc->num, channo);
- c->chan.chanpos = channo+1;
- c->chan.span = s;
- c->chan.pvt = wc;
- c->timeslot = channo;
- return c;
- }
- static void wcaxx_init_span(struct wcaxx *wc)
- {
- int x;
- struct wcaxx_chan *c;
- struct dahdi_echocan_state *ec[NUM_MODULES] = {NULL, };
- /* DAHDI stuff */
- wc->span.offset = 0;
- sprintf(wc->span.name, "WCTDM/%d", wc->num);
- snprintf(wc->span.desc, sizeof(wc->span.desc) - 1,
- "%s", wc->desc->name);
- if (wc->companding == DAHDI_LAW_DEFAULT) {
- wc->span.deflaw = DAHDI_LAW_MULAW;
- } else if (wc->companding == DAHDI_LAW_ALAW) {
- /* Force everything to alaw */
- wc->span.deflaw = DAHDI_LAW_ALAW;
- } else {
- /* Auto set to ulaw */
- wc->span.deflaw = DAHDI_LAW_MULAW;
- }
- wc->span.ops = &wcaxx_span_ops;
- wc->span.flags = DAHDI_FLAG_RBS;
- wc->span.spantype = SPANTYPE_ANALOG_MIXED;
- wc->span.chans = kmalloc(sizeof(wc->span.chans[0]) * wc->desc->ports,
- GFP_KERNEL);
- if (!wc->span.chans)
- return;
- /* allocate channels for the span */
- for (x = 0; x < wc->desc->ports; x++) {
- c = wcaxx_init_chan(wc, &wc->span, x);
- if (!c)
- return;
- wc->chans[x] = c;
- wc->span.chans[x] = &c->chan;
- /* TODO: Should echocan state hide under VPM_ENABLED or does
- * software ec use it? */
- ec[x] = kzalloc(sizeof(*ec[x]), GFP_KERNEL);
- }
- wc->span.channels = wc->desc->ports;
- memcpy(wc->ec, ec, sizeof(wc->ec));
- memset(ec, 0, sizeof(ec));
- }
- /**
- * should_set_alaw() - Should be called after all the spans are initialized.
- *
- * Returns true if the module companding should be set to alaw, otherwise
- * false.
- */
- static bool should_set_alaw(const struct wcaxx *wc)
- {
- if (DAHDI_LAW_ALAW == wc->companding)
- return true;
- else
- return false;
- }
- static void wcaxx_fixup_span(struct wcaxx *wc)
- {
- struct dahdi_span *s;
- int x, y;
- y = 0;
- s = &wc->span;
- for (x = 0; x < wc->desc->ports; x++) {
- struct wcaxx_module *const mod = &wc->mods[x];
- if (debug) {
- dev_info(&wc->xb.pdev->dev,
- "fixup_analog: x=%d, y=%d modtype=%d, "
- "s->chans[%d]=%p\n", x, y, mod->type,
- y, s->chans[y]);
- }
- if (mod->type == FXO) {
- int val;
- s->chans[y++]->sigcap = DAHDI_SIG_FXSKS |
- DAHDI_SIG_FXSLS | DAHDI_SIG_SF |
- DAHDI_SIG_CLEAR;
- val = should_set_alaw(wc) ? 0x20 : 0x28;
- #ifdef DEBUG
- val = (digitalloopback) ? 0x30 : val;
- #endif
- wcaxx_setreg(wc, mod, 33, val);
- wcaxx_voicedaa_set_ts(wc, mod, wc->chans[x]->timeslot);
- } else if (mod->type == FXS) {
- /* NOTE: Digital loopback does not work on the FXS
- * modules in the same way since the data is still
- * companded by the ProSLIC and doesn't appear to have
- * perfect symetry. */
- s->chans[y++]->sigcap = DAHDI_SIG_FXOKS |
- DAHDI_SIG_FXOLS | DAHDI_SIG_FXOGS |
- DAHDI_SIG_SF | DAHDI_SIG_EM | DAHDI_SIG_CLEAR;
- wcaxx_setreg(wc, mod, 1,
- (should_set_alaw(wc) ? 0x20 : 0x28));
- wcaxx_proslic_set_ts(wc, mod, wc->chans[x]->timeslot);
- } else {
- s->chans[y++]->sigcap = 0;
- }
- }
- }
- static bool wcaxx_init_fxs_port(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- u8 readi;
- enum {UNKNOWN = 0, SANE = 1};
- int ret = wcaxx_init_proslic(wc, mod, 0, 0, UNKNOWN);
- if (!ret) {
- if (debug & DEBUG_CARD) {
- readi = wcaxx_getreg(wc, mod, LOOP_I_LIMIT);
- dev_info(&wc->xb.pdev->dev,
- "Proslic module %d loop current is %dmA\n",
- mod->card, ((readi*3) + 20));
- }
- return true;
- }
- if (ret != -2) {
- /* Init with Manual Calibration */
- if (!wcaxx_init_proslic(wc, mod, 0, 1, SANE)) {
- if (debug & DEBUG_CARD) {
- readi = wcaxx_getreg(wc, mod, LOOP_I_LIMIT);
- dev_info(&wc->xb.pdev->dev,
- "Proslic module %d loop current is %dmA\n",
- mod->card, ((readi*3)+20));
- }
- } else {
- dev_notice(&wc->xb.pdev->dev,
- "Port %d: FAILED FXS (%s)\n", mod->card + 1,
- fxshonormode ? fxo_modes[_opermode].name :
- "FCC");
- }
- return true;
- }
- return false;
- }
- static void wcaxx_reset_module(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- u32 reg_val = (1UL << (mod->spi->chip_select + 12));
- wcxb_gpio_clear(&wc->xb, reg_val);
- udelay(500);
- wcxb_gpio_set(&wc->xb, reg_val);
- msleep(250); /* TODO: What should this value be? */
- }
- static bool check_for_single_fxs(struct wcaxx *wc, unsigned int port)
- {
- bool result;
- struct wcaxx_module *mod = &wc->mods[port];
- mod->spi = get_spi_device_for_port(wc, mod->card, false);
- mod->subaddr = 0;
- wcaxx_reset_module(wc, mod);
- wcaxx_fxsinit(mod->spi);
- result = wcaxx_init_fxs_port(wc, mod);
- if (!result)
- mod->type = NONE;
- /* It is currently unclear why this read is necessary for some of the
- * S100M modules to properly function. */
- wcaxx_getreg(wc, mod, 0x00);
- return result;
- }
- static bool check_for_single_fxo(struct wcaxx *wc, unsigned int port)
- {
- bool result;
- struct wcaxx_module *mod = &wc->mods[port];
- mod->spi = get_spi_device_for_port(wc, mod->card, false);
- mod->subaddr = 0;
- wcaxx_reset_module(wc, mod);
- result = (wcaxx_init_voicedaa(wc, mod, 0, 0, 0) == 0);
- if (!result)
- mod->type = NONE;
- return result;
- }
- static bool check_for_quad_fxs(struct wcaxx *wc, unsigned int base_port)
- {
- int port;
- int offset;
- struct wcaxx_module *mod = &wc->mods[base_port + 1];
- /* Cannot have quad port modules on the 4 port base cards. */
- if (is_four_port(wc))
- return false;
- /* We can assume that the base port has already been configured as an
- * FXS port if we're even in this function */
- mod->spi = get_spi_device_for_port(wc, mod->card, true);
- mod->subaddr = offset = 1;
- if (wcaxx_init_fxs_port(wc, mod)) {
- /* This must be a 4 port FXS module... */
- for (port = base_port + 2; port < base_port+4; ++port) {
- mod = &wc->mods[port];
- mod->spi = get_spi_device_for_port(wc, mod->card, true);
- mod->subaddr = ++offset;
- if (!wcaxx_init_fxs_port(wc, mod)) {
- /* This means that a quad-module failed to
- * setup ports 3 or 4? */
- dev_err(&wc->xb.pdev->dev,
- "Quad-FXS at base %d failed initialization.\n",
- base_port);
- goto error_exit;
- }
- }
- return true;
- }
- error_exit:
- for (port = base_port + 1; port < base_port + 4; ++port) {
- mod = &wc->mods[port];
- mod->type = NONE;
- }
- return false;
- }
- static bool check_for_quad_fxo(struct wcaxx *wc, unsigned int base_port)
- {
- int port;
- int offset;
- struct wcaxx_module *mod = &wc->mods[base_port + 1];
- /* Cannot have quad port modules on the 4 port base cards. */
- if (is_four_port(wc))
- return false;
- /* We can assume that the base port has already been configured as an
- * FXO port if we're even in this function */
- mod->spi = get_spi_device_for_port(wc, mod->card, true);
- mod->subaddr = offset = 1;
- if (!wcaxx_init_voicedaa(wc, mod, 0, 0, 0)) {
- /* This must be a 4 port FXO module. */
- for (port = base_port + 2; port < base_port + 4; ++port) {
- mod = &wc->mods[port];
- mod->spi = get_spi_device_for_port(wc, mod->card, true);
- mod->subaddr = ++offset;
- if (wcaxx_init_voicedaa(wc, mod, 0, 0, 0)) {
- dev_err(&wc->xb.pdev->dev,
- "Quad-FXO at base %d failed initialization.\n",
- base_port);
- goto error_exit;
- }
- }
- return true;
- }
- error_exit:
- for (port = base_port + 1; port < base_port + 4; ++port) {
- mod = &wc->mods[port];
- mod->type = NONE;
- }
- return false;
- }
- static void __wcaxx_identify_four_port_module_group(struct wcaxx *wc)
- {
- int i;
- for (i = 0; i < wc->desc->ports; i++) {
- if (!check_for_single_fxs(wc, i))
- check_for_single_fxo(wc, i);
- }
- return;
- }
- static void
- __wcaxx_identify_module_group(struct wcaxx *wc, unsigned long base)
- {
- if (check_for_single_fxs(wc, base)) {
- if (check_for_quad_fxs(wc, base)) {
- /* S400M installed */
- return;
- } else if (check_for_single_fxs(wc, base + 1)) {
- /* Two S110M installed */
- return;
- } else if (check_for_single_fxo(wc, base + 1)) {
- /* 1 S110M 1 X100M */
- return;
- } else {
- /* 1 S110M 1 Empty */
- return;
- }
- } else if (check_for_single_fxo(wc, base)) {
- if (check_for_quad_fxo(wc, base)) {
- /* X400M installed */
- return;
- } else if (check_for_single_fxo(wc, base + 1)) {
- /* Two X100M installed */
- return;
- } else if (check_for_single_fxs(wc, base + 1)) {
- /* 1 X100M 1 S100M installed */
- return;
- } else {
- /* 1 X100M 1 Empty */
- return;
- }
- } else if (check_for_single_fxs(wc, base + 1)) {
- /* 1 Empty 1 S110M installed */
- return;
- } else if (check_for_single_fxo(wc, base + 1)) {
- /* 1 Empty 1 X100M installed */
- return;
- }
- /* No module */
- return;
- }
- /**
- * wcaxx_print_moule_configuration - Print the configuration to the kernel log
- * @wc: The card we're interested in.
- *
- * This is to ensure that the module configuration from each card shows up
- * sequentially in the kernel log, as opposed to interleaved with one another.
- *
- */
- static void wcaxx_print_module_configuration(const struct wcaxx *const wc)
- {
- int i;
- static DEFINE_MUTEX(print);
- mutex_lock(&print);
- for (i = 0; i < wc->mods_per_board; ++i) {
- const struct wcaxx_module *const mod = &wc->mods[i];
- switch (mod->type) {
- case FXO:
- dev_info(&wc->xb.pdev->dev,
- "Port %d: Installed -- AUTO FXO (%s mode)\n",
- i + 1, fxo_modes[_opermode].name);
- break;
- case FXS:
- dev_info(&wc->xb.pdev->dev,
- "Port %d: Installed -- AUTO FXS/DPO\n", i + 1);
- break;
- case NONE:
- dev_info(&wc->xb.pdev->dev,
- "Port %d: Not installed\n", i + 1);
- break;
- }
- }
- mutex_unlock(&print);
- }
- static void wcaxx_identify_modules(struct wcaxx *wc)
- {
- int x;
- unsigned long flags;
- /* A8A/A8B - Reset the modules. */
- wcxb_gpio_clear(&wc->xb, 0xf000);
- msleep(50); /* TODO: what should these values be? */
- wcxb_gpio_set(&wc->xb, 0xf000);
- msleep(250); /* TODO: What should these values be? */
- /* Place all units in the daisy chain mode of operation. This allows
- * multiple devices to share a chip select (like on the X400 and S400
- * modules) */
- for (x = 0; x < ARRAY_SIZE(wc->spi_devices); ++x)
- wcaxx_fxsinit(wc->spi_devices[x]);
- spin_lock_irqsave(&wc->reglock, flags);
- wc->mods_per_board = wc->desc->ports;
- spin_unlock_irqrestore(&wc->reglock, flags);
- BUG_ON(wc->desc->ports % 4);
- if (is_four_port(wc)) {
- __wcaxx_identify_four_port_module_group(wc);
- } else {
- for (x = 0; x < wc->desc->ports/4; x++)
- __wcaxx_identify_module_group(wc, x*4);
- }
- wcaxx_print_module_configuration(wc);
- }
- static struct pci_driver wcaxx_driver;
- static void wcaxx_back_out_gracefully(struct wcaxx *wc)
- {
- int i;
- unsigned long flags;
- clear_bit(INITIALIZED, &wc->bit_flags);
- smp_mb__after_atomic();
- /* Make sure we're not on the card list anymore. */
- mutex_lock(&card_list_lock);
- list_del(&wc->card_node);
- mutex_unlock(&card_list_lock);
- wcxb_release(&wc->xb);
- for (i = 0; i < wc->mods_per_board; i++) {
- struct wcaxx_module *const mod = &wc->mods[i];
- kfree(mod->mod_poll);
- mod->mod_poll = NULL;
- }
- kfree(wc->span.chans);
- wc->span.chans = NULL;
- spin_lock_irqsave(&wc->reglock, flags);
- for (i = 0; i < wc->span.channels; ++i) {
- kfree(wc->chans[i]);
- kfree(wc->ec[i]);
- wc->chans[i] = NULL;
- wc->ec[i] = NULL;
- }
- spin_unlock_irqrestore(&wc->reglock, flags);
- for (i = 0; i < ARRAY_SIZE(wc->spi_devices); i++)
- wcxb_spi_device_destroy(wc->spi_devices[i]);
- wcxb_spi_master_destroy(wc->master);
- kfree(wc->board_name);
- if (wc->ddev) {
- kfree(wc->ddev->devicetype);
- kfree(wc->ddev->location);
- kfree(wc->ddev->hardware_id);
- dahdi_free_device(wc->ddev);
- }
- kfree(wc);
- }
- static void wcaxx_handle_error(struct wcxb *xb)
- {
- struct wcaxx *wc = container_of(xb, struct wcaxx, xb);
- wc->ddev->irqmisses++;
- }
- static const struct wcxb_operations wcxb_operations = {
- .handle_receive = wcaxx_handle_receive,
- .handle_transmit = wcaxx_handle_transmit,
- .handle_error = wcaxx_handle_error,
- };
- struct cmd_results {
- u8 results[8];
- };
- static int wcaxx_check_firmware(struct wcaxx *wc)
- {
- char *filename;
- u32 firmware_version;
- const bool force_firmware = false;
- const unsigned int A4A_VERSION = 0x0a0017;
- const unsigned int A4B_VERSION = 0x0d001e;
- const unsigned int A8A_VERSION = 0x1d0017;
- const unsigned int A8B_VERSION = 0x1f001e;
- if (wc->desc == &device_a8a) {
- firmware_version = A8A_VERSION;
- filename = "dahdi-fw-a8a.bin";
- } else if (wc->desc == &device_a8b) {
- firmware_version = A8B_VERSION;
- filename = "dahdi-fw-a8b.bin";
- } else if (wc->desc == &device_a4a) {
- firmware_version = A4A_VERSION;
- filename = "dahdi-fw-a4a.bin";
- } else if (wc->desc == &device_a4b) {
- firmware_version = A4B_VERSION;
- filename = "dahdi-fw-a4b.bin";
- } else {
- /* This is a bug in the driver code */
- WARN_ON(1);
- return 0;
- }
- return wcxb_check_firmware(&wc->xb, firmware_version,
- filename, force_firmware, WCXB_RESET_NOW);
- }
- static void wcaxx_check_sethook(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- if (mod->sethook) {
- wcaxx_setreg(wc, mod, ((mod->sethook >> 8) & 0xff),
- mod->sethook & 0xff);
- mod->sethook = 0;
- }
- }
- static void wcaxx_poll_fxs_complete(void *arg)
- {
- struct wcaxx_mod_poll *poll_fxs = arg;
- struct wcaxx *wc = poll_fxs->wc;
- struct wcaxx_module *const mod = poll_fxs->mod;
- if (!is_initialized(wc)) {
- kfree(poll_fxs);
- return;
- }
- mod->mod.fxs.hook_state_shadow = poll_fxs->buffer[2];
- mod->mod.fxs.linefeed_control_shadow = poll_fxs->buffer[5];
- wcaxx_isr_misc_fxs(poll_fxs->wc, poll_fxs->mod);
- memcpy(poll_fxs->buffer, poll_fxs->master_buffer,
- sizeof(poll_fxs->buffer));
- wcaxx_check_sethook(poll_fxs->wc, poll_fxs->mod);
- mod->mod_poll = poll_fxs;
- }
- /**
- * wcaxx_start_poll_fxs - Starts the interrupt polling loop for FXS modules.
- *
- * To stop the polling loop, clear the initialized bit and then flush the
- * pending wcxb_spi messages.
- *
- */
- static int wcaxx_start_poll_fxs(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- struct wcaxx_mod_poll *mod_poll = kzalloc(sizeof(*mod_poll),
- GFP_KERNEL);
- struct wcxb_spi_message *m = &mod_poll->m;
- struct wcxb_spi_transfer *t = &mod_poll->t;
- WARN_ON(!is_initialized(wc));
- if (!mod_poll)
- return -ENOMEM;
- memset(t, 0, sizeof(*t));
- wcxb_spi_message_init(m);
- t->tx_buf = t->rx_buf = mod_poll->buffer;
- t->len = sizeof(mod_poll->buffer);
- wcxb_spi_message_add_tail(t, m);
- mod_poll->wc = wc;
- mod_poll->mod = mod;
- mod_poll->master_buffer[0] = 1 << mod_poll->mod->subaddr;
- mod_poll->master_buffer[1] = (LOOP_STAT | 0x80) & 0xff;
- mod_poll->master_buffer[2] = 0;
- mod_poll->master_buffer[3] = mod_poll->master_buffer[0];
- mod_poll->master_buffer[4] = (LINE_STATE | 0x80) & 0xff;
- mod_poll->master_buffer[5] = 0;
- memcpy(mod_poll->buffer, mod_poll->master_buffer,
- sizeof(mod_poll->buffer));
- m->arg = mod_poll;
- m->complete = &wcaxx_poll_fxs_complete;
- wcxb_spi_async(mod->spi, m);
- return 0;
- }
- static void wcaxx_poll_fxo_complete(void *arg)
- {
- struct wcaxx_mod_poll *poll_fxo = arg;
- struct wcaxx *wc = poll_fxo->wc;
- struct wcaxx_module *const mod = poll_fxo->mod;
- if (!is_initialized(wc)) {
- kfree(poll_fxo);
- return;
- }
- mod->mod.fxo.hook_ring_shadow = poll_fxo->buffer[2];
- mod->mod.fxo.line_voltage_status = poll_fxo->buffer[5];
- wcaxx_voicedaa_check_hook(poll_fxo->wc, poll_fxo->mod);
- memcpy(poll_fxo->buffer, poll_fxo->master_buffer,
- sizeof(poll_fxo->buffer));
- wcaxx_check_sethook(poll_fxo->wc, poll_fxo->mod);
- mod->mod_poll = poll_fxo;
- }
- /**
- * wcaxx_start_poll_fxo - Starts the interrupt polling loop for FXS modules.
- *
- * To stop the polling loop, clear the initialized bit and then flush the
- * pending wcxb_spi messages.
- *
- */
- static int wcaxx_start_poll_fxo(struct wcaxx *wc, struct wcaxx_module *mod)
- {
- static const int ADDRS[4] = {0x00, 0x08, 0x04, 0x0c};
- struct wcaxx_mod_poll *poll_fxo = kzalloc(sizeof(*poll_fxo),
- GFP_KERNEL);
- struct wcxb_spi_message *m = &poll_fxo->m;
- struct wcxb_spi_transfer *t = &poll_fxo->t;
- WARN_ON(!is_initialized(wc));
- if (!poll_fxo)
- return -ENOMEM;
- memset(t, 0, sizeof(*t));
- wcxb_spi_message_init(m);
- t->tx_buf = t->rx_buf = poll_fxo->buffer;
- t->len = sizeof(poll_fxo->buffer);
- wcxb_spi_message_add_tail(t, m);
- poll_fxo->wc = wc;
- poll_fxo->mod = mod;
- poll_fxo->master_buffer[0] = 0x60 | ADDRS[poll_fxo->mod->subaddr];
- poll_fxo->master_buffer[1] = 5 & 0x7f; /* Hook / Ring State */
- poll_fxo->master_buffer[2] = 0;
- poll_fxo->master_buffer[3] = poll_fxo->master_buffer[0];
- poll_fxo->master_buffer[4] = 29 & 0x7f; /* Battery */
- poll_fxo->master_buffer[5] = 0;
- memcpy(poll_fxo->buffer, poll_fxo->master_buffer,
- sizeof(poll_fxo->buffer));
- m->arg = poll_fxo;
- m->complete = &wcaxx_poll_fxo_complete;
- wcxb_spi_async(mod->spi, m);
- return 0;
- }
- /**
- * wcaxx_read_serial - Returns the serial number of the board.
- * @wc: The board whos serial number we are reading.
- *
- * The buffer returned is dynamically allocated and must be kfree'd by the
- * caller. If memory could not be allocated, NULL is returned.
- *
- * Must be called in process context.
- *
- */
- static char *wcaxx_read_serial(struct wcaxx *wc)
- {
- int i;
- static const int MAX_SERIAL = 20*5;
- const unsigned int SERIAL_ADDRESS = 0x1f0000;
- unsigned char *serial = kzalloc(MAX_SERIAL + 1, GFP_KERNEL);
- struct wcxb const *xb = &wc->xb;
- struct wcxb_spi_master *flash_spi_master = NULL;
- struct wcxb_spi_device *flash_spi_device = NULL;
- const unsigned int FLASH_SPI_BASE = 0x200;
- if (!serial)
- return NULL;
- flash_spi_master = wcxb_spi_master_create(&xb->pdev->dev,
- xb->membase + FLASH_SPI_BASE,
- false);
- if (!flash_spi_master)
- return NULL;
- flash_spi_device = wcxb_spi_device_create(flash_spi_master, 0);
- if (!flash_spi_device)
- goto error_exit;
- wcxb_flash_read(flash_spi_device, SERIAL_ADDRESS,
- serial, MAX_SERIAL);
- for (i = 0; i < MAX_SERIAL; ++i) {
- if ((serial[i] < 0x20) || (serial[i] > 0x7e)) {
- serial[i] = '\0';
- break;
- }
- }
- if (!i) {
- kfree(serial);
- serial = NULL;
- } else {
- /* Limit the size of the buffer to just what is needed to
- * actually hold the serial number. */
- unsigned char *new_serial;
- new_serial = kasprintf(GFP_KERNEL, "%s", serial);
- kfree(serial);
- serial = new_serial;
- }
- error_exit:
- wcxb_spi_device_destroy(flash_spi_device);
- wcxb_spi_master_destroy(flash_spi_master);
- return serial;
- }
- static void wcaxx_start_module_polling(struct wcaxx *wc)
- {
- int x;
- WARN_ON(!is_initialized(wc));
- for (x = 0; x < wc->mods_per_board; x++) {
- struct wcaxx_module *const mod = &wc->mods[x];
- switch (mod->type) {
- case FXO:
- wcaxx_start_poll_fxo(wc, mod);
- break;
- case FXS:
- wcaxx_start_poll_fxs(wc, mod);
- break;
- case NONE:
- break;
- }
- }
- wc->module_poll_time = wc->framecount + MODULE_POLL_TIME_MS;
- }
- /**
- * t43x_assign_num - Assign wc->num a unique value and place on card_list
- *
- */
- static void wcaxx_assign_num(struct wcaxx *wc)
- {
- mutex_lock(&card_list_lock);
- if (list_empty(&card_list)) {
- wc->num = 0;
- list_add(&wc->card_node, &card_list);
- } else {
- struct wcaxx *cur;
- struct list_head *insert_pos;
- int new_num = 0;
- insert_pos = &card_list;
- list_for_each_entry(cur, &card_list, card_node) {
- if (new_num != cur->num)
- break;
- new_num++;
- insert_pos = &cur->card_node;
- }
- wc->num = new_num;
- list_add_tail(&wc->card_node, insert_pos);
- }
- mutex_unlock(&card_list_lock);
- }
- #ifdef USE_ASYNC_INIT
- struct async_data {
- struct pci_dev *pdev;
- const struct pci_device_id *ent;
- };
- static int __devinit
- __wcaxx_init_one(struct pci_dev *pdev, const struct pci_device_id *ent,
- async_cookie_t cookie)
- #else
- static int __devinit
- __wcaxx_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
- #endif
- {
- struct wcaxx *wc;
- int i, ret;
- int curchan;
- neonmwi_offlimit_cycles = neonmwi_offlimit / MS_PER_HOOKCHECK;
- wc = kzalloc(sizeof(*wc), GFP_KERNEL);
- if (!wc)
- return -ENOMEM;
- wcaxx_assign_num(wc);
- wc->desc = (struct _device_desc *)ent->driver_data;
- spin_lock_init(&wc->reglock);
- wc->board_name = kasprintf(GFP_KERNEL, "%s%d",
- wcaxx_driver.name, wc->num);
- if (!wc->board_name) {
- wcaxx_back_out_gracefully(wc);
- return -ENOMEM;
- }
- #ifdef CONFIG_VOICEBUS_DISABLE_ASPM
- if (is_pcie(wc)) {
- pci_disable_link_state(pdev->bus->self, PCIE_LINK_STATE_L0S |
- PCIE_LINK_STATE_L1 | PCIE_LINK_STATE_CLKPM);
- };
- #endif
- pci_set_drvdata(pdev, wc);
- wc->xb.ops = &wcxb_operations;
- wc->xb.pdev = pdev;
- wc->xb.debug = &debug;
- ret = wcxb_init(&wc->xb, wc->board_name, int_mode);
- if (ret) {
- wcaxx_back_out_gracefully(wc);
- return ret;
- }
- wcxb_set_minlatency(&wc->xb, latency);
- wcxb_set_maxlatency(&wc->xb, max_latency);
- ret = wcaxx_check_firmware(wc);
- if (ret) {
- wcaxx_back_out_gracefully(wc);
- return ret;
- }
- wcxb_lock_latency(&wc->xb);
- wc->mods_per_board = NUM_MODULES;
- if (alawoverride) {
- companding = "alaw";
- dev_info(&wc->xb.pdev->dev,
- "The module parameter alawoverride has been deprecated. Please use the parameter companding=alaw instead");
- }
- if (!strcasecmp(companding, "alaw"))
- /* Force this card's companding to alaw */
- wc->companding = DAHDI_LAW_ALAW;
- else if (!strcasecmp(companding, "ulaw"))
- /* Force this card's companding to ulaw */
- wc->companding = DAHDI_LAW_MULAW;
- else
- /* Auto detect this card's companding */
- wc->companding = DAHDI_LAW_DEFAULT;
- wc->master = wcxb_spi_master_create(&pdev->dev,
- wc->xb.membase + 0x280, true);
- for (i = 0; i < ARRAY_SIZE(wc->spi_devices); i++)
- wc->spi_devices[i] = wcxb_spi_device_create(wc->master, 3-i);
- for (i = 0; i < ARRAY_SIZE(wc->mods); i++) {
- struct wcaxx_module *const mod = &wc->mods[i];
- mod->dacssrc = -1;
- mod->card = i;
- mod->spi = NULL;
- mod->subaddr = 0;
- mod->type = NONE;
- }
- ret = wcaxx_vpm_init(wc);
- if (!ret)
- wcxb_enable_echocan(&wc->xb);
- /* Now track down what modules are installed */
- wcaxx_identify_modules(wc);
- /* Start the hardware processing. */
- if (wcxb_start(&wc->xb)) {
- WARN_ON(1);
- return -EIO;
- }
- if (fatal_signal_pending(current)) {
- wcaxx_back_out_gracefully(wc);
- return -EINTR;
- }
- curchan = 0;
- wcaxx_init_span(wc);
- wcaxx_fixup_span(wc);
- curchan += wc->desc->ports;
- #ifdef USE_ASYNC_INIT
- async_synchronize_cookie(cookie);
- #endif
- wc->ddev = dahdi_create_device();
- if (!wc->ddev) {
- wcaxx_back_out_gracefully(wc);
- return -ENOMEM;
- }
- wc->ddev->manufacturer = "Digium";
- wc->ddev->location = kasprintf(GFP_KERNEL, "PCI Bus %02d Slot %02d",
- pdev->bus->number,
- PCI_SLOT(pdev->devfn) + 1);
- if (!wc->ddev->location) {
- wcaxx_back_out_gracefully(wc);
- return -ENOMEM;
- }
- if (wc->vpm)
- wc->ddev->devicetype = kasprintf(GFP_KERNEL, "%s (%s)",
- wc->desc->name, "VPMOCT032");
- else
- wc->ddev->devicetype = kasprintf(GFP_KERNEL, "%s",
- wc->desc->name);
- if (!wc->ddev->devicetype) {
- wcaxx_back_out_gracefully(wc);
- return -ENOMEM;
- }
- wc->ddev->hardware_id = wcaxx_read_serial(wc);
- list_add_tail(&wc->span.device_node, &wc->ddev->spans);
- if (dahdi_register_device(wc->ddev, &wc->xb.pdev->dev)) {
- dev_notice(&wc->xb.pdev->dev, "Unable to register device with DAHDI\n");
- wcaxx_back_out_gracefully(wc);
- return -1;
- }
- dev_info(&wc->xb.pdev->dev, "Found a %s (SN: %s)\n",
- wc->desc->name, wc->ddev->hardware_id);
- set_bit(INITIALIZED, &wc->bit_flags);
- wcaxx_start_module_polling(wc);
- wcxb_unlock_latency(&wc->xb);
- return 0;
- }
- #ifdef USE_ASYNC_INIT
- static __devinit void
- wcaxx_init_one_async(void *data, async_cookie_t cookie)
- {
- struct async_data *dat = data;
- __wcaxx_init_one(dat->pdev, dat->ent, cookie);
- kfree(dat);
- }
- static int __devinit
- wcaxx_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
- {
- struct async_data *dat;
- dat = kmalloc(sizeof(*dat), GFP_KERNEL);
- /* If we can't allocate the memory for the async_data, odds are we won't
- * be able to initialize the device either, but let's try synchronously
- * anyway... */
- if (!dat)
- return __wcaxx_init_one(pdev, ent, 0);
- dat->pdev = pdev;
- dat->ent = ent;
- async_schedule(wcaxx_init_one_async, dat);
- return 0;
- }
- #else
- static int __devinit
- wcaxx_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
- {
- return __wcaxx_init_one(pdev, ent);
- }
- #endif
- static void wcaxx_release(struct wcaxx *wc)
- {
- if (is_initialized(wc))
- dahdi_unregister_device(wc->ddev);
- wcaxx_back_out_gracefully(wc);
- }
- static void __devexit wcaxx_remove_one(struct pci_dev *pdev)
- {
- struct wcaxx *wc = pci_get_drvdata(pdev);
- if (!wc)
- return;
- dev_info(&wc->xb.pdev->dev, "Removing a %s.\n", wc->desc->name);
- flush_scheduled_work();
- wcxb_stop(&wc->xb);
- if (wc->vpm)
- release_vpm450m(wc->vpm);
- wc->vpm = NULL;
- wcaxx_release(wc);
- }
- static DEFINE_PCI_DEVICE_TABLE(wcaxx_pci_tbl) = {
- { 0xd161, 0x800d, PCI_ANY_ID, PCI_ANY_ID, 0, 0,
- (unsigned long) &device_a8b
- },
- { 0xd161, 0x800c, PCI_ANY_ID, PCI_ANY_ID, 0, 0,
- (unsigned long) &device_a8a
- },
- { 0xd161, 0x8010, PCI_ANY_ID, PCI_ANY_ID, 0, 0,
- (unsigned long) &device_a4b
- },
- { 0xd161, 0x800f, PCI_ANY_ID, PCI_ANY_ID, 0, 0,
- (unsigned long) &device_a4a
- },
- { 0 }
- };
- MODULE_DEVICE_TABLE(pci, wcaxx_pci_tbl);
- static void wcaxx_shutdown(struct pci_dev *pdev)
- {
- struct wcaxx *wc = pci_get_drvdata(pdev);
- wcxb_stop(&wc->xb);
- }
- static int wcaxx_suspend(struct pci_dev *pdev, pm_message_t state)
- {
- return -ENOSYS;
- }
- static struct pci_driver wcaxx_driver = {
- .name = "wcaxx",
- .probe = wcaxx_init_one,
- .remove = __devexit_p(wcaxx_remove_one),
- .shutdown = wcaxx_shutdown,
- .suspend = wcaxx_suspend,
- .id_table = wcaxx_pci_tbl,
- };
- static int __init wcaxx_init(void)
- {
- int res;
- int x;
- for (x = 0; x < ARRAY_SIZE(fxo_modes); x++) {
- if (!strcmp(fxo_modes[x].name, opermode))
- break;
- }
- if (x < ARRAY_SIZE(fxo_modes)) {
- _opermode = x;
- } else {
- pr_notice("Invalid/unknown operating mode '%s' specified. Please choose one of:\n",
- opermode);
- for (x = 0; x < ARRAY_SIZE(fxo_modes); x++)
- pr_notice(" %s\n", fxo_modes[x].name);
- pr_notice("Note this option is CASE SENSITIVE!\n");
- return -ENODEV;
- }
- if (!strcmp(opermode, "AUSTRALIA")) {
- boostringer = 1;
- fxshonormode = 1;
- }
- if (-1 == fastpickup) {
- if (!strcmp(opermode, "JAPAN"))
- fastpickup = 1;
- else
- fastpickup = 0;
- }
- /* for the voicedaa_check_hook defaults, if the user has not
- * overridden them by specifying them as module parameters, then get
- * the values from the selected operating mode */
- if (!battdebounce)
- battdebounce = fxo_modes[_opermode].battdebounce;
- if (!battalarm)
- battalarm = fxo_modes[_opermode].battalarm;
- if (!battthresh)
- battthresh = fxo_modes[_opermode].battthresh;
- res = dahdi_pci_module(&wcaxx_driver);
- if (res)
- return -ENODEV;
- #ifdef USE_ASYNC_INIT
- async_synchronize_full();
- #endif
- return 0;
- }
- static void __exit wcaxx_cleanup(void)
- {
- pci_unregister_driver(&wcaxx_driver);
- }
- module_param(debug, int, 0600);
- module_param(int_mode, int, 0400);
- MODULE_PARM_DESC(int_mode,
- "0 = Use MSI interrupt if available. 1 = Legacy interrupt only.\n");
- module_param(fastpickup, int, 0400);
- MODULE_PARM_DESC(fastpickup,
- "Set to 1 to shorten the calibration delay when taking an FXO port off "
- "hook. This can be required for Type-II CID. If -1 the calibration "
- "delay will depend on the current opermode.\n");
- module_param(fxovoltage, int, 0600);
- module_param(loopcurrent, int, 0600);
- module_param(reversepolarity, int, 0600);
- #ifdef DEBUG
- module_param(robust, int, 0600);
- module_param(digitalloopback, int, 0400);
- MODULE_PARM_DESC(digitalloopback,
- "Set to 1 to place FXO modules into loopback mode for troubleshooting.");
- #endif
- module_param(opermode, charp, 0600);
- module_param(lowpower, int, 0600);
- module_param(boostringer, int, 0600);
- module_param(fastringer, int, 0600);
- module_param(fxshonormode, int, 0600);
- module_param(battdebounce, uint, 0600);
- module_param(battalarm, uint, 0600);
- module_param(battthresh, uint, 0600);
- module_param(nativebridge, int, 0600);
- module_param(fxotxgain, int, 0600);
- module_param(fxorxgain, int, 0600);
- module_param(fxstxgain, int, 0600);
- module_param(fxsrxgain, int, 0600);
- module_param(ringdebounce, int, 0600);
- module_param(latency, int, 0400);
- module_param(max_latency, int, 0400);
- module_param(neonmwi_monitor, int, 0600);
- module_param(neonmwi_level, int, 0600);
- module_param(neonmwi_envelope, int, 0600);
- module_param(neonmwi_offlimit, int, 0600);
- module_param(vpmsupport, int, 0400);
- module_param(forceload, int, 0600);
- MODULE_PARM_DESC(forceload,
- "Set to 1 in order to force an FPGA reload after power on.");
- module_param(companding, charp, 0400);
- MODULE_PARM_DESC(companding,
- "Change the companding to \"auto\" or \"alaw\" or \"ulaw\". Auto "
- "(default) will set everything to ulaw unless a BRI module is "
- "installed. It will use alaw in that case.");
- MODULE_DESCRIPTION("A4A,A4B,A8A,A8B Driver for Analog Telephony Cards");
- MODULE_AUTHOR("Digium Incorporated <support@digium.com>");
- MODULE_LICENSE("GPL v2");
- module_init(wcaxx_init);
- module_exit(wcaxx_cleanup);
|