| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203102041020510206102071020810209102101021110212102131021410215102161021710218102191022010221102221022310224102251022610227102281022910230102311023210233102341023510236102371023810239102401024110242102431024410245102461024710248102491025010251102521025310254102551025610257102581025910260102611026210263102641026510266102671026810269102701027110272102731027410275102761027710278102791028010281102821028310284102851028610287102881028910290102911029210293102941029510296102971029810299103001030110302103031030410305103061030710308103091031010311103121031310314103151031610317103181031910320103211032210323103241032510326103271032810329103301033110332103331033410335103361033710338103391034010341103421034310344103451034610347103481034910350103511035210353103541035510356103571035810359103601036110362103631036410365103661036710368103691037010371103721037310374103751037610377103781037910380103811038210383103841038510386103871038810389103901039110392103931039410395103961039710398103991040010401104021040310404104051040610407104081040910410104111041210413104141041510416104171041810419104201042110422104231042410425104261042710428104291043010431104321043310434104351043610437104381043910440104411044210443104441044510446104471044810449104501045110452104531045410455104561045710458104591046010461104621046310464104651046610467104681046910470104711047210473104741047510476104771047810479104801048110482104831048410485104861048710488104891049010491104921049310494104951049610497104981049910500105011050210503105041050510506105071050810509105101051110512105131051410515105161051710518105191052010521105221052310524105251052610527105281052910530105311053210533105341053510536105371053810539105401054110542105431054410545105461054710548105491055010551105521055310554105551055610557105581055910560105611056210563105641056510566105671056810569105701057110572105731057410575105761057710578105791058010581105821058310584105851058610587105881058910590105911059210593105941059510596105971059810599106001060110602106031060410605106061060710608106091061010611106121061310614106151061610617106181061910620106211062210623106241062510626106271062810629106301063110632106331063410635106361063710638106391064010641106421064310644106451064610647106481064910650106511065210653106541065510656106571065810659106601066110662106631066410665106661066710668106691067010671106721067310674106751067610677106781067910680106811068210683106841068510686106871068810689106901069110692106931069410695106961069710698106991070010701107021070310704107051070610707107081070910710107111071210713107141071510716107171071810719107201072110722107231072410725107261072710728107291073010731107321073310734107351073610737107381073910740107411074210743107441074510746107471074810749107501075110752107531075410755107561075710758107591076010761107621076310764107651076610767107681076910770107711077210773107741077510776107771077810779107801078110782107831078410785107861078710788107891079010791107921079310794107951079610797107981079910800108011080210803108041080510806108071080810809108101081110812108131081410815108161081710818108191082010821108221082310824108251082610827108281082910830108311083210833108341083510836108371083810839108401084110842108431084410845108461084710848108491085010851108521085310854108551085610857108581085910860108611086210863108641086510866108671086810869108701087110872108731087410875108761087710878108791088010881108821088310884108851088610887108881088910890108911089210893108941089510896108971089810899109001090110902109031090410905109061090710908109091091010911109121091310914109151091610917109181091910920109211092210923109241092510926109271092810929109301093110932109331093410935109361093710938109391094010941109421094310944109451094610947109481094910950109511095210953109541095510956109571095810959109601096110962109631096410965109661096710968109691097010971109721097310974109751097610977109781097910980109811098210983109841098510986109871098810989109901099110992109931099410995109961099710998109991100011001110021100311004110051100611007110081100911010110111101211013110141101511016110171101811019110201102111022110231102411025110261102711028110291103011031110321103311034110351103611037110381103911040110411104211043110441104511046110471104811049110501105111052110531105411055110561105711058110591106011061110621106311064110651106611067110681106911070110711107211073110741107511076110771107811079110801108111082110831108411085110861108711088110891109011091110921109311094110951109611097110981109911100111011110211103111041110511106111071110811109111101111111112111131111411115111161111711118111191112011121111221112311124111251112611127111281112911130111311113211133111341113511136111371113811139111401114111142111431114411145111461114711148111491115011151111521115311154111551115611157111581115911160111611116211163111641116511166111671116811169111701117111172111731117411175111761117711178111791118011181111821118311184111851118611187111881118911190111911119211193111941119511196111971119811199112001120111202112031120411205112061120711208112091121011211112121121311214112151121611217112181121911220112211122211223112241122511226112271122811229112301123111232112331123411235112361123711238112391124011241112421124311244112451124611247112481124911250112511125211253112541125511256112571125811259112601126111262112631126411265112661126711268112691127011271112721127311274112751127611277112781127911280112811128211283112841128511286112871128811289112901129111292112931129411295112961129711298112991130011301113021130311304113051130611307113081130911310113111131211313113141131511316113171131811319113201132111322113231132411325113261132711328113291133011331113321133311334113351133611337113381133911340113411134211343113441134511346113471134811349113501135111352113531135411355113561135711358113591136011361113621136311364113651136611367113681136911370113711137211373113741137511376113771137811379113801138111382113831138411385113861138711388113891139011391113921139311394113951139611397113981139911400114011140211403114041140511406114071140811409114101141111412114131141411415114161141711418114191142011421114221142311424114251142611427114281142911430114311143211433114341143511436114371143811439114401144111442114431144411445114461144711448114491145011451114521145311454114551145611457114581145911460114611146211463114641146511466114671146811469114701147111472114731147411475114761147711478114791148011481114821148311484114851148611487114881148911490114911149211493114941149511496114971149811499115001150111502115031150411505115061150711508115091151011511115121151311514115151151611517115181151911520115211152211523115241152511526115271152811529115301153111532115331153411535115361153711538115391154011541115421154311544115451154611547115481154911550115511155211553115541155511556115571155811559115601156111562115631156411565115661156711568115691157011571115721157311574115751157611577115781157911580115811158211583115841158511586115871158811589115901159111592115931159411595115961159711598115991160011601116021160311604116051160611607116081160911610116111161211613116141161511616116171161811619116201162111622116231162411625116261162711628116291163011631116321163311634116351163611637116381163911640116411164211643116441164511646116471164811649116501165111652116531165411655116561165711658116591166011661116621166311664116651166611667116681166911670116711167211673116741167511676116771167811679116801168111682116831168411685116861168711688116891169011691116921169311694116951169611697116981169911700117011170211703117041170511706117071170811709117101171111712117131171411715117161171711718117191172011721117221172311724117251172611727117281172911730117311173211733117341173511736117371173811739117401174111742117431174411745117461174711748117491175011751117521175311754117551175611757117581175911760117611176211763117641176511766117671176811769117701177111772117731177411775117761177711778117791178011781117821178311784117851178611787117881178911790117911179211793117941179511796117971179811799118001180111802118031180411805118061180711808118091181011811118121181311814118151181611817118181181911820118211182211823118241182511826118271182811829118301183111832118331183411835118361183711838118391184011841118421184311844118451184611847118481184911850118511185211853118541185511856118571185811859118601186111862118631186411865118661186711868118691187011871118721187311874118751187611877118781187911880118811188211883118841188511886118871188811889118901189111892118931189411895118961189711898118991190011901119021190311904119051190611907119081190911910119111191211913119141191511916119171191811919119201192111922119231192411925119261192711928119291193011931119321193311934119351193611937119381193911940119411194211943119441194511946119471194811949119501195111952119531195411955119561195711958119591196011961119621196311964119651196611967119681196911970119711197211973119741197511976119771197811979119801198111982119831198411985119861198711988119891199011991119921199311994119951199611997119981199912000120011200212003120041200512006120071200812009120101201112012120131201412015120161201712018120191202012021120221202312024120251202612027120281202912030120311203212033120341203512036120371203812039120401204112042120431204412045120461204712048120491205012051120521205312054120551205612057120581205912060120611206212063120641206512066120671206812069120701207112072120731207412075120761207712078120791208012081120821208312084120851208612087120881208912090120911209212093120941209512096120971209812099121001210112102121031210412105121061210712108121091211012111121121211312114121151211612117121181211912120121211212212123121241212512126121271212812129121301213112132121331213412135121361213712138121391214012141121421214312144121451214612147121481214912150121511215212153121541215512156121571215812159121601216112162121631216412165121661216712168121691217012171121721217312174121751217612177121781217912180121811218212183121841218512186121871218812189121901219112192121931219412195121961219712198121991220012201122021220312204122051220612207122081220912210122111221212213122141221512216122171221812219122201222112222122231222412225122261222712228122291223012231122321223312234122351223612237122381223912240122411224212243122441224512246122471224812249122501225112252122531225412255122561225712258122591226012261122621226312264122651226612267122681226912270122711227212273122741227512276122771227812279122801228112282122831228412285122861228712288122891229012291122921229312294122951229612297122981229912300123011230212303123041230512306123071230812309123101231112312123131231412315123161231712318123191232012321123221232312324123251232612327123281232912330123311233212333123341233512336123371233812339123401234112342123431234412345123461234712348123491235012351123521235312354123551235612357123581235912360123611236212363123641236512366123671236812369123701237112372123731237412375123761237712378123791238012381123821238312384123851238612387123881238912390123911239212393123941239512396 |
- /*
- Copyright (c), 2004-2005,2007-2010 Trident Microsystems, Inc.
- All rights reserved.
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are met:
- * Redistributions of source code must retain the above copyright notice,
- this list of conditions and the following disclaimer.
- * Redistributions in binary form must reproduce the above copyright notice,
- this list of conditions and the following disclaimer in the documentation
- and/or other materials provided with the distribution.
- * Neither the name of Trident Microsystems nor Hauppauge Computer Works
- nor the names of its contributors may be used to endorse or promote
- products derived from this software without specific prior written
- permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
- AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
- LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
- CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
- SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
- INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
- CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
- ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- POSSIBILITY OF SUCH DAMAGE.
- DRXJ specific implementation of DRX driver
- authors: Dragan Savic, Milos Nikolic, Mihajlo Katona, Tao Ding, Paul Janssen
- The Linux DVB Driver for Micronas DRX39xx family (drx3933j) was
- written by Devin Heitmueller <devin.heitmueller@kernellabs.com>
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 2 of the License, or
- (at your option) any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
- /*-----------------------------------------------------------------------------
- INCLUDE FILES
- ----------------------------------------------------------------------------*/
- #define pr_fmt(fmt) KBUILD_MODNAME ":%s: " fmt, __func__
- #include <linux/module.h>
- #include <linux/init.h>
- #include <linux/string.h>
- #include <linux/slab.h>
- #include <asm/div64.h>
- #include "dvb_frontend.h"
- #include "drx39xxj.h"
- #include "drxj.h"
- #include "drxj_map.h"
- /*============================================================================*/
- /*=== DEFINES ================================================================*/
- /*============================================================================*/
- #define DRX39XX_MAIN_FIRMWARE "dvb-fe-drxj-mc-1.0.8.fw"
- /*
- * \brief Maximum u32 value.
- */
- #ifndef MAX_U32
- #define MAX_U32 ((u32) (0xFFFFFFFFL))
- #endif
- /* Customer configurable hardware settings, etc */
- #ifndef MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH
- #define MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH 0x02
- #endif
- #ifndef MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH
- #define MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH 0x02
- #endif
- #ifndef MPEG_OUTPUT_CLK_DRIVE_STRENGTH
- #define MPEG_OUTPUT_CLK_DRIVE_STRENGTH 0x06
- #endif
- #ifndef OOB_CRX_DRIVE_STRENGTH
- #define OOB_CRX_DRIVE_STRENGTH 0x02
- #endif
- #ifndef OOB_DRX_DRIVE_STRENGTH
- #define OOB_DRX_DRIVE_STRENGTH 0x02
- #endif
- /*** START DJCOMBO patches to DRXJ registermap constants *********************/
- /*** registermap 200706071303 from drxj **************************************/
- #define ATV_TOP_CR_AMP_TH_FM 0x0
- #define ATV_TOP_CR_AMP_TH_L 0xA
- #define ATV_TOP_CR_AMP_TH_LP 0xA
- #define ATV_TOP_CR_AMP_TH_BG 0x8
- #define ATV_TOP_CR_AMP_TH_DK 0x8
- #define ATV_TOP_CR_AMP_TH_I 0x8
- #define ATV_TOP_CR_CONT_CR_D_MN 0x18
- #define ATV_TOP_CR_CONT_CR_D_FM 0x0
- #define ATV_TOP_CR_CONT_CR_D_L 0x20
- #define ATV_TOP_CR_CONT_CR_D_LP 0x20
- #define ATV_TOP_CR_CONT_CR_D_BG 0x18
- #define ATV_TOP_CR_CONT_CR_D_DK 0x18
- #define ATV_TOP_CR_CONT_CR_D_I 0x18
- #define ATV_TOP_CR_CONT_CR_I_MN 0x80
- #define ATV_TOP_CR_CONT_CR_I_FM 0x0
- #define ATV_TOP_CR_CONT_CR_I_L 0x80
- #define ATV_TOP_CR_CONT_CR_I_LP 0x80
- #define ATV_TOP_CR_CONT_CR_I_BG 0x80
- #define ATV_TOP_CR_CONT_CR_I_DK 0x80
- #define ATV_TOP_CR_CONT_CR_I_I 0x80
- #define ATV_TOP_CR_CONT_CR_P_MN 0x4
- #define ATV_TOP_CR_CONT_CR_P_FM 0x0
- #define ATV_TOP_CR_CONT_CR_P_L 0x4
- #define ATV_TOP_CR_CONT_CR_P_LP 0x4
- #define ATV_TOP_CR_CONT_CR_P_BG 0x4
- #define ATV_TOP_CR_CONT_CR_P_DK 0x4
- #define ATV_TOP_CR_CONT_CR_P_I 0x4
- #define ATV_TOP_CR_OVM_TH_MN 0xA0
- #define ATV_TOP_CR_OVM_TH_FM 0x0
- #define ATV_TOP_CR_OVM_TH_L 0xA0
- #define ATV_TOP_CR_OVM_TH_LP 0xA0
- #define ATV_TOP_CR_OVM_TH_BG 0xA0
- #define ATV_TOP_CR_OVM_TH_DK 0xA0
- #define ATV_TOP_CR_OVM_TH_I 0xA0
- #define ATV_TOP_EQU0_EQU_C0_FM 0x0
- #define ATV_TOP_EQU0_EQU_C0_L 0x3
- #define ATV_TOP_EQU0_EQU_C0_LP 0x3
- #define ATV_TOP_EQU0_EQU_C0_BG 0x7
- #define ATV_TOP_EQU0_EQU_C0_DK 0x0
- #define ATV_TOP_EQU0_EQU_C0_I 0x3
- #define ATV_TOP_EQU1_EQU_C1_FM 0x0
- #define ATV_TOP_EQU1_EQU_C1_L 0x1F6
- #define ATV_TOP_EQU1_EQU_C1_LP 0x1F6
- #define ATV_TOP_EQU1_EQU_C1_BG 0x197
- #define ATV_TOP_EQU1_EQU_C1_DK 0x198
- #define ATV_TOP_EQU1_EQU_C1_I 0x1F6
- #define ATV_TOP_EQU2_EQU_C2_FM 0x0
- #define ATV_TOP_EQU2_EQU_C2_L 0x28
- #define ATV_TOP_EQU2_EQU_C2_LP 0x28
- #define ATV_TOP_EQU2_EQU_C2_BG 0xC5
- #define ATV_TOP_EQU2_EQU_C2_DK 0xB0
- #define ATV_TOP_EQU2_EQU_C2_I 0x28
- #define ATV_TOP_EQU3_EQU_C3_FM 0x0
- #define ATV_TOP_EQU3_EQU_C3_L 0x192
- #define ATV_TOP_EQU3_EQU_C3_LP 0x192
- #define ATV_TOP_EQU3_EQU_C3_BG 0x12E
- #define ATV_TOP_EQU3_EQU_C3_DK 0x18E
- #define ATV_TOP_EQU3_EQU_C3_I 0x192
- #define ATV_TOP_STD_MODE_MN 0x0
- #define ATV_TOP_STD_MODE_FM 0x1
- #define ATV_TOP_STD_MODE_L 0x0
- #define ATV_TOP_STD_MODE_LP 0x0
- #define ATV_TOP_STD_MODE_BG 0x0
- #define ATV_TOP_STD_MODE_DK 0x0
- #define ATV_TOP_STD_MODE_I 0x0
- #define ATV_TOP_STD_VID_POL_MN 0x0
- #define ATV_TOP_STD_VID_POL_FM 0x0
- #define ATV_TOP_STD_VID_POL_L 0x2
- #define ATV_TOP_STD_VID_POL_LP 0x2
- #define ATV_TOP_STD_VID_POL_BG 0x0
- #define ATV_TOP_STD_VID_POL_DK 0x0
- #define ATV_TOP_STD_VID_POL_I 0x0
- #define ATV_TOP_VID_AMP_MN 0x380
- #define ATV_TOP_VID_AMP_FM 0x0
- #define ATV_TOP_VID_AMP_L 0xF50
- #define ATV_TOP_VID_AMP_LP 0xF50
- #define ATV_TOP_VID_AMP_BG 0x380
- #define ATV_TOP_VID_AMP_DK 0x394
- #define ATV_TOP_VID_AMP_I 0x3D8
- #define IQM_CF_OUT_ENA_OFDM__M 0x4
- #define IQM_FS_ADJ_SEL_B_QAM 0x1
- #define IQM_FS_ADJ_SEL_B_OFF 0x0
- #define IQM_FS_ADJ_SEL_B_VSB 0x2
- #define IQM_RC_ADJ_SEL_B_OFF 0x0
- #define IQM_RC_ADJ_SEL_B_QAM 0x1
- #define IQM_RC_ADJ_SEL_B_VSB 0x2
- /*** END DJCOMBO patches to DRXJ registermap *********************************/
- #include "drx_driver_version.h"
- /* #define DRX_DEBUG */
- #ifdef DRX_DEBUG
- #include <stdio.h>
- #endif
- /*-----------------------------------------------------------------------------
- ENUMS
- ----------------------------------------------------------------------------*/
- /*-----------------------------------------------------------------------------
- DEFINES
- ----------------------------------------------------------------------------*/
- #ifndef DRXJ_WAKE_UP_KEY
- #define DRXJ_WAKE_UP_KEY (demod->my_i2c_dev_addr->i2c_addr)
- #endif
- /*
- * \def DRXJ_DEF_I2C_ADDR
- * \brief Default I2C address of a demodulator instance.
- */
- #define DRXJ_DEF_I2C_ADDR (0x52)
- /*
- * \def DRXJ_DEF_DEMOD_DEV_ID
- * \brief Default device identifier of a demodultor instance.
- */
- #define DRXJ_DEF_DEMOD_DEV_ID (1)
- /*
- * \def DRXJ_SCAN_TIMEOUT
- * \brief Timeout value for waiting on demod lock during channel scan (millisec).
- */
- #define DRXJ_SCAN_TIMEOUT 1000
- /*
- * \def HI_I2C_DELAY
- * \brief HI timing delay for I2C timing (in nano seconds)
- *
- * Used to compute HI_CFG_DIV
- */
- #define HI_I2C_DELAY 42
- /*
- * \def HI_I2C_BRIDGE_DELAY
- * \brief HI timing delay for I2C timing (in nano seconds)
- *
- * Used to compute HI_CFG_BDL
- */
- #define HI_I2C_BRIDGE_DELAY 750
- /*
- * \brief Time Window for MER and SER Measurement in Units of Segment duration.
- */
- #define VSB_TOP_MEASUREMENT_PERIOD 64
- #define SYMBOLS_PER_SEGMENT 832
- /*
- * \brief bit rate and segment rate constants used for SER and BER.
- */
- /* values taken from the QAM microcode */
- #define DRXJ_QAM_SL_SIG_POWER_QAM_UNKNOWN 0
- #define DRXJ_QAM_SL_SIG_POWER_QPSK 32768
- #define DRXJ_QAM_SL_SIG_POWER_QAM8 24576
- #define DRXJ_QAM_SL_SIG_POWER_QAM16 40960
- #define DRXJ_QAM_SL_SIG_POWER_QAM32 20480
- #define DRXJ_QAM_SL_SIG_POWER_QAM64 43008
- #define DRXJ_QAM_SL_SIG_POWER_QAM128 20992
- #define DRXJ_QAM_SL_SIG_POWER_QAM256 43520
- /*
- * \brief Min supported symbolrates.
- */
- #ifndef DRXJ_QAM_SYMBOLRATE_MIN
- #define DRXJ_QAM_SYMBOLRATE_MIN (520000)
- #endif
- /*
- * \brief Max supported symbolrates.
- */
- #ifndef DRXJ_QAM_SYMBOLRATE_MAX
- #define DRXJ_QAM_SYMBOLRATE_MAX (7233000)
- #endif
- /*
- * \def DRXJ_QAM_MAX_WAITTIME
- * \brief Maximal wait time for QAM auto constellation in ms
- */
- #ifndef DRXJ_QAM_MAX_WAITTIME
- #define DRXJ_QAM_MAX_WAITTIME 900
- #endif
- #ifndef DRXJ_QAM_FEC_LOCK_WAITTIME
- #define DRXJ_QAM_FEC_LOCK_WAITTIME 150
- #endif
- #ifndef DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME
- #define DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME 200
- #endif
- /*
- * \def SCU status and results
- * \brief SCU
- */
- #define DRX_SCU_READY 0
- #define DRXJ_MAX_WAITTIME 100 /* ms */
- #define FEC_RS_MEASUREMENT_PERIOD 12894 /* 1 sec */
- #define FEC_RS_MEASUREMENT_PRESCALE 1 /* n sec */
- /*
- * \def DRX_AUD_MAX_DEVIATION
- * \brief Needed for calculation of prescale feature in AUD
- */
- #ifndef DRXJ_AUD_MAX_FM_DEVIATION
- #define DRXJ_AUD_MAX_FM_DEVIATION 100 /* kHz */
- #endif
- /*
- * \brief Needed for calculation of NICAM prescale feature in AUD
- */
- #ifndef DRXJ_AUD_MAX_NICAM_PRESCALE
- #define DRXJ_AUD_MAX_NICAM_PRESCALE (9) /* dB */
- #endif
- /*
- * \brief Needed for calculation of NICAM prescale feature in AUD
- */
- #ifndef DRXJ_AUD_MAX_WAITTIME
- #define DRXJ_AUD_MAX_WAITTIME 250 /* ms */
- #endif
- /* ATV config changed flags */
- #define DRXJ_ATV_CHANGED_COEF (0x00000001UL)
- #define DRXJ_ATV_CHANGED_PEAK_FLT (0x00000008UL)
- #define DRXJ_ATV_CHANGED_NOISE_FLT (0x00000010UL)
- #define DRXJ_ATV_CHANGED_OUTPUT (0x00000020UL)
- #define DRXJ_ATV_CHANGED_SIF_ATT (0x00000040UL)
- /* UIO define */
- #define DRX_UIO_MODE_FIRMWARE_SMA DRX_UIO_MODE_FIRMWARE0
- #define DRX_UIO_MODE_FIRMWARE_SAW DRX_UIO_MODE_FIRMWARE1
- /*
- * MICROCODE RELATED DEFINES
- */
- /* Magic word for checking correct Endianness of microcode data */
- #define DRX_UCODE_MAGIC_WORD ((((u16)'H')<<8)+((u16)'L'))
- /* CRC flag in ucode header, flags field. */
- #define DRX_UCODE_CRC_FLAG (0x0001)
- /*
- * Maximum size of buffer used to verify the microcode.
- * Must be an even number
- */
- #define DRX_UCODE_MAX_BUF_SIZE (DRXDAP_MAX_RCHUNKSIZE)
- #if DRX_UCODE_MAX_BUF_SIZE & 1
- #error DRX_UCODE_MAX_BUF_SIZE must be an even number
- #endif
- /*
- * Power mode macros
- */
- #define DRX_ISPOWERDOWNMODE(mode) ((mode == DRX_POWER_MODE_9) || \
- (mode == DRX_POWER_MODE_10) || \
- (mode == DRX_POWER_MODE_11) || \
- (mode == DRX_POWER_MODE_12) || \
- (mode == DRX_POWER_MODE_13) || \
- (mode == DRX_POWER_MODE_14) || \
- (mode == DRX_POWER_MODE_15) || \
- (mode == DRX_POWER_MODE_16) || \
- (mode == DRX_POWER_DOWN))
- /* Pin safe mode macro */
- #define DRXJ_PIN_SAFE_MODE 0x0000
- /*============================================================================*/
- /*=== GLOBAL VARIABLEs =======================================================*/
- /*============================================================================*/
- /*
- */
- /*
- * \brief Temporary register definitions.
- * (register definitions that are not yet available in register master)
- */
- /*****************************************************************************/
- /* Audio block 0x103 is write only. To avoid shadowing in driver accessing */
- /* RAM adresses directly. This must be READ ONLY to avoid problems. */
- /* Writing to the interface adresses is more than only writing the RAM */
- /* locations */
- /*****************************************************************************/
- /*
- * \brief RAM location of MODUS registers
- */
- #define AUD_DEM_RAM_MODUS_HI__A 0x10204A3
- #define AUD_DEM_RAM_MODUS_HI__M 0xF000
- #define AUD_DEM_RAM_MODUS_LO__A 0x10204A4
- #define AUD_DEM_RAM_MODUS_LO__M 0x0FFF
- /*
- * \brief RAM location of I2S config registers
- */
- #define AUD_DEM_RAM_I2S_CONFIG1__A 0x10204B1
- #define AUD_DEM_RAM_I2S_CONFIG2__A 0x10204B2
- /*
- * \brief RAM location of DCO config registers
- */
- #define AUD_DEM_RAM_DCO_B_HI__A 0x1020461
- #define AUD_DEM_RAM_DCO_B_LO__A 0x1020462
- #define AUD_DEM_RAM_DCO_A_HI__A 0x1020463
- #define AUD_DEM_RAM_DCO_A_LO__A 0x1020464
- /*
- * \brief RAM location of Threshold registers
- */
- #define AUD_DEM_RAM_NICAM_THRSHLD__A 0x102045A
- #define AUD_DEM_RAM_A2_THRSHLD__A 0x10204BB
- #define AUD_DEM_RAM_BTSC_THRSHLD__A 0x10204A6
- /*
- * \brief RAM location of Carrier Threshold registers
- */
- #define AUD_DEM_RAM_CM_A_THRSHLD__A 0x10204AF
- #define AUD_DEM_RAM_CM_B_THRSHLD__A 0x10204B0
- /*
- * \brief FM Matrix register fix
- */
- #ifdef AUD_DEM_WR_FM_MATRIX__A
- #undef AUD_DEM_WR_FM_MATRIX__A
- #endif
- #define AUD_DEM_WR_FM_MATRIX__A 0x105006F
- /*============================================================================*/
- /*
- * \brief Defines required for audio
- */
- #define AUD_VOLUME_ZERO_DB 115
- #define AUD_VOLUME_DB_MIN -60
- #define AUD_VOLUME_DB_MAX 12
- #define AUD_CARRIER_STRENGTH_QP_0DB 0x4000
- #define AUD_CARRIER_STRENGTH_QP_0DB_LOG10T100 421
- #define AUD_MAX_AVC_REF_LEVEL 15
- #define AUD_I2S_FREQUENCY_MAX 48000UL
- #define AUD_I2S_FREQUENCY_MIN 12000UL
- #define AUD_RDS_ARRAY_SIZE 18
- /*
- * \brief Needed for calculation of prescale feature in AUD
- */
- #ifndef DRX_AUD_MAX_FM_DEVIATION
- #define DRX_AUD_MAX_FM_DEVIATION (100) /* kHz */
- #endif
- /*
- * \brief Needed for calculation of NICAM prescale feature in AUD
- */
- #ifndef DRX_AUD_MAX_NICAM_PRESCALE
- #define DRX_AUD_MAX_NICAM_PRESCALE (9) /* dB */
- #endif
- /*============================================================================*/
- /* Values for I2S Master/Slave pin configurations */
- #define SIO_PDR_I2S_CL_CFG_MODE__MASTER 0x0004
- #define SIO_PDR_I2S_CL_CFG_DRIVE__MASTER 0x0008
- #define SIO_PDR_I2S_CL_CFG_MODE__SLAVE 0x0004
- #define SIO_PDR_I2S_CL_CFG_DRIVE__SLAVE 0x0000
- #define SIO_PDR_I2S_DA_CFG_MODE__MASTER 0x0003
- #define SIO_PDR_I2S_DA_CFG_DRIVE__MASTER 0x0008
- #define SIO_PDR_I2S_DA_CFG_MODE__SLAVE 0x0003
- #define SIO_PDR_I2S_DA_CFG_DRIVE__SLAVE 0x0008
- #define SIO_PDR_I2S_WS_CFG_MODE__MASTER 0x0004
- #define SIO_PDR_I2S_WS_CFG_DRIVE__MASTER 0x0008
- #define SIO_PDR_I2S_WS_CFG_MODE__SLAVE 0x0004
- #define SIO_PDR_I2S_WS_CFG_DRIVE__SLAVE 0x0000
- /*============================================================================*/
- /*=== REGISTER ACCESS MACROS =================================================*/
- /*============================================================================*/
- /*
- * This macro is used to create byte arrays for block writes.
- * Block writes speed up I2C traffic between host and demod.
- * The macro takes care of the required byte order in a 16 bits word.
- * x -> lowbyte(x), highbyte(x)
- */
- #define DRXJ_16TO8(x) ((u8) (((u16)x) & 0xFF)), \
- ((u8)((((u16)x)>>8)&0xFF))
- /*
- * This macro is used to convert byte array to 16 bit register value for block read.
- * Block read speed up I2C traffic between host and demod.
- * The macro takes care of the required byte order in a 16 bits word.
- */
- #define DRXJ_8TO16(x) ((u16) (x[0] | (x[1] << 8)))
- /*============================================================================*/
- /*=== MISC DEFINES ===========================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*=== HI COMMAND RELATED DEFINES =============================================*/
- /*============================================================================*/
- /*
- * \brief General maximum number of retries for ucode command interfaces
- */
- #define DRXJ_MAX_RETRIES (100)
- /*============================================================================*/
- /*=== STANDARD RELATED MACROS ================================================*/
- /*============================================================================*/
- #define DRXJ_ISATVSTD(std) ((std == DRX_STANDARD_PAL_SECAM_BG) || \
- (std == DRX_STANDARD_PAL_SECAM_DK) || \
- (std == DRX_STANDARD_PAL_SECAM_I) || \
- (std == DRX_STANDARD_PAL_SECAM_L) || \
- (std == DRX_STANDARD_PAL_SECAM_LP) || \
- (std == DRX_STANDARD_NTSC) || \
- (std == DRX_STANDARD_FM))
- #define DRXJ_ISQAMSTD(std) ((std == DRX_STANDARD_ITU_A) || \
- (std == DRX_STANDARD_ITU_B) || \
- (std == DRX_STANDARD_ITU_C) || \
- (std == DRX_STANDARD_ITU_D))
- /*-----------------------------------------------------------------------------
- GLOBAL VARIABLES
- ----------------------------------------------------------------------------*/
- /*
- * DRXJ DAP structures
- */
- static int drxdap_fasi_read_block(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 datasize,
- u8 *data, u32 flags);
- static int drxj_dap_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
- u32 waddr,
- u32 raddr,
- u16 wdata, u16 *rdata);
- static int drxj_dap_read_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 *data, u32 flags);
- static int drxdap_fasi_read_reg32(struct i2c_device_addr *dev_addr,
- u32 addr,
- u32 *data, u32 flags);
- static int drxdap_fasi_write_block(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 datasize,
- u8 *data, u32 flags);
- static int drxj_dap_write_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 data, u32 flags);
- static int drxdap_fasi_write_reg32(struct i2c_device_addr *dev_addr,
- u32 addr,
- u32 data, u32 flags);
- static struct drxj_data drxj_data_g = {
- false, /* has_lna : true if LNA (aka PGA) present */
- false, /* has_oob : true if OOB supported */
- false, /* has_ntsc: true if NTSC supported */
- false, /* has_btsc: true if BTSC supported */
- false, /* has_smatx: true if SMA_TX pin is available */
- false, /* has_smarx: true if SMA_RX pin is available */
- false, /* has_gpio : true if GPIO pin is available */
- false, /* has_irqn : true if IRQN pin is available */
- 0, /* mfx A1/A2/A... */
- /* tuner settings */
- false, /* tuner mirrors RF signal */
- /* standard/channel settings */
- DRX_STANDARD_UNKNOWN, /* current standard */
- DRX_CONSTELLATION_AUTO, /* constellation */
- 0, /* frequency in KHz */
- DRX_BANDWIDTH_UNKNOWN, /* curr_bandwidth */
- DRX_MIRROR_NO, /* mirror */
- /* signal quality information: */
- /* default values taken from the QAM Programming guide */
- /* fec_bits_desired should not be less than 4000000 */
- 4000000, /* fec_bits_desired */
- 5, /* fec_vd_plen */
- 4, /* qam_vd_prescale */
- 0xFFFF, /* qamVDPeriod */
- 204 * 8, /* fec_rs_plen annex A */
- 1, /* fec_rs_prescale */
- FEC_RS_MEASUREMENT_PERIOD, /* fec_rs_period */
- true, /* reset_pkt_err_acc */
- 0, /* pkt_err_acc_start */
- /* HI configuration */
- 0, /* hi_cfg_timing_div */
- 0, /* hi_cfg_bridge_delay */
- 0, /* hi_cfg_wake_up_key */
- 0, /* hi_cfg_ctrl */
- 0, /* HICfgTimeout */
- /* UIO configuration */
- DRX_UIO_MODE_DISABLE, /* uio_sma_rx_mode */
- DRX_UIO_MODE_DISABLE, /* uio_sma_tx_mode */
- DRX_UIO_MODE_DISABLE, /* uioASELMode */
- DRX_UIO_MODE_DISABLE, /* uio_irqn_mode */
- /* FS setting */
- 0UL, /* iqm_fs_rate_ofs */
- false, /* pos_image */
- /* RC setting */
- 0UL, /* iqm_rc_rate_ofs */
- /* AUD information */
- /* false, * flagSetAUDdone */
- /* false, * detectedRDS */
- /* true, * flagASDRequest */
- /* false, * flagHDevClear */
- /* false, * flagHDevSet */
- /* (u16) 0xFFF, * rdsLastCount */
- /* ATV configuration */
- 0UL, /* flags cfg changes */
- /* shadow of ATV_TOP_EQU0__A */
- {-5,
- ATV_TOP_EQU0_EQU_C0_FM,
- ATV_TOP_EQU0_EQU_C0_L,
- ATV_TOP_EQU0_EQU_C0_LP,
- ATV_TOP_EQU0_EQU_C0_BG,
- ATV_TOP_EQU0_EQU_C0_DK,
- ATV_TOP_EQU0_EQU_C0_I},
- /* shadow of ATV_TOP_EQU1__A */
- {-50,
- ATV_TOP_EQU1_EQU_C1_FM,
- ATV_TOP_EQU1_EQU_C1_L,
- ATV_TOP_EQU1_EQU_C1_LP,
- ATV_TOP_EQU1_EQU_C1_BG,
- ATV_TOP_EQU1_EQU_C1_DK,
- ATV_TOP_EQU1_EQU_C1_I},
- /* shadow of ATV_TOP_EQU2__A */
- {210,
- ATV_TOP_EQU2_EQU_C2_FM,
- ATV_TOP_EQU2_EQU_C2_L,
- ATV_TOP_EQU2_EQU_C2_LP,
- ATV_TOP_EQU2_EQU_C2_BG,
- ATV_TOP_EQU2_EQU_C2_DK,
- ATV_TOP_EQU2_EQU_C2_I},
- /* shadow of ATV_TOP_EQU3__A */
- {-160,
- ATV_TOP_EQU3_EQU_C3_FM,
- ATV_TOP_EQU3_EQU_C3_L,
- ATV_TOP_EQU3_EQU_C3_LP,
- ATV_TOP_EQU3_EQU_C3_BG,
- ATV_TOP_EQU3_EQU_C3_DK,
- ATV_TOP_EQU3_EQU_C3_I},
- false, /* flag: true=bypass */
- ATV_TOP_VID_PEAK__PRE, /* shadow of ATV_TOP_VID_PEAK__A */
- ATV_TOP_NOISE_TH__PRE, /* shadow of ATV_TOP_NOISE_TH__A */
- true, /* flag CVBS ouput enable */
- false, /* flag SIF ouput enable */
- DRXJ_SIF_ATTENUATION_0DB, /* current SIF att setting */
- { /* qam_rf_agc_cfg */
- DRX_STANDARD_ITU_B, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level */
- 0xFFFF, /* max_output_level */
- 0x0000, /* speed */
- 0x0000, /* top */
- 0x0000 /* c.o.c. */
- },
- { /* qam_if_agc_cfg */
- DRX_STANDARD_ITU_B, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level */
- 0xFFFF, /* max_output_level */
- 0x0000, /* speed */
- 0x0000, /* top (don't care) */
- 0x0000 /* c.o.c. (don't care) */
- },
- { /* vsb_rf_agc_cfg */
- DRX_STANDARD_8VSB, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level */
- 0xFFFF, /* max_output_level */
- 0x0000, /* speed */
- 0x0000, /* top (don't care) */
- 0x0000 /* c.o.c. (don't care) */
- },
- { /* vsb_if_agc_cfg */
- DRX_STANDARD_8VSB, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level */
- 0xFFFF, /* max_output_level */
- 0x0000, /* speed */
- 0x0000, /* top (don't care) */
- 0x0000 /* c.o.c. (don't care) */
- },
- 0, /* qam_pga_cfg */
- 0, /* vsb_pga_cfg */
- { /* qam_pre_saw_cfg */
- DRX_STANDARD_ITU_B, /* standard */
- 0, /* reference */
- false /* use_pre_saw */
- },
- { /* vsb_pre_saw_cfg */
- DRX_STANDARD_8VSB, /* standard */
- 0, /* reference */
- false /* use_pre_saw */
- },
- /* Version information */
- #ifndef _CH_
- {
- "01234567890", /* human readable version microcode */
- "01234567890" /* human readable version device specific code */
- },
- {
- { /* struct drx_version for microcode */
- DRX_MODULE_UNKNOWN,
- (char *)(NULL),
- 0,
- 0,
- 0,
- (char *)(NULL)
- },
- { /* struct drx_version for device specific code */
- DRX_MODULE_UNKNOWN,
- (char *)(NULL),
- 0,
- 0,
- 0,
- (char *)(NULL)
- }
- },
- {
- { /* struct drx_version_list for microcode */
- (struct drx_version *) (NULL),
- (struct drx_version_list *) (NULL)
- },
- { /* struct drx_version_list for device specific code */
- (struct drx_version *) (NULL),
- (struct drx_version_list *) (NULL)
- }
- },
- #endif
- false, /* smart_ant_inverted */
- /* Tracking filter setting for OOB */
- {
- 12000,
- 9300,
- 6600,
- 5280,
- 3700,
- 3000,
- 2000,
- 0},
- false, /* oob_power_on */
- 0, /* mpeg_ts_static_bitrate */
- false, /* disable_te_ihandling */
- false, /* bit_reverse_mpeg_outout */
- DRXJ_MPEGOUTPUT_CLOCK_RATE_AUTO, /* mpeg_output_clock_rate */
- DRXJ_MPEG_START_WIDTH_1CLKCYC, /* mpeg_start_width */
- /* Pre SAW & Agc configuration for ATV */
- {
- DRX_STANDARD_NTSC, /* standard */
- 7, /* reference */
- true /* use_pre_saw */
- },
- { /* ATV RF-AGC */
- DRX_STANDARD_NTSC, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level (d.c.) */
- 0, /* max_output_level (d.c.) */
- 3, /* speed */
- 9500, /* top */
- 4000 /* cut-off current */
- },
- { /* ATV IF-AGC */
- DRX_STANDARD_NTSC, /* standard */
- DRX_AGC_CTRL_AUTO, /* ctrl_mode */
- 0, /* output_level */
- 0, /* min_output_level (d.c.) */
- 0, /* max_output_level (d.c.) */
- 3, /* speed */
- 2400, /* top */
- 0 /* c.o.c. (d.c.) */
- },
- 140, /* ATV PGA config */
- 0, /* curr_symbol_rate */
- false, /* pdr_safe_mode */
- SIO_PDR_GPIO_CFG__PRE, /* pdr_safe_restore_val_gpio */
- SIO_PDR_VSYNC_CFG__PRE, /* pdr_safe_restore_val_v_sync */
- SIO_PDR_SMA_RX_CFG__PRE, /* pdr_safe_restore_val_sma_rx */
- SIO_PDR_SMA_TX_CFG__PRE, /* pdr_safe_restore_val_sma_tx */
- 4, /* oob_pre_saw */
- DRXJ_OOB_LO_POW_MINUS10DB, /* oob_lo_pow */
- {
- false /* aud_data, only first member */
- },
- };
- /*
- * \var drxj_default_addr_g
- * \brief Default I2C address and device identifier.
- */
- static struct i2c_device_addr drxj_default_addr_g = {
- DRXJ_DEF_I2C_ADDR, /* i2c address */
- DRXJ_DEF_DEMOD_DEV_ID /* device id */
- };
- /*
- * \var drxj_default_comm_attr_g
- * \brief Default common attributes of a drxj demodulator instance.
- */
- static struct drx_common_attr drxj_default_comm_attr_g = {
- NULL, /* ucode file */
- true, /* ucode verify switch */
- {0}, /* version record */
- 44000, /* IF in kHz in case no tuner instance is used */
- (151875 - 0), /* system clock frequency in kHz */
- 0, /* oscillator frequency kHz */
- 0, /* oscillator deviation in ppm, signed */
- false, /* If true mirror frequency spectrum */
- {
- /* MPEG output configuration */
- true, /* If true, enable MPEG ouput */
- false, /* If true, insert RS byte */
- false, /* If true, parallel out otherwise serial */
- false, /* If true, invert DATA signals */
- false, /* If true, invert ERR signal */
- false, /* If true, invert STR signals */
- false, /* If true, invert VAL signals */
- false, /* If true, invert CLK signals */
- true, /* If true, static MPEG clockrate will
- be used, otherwise clockrate will
- adapt to the bitrate of the TS */
- 19392658UL, /* Maximum bitrate in b/s in case
- static clockrate is selected */
- DRX_MPEG_STR_WIDTH_1 /* MPEG Start width in clock cycles */
- },
- /* Initilisations below can be omitted, they require no user input and
- are initialy 0, NULL or false. The compiler will initialize them to these
- values when omitted. */
- false, /* is_opened */
- /* SCAN */
- NULL, /* no scan params yet */
- 0, /* current scan index */
- 0, /* next scan frequency */
- false, /* scan ready flag */
- 0, /* max channels to scan */
- 0, /* nr of channels scanned */
- NULL, /* default scan function */
- NULL, /* default context pointer */
- 0, /* millisec to wait for demod lock */
- DRXJ_DEMOD_LOCK, /* desired lock */
- false,
- /* Power management */
- DRX_POWER_UP,
- /* Tuner */
- 1, /* nr of I2C port to wich tuner is */
- 0L, /* minimum RF input frequency, in kHz */
- 0L, /* maximum RF input frequency, in kHz */
- false, /* Rf Agc Polarity */
- false, /* If Agc Polarity */
- false, /* tuner slow mode */
- { /* current channel (all 0) */
- 0UL /* channel.frequency */
- },
- DRX_STANDARD_UNKNOWN, /* current standard */
- DRX_STANDARD_UNKNOWN, /* previous standard */
- DRX_STANDARD_UNKNOWN, /* di_cache_standard */
- false, /* use_bootloader */
- 0UL, /* capabilities */
- 0 /* mfx */
- };
- /*
- * \var drxj_default_demod_g
- * \brief Default drxj demodulator instance.
- */
- static struct drx_demod_instance drxj_default_demod_g = {
- &drxj_default_addr_g, /* i2c address & device id */
- &drxj_default_comm_attr_g, /* demod common attributes */
- &drxj_data_g /* demod device specific attributes */
- };
- /*
- * \brief Default audio data structure for DRK demodulator instance.
- *
- * This structure is DRXK specific.
- *
- */
- static struct drx_aud_data drxj_default_aud_data_g = {
- false, /* audio_is_active */
- DRX_AUD_STANDARD_AUTO, /* audio_standard */
- /* i2sdata */
- {
- false, /* output_enable */
- 48000, /* frequency */
- DRX_I2S_MODE_MASTER, /* mode */
- DRX_I2S_WORDLENGTH_32, /* word_length */
- DRX_I2S_POLARITY_RIGHT, /* polarity */
- DRX_I2S_FORMAT_WS_WITH_DATA /* format */
- },
- /* volume */
- {
- true, /* mute; */
- 0, /* volume */
- DRX_AUD_AVC_OFF, /* avc_mode */
- 0, /* avc_ref_level */
- DRX_AUD_AVC_MAX_GAIN_12DB, /* avc_max_gain */
- DRX_AUD_AVC_MAX_ATTEN_24DB, /* avc_max_atten */
- 0, /* strength_left */
- 0 /* strength_right */
- },
- DRX_AUD_AUTO_SOUND_SELECT_ON_CHANGE_ON, /* auto_sound */
- /* ass_thresholds */
- {
- 440, /* A2 */
- 12, /* BTSC */
- 700, /* NICAM */
- },
- /* carrier */
- {
- /* a */
- {
- 42, /* thres */
- DRX_NO_CARRIER_NOISE, /* opt */
- 0, /* shift */
- 0 /* dco */
- },
- /* b */
- {
- 42, /* thres */
- DRX_NO_CARRIER_MUTE, /* opt */
- 0, /* shift */
- 0 /* dco */
- },
- },
- /* mixer */
- {
- DRX_AUD_SRC_STEREO_OR_A, /* source_i2s */
- DRX_AUD_I2S_MATRIX_STEREO, /* matrix_i2s */
- DRX_AUD_FM_MATRIX_SOUND_A /* matrix_fm */
- },
- DRX_AUD_DEVIATION_NORMAL, /* deviation */
- DRX_AUD_AVSYNC_OFF, /* av_sync */
- /* prescale */
- {
- DRX_AUD_MAX_FM_DEVIATION, /* fm_deviation */
- DRX_AUD_MAX_NICAM_PRESCALE /* nicam_gain */
- },
- DRX_AUD_FM_DEEMPH_75US, /* deemph */
- DRX_BTSC_STEREO, /* btsc_detect */
- 0, /* rds_data_counter */
- false /* rds_data_present */
- };
- /*-----------------------------------------------------------------------------
- STRUCTURES
- ----------------------------------------------------------------------------*/
- struct drxjeq_stat {
- u16 eq_mse;
- u8 eq_mode;
- u8 eq_ctrl;
- u8 eq_stat;
- };
- /* HI command */
- struct drxj_hi_cmd {
- u16 cmd;
- u16 param1;
- u16 param2;
- u16 param3;
- u16 param4;
- u16 param5;
- u16 param6;
- };
- /*============================================================================*/
- /*=== MICROCODE RELATED STRUCTURES ===========================================*/
- /*============================================================================*/
- /*
- * struct drxu_code_block_hdr - Structure of the microcode block headers
- *
- * @addr: Destination address of the data in this block
- * @size: Size of the block data following this header counted in
- * 16 bits words
- * @CRC: CRC value of the data block, only valid if CRC flag is
- * set.
- */
- struct drxu_code_block_hdr {
- u32 addr;
- u16 size;
- u16 flags;
- u16 CRC;
- };
- /*-----------------------------------------------------------------------------
- FUNCTIONS
- ----------------------------------------------------------------------------*/
- /* Some prototypes */
- static int
- hi_command(struct i2c_device_addr *dev_addr,
- const struct drxj_hi_cmd *cmd, u16 *result);
- static int
- ctrl_lock_status(struct drx_demod_instance *demod, enum drx_lock_status *lock_stat);
- static int
- ctrl_power_mode(struct drx_demod_instance *demod, enum drx_power_mode *mode);
- static int power_down_aud(struct drx_demod_instance *demod);
- static int
- ctrl_set_cfg_pre_saw(struct drx_demod_instance *demod, struct drxj_cfg_pre_saw *pre_saw);
- static int
- ctrl_set_cfg_afe_gain(struct drx_demod_instance *demod, struct drxj_cfg_afe_gain *afe_gain);
- /*============================================================================*/
- /*============================================================================*/
- /*== HELPER FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn u32 frac28(u32 N, u32 D)
- * \brief Compute: (1<<28)*N/D
- * \param N 32 bits
- * \param D 32 bits
- * \return (1<<28)*N/D
- * This function is used to avoid floating-point calculations as they may
- * not be present on the target platform.
- * frac28 performs an unsigned 28/28 bits division to 32-bit fixed point
- * fraction used for setting the Frequency Shifter registers.
- * N and D can hold numbers up to width: 28-bits.
- * The 4 bits integer part and the 28 bits fractional part are calculated.
- * Usage condition: ((1<<28)*n)/d < ((1<<32)-1) => (n/d) < 15.999
- * N: 0...(1<<28)-1 = 268435454
- * D: 0...(1<<28)-1
- * Q: 0...(1<<32)-1
- */
- static u32 frac28(u32 N, u32 D)
- {
- int i = 0;
- u32 Q1 = 0;
- u32 R0 = 0;
- R0 = (N % D) << 4; /* 32-28 == 4 shifts possible at max */
- Q1 = N / D; /* integer part, only the 4 least significant bits
- will be visible in the result */
- /* division using radix 16, 7 nibbles in the result */
- for (i = 0; i < 7; i++) {
- Q1 = (Q1 << 4) | R0 / D;
- R0 = (R0 % D) << 4;
- }
- /* rounding */
- if ((R0 >> 3) >= D)
- Q1++;
- return Q1;
- }
- /*
- * \fn u32 log1_times100( u32 x)
- * \brief Compute: 100*log10(x)
- * \param x 32 bits
- * \return 100*log10(x)
- *
- * 100*log10(x)
- * = 100*(log2(x)/log2(10)))
- * = (100*(2^15)*log2(x))/((2^15)*log2(10))
- * = ((200*(2^15)*log2(x))/((2^15)*log2(10)))/2
- * = ((200*(2^15)*(log2(x/y)+log2(y)))/((2^15)*log2(10)))/2
- * = ((200*(2^15)*log2(x/y))+(200*(2^15)*log2(y)))/((2^15)*log2(10)))/2
- *
- * where y = 2^k and 1<= (x/y) < 2
- */
- static u32 log1_times100(u32 x)
- {
- static const u8 scale = 15;
- static const u8 index_width = 5;
- /*
- log2lut[n] = (1<<scale) * 200 * log2( 1.0 + ( (1.0/(1<<INDEXWIDTH)) * n ))
- 0 <= n < ((1<<INDEXWIDTH)+1)
- */
- static const u32 log2lut[] = {
- 0, /* 0.000000 */
- 290941, /* 290941.300628 */
- 573196, /* 573196.476418 */
- 847269, /* 847269.179851 */
- 1113620, /* 1113620.489452 */
- 1372674, /* 1372673.576986 */
- 1624818, /* 1624817.752104 */
- 1870412, /* 1870411.981536 */
- 2109788, /* 2109787.962654 */
- 2343253, /* 2343252.817465 */
- 2571091, /* 2571091.461923 */
- 2793569, /* 2793568.696416 */
- 3010931, /* 3010931.055901 */
- 3223408, /* 3223408.452106 */
- 3431216, /* 3431215.635215 */
- 3634553, /* 3634553.498355 */
- 3833610, /* 3833610.244726 */
- 4028562, /* 4028562.434393 */
- 4219576, /* 4219575.925308 */
- 4406807, /* 4406806.721144 */
- 4590402, /* 4590401.736809 */
- 4770499, /* 4770499.491025 */
- 4947231, /* 4947230.734179 */
- 5120719, /* 5120719.018555 */
- 5291081, /* 5291081.217197 */
- 5458428, /* 5458427.996830 */
- 5622864, /* 5622864.249668 */
- 5784489, /* 5784489.488298 */
- 5943398, /* 5943398.207380 */
- 6099680, /* 6099680.215452 */
- 6253421, /* 6253420.939751 */
- 6404702, /* 6404701.706649 */
- 6553600, /* 6553600.000000 */
- };
- u8 i = 0;
- u32 y = 0;
- u32 d = 0;
- u32 k = 0;
- u32 r = 0;
- if (x == 0)
- return 0;
- /* Scale x (normalize) */
- /* computing y in log(x/y) = log(x) - log(y) */
- if ((x & (((u32) (-1)) << (scale + 1))) == 0) {
- for (k = scale; k > 0; k--) {
- if (x & (((u32) 1) << scale))
- break;
- x <<= 1;
- }
- } else {
- for (k = scale; k < 31; k++) {
- if ((x & (((u32) (-1)) << (scale + 1))) == 0)
- break;
- x >>= 1;
- }
- }
- /*
- Now x has binary point between bit[scale] and bit[scale-1]
- and 1.0 <= x < 2.0 */
- /* correction for division: log(x) = log(x/y)+log(y) */
- y = k * ((((u32) 1) << scale) * 200);
- /* remove integer part */
- x &= ((((u32) 1) << scale) - 1);
- /* get index */
- i = (u8) (x >> (scale - index_width));
- /* compute delta (x-a) */
- d = x & ((((u32) 1) << (scale - index_width)) - 1);
- /* compute log, multiplication ( d* (.. )) must be within range ! */
- y += log2lut[i] +
- ((d * (log2lut[i + 1] - log2lut[i])) >> (scale - index_width));
- /* Conver to log10() */
- y /= 108853; /* (log2(10) << scale) */
- r = (y >> 1);
- /* rounding */
- if (y & ((u32)1))
- r++;
- return r;
- }
- /*
- * \fn u32 frac_times1e6( u16 N, u32 D)
- * \brief Compute: (N/D) * 1000000.
- * \param N nominator 16-bits.
- * \param D denominator 32-bits.
- * \return u32
- * \retval ((N/D) * 1000000), 32 bits
- *
- * No check on D=0!
- */
- static u32 frac_times1e6(u32 N, u32 D)
- {
- u32 remainder = 0;
- u32 frac = 0;
- /*
- frac = (N * 1000000) / D
- To let it fit in a 32 bits computation:
- frac = (N * (1000000 >> 4)) / (D >> 4)
- This would result in a problem in case D < 16 (div by 0).
- So we do it more elaborate as shown below.
- */
- frac = (((u32) N) * (1000000 >> 4)) / D;
- frac <<= 4;
- remainder = (((u32) N) * (1000000 >> 4)) % D;
- remainder <<= 4;
- frac += remainder / D;
- remainder = remainder % D;
- if ((remainder * 2) > D)
- frac++;
- return frac;
- }
- /*============================================================================*/
- /*
- * \brief Values for NICAM prescaler gain. Computed from dB to integer
- * and rounded. For calc used formula: 16*10^(prescaleGain[dB]/20).
- *
- */
- #if 0
- /* Currently, unused as we lack support for analog TV */
- static const u16 nicam_presc_table_val[43] = {
- 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 4, 4,
- 5, 5, 6, 6, 7, 8, 9, 10, 11, 13, 14, 16,
- 18, 20, 23, 25, 28, 32, 36, 40, 45,
- 51, 57, 64, 71, 80, 90, 101, 113, 127
- };
- #endif
- /*============================================================================*/
- /*== END HELPER FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== DRXJ DAP FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- This layer takes care of some device specific register access protocols:
- -conversion to short address format
- -access to audio block
- This layer is placed between the drx_dap_fasi and the rest of the drxj
- specific implementation. This layer can use address map knowledge whereas
- dap_fasi may not use memory map knowledge.
- * For audio currently only 16 bits read and write register access is
- supported. More is not needed. RMW and 32 or 8 bit access on audio
- registers will have undefined behaviour. Flags (RMW, CRC reset, broadcast
- single/multi master) will be ignored.
- TODO: check ignoring single/multimaster is ok for AUD access ?
- */
- #define DRXJ_ISAUDWRITE(addr) (((((addr)>>16)&1) == 1) ? true : false)
- #define DRXJ_DAP_AUDTRIF_TIMEOUT 80 /* millisec */
- /*============================================================================*/
- /*
- * \fn bool is_handled_by_aud_tr_if( u32 addr )
- * \brief Check if this address is handled by the audio token ring interface.
- * \param addr
- * \return bool
- * \retval true Yes, handled by audio token ring interface
- * \retval false No, not handled by audio token ring interface
- *
- */
- static
- bool is_handled_by_aud_tr_if(u32 addr)
- {
- bool retval = false;
- if ((DRXDAP_FASI_ADDR2BLOCK(addr) == 4) &&
- (DRXDAP_FASI_ADDR2BANK(addr) > 1) &&
- (DRXDAP_FASI_ADDR2BANK(addr) < 6)) {
- retval = true;
- }
- return retval;
- }
- /*============================================================================*/
- int drxbsp_i2c_write_read(struct i2c_device_addr *w_dev_addr,
- u16 w_count,
- u8 *wData,
- struct i2c_device_addr *r_dev_addr,
- u16 r_count, u8 *r_data)
- {
- struct drx39xxj_state *state;
- struct i2c_msg msg[2];
- unsigned int num_msgs;
- if (w_dev_addr == NULL) {
- /* Read only */
- state = r_dev_addr->user_data;
- msg[0].addr = r_dev_addr->i2c_addr >> 1;
- msg[0].flags = I2C_M_RD;
- msg[0].buf = r_data;
- msg[0].len = r_count;
- num_msgs = 1;
- } else if (r_dev_addr == NULL) {
- /* Write only */
- state = w_dev_addr->user_data;
- msg[0].addr = w_dev_addr->i2c_addr >> 1;
- msg[0].flags = 0;
- msg[0].buf = wData;
- msg[0].len = w_count;
- num_msgs = 1;
- } else {
- /* Both write and read */
- state = w_dev_addr->user_data;
- msg[0].addr = w_dev_addr->i2c_addr >> 1;
- msg[0].flags = 0;
- msg[0].buf = wData;
- msg[0].len = w_count;
- msg[1].addr = r_dev_addr->i2c_addr >> 1;
- msg[1].flags = I2C_M_RD;
- msg[1].buf = r_data;
- msg[1].len = r_count;
- num_msgs = 2;
- }
- if (state->i2c == NULL) {
- pr_err("i2c was zero, aborting\n");
- return 0;
- }
- if (i2c_transfer(state->i2c, msg, num_msgs) != num_msgs) {
- pr_warn("drx3933: I2C write/read failed\n");
- return -EREMOTEIO;
- }
- #ifdef DJH_DEBUG
- if (w_dev_addr == NULL || r_dev_addr == NULL)
- return 0;
- state = w_dev_addr->user_data;
- if (state->i2c == NULL)
- return 0;
- msg[0].addr = w_dev_addr->i2c_addr;
- msg[0].flags = 0;
- msg[0].buf = wData;
- msg[0].len = w_count;
- msg[1].addr = r_dev_addr->i2c_addr;
- msg[1].flags = I2C_M_RD;
- msg[1].buf = r_data;
- msg[1].len = r_count;
- num_msgs = 2;
- pr_debug("drx3933 i2c operation addr=%x i2c=%p, wc=%x rc=%x\n",
- w_dev_addr->i2c_addr, state->i2c, w_count, r_count);
- if (i2c_transfer(state->i2c, msg, 2) != 2) {
- pr_warn("drx3933: I2C write/read failed\n");
- return -EREMOTEIO;
- }
- #endif
- return 0;
- }
- /*============================================================================*/
- /*****************************
- *
- * int drxdap_fasi_read_block (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u16 datasize, -- number of bytes to read
- * u8 *data, -- data to receive
- * u32 flags) -- special device flags
- *
- * Read block data from chip address. Because the chip is word oriented,
- * the number of bytes to read must be even.
- *
- * Make sure that the buffer to receive the data is large enough.
- *
- * Although this function expects an even number of bytes, it is still byte
- * oriented, and the data read back is NOT translated to the endianness of
- * the target platform.
- *
- * Output:
- * - 0 if reading was successful
- * in that case: data read is in *data.
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_read_block(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 datasize,
- u8 *data, u32 flags)
- {
- u8 buf[4];
- u16 bufx;
- int rc;
- u16 overhead_size = 0;
- /* Check parameters ******************************************************* */
- if (dev_addr == NULL)
- return -EINVAL;
- overhead_size = (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1) +
- (DRXDAP_FASI_LONG_FORMAT(addr) ? 4 : 2);
- if ((DRXDAP_FASI_OFFSET_TOO_LARGE(addr)) ||
- ((!(DRXDAPFASI_LONG_ADDR_ALLOWED)) &&
- DRXDAP_FASI_LONG_FORMAT(addr)) ||
- (overhead_size > (DRXDAP_MAX_WCHUNKSIZE)) ||
- ((datasize != 0) && (data == NULL)) || ((datasize & 1) == 1)) {
- return -EINVAL;
- }
- /* ReadModifyWrite & mode flag bits are not allowed */
- flags &= (~DRXDAP_FASI_RMW & ~DRXDAP_FASI_MODEFLAGS);
- #if DRXDAP_SINGLE_MASTER
- flags |= DRXDAP_FASI_SINGLE_MASTER;
- #endif
- /* Read block from I2C **************************************************** */
- do {
- u16 todo = (datasize < DRXDAP_MAX_RCHUNKSIZE ?
- datasize : DRXDAP_MAX_RCHUNKSIZE);
- bufx = 0;
- addr &= ~DRXDAP_FASI_FLAGS;
- addr |= flags;
- #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
- /* short format address preferred but long format otherwise */
- if (DRXDAP_FASI_LONG_FORMAT(addr)) {
- #endif
- #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
- buf[bufx++] = (u8) (((addr << 1) & 0xFF) | 0x01);
- buf[bufx++] = (u8) ((addr >> 16) & 0xFF);
- buf[bufx++] = (u8) ((addr >> 24) & 0xFF);
- buf[bufx++] = (u8) ((addr >> 7) & 0xFF);
- #endif
- #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
- } else {
- #endif
- #if (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1)
- buf[bufx++] = (u8) ((addr << 1) & 0xFF);
- buf[bufx++] =
- (u8) (((addr >> 16) & 0x0F) |
- ((addr >> 18) & 0xF0));
- #endif
- #if ((DRXDAPFASI_LONG_ADDR_ALLOWED == 1) && (DRXDAPFASI_SHORT_ADDR_ALLOWED == 1))
- }
- #endif
- #if DRXDAP_SINGLE_MASTER
- /*
- * In single master mode, split the read and write actions.
- * No special action is needed for write chunks here.
- */
- rc = drxbsp_i2c_write_read(dev_addr, bufx, buf,
- NULL, 0, NULL);
- if (rc == 0)
- rc = drxbsp_i2c_write_read(NULL, 0, NULL, dev_addr, todo, data);
- #else
- /* In multi master mode, do everything in one RW action */
- rc = drxbsp_i2c_write_read(dev_addr, bufx, buf, dev_addr, todo,
- data);
- #endif
- data += todo;
- addr += (todo >> 1);
- datasize -= todo;
- } while (datasize && rc == 0);
- return rc;
- }
- /*****************************
- *
- * int drxdap_fasi_read_reg16 (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u16 *data, -- data to receive
- * u32 flags) -- special device flags
- *
- * Read one 16-bit register or memory location. The data received back is
- * converted back to the target platform's endianness.
- *
- * Output:
- * - 0 if reading was successful
- * in that case: read data is at *data
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_read_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 *data, u32 flags)
- {
- u8 buf[sizeof(*data)];
- int rc;
- if (!data)
- return -EINVAL;
- rc = drxdap_fasi_read_block(dev_addr, addr, sizeof(*data), buf, flags);
- *data = buf[0] + (((u16) buf[1]) << 8);
- return rc;
- }
- /*****************************
- *
- * int drxdap_fasi_read_reg32 (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u32 *data, -- data to receive
- * u32 flags) -- special device flags
- *
- * Read one 32-bit register or memory location. The data received back is
- * converted back to the target platform's endianness.
- *
- * Output:
- * - 0 if reading was successful
- * in that case: read data is at *data
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_read_reg32(struct i2c_device_addr *dev_addr,
- u32 addr,
- u32 *data, u32 flags)
- {
- u8 buf[sizeof(*data)];
- int rc;
- if (!data)
- return -EINVAL;
- rc = drxdap_fasi_read_block(dev_addr, addr, sizeof(*data), buf, flags);
- *data = (((u32) buf[0]) << 0) +
- (((u32) buf[1]) << 8) +
- (((u32) buf[2]) << 16) + (((u32) buf[3]) << 24);
- return rc;
- }
- /*****************************
- *
- * int drxdap_fasi_write_block (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u16 datasize, -- number of bytes to read
- * u8 *data, -- data to receive
- * u32 flags) -- special device flags
- *
- * Write block data to chip address. Because the chip is word oriented,
- * the number of bytes to write must be even.
- *
- * Although this function expects an even number of bytes, it is still byte
- * oriented, and the data being written is NOT translated from the endianness of
- * the target platform.
- *
- * Output:
- * - 0 if writing was successful
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_write_block(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 datasize,
- u8 *data, u32 flags)
- {
- u8 buf[DRXDAP_MAX_WCHUNKSIZE];
- int st = -EIO;
- int first_err = 0;
- u16 overhead_size = 0;
- u16 block_size = 0;
- /* Check parameters ******************************************************* */
- if (dev_addr == NULL)
- return -EINVAL;
- overhead_size = (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1) +
- (DRXDAP_FASI_LONG_FORMAT(addr) ? 4 : 2);
- if ((DRXDAP_FASI_OFFSET_TOO_LARGE(addr)) ||
- ((!(DRXDAPFASI_LONG_ADDR_ALLOWED)) &&
- DRXDAP_FASI_LONG_FORMAT(addr)) ||
- (overhead_size > (DRXDAP_MAX_WCHUNKSIZE)) ||
- ((datasize != 0) && (data == NULL)) || ((datasize & 1) == 1))
- return -EINVAL;
- flags &= DRXDAP_FASI_FLAGS;
- flags &= ~DRXDAP_FASI_MODEFLAGS;
- #if DRXDAP_SINGLE_MASTER
- flags |= DRXDAP_FASI_SINGLE_MASTER;
- #endif
- /* Write block to I2C ***************************************************** */
- block_size = ((DRXDAP_MAX_WCHUNKSIZE) - overhead_size) & ~1;
- do {
- u16 todo = 0;
- u16 bufx = 0;
- /* Buffer device address */
- addr &= ~DRXDAP_FASI_FLAGS;
- addr |= flags;
- #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
- /* short format address preferred but long format otherwise */
- if (DRXDAP_FASI_LONG_FORMAT(addr)) {
- #endif
- #if ((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1)
- buf[bufx++] = (u8) (((addr << 1) & 0xFF) | 0x01);
- buf[bufx++] = (u8) ((addr >> 16) & 0xFF);
- buf[bufx++] = (u8) ((addr >> 24) & 0xFF);
- buf[bufx++] = (u8) ((addr >> 7) & 0xFF);
- #endif
- #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
- } else {
- #endif
- #if ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1)
- buf[bufx++] = (u8) ((addr << 1) & 0xFF);
- buf[bufx++] =
- (u8) (((addr >> 16) & 0x0F) |
- ((addr >> 18) & 0xF0));
- #endif
- #if (((DRXDAPFASI_LONG_ADDR_ALLOWED) == 1) && ((DRXDAPFASI_SHORT_ADDR_ALLOWED) == 1))
- }
- #endif
- /*
- In single master mode block_size can be 0. In such a case this I2C
- sequense will be visible: (1) write address {i2c addr,
- 4 bytes chip address} (2) write data {i2c addr, 4 bytes data }
- (3) write address (4) write data etc...
- Address must be rewriten because HI is reset after data transport and
- expects an address.
- */
- todo = (block_size < datasize ? block_size : datasize);
- if (todo == 0) {
- u16 overhead_size_i2c_addr = 0;
- u16 data_block_size = 0;
- overhead_size_i2c_addr =
- (IS_I2C_10BIT(dev_addr->i2c_addr) ? 2 : 1);
- data_block_size =
- (DRXDAP_MAX_WCHUNKSIZE - overhead_size_i2c_addr) & ~1;
- /* write device address */
- st = drxbsp_i2c_write_read(dev_addr,
- (u16) (bufx),
- buf,
- (struct i2c_device_addr *)(NULL),
- 0, (u8 *)(NULL));
- if ((st != 0) && (first_err == 0)) {
- /* at the end, return the first error encountered */
- first_err = st;
- }
- bufx = 0;
- todo =
- (data_block_size <
- datasize ? data_block_size : datasize);
- }
- memcpy(&buf[bufx], data, todo);
- /* write (address if can do and) data */
- st = drxbsp_i2c_write_read(dev_addr,
- (u16) (bufx + todo),
- buf,
- (struct i2c_device_addr *)(NULL),
- 0, (u8 *)(NULL));
- if ((st != 0) && (first_err == 0)) {
- /* at the end, return the first error encountered */
- first_err = st;
- }
- datasize -= todo;
- data += todo;
- addr += (todo >> 1);
- } while (datasize);
- return first_err;
- }
- /*****************************
- *
- * int drxdap_fasi_write_reg16 (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u16 data, -- data to send
- * u32 flags) -- special device flags
- *
- * Write one 16-bit register or memory location. The data being written is
- * converted from the target platform's endianness to little endian.
- *
- * Output:
- * - 0 if writing was successful
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_write_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 data, u32 flags)
- {
- u8 buf[sizeof(data)];
- buf[0] = (u8) ((data >> 0) & 0xFF);
- buf[1] = (u8) ((data >> 8) & 0xFF);
- return drxdap_fasi_write_block(dev_addr, addr, sizeof(data), buf, flags);
- }
- /*****************************
- *
- * int drxdap_fasi_read_modify_write_reg16 (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 waddr, -- address of chip register/memory
- * u32 raddr, -- chip address to read back from
- * u16 wdata, -- data to send
- * u16 *rdata) -- data to receive back
- *
- * Write 16-bit data, then read back the original contents of that location.
- * Requires long addressing format to be allowed.
- *
- * Before sending data, the data is converted to little endian. The
- * data received back is converted back to the target platform's endianness.
- *
- * WARNING: This function is only guaranteed to work if there is one
- * master on the I2C bus.
- *
- * Output:
- * - 0 if reading was successful
- * in that case: read back data is at *rdata
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
- u32 waddr,
- u32 raddr,
- u16 wdata, u16 *rdata)
- {
- int rc = -EIO;
- #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
- if (rdata == NULL)
- return -EINVAL;
- rc = drxdap_fasi_write_reg16(dev_addr, waddr, wdata, DRXDAP_FASI_RMW);
- if (rc == 0)
- rc = drxdap_fasi_read_reg16(dev_addr, raddr, rdata, 0);
- #endif
- return rc;
- }
- /*****************************
- *
- * int drxdap_fasi_write_reg32 (
- * struct i2c_device_addr *dev_addr, -- address of I2C device
- * u32 addr, -- address of chip register/memory
- * u32 data, -- data to send
- * u32 flags) -- special device flags
- *
- * Write one 32-bit register or memory location. The data being written is
- * converted from the target platform's endianness to little endian.
- *
- * Output:
- * - 0 if writing was successful
- * - -EIO if anything went wrong
- *
- ******************************/
- static int drxdap_fasi_write_reg32(struct i2c_device_addr *dev_addr,
- u32 addr,
- u32 data, u32 flags)
- {
- u8 buf[sizeof(data)];
- buf[0] = (u8) ((data >> 0) & 0xFF);
- buf[1] = (u8) ((data >> 8) & 0xFF);
- buf[2] = (u8) ((data >> 16) & 0xFF);
- buf[3] = (u8) ((data >> 24) & 0xFF);
- return drxdap_fasi_write_block(dev_addr, addr, sizeof(data), buf, flags);
- }
- /*============================================================================*/
- /*
- * \fn int drxj_dap_rm_write_reg16short
- * \brief Read modify write 16 bits audio register using short format only.
- * \param dev_addr
- * \param waddr Address to write to
- * \param raddr Address to read from (usually SIO_HI_RA_RAM_S0_RMWBUF__A)
- * \param wdata Data to write
- * \param rdata Buffer for data to read
- * \return int
- * \retval 0 Succes
- * \retval -EIO Timeout, I2C error, illegal bank
- *
- * 16 bits register read modify write access using short addressing format only.
- * Requires knowledge of the registermap, thus device dependent.
- * Using DAP FASI directly to avoid endless recursion of RMWs to audio registers.
- *
- */
- /* TODO correct define should be #if ( DRXDAPFASI_SHORT_ADDR_ALLOWED==1 )
- See comments drxj_dap_read_modify_write_reg16 */
- #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 0)
- static int drxj_dap_rm_write_reg16short(struct i2c_device_addr *dev_addr,
- u32 waddr,
- u32 raddr,
- u16 wdata, u16 *rdata)
- {
- int rc;
- if (rdata == NULL)
- return -EINVAL;
- /* Set RMW flag */
- rc = drxdap_fasi_write_reg16(dev_addr,
- SIO_HI_RA_RAM_S0_FLG_ACC__A,
- SIO_HI_RA_RAM_S0_FLG_ACC_S0_RWM__M,
- 0x0000);
- if (rc == 0) {
- /* Write new data: triggers RMW */
- rc = drxdap_fasi_write_reg16(dev_addr, waddr, wdata,
- 0x0000);
- }
- if (rc == 0) {
- /* Read old data */
- rc = drxdap_fasi_read_reg16(dev_addr, raddr, rdata,
- 0x0000);
- }
- if (rc == 0) {
- /* Reset RMW flag */
- rc = drxdap_fasi_write_reg16(dev_addr,
- SIO_HI_RA_RAM_S0_FLG_ACC__A,
- 0, 0x0000);
- }
- return rc;
- }
- #endif
- /*============================================================================*/
- static int drxj_dap_read_modify_write_reg16(struct i2c_device_addr *dev_addr,
- u32 waddr,
- u32 raddr,
- u16 wdata, u16 *rdata)
- {
- /* TODO: correct short/long addressing format decision,
- now long format has higher prio then short because short also
- needs virt bnks (not impl yet) for certain audio registers */
- #if (DRXDAPFASI_LONG_ADDR_ALLOWED == 1)
- return drxdap_fasi_read_modify_write_reg16(dev_addr,
- waddr,
- raddr, wdata, rdata);
- #else
- return drxj_dap_rm_write_reg16short(dev_addr, waddr, raddr, wdata, rdata);
- #endif
- }
- /*============================================================================*/
- /*
- * \fn int drxj_dap_read_aud_reg16
- * \brief Read 16 bits audio register
- * \param dev_addr
- * \param addr
- * \param data
- * \return int
- * \retval 0 Succes
- * \retval -EIO Timeout, I2C error, illegal bank
- *
- * 16 bits register read access via audio token ring interface.
- *
- */
- static int drxj_dap_read_aud_reg16(struct i2c_device_addr *dev_addr,
- u32 addr, u16 *data)
- {
- u32 start_timer = 0;
- u32 current_timer = 0;
- u32 delta_timer = 0;
- u16 tr_status = 0;
- int stat = -EIO;
- /* No read possible for bank 3, return with error */
- if (DRXDAP_FASI_ADDR2BANK(addr) == 3) {
- stat = -EINVAL;
- } else {
- const u32 write_bit = ((dr_xaddr_t) 1) << 16;
- /* Force reset write bit */
- addr &= (~write_bit);
- /* Set up read */
- start_timer = jiffies_to_msecs(jiffies);
- do {
- /* RMW to aud TR IF until request is granted or timeout */
- stat = drxj_dap_read_modify_write_reg16(dev_addr,
- addr,
- SIO_HI_RA_RAM_S0_RMWBUF__A,
- 0x0000, &tr_status);
- if (stat != 0)
- break;
- current_timer = jiffies_to_msecs(jiffies);
- delta_timer = current_timer - start_timer;
- if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
- stat = -EIO;
- break;
- }
- } while (((tr_status & AUD_TOP_TR_CTR_FIFO_LOCK__M) ==
- AUD_TOP_TR_CTR_FIFO_LOCK_LOCKED) ||
- ((tr_status & AUD_TOP_TR_CTR_FIFO_FULL__M) ==
- AUD_TOP_TR_CTR_FIFO_FULL_FULL));
- } /* if ( DRXDAP_FASI_ADDR2BANK(addr)!=3 ) */
- /* Wait for read ready status or timeout */
- if (stat == 0) {
- start_timer = jiffies_to_msecs(jiffies);
- while ((tr_status & AUD_TOP_TR_CTR_FIFO_RD_RDY__M) !=
- AUD_TOP_TR_CTR_FIFO_RD_RDY_READY) {
- stat = drxj_dap_read_reg16(dev_addr,
- AUD_TOP_TR_CTR__A,
- &tr_status, 0x0000);
- if (stat != 0)
- break;
- current_timer = jiffies_to_msecs(jiffies);
- delta_timer = current_timer - start_timer;
- if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
- stat = -EIO;
- break;
- }
- } /* while ( ... ) */
- }
- /* Read value */
- if (stat == 0)
- stat = drxj_dap_read_modify_write_reg16(dev_addr,
- AUD_TOP_TR_RD_REG__A,
- SIO_HI_RA_RAM_S0_RMWBUF__A,
- 0x0000, data);
- return stat;
- }
- /*============================================================================*/
- static int drxj_dap_read_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 *data, u32 flags)
- {
- int stat = -EIO;
- /* Check param */
- if ((dev_addr == NULL) || (data == NULL))
- return -EINVAL;
- if (is_handled_by_aud_tr_if(addr))
- stat = drxj_dap_read_aud_reg16(dev_addr, addr, data);
- else
- stat = drxdap_fasi_read_reg16(dev_addr, addr, data, flags);
- return stat;
- }
- /*============================================================================*/
- /*
- * \fn int drxj_dap_write_aud_reg16
- * \brief Write 16 bits audio register
- * \param dev_addr
- * \param addr
- * \param data
- * \return int
- * \retval 0 Succes
- * \retval -EIO Timeout, I2C error, illegal bank
- *
- * 16 bits register write access via audio token ring interface.
- *
- */
- static int drxj_dap_write_aud_reg16(struct i2c_device_addr *dev_addr,
- u32 addr, u16 data)
- {
- int stat = -EIO;
- /* No write possible for bank 2, return with error */
- if (DRXDAP_FASI_ADDR2BANK(addr) == 2) {
- stat = -EINVAL;
- } else {
- u32 start_timer = 0;
- u32 current_timer = 0;
- u32 delta_timer = 0;
- u16 tr_status = 0;
- const u32 write_bit = ((dr_xaddr_t) 1) << 16;
- /* Force write bit */
- addr |= write_bit;
- start_timer = jiffies_to_msecs(jiffies);
- do {
- /* RMW to aud TR IF until request is granted or timeout */
- stat = drxj_dap_read_modify_write_reg16(dev_addr,
- addr,
- SIO_HI_RA_RAM_S0_RMWBUF__A,
- data, &tr_status);
- if (stat != 0)
- break;
- current_timer = jiffies_to_msecs(jiffies);
- delta_timer = current_timer - start_timer;
- if (delta_timer > DRXJ_DAP_AUDTRIF_TIMEOUT) {
- stat = -EIO;
- break;
- }
- } while (((tr_status & AUD_TOP_TR_CTR_FIFO_LOCK__M) ==
- AUD_TOP_TR_CTR_FIFO_LOCK_LOCKED) ||
- ((tr_status & AUD_TOP_TR_CTR_FIFO_FULL__M) ==
- AUD_TOP_TR_CTR_FIFO_FULL_FULL));
- } /* if ( DRXDAP_FASI_ADDR2BANK(addr)!=2 ) */
- return stat;
- }
- /*============================================================================*/
- static int drxj_dap_write_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 data, u32 flags)
- {
- int stat = -EIO;
- /* Check param */
- if (dev_addr == NULL)
- return -EINVAL;
- if (is_handled_by_aud_tr_if(addr))
- stat = drxj_dap_write_aud_reg16(dev_addr, addr, data);
- else
- stat = drxdap_fasi_write_reg16(dev_addr,
- addr, data, flags);
- return stat;
- }
- /*============================================================================*/
- /* Free data ram in SIO HI */
- #define SIO_HI_RA_RAM_USR_BEGIN__A 0x420040
- #define SIO_HI_RA_RAM_USR_END__A 0x420060
- #define DRXJ_HI_ATOMIC_BUF_START (SIO_HI_RA_RAM_USR_BEGIN__A)
- #define DRXJ_HI_ATOMIC_BUF_END (SIO_HI_RA_RAM_USR_BEGIN__A + 7)
- #define DRXJ_HI_ATOMIC_READ SIO_HI_RA_RAM_PAR_3_ACP_RW_READ
- #define DRXJ_HI_ATOMIC_WRITE SIO_HI_RA_RAM_PAR_3_ACP_RW_WRITE
- /*
- * \fn int drxj_dap_atomic_read_write_block()
- * \brief Basic access routine for atomic read or write access
- * \param dev_addr pointer to i2c dev address
- * \param addr destination/source address
- * \param datasize size of data buffer in bytes
- * \param data pointer to data buffer
- * \return int
- * \retval 0 Succes
- * \retval -EIO Timeout, I2C error, illegal bank
- *
- */
- static
- int drxj_dap_atomic_read_write_block(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 datasize,
- u8 *data, bool read_flag)
- {
- struct drxj_hi_cmd hi_cmd;
- int rc;
- u16 word;
- u16 dummy = 0;
- u16 i = 0;
- /* Parameter check */
- if (!data || !dev_addr || ((datasize % 2)) || ((datasize / 2) > 8))
- return -EINVAL;
- /* Set up HI parameters to read or write n bytes */
- hi_cmd.cmd = SIO_HI_RA_RAM_CMD_ATOMIC_COPY;
- hi_cmd.param1 =
- (u16) ((DRXDAP_FASI_ADDR2BLOCK(DRXJ_HI_ATOMIC_BUF_START) << 6) +
- DRXDAP_FASI_ADDR2BANK(DRXJ_HI_ATOMIC_BUF_START));
- hi_cmd.param2 =
- (u16) DRXDAP_FASI_ADDR2OFFSET(DRXJ_HI_ATOMIC_BUF_START);
- hi_cmd.param3 = (u16) ((datasize / 2) - 1);
- if (!read_flag)
- hi_cmd.param3 |= DRXJ_HI_ATOMIC_WRITE;
- else
- hi_cmd.param3 |= DRXJ_HI_ATOMIC_READ;
- hi_cmd.param4 = (u16) ((DRXDAP_FASI_ADDR2BLOCK(addr) << 6) +
- DRXDAP_FASI_ADDR2BANK(addr));
- hi_cmd.param5 = (u16) DRXDAP_FASI_ADDR2OFFSET(addr);
- if (!read_flag) {
- /* write data to buffer */
- for (i = 0; i < (datasize / 2); i++) {
- word = ((u16) data[2 * i]);
- word += (((u16) data[(2 * i) + 1]) << 8);
- drxj_dap_write_reg16(dev_addr,
- (DRXJ_HI_ATOMIC_BUF_START + i),
- word, 0);
- }
- }
- rc = hi_command(dev_addr, &hi_cmd, &dummy);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (read_flag) {
- /* read data from buffer */
- for (i = 0; i < (datasize / 2); i++) {
- drxj_dap_read_reg16(dev_addr,
- (DRXJ_HI_ATOMIC_BUF_START + i),
- &word, 0);
- data[2 * i] = (u8) (word & 0xFF);
- data[(2 * i) + 1] = (u8) (word >> 8);
- }
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int drxj_dap_atomic_read_reg32()
- * \brief Atomic read of 32 bits words
- */
- static
- int drxj_dap_atomic_read_reg32(struct i2c_device_addr *dev_addr,
- u32 addr,
- u32 *data, u32 flags)
- {
- u8 buf[sizeof(*data)] = { 0 };
- int rc = -EIO;
- u32 word = 0;
- if (!data)
- return -EINVAL;
- rc = drxj_dap_atomic_read_write_block(dev_addr, addr,
- sizeof(*data), buf, true);
- if (rc < 0)
- return 0;
- word = (u32) buf[3];
- word <<= 8;
- word |= (u32) buf[2];
- word <<= 8;
- word |= (u32) buf[1];
- word <<= 8;
- word |= (u32) buf[0];
- *data = word;
- return rc;
- }
- /*============================================================================*/
- /*============================================================================*/
- /*== END DRXJ DAP FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== HOST INTERFACE FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn int hi_cfg_command()
- * \brief Configure HI with settings stored in the demod structure.
- * \param demod Demodulator.
- * \return int.
- *
- * This routine was created because to much orthogonal settings have
- * been put into one HI API function (configure). Especially the I2C bridge
- * enable/disable should not need re-configuration of the HI.
- *
- */
- static int hi_cfg_command(const struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct drxj_hi_cmd hi_cmd;
- u16 result = 0;
- int rc;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- hi_cmd.cmd = SIO_HI_RA_RAM_CMD_CONFIG;
- hi_cmd.param1 = SIO_HI_RA_RAM_PAR_1_PAR1_SEC_KEY;
- hi_cmd.param2 = ext_attr->hi_cfg_timing_div;
- hi_cmd.param3 = ext_attr->hi_cfg_bridge_delay;
- hi_cmd.param4 = ext_attr->hi_cfg_wake_up_key;
- hi_cmd.param5 = ext_attr->hi_cfg_ctrl;
- hi_cmd.param6 = ext_attr->hi_cfg_transmit;
- rc = hi_command(demod->my_i2c_dev_addr, &hi_cmd, &result);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Reset power down flag (set one call only) */
- ext_attr->hi_cfg_ctrl &= (~(SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ));
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int hi_command()
- * \brief Configure HI with settings stored in the demod structure.
- * \param dev_addr I2C address.
- * \param cmd HI command.
- * \param result HI command result.
- * \return int.
- *
- * Sends command to HI
- *
- */
- static int
- hi_command(struct i2c_device_addr *dev_addr, const struct drxj_hi_cmd *cmd, u16 *result)
- {
- u16 wait_cmd = 0;
- u16 nr_retries = 0;
- bool powerdown_cmd = false;
- int rc;
- /* Write parameters */
- switch (cmd->cmd) {
- case SIO_HI_RA_RAM_CMD_CONFIG:
- case SIO_HI_RA_RAM_CMD_ATOMIC_COPY:
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_6__A, cmd->param6, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_5__A, cmd->param5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_4__A, cmd->param4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_3__A, cmd->param3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* fallthrough */
- case SIO_HI_RA_RAM_CMD_BRDCTRL:
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_2__A, cmd->param2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_PAR_1__A, cmd->param1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* fallthrough */
- case SIO_HI_RA_RAM_CMD_NULL:
- /* No parameters */
- break;
- default:
- return -EINVAL;
- break;
- }
- /* Write command */
- rc = drxj_dap_write_reg16(dev_addr, SIO_HI_RA_RAM_CMD__A, cmd->cmd, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if ((cmd->cmd) == SIO_HI_RA_RAM_CMD_RESET)
- msleep(1);
- /* Detect power down to ommit reading result */
- powerdown_cmd = (bool) ((cmd->cmd == SIO_HI_RA_RAM_CMD_CONFIG) &&
- (((cmd->
- param5) & SIO_HI_RA_RAM_PAR_5_CFG_SLEEP__M)
- == SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ));
- if (!powerdown_cmd) {
- /* Wait until command rdy */
- do {
- nr_retries++;
- if (nr_retries > DRXJ_MAX_RETRIES) {
- pr_err("timeout\n");
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, SIO_HI_RA_RAM_CMD__A, &wait_cmd, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } while (wait_cmd != 0);
- /* Read result */
- rc = drxj_dap_read_reg16(dev_addr, SIO_HI_RA_RAM_RES__A, result, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* if ( powerdown_cmd == true ) */
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int init_hi( const struct drx_demod_instance *demod )
- * \brief Initialise and configurate HI.
- * \param demod pointer to demod data.
- * \return int Return status.
- * \retval 0 Success.
- * \retval -EIO Failure.
- *
- * Needs to know Psys (System Clock period) and Posc (Osc Clock period)
- * Need to store configuration in driver because of the way I2C
- * bridging is controlled.
- *
- */
- static int init_hi(const struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- int rc;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- dev_addr = demod->my_i2c_dev_addr;
- /* PATCH for bug 5003, HI ucode v3.1.0 */
- rc = drxj_dap_write_reg16(dev_addr, 0x4301D7, 0x801, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Timing div, 250ns/Psys */
- /* Timing div, = ( delay (nano seconds) * sysclk (kHz) )/ 1000 */
- ext_attr->hi_cfg_timing_div =
- (u16) ((common_attr->sys_clock_freq / 1000) * HI_I2C_DELAY) / 1000;
- /* Clipping */
- if ((ext_attr->hi_cfg_timing_div) > SIO_HI_RA_RAM_PAR_2_CFG_DIV__M)
- ext_attr->hi_cfg_timing_div = SIO_HI_RA_RAM_PAR_2_CFG_DIV__M;
- /* Bridge delay, uses oscilator clock */
- /* Delay = ( delay (nano seconds) * oscclk (kHz) )/ 1000 */
- /* SDA brdige delay */
- ext_attr->hi_cfg_bridge_delay =
- (u16) ((common_attr->osc_clock_freq / 1000) * HI_I2C_BRIDGE_DELAY) /
- 1000;
- /* Clipping */
- if ((ext_attr->hi_cfg_bridge_delay) > SIO_HI_RA_RAM_PAR_3_CFG_DBL_SDA__M)
- ext_attr->hi_cfg_bridge_delay = SIO_HI_RA_RAM_PAR_3_CFG_DBL_SDA__M;
- /* SCL bridge delay, same as SDA for now */
- ext_attr->hi_cfg_bridge_delay += ((ext_attr->hi_cfg_bridge_delay) <<
- SIO_HI_RA_RAM_PAR_3_CFG_DBL_SCL__B);
- /* Wakeup key, setting the read flag (as suggest in the documentation) does
- not always result into a working solution (barebones worked VI2C failed).
- Not setting the bit works in all cases . */
- ext_attr->hi_cfg_wake_up_key = DRXJ_WAKE_UP_KEY;
- /* port/bridge/power down ctrl */
- ext_attr->hi_cfg_ctrl = (SIO_HI_RA_RAM_PAR_5_CFG_SLV0_SLAVE);
- /* transit mode time out delay and watch dog divider */
- ext_attr->hi_cfg_transmit = SIO_HI_RA_RAM_PAR_6__PRE;
- rc = hi_cfg_command(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== END HOST INTERFACE FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== AUXILIARY FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn int get_device_capabilities()
- * \brief Get and store device capabilities.
- * \param demod Pointer to demodulator instance.
- * \return int.
- * \return 0 Success
- * \retval -EIO Failure
- *
- * Depending on pulldowns on MDx pins the following internals are set:
- * * common_attr->osc_clock_freq
- * * ext_attr->has_lna
- * * ext_attr->has_ntsc
- * * ext_attr->has_btsc
- * * ext_attr->has_oob
- *
- */
- static int get_device_capabilities(struct drx_demod_instance *demod)
- {
- struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
- struct drxj_data *ext_attr = (struct drxj_data *) NULL;
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- u16 sio_pdr_ohw_cfg = 0;
- u32 sio_top_jtagid_lo = 0;
- u16 bid = 0;
- int rc;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- dev_addr = demod->my_i2c_dev_addr;
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, SIO_PDR_OHW_CFG__A, &sio_pdr_ohw_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch ((sio_pdr_ohw_cfg & SIO_PDR_OHW_CFG_FREF_SEL__M)) {
- case 0:
- /* ignore (bypass ?) */
- break;
- case 1:
- /* 27 MHz */
- common_attr->osc_clock_freq = 27000;
- break;
- case 2:
- /* 20.25 MHz */
- common_attr->osc_clock_freq = 20250;
- break;
- case 3:
- /* 4 MHz */
- common_attr->osc_clock_freq = 4000;
- break;
- default:
- return -EIO;
- }
- /*
- Determine device capabilities
- Based on pinning v47
- */
- rc = drxdap_fasi_read_reg32(dev_addr, SIO_TOP_JTAGID_LO__A, &sio_top_jtagid_lo, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->mfx = (u8) ((sio_top_jtagid_lo >> 29) & 0xF);
- switch ((sio_top_jtagid_lo >> 12) & 0xFF) {
- case 0x31:
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, SIO_PDR_UIO_IN_HI__A, &bid, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- bid = (bid >> 10) & 0xf;
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->has_lna = true;
- ext_attr->has_ntsc = false;
- ext_attr->has_btsc = false;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = false;
- ext_attr->has_gpio = false;
- ext_attr->has_irqn = false;
- break;
- case 0x33:
- ext_attr->has_lna = false;
- ext_attr->has_ntsc = false;
- ext_attr->has_btsc = false;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = false;
- ext_attr->has_gpio = false;
- ext_attr->has_irqn = false;
- break;
- case 0x45:
- ext_attr->has_lna = true;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = false;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = false;
- break;
- case 0x46:
- ext_attr->has_lna = false;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = false;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = false;
- break;
- case 0x41:
- ext_attr->has_lna = true;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = true;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = false;
- break;
- case 0x43:
- ext_attr->has_lna = false;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = true;
- ext_attr->has_oob = false;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = false;
- break;
- case 0x32:
- ext_attr->has_lna = true;
- ext_attr->has_ntsc = false;
- ext_attr->has_btsc = false;
- ext_attr->has_oob = true;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = true;
- break;
- case 0x34:
- ext_attr->has_lna = false;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = true;
- ext_attr->has_oob = true;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = true;
- break;
- case 0x42:
- ext_attr->has_lna = true;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = true;
- ext_attr->has_oob = true;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = true;
- break;
- case 0x44:
- ext_attr->has_lna = false;
- ext_attr->has_ntsc = true;
- ext_attr->has_btsc = true;
- ext_attr->has_oob = true;
- ext_attr->has_smatx = true;
- ext_attr->has_smarx = true;
- ext_attr->has_gpio = true;
- ext_attr->has_irqn = true;
- break;
- default:
- /* Unknown device variant */
- return -EIO;
- break;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int power_up_device()
- * \brief Power up device.
- * \param demod Pointer to demodulator instance.
- * \return int.
- * \return 0 Success
- * \retval -EIO Failure, I2C or max retries reached
- *
- */
- #ifndef DRXJ_MAX_RETRIES_POWERUP
- #define DRXJ_MAX_RETRIES_POWERUP 10
- #endif
- static int power_up_device(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- u8 data = 0;
- u16 retry_count = 0;
- struct i2c_device_addr wake_up_addr;
- dev_addr = demod->my_i2c_dev_addr;
- wake_up_addr.i2c_addr = DRXJ_WAKE_UP_KEY;
- wake_up_addr.i2c_dev_id = dev_addr->i2c_dev_id;
- wake_up_addr.user_data = dev_addr->user_data;
- /*
- * I2C access may fail in this case: no ack
- * dummy write must be used to wake uop device, dummy read must be used to
- * reset HI state machine (avoiding actual writes)
- */
- do {
- data = 0;
- drxbsp_i2c_write_read(&wake_up_addr, 1, &data,
- (struct i2c_device_addr *)(NULL), 0,
- (u8 *)(NULL));
- msleep(10);
- retry_count++;
- } while ((drxbsp_i2c_write_read
- ((struct i2c_device_addr *) (NULL), 0, (u8 *)(NULL), dev_addr, 1,
- &data)
- != 0) && (retry_count < DRXJ_MAX_RETRIES_POWERUP));
- /* Need some recovery time .... */
- msleep(10);
- if (retry_count == DRXJ_MAX_RETRIES_POWERUP)
- return -EIO;
- return 0;
- }
- /*----------------------------------------------------------------------------*/
- /* MPEG Output Configuration Functions - begin */
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int ctrl_set_cfg_mpeg_output()
- * \brief Set MPEG output configuration of the device.
- * \param devmod Pointer to demodulator instance.
- * \param cfg_data Pointer to mpeg output configuaration.
- * \return int.
- *
- * Configure MPEG output parameters.
- *
- */
- static int
- ctrl_set_cfg_mpeg_output(struct drx_demod_instance *demod, struct drx_cfg_mpeg_output *cfg_data)
- {
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct drx_common_attr *common_attr = (struct drx_common_attr *) (NULL);
- int rc;
- u16 fec_oc_reg_mode = 0;
- u16 fec_oc_reg_ipr_mode = 0;
- u16 fec_oc_reg_ipr_invert = 0;
- u32 max_bit_rate = 0;
- u32 rcn_rate = 0;
- u32 nr_bits = 0;
- u16 sio_pdr_md_cfg = 0;
- /* data mask for the output data byte */
- u16 invert_data_mask =
- FEC_OC_IPR_INVERT_MD7__M | FEC_OC_IPR_INVERT_MD6__M |
- FEC_OC_IPR_INVERT_MD5__M | FEC_OC_IPR_INVERT_MD4__M |
- FEC_OC_IPR_INVERT_MD3__M | FEC_OC_IPR_INVERT_MD2__M |
- FEC_OC_IPR_INVERT_MD1__M | FEC_OC_IPR_INVERT_MD0__M;
- /* check arguments */
- if ((demod == NULL) || (cfg_data == NULL))
- return -EINVAL;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- if (cfg_data->enable_mpeg_output == true) {
- /* quick and dirty patch to set MPEG incase current std is not
- producing MPEG */
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- break;
- default:
- return 0;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_OCR_INVERT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_USAGE__A, 7, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* 2048 bytes fifo ram */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_CTL_UPD_RATE__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_INT_UPD_RATE__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_A__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_B__A, 7, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Low Water Mark for synchronization */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_LWM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* High Water Mark for synchronization */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_HWM__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- switch (ext_attr->constellation) {
- case DRX_CONSTELLATION_QAM256:
- nr_bits = 8;
- break;
- case DRX_CONSTELLATION_QAM128:
- nr_bits = 7;
- break;
- case DRX_CONSTELLATION_QAM64:
- nr_bits = 6;
- break;
- case DRX_CONSTELLATION_QAM32:
- nr_bits = 5;
- break;
- case DRX_CONSTELLATION_QAM16:
- nr_bits = 4;
- break;
- default:
- return -EIO;
- } /* ext_attr->constellation */
- /* max_bit_rate = symbol_rate * nr_bits * coef */
- /* coef = 188/204 */
- max_bit_rate =
- (ext_attr->curr_symbol_rate / 8) * nr_bits * 188;
- /* pass through as b/c Annex A/c need following settings */
- /* fall-through */
- case DRX_STANDARD_ITU_B:
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_USAGE__A, FEC_OC_FCT_USAGE__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_CTL_UPD_RATE__A, FEC_OC_TMD_CTL_UPD_RATE__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_TMD_INT_UPD_RATE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_A__A, FEC_OC_AVR_PARM_A__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_AVR_PARM_B__A, FEC_OC_AVR_PARM_B__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (cfg_data->static_clk == true) {
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, 0xD, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_RCN_GAIN__A, FEC_OC_RCN_GAIN__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_LWM__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_HWM__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- break;
- } /* swtich (standard) */
- /* Check insertion of the Reed-Solomon parity bytes */
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_MODE__A, &fec_oc_reg_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_IPR_MODE__A, &fec_oc_reg_ipr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (cfg_data->insert_rs_byte == true) {
- /* enable parity symbol forward */
- fec_oc_reg_mode |= FEC_OC_MODE_PARITY__M;
- /* MVAL disable during parity bytes */
- fec_oc_reg_ipr_mode |= FEC_OC_IPR_MODE_MVAL_DIS_PAR__M;
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- rcn_rate = 0x004854D3;
- break;
- case DRX_STANDARD_ITU_B:
- fec_oc_reg_mode |= FEC_OC_MODE_TRANSPARENT__M;
- switch (ext_attr->constellation) {
- case DRX_CONSTELLATION_QAM256:
- rcn_rate = 0x008945E7;
- break;
- case DRX_CONSTELLATION_QAM64:
- rcn_rate = 0x005F64D4;
- break;
- default:
- return -EIO;
- }
- break;
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- /* insert_rs_byte = true -> coef = 188/188 -> 1, RS bits are in MPEG output */
- rcn_rate =
- (frac28
- (max_bit_rate,
- (u32) (common_attr->sys_clock_freq / 8))) /
- 188;
- break;
- default:
- return -EIO;
- } /* ext_attr->standard */
- } else { /* insert_rs_byte == false */
- /* disable parity symbol forward */
- fec_oc_reg_mode &= (~FEC_OC_MODE_PARITY__M);
- /* MVAL enable during parity bytes */
- fec_oc_reg_ipr_mode &= (~FEC_OC_IPR_MODE_MVAL_DIS_PAR__M);
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- rcn_rate = 0x0041605C;
- break;
- case DRX_STANDARD_ITU_B:
- fec_oc_reg_mode &= (~FEC_OC_MODE_TRANSPARENT__M);
- switch (ext_attr->constellation) {
- case DRX_CONSTELLATION_QAM256:
- rcn_rate = 0x0082D6A0;
- break;
- case DRX_CONSTELLATION_QAM64:
- rcn_rate = 0x005AEC1A;
- break;
- default:
- return -EIO;
- }
- break;
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- /* insert_rs_byte = false -> coef = 188/204, RS bits not in MPEG output */
- rcn_rate =
- (frac28
- (max_bit_rate,
- (u32) (common_attr->sys_clock_freq / 8))) /
- 204;
- break;
- default:
- return -EIO;
- } /* ext_attr->standard */
- }
- if (cfg_data->enable_parallel == true) { /* MPEG data output is parallel -> clear ipr_mode[0] */
- fec_oc_reg_ipr_mode &= (~(FEC_OC_IPR_MODE_SERIAL__M));
- } else { /* MPEG data output is serial -> set ipr_mode[0] */
- fec_oc_reg_ipr_mode |= FEC_OC_IPR_MODE_SERIAL__M;
- }
- /* Control slective inversion of output bits */
- if (cfg_data->invert_data == true)
- fec_oc_reg_ipr_invert |= invert_data_mask;
- else
- fec_oc_reg_ipr_invert &= (~(invert_data_mask));
- if (cfg_data->invert_err == true)
- fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MERR__M;
- else
- fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MERR__M));
- if (cfg_data->invert_str == true)
- fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MSTRT__M;
- else
- fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MSTRT__M));
- if (cfg_data->invert_val == true)
- fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MVAL__M;
- else
- fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MVAL__M));
- if (cfg_data->invert_clk == true)
- fec_oc_reg_ipr_invert |= FEC_OC_IPR_INVERT_MCLK__M;
- else
- fec_oc_reg_ipr_invert &= (~(FEC_OC_IPR_INVERT_MCLK__M));
- if (cfg_data->static_clk == true) { /* Static mode */
- u32 dto_rate = 0;
- u32 bit_rate = 0;
- u16 fec_oc_dto_burst_len = 0;
- u16 fec_oc_dto_period = 0;
- fec_oc_dto_burst_len = FEC_OC_DTO_BURST_LEN__PRE;
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- fec_oc_dto_period = 4;
- if (cfg_data->insert_rs_byte == true)
- fec_oc_dto_burst_len = 208;
- break;
- case DRX_STANDARD_ITU_A:
- {
- u32 symbol_rate_th = 6400000;
- if (cfg_data->insert_rs_byte == true) {
- fec_oc_dto_burst_len = 204;
- symbol_rate_th = 5900000;
- }
- if (ext_attr->curr_symbol_rate >=
- symbol_rate_th) {
- fec_oc_dto_period = 0;
- } else {
- fec_oc_dto_period = 1;
- }
- }
- break;
- case DRX_STANDARD_ITU_B:
- fec_oc_dto_period = 1;
- if (cfg_data->insert_rs_byte == true)
- fec_oc_dto_burst_len = 128;
- break;
- case DRX_STANDARD_ITU_C:
- fec_oc_dto_period = 1;
- if (cfg_data->insert_rs_byte == true)
- fec_oc_dto_burst_len = 204;
- break;
- default:
- return -EIO;
- }
- bit_rate =
- common_attr->sys_clock_freq * 1000 / (fec_oc_dto_period +
- 2);
- dto_rate =
- frac28(bit_rate, common_attr->sys_clock_freq * 1000);
- dto_rate >>= 3;
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_RATE_HI__A, (u16)((dto_rate >> 16) & FEC_OC_DTO_RATE_HI__M), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_RATE_LO__A, (u16)(dto_rate & FEC_OC_DTO_RATE_LO_RATE_LO__M), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_MODE__A, FEC_OC_DTO_MODE_DYNAMIC__M | FEC_OC_DTO_MODE_OFFSET_ENABLE__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_MODE__A, FEC_OC_FCT_MODE_RAT_ENA__M | FEC_OC_FCT_MODE_VIRT_ENA__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_BURST_LEN__A, fec_oc_dto_burst_len, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (ext_attr->mpeg_output_clock_rate != DRXJ_MPEGOUTPUT_CLOCK_RATE_AUTO)
- fec_oc_dto_period = ext_attr->mpeg_output_clock_rate - 1;
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_PERIOD__A, fec_oc_dto_period, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else { /* Dynamic mode */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DTO_MODE__A, FEC_OC_DTO_MODE_DYNAMIC__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_FCT_MODE__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxdap_fasi_write_reg32(dev_addr, FEC_OC_RCN_CTL_RATE_LO__A, rcn_rate, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write appropriate registers with requested configuration */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_MODE__A, fec_oc_reg_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_MODE__A, fec_oc_reg_ipr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_INVERT__A, fec_oc_reg_ipr_invert, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* enabling for both parallel and serial now */
- /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Set MPEG TS pads to outputmode */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MSTRT_CFG__A, 0x0013, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MERR_CFG__A, 0x0013, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MCLK_CFG__A, MPEG_OUTPUT_CLK_DRIVE_STRENGTH << SIO_PDR_MCLK_CFG_DRIVE__B | 0x03 << SIO_PDR_MCLK_CFG_MODE__B, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MVAL_CFG__A, 0x0013, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- sio_pdr_md_cfg =
- MPEG_SERIAL_OUTPUT_PIN_DRIVE_STRENGTH <<
- SIO_PDR_MD0_CFG_DRIVE__B | 0x03 << SIO_PDR_MD0_CFG_MODE__B;
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (cfg_data->enable_parallel == true) { /* MPEG data output is parallel -> set MD1 to MD7 to output mode */
- sio_pdr_md_cfg =
- MPEG_PARALLEL_OUTPUT_PIN_DRIVE_STRENGTH <<
- SIO_PDR_MD0_CFG_DRIVE__B | 0x03 <<
- SIO_PDR_MD0_CFG_MODE__B;
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, sio_pdr_md_cfg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else { /* MPEG data output is serial -> set MD1 to MD7 to tri-state */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* Enable Monitor Bus output over MPEG pads and ctl input */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MON_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write nomagic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Set MPEG TS pads to inputmode */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MSTRT_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MERR_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MCLK_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MVAL_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD0_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD1_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD2_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD3_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD4_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD5_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD6_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MD7_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Enable Monitor Bus output over MPEG pads and ctl input */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_MON_CFG__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write nomagic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* save values for restore after re-acquire */
- common_attr->mpeg_cfg.enable_mpeg_output = cfg_data->enable_mpeg_output;
- return 0;
- rw_error:
- return rc;
- }
- /*----------------------------------------------------------------------------*/
- /*----------------------------------------------------------------------------*/
- /* MPEG Output Configuration Functions - end */
- /*----------------------------------------------------------------------------*/
- /*----------------------------------------------------------------------------*/
- /* miscellaneous configurations - begin */
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int set_mpegtei_handling()
- * \brief Activate MPEG TEI handling settings.
- * \param devmod Pointer to demodulator instance.
- * \return int.
- *
- * This routine should be called during a set channel of QAM/VSB
- *
- */
- static int set_mpegtei_handling(struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- int rc;
- u16 fec_oc_dpr_mode = 0;
- u16 fec_oc_snc_mode = 0;
- u16 fec_oc_ems_mode = 0;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_DPR_MODE__A, &fec_oc_dpr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_MODE__A, &fec_oc_snc_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_EMS_MODE__A, &fec_oc_ems_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* reset to default, allow TEI bit to be changed */
- fec_oc_dpr_mode &= (~FEC_OC_DPR_MODE_ERR_DISABLE__M);
- fec_oc_snc_mode &= (~(FEC_OC_SNC_MODE_ERROR_CTL__M |
- FEC_OC_SNC_MODE_CORR_DISABLE__M));
- fec_oc_ems_mode &= (~FEC_OC_EMS_MODE_MODE__M);
- if (ext_attr->disable_te_ihandling) {
- /* do not change TEI bit */
- fec_oc_dpr_mode |= FEC_OC_DPR_MODE_ERR_DISABLE__M;
- fec_oc_snc_mode |= FEC_OC_SNC_MODE_CORR_DISABLE__M |
- ((0x2) << (FEC_OC_SNC_MODE_ERROR_CTL__B));
- fec_oc_ems_mode |= ((0x01) << (FEC_OC_EMS_MODE_MODE__B));
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_DPR_MODE__A, fec_oc_dpr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_MODE__A, fec_oc_snc_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_EMS_MODE__A, fec_oc_ems_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int bit_reverse_mpeg_output()
- * \brief Set MPEG output bit-endian settings.
- * \param devmod Pointer to demodulator instance.
- * \return int.
- *
- * This routine should be called during a set channel of QAM/VSB
- *
- */
- static int bit_reverse_mpeg_output(struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- int rc;
- u16 fec_oc_ipr_mode = 0;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_IPR_MODE__A, &fec_oc_ipr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* reset to default (normal bit order) */
- fec_oc_ipr_mode &= (~FEC_OC_IPR_MODE_REVERSE_ORDER__M);
- if (ext_attr->bit_reverse_mpeg_outout)
- fec_oc_ipr_mode |= FEC_OC_IPR_MODE_REVERSE_ORDER__M;
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_IPR_MODE__A, fec_oc_ipr_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int set_mpeg_start_width()
- * \brief Set MPEG start width.
- * \param devmod Pointer to demodulator instance.
- * \return int.
- *
- * This routine should be called during a set channel of QAM/VSB
- *
- */
- static int set_mpeg_start_width(struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)(NULL);
- struct drx_common_attr *common_attr = (struct drx_common_attr *) NULL;
- int rc;
- u16 fec_oc_comm_mb = 0;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- common_attr = demod->my_common_attr;
- if ((common_attr->mpeg_cfg.static_clk == true)
- && (common_attr->mpeg_cfg.enable_parallel == false)) {
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_COMM_MB__A, &fec_oc_comm_mb, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- fec_oc_comm_mb &= ~FEC_OC_COMM_MB_CTL_ON;
- if (ext_attr->mpeg_start_width == DRXJ_MPEG_START_WIDTH_8CLKCYC)
- fec_oc_comm_mb |= FEC_OC_COMM_MB_CTL_ON;
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_COMM_MB__A, fec_oc_comm_mb, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- return 0;
- rw_error:
- return rc;
- }
- /*----------------------------------------------------------------------------*/
- /* miscellaneous configurations - end */
- /*----------------------------------------------------------------------------*/
- /*----------------------------------------------------------------------------*/
- /* UIO Configuration Functions - begin */
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int ctrl_set_uio_cfg()
- * \brief Configure modus oprandi UIO.
- * \param demod Pointer to demodulator instance.
- * \param uio_cfg Pointer to a configuration setting for a certain UIO.
- * \return int.
- */
- static int ctrl_set_uio_cfg(struct drx_demod_instance *demod, struct drxuio_cfg *uio_cfg)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- int rc;
- if ((uio_cfg == NULL) || (demod == NULL))
- return -EINVAL;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch (uio_cfg->uio) {
- /*====================================================================*/
- case DRX_UIO1:
- /* DRX_UIO1: SMA_TX UIO-1 */
- if (!ext_attr->has_smatx)
- return -EIO;
- switch (uio_cfg->mode) {
- case DRX_UIO_MODE_FIRMWARE_SMA: /* falltrough */
- case DRX_UIO_MODE_FIRMWARE_SAW: /* falltrough */
- case DRX_UIO_MODE_READWRITE:
- ext_attr->uio_sma_tx_mode = uio_cfg->mode;
- break;
- case DRX_UIO_MODE_DISABLE:
- ext_attr->uio_sma_tx_mode = uio_cfg->mode;
- /* pad configuration register is set 0 - input mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EINVAL;
- } /* switch ( uio_cfg->mode ) */
- break;
- /*====================================================================*/
- case DRX_UIO2:
- /* DRX_UIO2: SMA_RX UIO-2 */
- if (!ext_attr->has_smarx)
- return -EIO;
- switch (uio_cfg->mode) {
- case DRX_UIO_MODE_FIRMWARE0: /* falltrough */
- case DRX_UIO_MODE_READWRITE:
- ext_attr->uio_sma_rx_mode = uio_cfg->mode;
- break;
- case DRX_UIO_MODE_DISABLE:
- ext_attr->uio_sma_rx_mode = uio_cfg->mode;
- /* pad configuration register is set 0 - input mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_RX_CFG__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EINVAL;
- break;
- } /* switch ( uio_cfg->mode ) */
- break;
- /*====================================================================*/
- case DRX_UIO3:
- /* DRX_UIO3: GPIO UIO-3 */
- if (!ext_attr->has_gpio)
- return -EIO;
- switch (uio_cfg->mode) {
- case DRX_UIO_MODE_FIRMWARE0: /* falltrough */
- case DRX_UIO_MODE_READWRITE:
- ext_attr->uio_gpio_mode = uio_cfg->mode;
- break;
- case DRX_UIO_MODE_DISABLE:
- ext_attr->uio_gpio_mode = uio_cfg->mode;
- /* pad configuration register is set 0 - input mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_GPIO_CFG__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EINVAL;
- break;
- } /* switch ( uio_cfg->mode ) */
- break;
- /*====================================================================*/
- case DRX_UIO4:
- /* DRX_UIO4: IRQN UIO-4 */
- if (!ext_attr->has_irqn)
- return -EIO;
- switch (uio_cfg->mode) {
- case DRX_UIO_MODE_READWRITE:
- ext_attr->uio_irqn_mode = uio_cfg->mode;
- break;
- case DRX_UIO_MODE_DISABLE:
- /* pad configuration register is set 0 - input mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_IRQN_CFG__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->uio_irqn_mode = uio_cfg->mode;
- break;
- case DRX_UIO_MODE_FIRMWARE0: /* falltrough */
- default:
- return -EINVAL;
- break;
- } /* switch ( uio_cfg->mode ) */
- break;
- /*====================================================================*/
- default:
- return -EINVAL;
- } /* switch ( uio_cfg->uio ) */
- /* Write magic word to disable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int ctrl_uio_write()
- * \brief Write to a UIO.
- * \param demod Pointer to demodulator instance.
- * \param uio_data Pointer to data container for a certain UIO.
- * \return int.
- */
- static int
- ctrl_uio_write(struct drx_demod_instance *demod, struct drxuio_data *uio_data)
- {
- struct drxj_data *ext_attr = (struct drxj_data *) (NULL);
- int rc;
- u16 pin_cfg_value = 0;
- u16 value = 0;
- if ((uio_data == NULL) || (demod == NULL))
- return -EINVAL;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch (uio_data->uio) {
- /*====================================================================*/
- case DRX_UIO1:
- /* DRX_UIO1: SMA_TX UIO-1 */
- if (!ext_attr->has_smatx)
- return -EIO;
- if ((ext_attr->uio_sma_tx_mode != DRX_UIO_MODE_READWRITE)
- && (ext_attr->uio_sma_tx_mode != DRX_UIO_MODE_FIRMWARE_SAW)) {
- return -EIO;
- }
- pin_cfg_value = 0;
- /* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
- pin_cfg_value |= 0x0113;
- /* io_pad_cfg_mode output mode is drive always */
- /* io_pad_cfg_drive is set to power 2 (23 mA) */
- /* write to io pad configuration register - output mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, pin_cfg_value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* use corresponding bit in io data output registar */
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (!uio_data->value)
- value &= 0x7FFF; /* write zero to 15th bit - 1st UIO */
- else
- value |= 0x8000; /* write one to 15th bit - 1st UIO */
- /* write back to io data output register */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- /*======================================================================*/
- case DRX_UIO2:
- /* DRX_UIO2: SMA_RX UIO-2 */
- if (!ext_attr->has_smarx)
- return -EIO;
- if (ext_attr->uio_sma_rx_mode != DRX_UIO_MODE_READWRITE)
- return -EIO;
- pin_cfg_value = 0;
- /* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
- pin_cfg_value |= 0x0113;
- /* io_pad_cfg_mode output mode is drive always */
- /* io_pad_cfg_drive is set to power 2 (23 mA) */
- /* write to io pad configuration register - output mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_RX_CFG__A, pin_cfg_value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* use corresponding bit in io data output registar */
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (!uio_data->value)
- value &= 0xBFFF; /* write zero to 14th bit - 2nd UIO */
- else
- value |= 0x4000; /* write one to 14th bit - 2nd UIO */
- /* write back to io data output register */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- /*====================================================================*/
- case DRX_UIO3:
- /* DRX_UIO3: ASEL UIO-3 */
- if (!ext_attr->has_gpio)
- return -EIO;
- if (ext_attr->uio_gpio_mode != DRX_UIO_MODE_READWRITE)
- return -EIO;
- pin_cfg_value = 0;
- /* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
- pin_cfg_value |= 0x0113;
- /* io_pad_cfg_mode output mode is drive always */
- /* io_pad_cfg_drive is set to power 2 (23 mA) */
- /* write to io pad configuration register - output mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_GPIO_CFG__A, pin_cfg_value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* use corresponding bit in io data output registar */
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_HI__A, &value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (!uio_data->value)
- value &= 0xFFFB; /* write zero to 2nd bit - 3rd UIO */
- else
- value |= 0x0004; /* write one to 2nd bit - 3rd UIO */
- /* write back to io data output register */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_HI__A, value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- /*=====================================================================*/
- case DRX_UIO4:
- /* DRX_UIO4: IRQN UIO-4 */
- if (!ext_attr->has_irqn)
- return -EIO;
- if (ext_attr->uio_irqn_mode != DRX_UIO_MODE_READWRITE)
- return -EIO;
- pin_cfg_value = 0;
- /* io_pad_cfg register (8 bit reg.) MSB bit is 1 (default value) */
- pin_cfg_value |= 0x0113;
- /* io_pad_cfg_mode output mode is drive always */
- /* io_pad_cfg_drive is set to power 2 (23 mA) */
- /* write to io pad configuration register - output mode */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_IRQN_CFG__A, pin_cfg_value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* use corresponding bit in io data output registar */
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, &value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (uio_data->value == false)
- value &= 0xEFFF; /* write zero to 12th bit - 4th UIO */
- else
- value |= 0x1000; /* write one to 12th bit - 4th UIO */
- /* write back to io data output register */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_UIO_OUT_LO__A, value, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- /*=====================================================================*/
- default:
- return -EINVAL;
- } /* switch ( uio_data->uio ) */
- /* Write magic word to disable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*---------------------------------------------------------------------------*/
- /* UIO Configuration Functions - end */
- /*---------------------------------------------------------------------------*/
- /*----------------------------------------------------------------------------*/
- /* I2C Bridge Functions - begin */
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int ctrl_i2c_bridge()
- * \brief Open or close the I2C switch to tuner.
- * \param demod Pointer to demodulator instance.
- * \param bridge_closed Pointer to bool indication if bridge is closed not.
- * \return int.
- */
- static int
- ctrl_i2c_bridge(struct drx_demod_instance *demod, bool *bridge_closed)
- {
- struct drxj_hi_cmd hi_cmd;
- u16 result = 0;
- /* check arguments */
- if (bridge_closed == NULL)
- return -EINVAL;
- hi_cmd.cmd = SIO_HI_RA_RAM_CMD_BRDCTRL;
- hi_cmd.param1 = SIO_HI_RA_RAM_PAR_1_PAR1_SEC_KEY;
- if (*bridge_closed)
- hi_cmd.param2 = SIO_HI_RA_RAM_PAR_2_BRD_CFG_CLOSED;
- else
- hi_cmd.param2 = SIO_HI_RA_RAM_PAR_2_BRD_CFG_OPEN;
- return hi_command(demod->my_i2c_dev_addr, &hi_cmd, &result);
- }
- /*----------------------------------------------------------------------------*/
- /* I2C Bridge Functions - end */
- /*----------------------------------------------------------------------------*/
- /*----------------------------------------------------------------------------*/
- /* Smart antenna Functions - begin */
- /*----------------------------------------------------------------------------*/
- /*
- * \fn int smart_ant_init()
- * \brief Initialize Smart Antenna.
- * \param pointer to struct drx_demod_instance.
- * \return int.
- *
- */
- static int smart_ant_init(struct drx_demod_instance *demod)
- {
- struct drxj_data *ext_attr = NULL;
- struct i2c_device_addr *dev_addr = NULL;
- struct drxuio_cfg uio_cfg = { DRX_UIO1, DRX_UIO_MODE_FIRMWARE_SMA };
- int rc;
- u16 data = 0;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, SIO_TOP_COMM_KEY_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* init smart antenna */
- rc = drxj_dap_read_reg16(dev_addr, SIO_SA_TX_COMMAND__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (ext_attr->smart_ant_inverted) {
- rc = drxj_dap_write_reg16(dev_addr, SIO_SA_TX_COMMAND__A, (data | SIO_SA_TX_COMMAND_TX_INVERT__M) | SIO_SA_TX_COMMAND_TX_ENABLE__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = drxj_dap_write_reg16(dev_addr, SIO_SA_TX_COMMAND__A, (data & (~SIO_SA_TX_COMMAND_TX_INVERT__M)) | SIO_SA_TX_COMMAND_TX_ENABLE__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* config SMA_TX pin to smart antenna mode */
- rc = ctrl_set_uio_cfg(demod, &uio_cfg);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_CFG__A, 0x13, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_PDR_SMA_TX_GPIO_FNC__A, 0x03, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write magic word to disable pdr reg write */
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- static int scu_command(struct i2c_device_addr *dev_addr, struct drxjscu_cmd *cmd)
- {
- int rc;
- u16 cur_cmd = 0;
- unsigned long timeout;
- /* Check param */
- if (cmd == NULL)
- return -EINVAL;
- /* Wait until SCU command interface is ready to receive command */
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_COMMAND__A, &cur_cmd, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (cur_cmd != DRX_SCU_READY)
- return -EIO;
- switch (cmd->parameter_len) {
- case 5:
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_4__A, *(cmd->parameter + 4), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 4:
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_3__A, *(cmd->parameter + 3), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 3:
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_2__A, *(cmd->parameter + 2), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 2:
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_1__A, *(cmd->parameter + 1), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 1:
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_PARAM_0__A, *(cmd->parameter + 0), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 0:
- /* do nothing */
- break;
- default:
- /* this number of parameters is not supported */
- return -EIO;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_COMMAND__A, cmd->command, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Wait until SCU has processed command */
- timeout = jiffies + msecs_to_jiffies(DRXJ_MAX_WAITTIME);
- while (time_is_after_jiffies(timeout)) {
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_COMMAND__A, &cur_cmd, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (cur_cmd == DRX_SCU_READY)
- break;
- usleep_range(1000, 2000);
- }
- if (cur_cmd != DRX_SCU_READY)
- return -EIO;
- /* read results */
- if ((cmd->result_len > 0) && (cmd->result != NULL)) {
- s16 err;
- switch (cmd->result_len) {
- case 4:
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_3__A, cmd->result + 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 3:
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_2__A, cmd->result + 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 2:
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_1__A, cmd->result + 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 1:
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_PARAM_0__A, cmd->result + 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* fallthrough */
- case 0:
- /* do nothing */
- break;
- default:
- /* this number of parameters is not supported */
- return -EIO;
- }
- /* Check if an error was reported by SCU */
- err = cmd->result[0];
- /* check a few fixed error codes */
- if ((err == (s16) SCU_RAM_PARAM_0_RESULT_UNKSTD)
- || (err == (s16) SCU_RAM_PARAM_0_RESULT_UNKCMD)
- || (err == (s16) SCU_RAM_PARAM_0_RESULT_INVPAR)
- || (err == (s16) SCU_RAM_PARAM_0_RESULT_SIZE)
- ) {
- return -EINVAL;
- }
- /* here it is assumed that negative means error, and positive no error */
- else if (err < 0)
- return -EIO;
- else
- return 0;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int DRXJ_DAP_SCUAtomicReadWriteBlock()
- * \brief Basic access routine for SCU atomic read or write access
- * \param dev_addr pointer to i2c dev address
- * \param addr destination/source address
- * \param datasize size of data buffer in bytes
- * \param data pointer to data buffer
- * \return int
- * \retval 0 Succes
- * \retval -EIO Timeout, I2C error, illegal bank
- *
- */
- #define ADDR_AT_SCU_SPACE(x) ((x - 0x82E000) * 2)
- static
- int drxj_dap_scu_atomic_read_write_block(struct i2c_device_addr *dev_addr, u32 addr, u16 datasize, /* max 30 bytes because the limit of SCU parameter */
- u8 *data, bool read_flag)
- {
- struct drxjscu_cmd scu_cmd;
- int rc;
- u16 set_param_parameters[18];
- u16 cmd_result[15];
- /* Parameter check */
- if (!data || !dev_addr || (datasize % 2) || ((datasize / 2) > 16))
- return -EINVAL;
- set_param_parameters[1] = (u16) ADDR_AT_SCU_SPACE(addr);
- if (read_flag) { /* read */
- set_param_parameters[0] = ((~(0x0080)) & datasize);
- scu_cmd.parameter_len = 2;
- scu_cmd.result_len = datasize / 2 + 2;
- } else {
- int i = 0;
- set_param_parameters[0] = 0x0080 | datasize;
- for (i = 0; i < (datasize / 2); i++) {
- set_param_parameters[i + 2] =
- (data[2 * i] | (data[(2 * i) + 1] << 8));
- }
- scu_cmd.parameter_len = datasize / 2 + 2;
- scu_cmd.result_len = 1;
- }
- scu_cmd.command =
- SCU_RAM_COMMAND_STANDARD_TOP |
- SCU_RAM_COMMAND_CMD_AUX_SCU_ATOMIC_ACCESS;
- scu_cmd.result = cmd_result;
- scu_cmd.parameter = set_param_parameters;
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (read_flag) {
- int i = 0;
- /* read data from buffer */
- for (i = 0; i < (datasize / 2); i++) {
- data[2 * i] = (u8) (scu_cmd.result[i + 2] & 0xFF);
- data[(2 * i) + 1] = (u8) (scu_cmd.result[i + 2] >> 8);
- }
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int DRXJ_DAP_AtomicReadReg16()
- * \brief Atomic read of 16 bits words
- */
- static
- int drxj_dap_scu_atomic_read_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 *data, u32 flags)
- {
- u8 buf[2] = { 0 };
- int rc = -EIO;
- u16 word = 0;
- if (!data)
- return -EINVAL;
- rc = drxj_dap_scu_atomic_read_write_block(dev_addr, addr, 2, buf, true);
- if (rc < 0)
- return rc;
- word = (u16) (buf[0] + (buf[1] << 8));
- *data = word;
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int drxj_dap_scu_atomic_write_reg16()
- * \brief Atomic read of 16 bits words
- */
- static
- int drxj_dap_scu_atomic_write_reg16(struct i2c_device_addr *dev_addr,
- u32 addr,
- u16 data, u32 flags)
- {
- u8 buf[2];
- int rc = -EIO;
- buf[0] = (u8) (data & 0xff);
- buf[1] = (u8) ((data >> 8) & 0xff);
- rc = drxj_dap_scu_atomic_read_write_block(dev_addr, addr, 2, buf, false);
- return rc;
- }
- /* -------------------------------------------------------------------------- */
- /*
- * \brief Measure result of ADC synchronisation
- * \param demod demod instance
- * \param count (returned) count
- * \return int.
- * \retval 0 Success
- * \retval -EIO Failure: I2C error
- *
- */
- static int adc_sync_measurement(struct drx_demod_instance *demod, u16 *count)
- {
- struct i2c_device_addr *dev_addr = NULL;
- int rc;
- u16 data = 0;
- dev_addr = demod->my_i2c_dev_addr;
- /* Start measurement */
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_COMM_EXEC__A, IQM_AF_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_START_LOCK__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Wait at least 3*128*(1/sysclk) <<< 1 millisec */
- msleep(1);
- *count = 0;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE0__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (data == 127)
- *count = *count + 1;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE1__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (data == 127)
- *count = *count + 1;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_PHASE2__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (data == 127)
- *count = *count + 1;
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \brief Synchronize analog and digital clock domains
- * \param demod demod instance
- * \return int.
- * \retval 0 Success
- * \retval -EIO Failure: I2C error or failure to synchronize
- *
- * An IQM reset will also reset the results of this synchronization.
- * After an IQM reset this routine needs to be called again.
- *
- */
- static int adc_synchronization(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- int rc;
- u16 count = 0;
- dev_addr = demod->my_i2c_dev_addr;
- rc = adc_sync_measurement(demod, &count);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (count == 1) {
- /* Try sampling on a different edge */
- u16 clk_neg = 0;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_CLKNEG__A, &clk_neg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- clk_neg ^= IQM_AF_CLKNEG_CLKNEGDATA__M;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLKNEG__A, clk_neg, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = adc_sync_measurement(demod, &count);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* TODO: implement fallback scenarios */
- if (count < 2)
- return -EIO;
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== END AUXILIARY FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== 8VSB & QAM COMMON DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn int init_agc ()
- * \brief Initialize AGC for all standards.
- * \param demod instance of demodulator.
- * \param channel pointer to channel data.
- * \return int.
- */
- static int init_agc(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drx_common_attr *common_attr = NULL;
- struct drxj_data *ext_attr = NULL;
- struct drxj_cfg_agc *p_agc_rf_settings = NULL;
- struct drxj_cfg_agc *p_agc_if_settings = NULL;
- int rc;
- u16 ingain_tgt_max = 0;
- u16 clp_dir_to = 0;
- u16 sns_sum_max = 0;
- u16 clp_sum_max = 0;
- u16 sns_dir_to = 0;
- u16 ki_innergain_min = 0;
- u16 agc_ki = 0;
- u16 ki_max = 0;
- u16 if_iaccu_hi_tgt_min = 0;
- u16 data = 0;
- u16 agc_ki_dgain = 0;
- u16 ki_min = 0;
- u16 clp_ctrl_mode = 0;
- u16 agc_rf = 0;
- u16 agc_if = 0;
- dev_addr = demod->my_i2c_dev_addr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- switch (ext_attr->standard) {
- case DRX_STANDARD_8VSB:
- clp_sum_max = 1023;
- clp_dir_to = (u16) (-9);
- sns_sum_max = 1023;
- sns_dir_to = (u16) (-9);
- ki_innergain_min = (u16) (-32768);
- ki_max = 0x032C;
- agc_ki_dgain = 0xC;
- if_iaccu_hi_tgt_min = 2047;
- ki_min = 0x0117;
- ingain_tgt_max = 16383;
- clp_ctrl_mode = 0;
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MINGAIN__A, 0x7fff, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXGAIN__A, 0x0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCCNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_WD__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_STP__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCCNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_WD__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_STP__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN__A, 1024, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_VSB_AGC_POW_TGT__A, 22600, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT__A, 13200, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- p_agc_if_settings = &(ext_attr->vsb_if_agc_cfg);
- p_agc_rf_settings = &(ext_attr->vsb_rf_agc_cfg);
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- case DRX_STANDARD_ITU_B:
- ingain_tgt_max = 5119;
- clp_sum_max = 1023;
- clp_dir_to = (u16) (-5);
- sns_sum_max = 127;
- sns_dir_to = (u16) (-3);
- ki_innergain_min = 0;
- ki_max = 0x0657;
- if_iaccu_hi_tgt_min = 2047;
- agc_ki_dgain = 0x7;
- ki_min = 0x0117;
- clp_ctrl_mode = 0;
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MINGAIN__A, 0x7fff, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXGAIN__A, 0x0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCCNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_WD__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_STP__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCCNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_WD__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_STP__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- p_agc_if_settings = &(ext_attr->qam_if_agc_cfg);
- p_agc_rf_settings = &(ext_attr->qam_rf_agc_cfg);
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT__A, p_agc_if_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_AGC_KI__A, &agc_ki, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- agc_ki &= 0xf000;
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI__A, agc_ki, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- #endif
- default:
- return -EINVAL;
- }
- /* for new AGC interface */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT_MIN__A, p_agc_if_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN__A, p_agc_if_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* Gain fed from inner to outer AGC */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_INGAIN_TGT_MAX__A, ingain_tgt_max, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MIN__A, if_iaccu_hi_tgt_min, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_HI__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* set to p_agc_settings->top before */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_IF_IACCU_LO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_IACCU_HI__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_IACCU_LO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_RF_MAX__A, 32767, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM_MAX__A, clp_sum_max, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM_MAX__A, sns_sum_max, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_INNERGAIN_MIN__A, ki_innergain_min, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_FAST_SNS_CTRL_DELAY__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_CYCLEN__A, 500, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_CYCLEN__A, 500, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAXMINGAIN_TH__A, 20, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MIN__A, ki_min, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_MAX__A, ki_max, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI_RED__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_SUM_MIN__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CYCLEN__A, 500, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_DIR_TO__A, clp_dir_to, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_SUM_MIN__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_SNS_DIR_TO__A, sns_dir_to, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_FAST_CLP_CTRL_DELAY__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_CLP_CTRL_MODE__A, clp_ctrl_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- agc_rf = 0x800 + p_agc_rf_settings->cut_off_current;
- if (common_attr->tuner_rf_agc_pol == true)
- agc_rf = 0x87ff - agc_rf;
- agc_if = 0x800;
- if (common_attr->tuner_if_agc_pol == true)
- agc_rf = 0x87ff - agc_rf;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AGC_RF__A, agc_rf, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AGC_IF__A, agc_if, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Set/restore Ki DGAIN factor */
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_DGAIN__M;
- data |= (agc_ki_dgain << SCU_RAM_AGC_KI_DGAIN__B);
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_frequency ()
- * \brief Set frequency shift.
- * \param demod instance of demodulator.
- * \param channel pointer to channel data.
- * \param tuner_freq_offset residual frequency from tuner.
- * \return int.
- */
- static int
- set_frequency(struct drx_demod_instance *demod,
- struct drx_channel *channel, s32 tuner_freq_offset)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxj_data *ext_attr = demod->my_ext_attr;
- int rc;
- s32 sampling_frequency = 0;
- s32 frequency_shift = 0;
- s32 if_freq_actual = 0;
- s32 rf_freq_residual = -1 * tuner_freq_offset;
- s32 adc_freq = 0;
- s32 intermediate_freq = 0;
- u32 iqm_fs_rate_ofs = 0;
- bool adc_flip = true;
- bool select_pos_image = false;
- bool rf_mirror;
- bool tuner_mirror;
- bool image_to_select = true;
- s32 fm_frequency_shift = 0;
- rf_mirror = (ext_attr->mirror == DRX_MIRROR_YES) ? true : false;
- tuner_mirror = demod->my_common_attr->mirror_freq_spect ? false : true;
- /*
- Program frequency shifter
- No need to account for mirroring on RF
- */
- switch (ext_attr->standard) {
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- case DRX_STANDARD_PAL_SECAM_LP:
- case DRX_STANDARD_8VSB:
- select_pos_image = true;
- break;
- case DRX_STANDARD_FM:
- /* After IQM FS sound carrier must appear at 4 Mhz in spect.
- Sound carrier is already 3Mhz above centre frequency due
- to tuner setting so now add an extra shift of 1MHz... */
- fm_frequency_shift = 1000;
- /*fall through */
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_NTSC:
- case DRX_STANDARD_PAL_SECAM_BG:
- case DRX_STANDARD_PAL_SECAM_DK:
- case DRX_STANDARD_PAL_SECAM_I:
- case DRX_STANDARD_PAL_SECAM_L:
- select_pos_image = false;
- break;
- default:
- return -EINVAL;
- }
- intermediate_freq = demod->my_common_attr->intermediate_freq;
- sampling_frequency = demod->my_common_attr->sys_clock_freq / 3;
- if (tuner_mirror)
- if_freq_actual = intermediate_freq + rf_freq_residual + fm_frequency_shift;
- else
- if_freq_actual = intermediate_freq - rf_freq_residual - fm_frequency_shift;
- if (if_freq_actual > sampling_frequency / 2) {
- /* adc mirrors */
- adc_freq = sampling_frequency - if_freq_actual;
- adc_flip = true;
- } else {
- /* adc doesn't mirror */
- adc_freq = if_freq_actual;
- adc_flip = false;
- }
- frequency_shift = adc_freq;
- image_to_select =
- (bool) (rf_mirror ^ tuner_mirror ^ adc_flip ^ select_pos_image);
- iqm_fs_rate_ofs = frac28(frequency_shift, sampling_frequency);
- if (image_to_select)
- iqm_fs_rate_ofs = ~iqm_fs_rate_ofs + 1;
- /* Program frequency shifter with tuner offset compensation */
- /* frequency_shift += tuner_freq_offset; TODO */
- rc = drxdap_fasi_write_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, iqm_fs_rate_ofs, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->iqm_fs_rate_ofs = iqm_fs_rate_ofs;
- ext_attr->pos_image = (bool) (rf_mirror ^ tuner_mirror ^ select_pos_image);
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int get_acc_pkt_err()
- * \brief Retrieve signal strength for VSB and QAM.
- * \param demod Pointer to demod instance
- * \param packet_err Pointer to packet error
- * \return int.
- * \retval 0 sig_strength contains valid data.
- * \retval -EINVAL sig_strength is NULL.
- * \retval -EIO Erroneous data, sig_strength contains invalid data.
- */
- #ifdef DRXJ_SIGNAL_ACCUM_ERR
- static int get_acc_pkt_err(struct drx_demod_instance *demod, u16 *packet_err)
- {
- int rc;
- static u16 pkt_err;
- static u16 last_pkt_err;
- u16 data = 0;
- struct drxj_data *ext_attr = NULL;
- struct i2c_device_addr *dev_addr = NULL;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- dev_addr = demod->my_i2c_dev_addr;
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (ext_attr->reset_pkt_err_acc) {
- last_pkt_err = data;
- pkt_err = 0;
- ext_attr->reset_pkt_err_acc = false;
- }
- if (data < last_pkt_err) {
- pkt_err += 0xffff - last_pkt_err;
- pkt_err += data;
- } else {
- pkt_err += (data - last_pkt_err);
- }
- *packet_err = pkt_err;
- last_pkt_err = data;
- return 0;
- rw_error:
- return rc;
- }
- #endif
- /*============================================================================*/
- /*
- * \fn int set_agc_rf ()
- * \brief Configure RF AGC
- * \param demod instance of demodulator.
- * \param agc_settings AGC configuration structure
- * \return int.
- */
- static int
- set_agc_rf(struct drx_demod_instance *demod, struct drxj_cfg_agc *agc_settings, bool atomic)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- struct drxj_cfg_agc *p_agc_settings = NULL;
- struct drx_common_attr *common_attr = NULL;
- int rc;
- drx_write_reg16func_t scu_wr16 = NULL;
- drx_read_reg16func_t scu_rr16 = NULL;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- if (atomic) {
- scu_rr16 = drxj_dap_scu_atomic_read_reg16;
- scu_wr16 = drxj_dap_scu_atomic_write_reg16;
- } else {
- scu_rr16 = drxj_dap_read_reg16;
- scu_wr16 = drxj_dap_write_reg16;
- }
- /* Configure AGC only if standard is currently active */
- if ((ext_attr->standard == agc_settings->standard) ||
- (DRXJ_ISQAMSTD(ext_attr->standard) &&
- DRXJ_ISQAMSTD(agc_settings->standard)) ||
- (DRXJ_ISATVSTD(ext_attr->standard) &&
- DRXJ_ISATVSTD(agc_settings->standard))) {
- u16 data = 0;
- switch (agc_settings->ctrl_mode) {
- case DRX_AGC_CTRL_AUTO:
- /* Enable RF AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data |= IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Enable SCU RF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_RF__M;
- if (ext_attr->standard == DRX_STANDARD_8VSB)
- data |= (2 << SCU_RAM_AGC_KI_RF__B);
- else if (DRXJ_ISQAMSTD(ext_attr->standard))
- data |= (5 << SCU_RAM_AGC_KI_RF__B);
- else
- data |= (4 << SCU_RAM_AGC_KI_RF__B);
- if (common_attr->tuner_rf_agc_pol)
- data |= SCU_RAM_AGC_KI_INV_RF_POL__M;
- else
- data &= ~SCU_RAM_AGC_KI_INV_RF_POL__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Set speed ( using complementary reduction value ) */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_RED_RAGC_RED__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, (~(agc_settings->speed << SCU_RAM_AGC_KI_RED_RAGC_RED__B) & SCU_RAM_AGC_KI_RED_RAGC_RED__M) | data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (agc_settings->standard == DRX_STANDARD_8VSB)
- p_agc_settings = &(ext_attr->vsb_if_agc_cfg);
- else if (DRXJ_ISQAMSTD(agc_settings->standard))
- p_agc_settings = &(ext_attr->qam_if_agc_cfg);
- else if (DRXJ_ISATVSTD(agc_settings->standard))
- p_agc_settings = &(ext_attr->atv_if_agc_cfg);
- else
- return -EINVAL;
- /* Set TOP, only if IF-AGC is in AUTO mode */
- if (p_agc_settings->ctrl_mode == DRX_AGC_CTRL_AUTO) {
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, agc_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, agc_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* Cut-Off current */
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_RF_IACCU_HI_CO__A, agc_settings->cut_off_current, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_AGC_CTRL_USER:
- /* Enable RF AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data |= IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Disable SCU RF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_RF__M;
- if (common_attr->tuner_rf_agc_pol)
- data |= SCU_RAM_AGC_KI_INV_RF_POL__M;
- else
- data &= ~SCU_RAM_AGC_KI_INV_RF_POL__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write value to output pin */
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_RF_IACCU_HI__A, agc_settings->output_level, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_AGC_CTRL_OFF:
- /* Disable RF AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= (~IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE);
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Disable SCU RF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_RF__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EINVAL;
- } /* switch ( agcsettings->ctrl_mode ) */
- }
- /* Store rf agc settings */
- switch (agc_settings->standard) {
- case DRX_STANDARD_8VSB:
- ext_attr->vsb_rf_agc_cfg = *agc_settings;
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- ext_attr->qam_rf_agc_cfg = *agc_settings;
- break;
- #endif
- default:
- return -EIO;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_agc_if ()
- * \brief Configure If AGC
- * \param demod instance of demodulator.
- * \param agc_settings AGC configuration structure
- * \return int.
- */
- static int
- set_agc_if(struct drx_demod_instance *demod, struct drxj_cfg_agc *agc_settings, bool atomic)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- struct drxj_cfg_agc *p_agc_settings = NULL;
- struct drx_common_attr *common_attr = NULL;
- drx_write_reg16func_t scu_wr16 = NULL;
- drx_read_reg16func_t scu_rr16 = NULL;
- int rc;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- if (atomic) {
- scu_rr16 = drxj_dap_scu_atomic_read_reg16;
- scu_wr16 = drxj_dap_scu_atomic_write_reg16;
- } else {
- scu_rr16 = drxj_dap_read_reg16;
- scu_wr16 = drxj_dap_write_reg16;
- }
- /* Configure AGC only if standard is currently active */
- if ((ext_attr->standard == agc_settings->standard) ||
- (DRXJ_ISQAMSTD(ext_attr->standard) &&
- DRXJ_ISQAMSTD(agc_settings->standard)) ||
- (DRXJ_ISATVSTD(ext_attr->standard) &&
- DRXJ_ISATVSTD(agc_settings->standard))) {
- u16 data = 0;
- switch (agc_settings->ctrl_mode) {
- case DRX_AGC_CTRL_AUTO:
- /* Enable IF AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data |= IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Enable SCU IF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
- data &= ~SCU_RAM_AGC_KI_IF__M;
- if (ext_attr->standard == DRX_STANDARD_8VSB)
- data |= (3 << SCU_RAM_AGC_KI_IF__B);
- else if (DRXJ_ISQAMSTD(ext_attr->standard))
- data |= (6 << SCU_RAM_AGC_KI_IF__B);
- else
- data |= (5 << SCU_RAM_AGC_KI_IF__B);
- if (common_attr->tuner_if_agc_pol)
- data |= SCU_RAM_AGC_KI_INV_IF_POL__M;
- else
- data &= ~SCU_RAM_AGC_KI_INV_IF_POL__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Set speed (using complementary reduction value) */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI_RED__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_RED_IAGC_RED__M;
- rc = (*scu_wr16) (dev_addr, SCU_RAM_AGC_KI_RED__A, (~(agc_settings->speed << SCU_RAM_AGC_KI_RED_IAGC_RED__B) & SCU_RAM_AGC_KI_RED_IAGC_RED__M) | data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (agc_settings->standard == DRX_STANDARD_8VSB)
- p_agc_settings = &(ext_attr->vsb_rf_agc_cfg);
- else if (DRXJ_ISQAMSTD(agc_settings->standard))
- p_agc_settings = &(ext_attr->qam_rf_agc_cfg);
- else if (DRXJ_ISATVSTD(agc_settings->standard))
- p_agc_settings = &(ext_attr->atv_rf_agc_cfg);
- else
- return -EINVAL;
- /* Restore TOP */
- if (p_agc_settings->ctrl_mode == DRX_AGC_CTRL_AUTO) {
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, p_agc_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, p_agc_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- break;
- case DRX_AGC_CTRL_USER:
- /* Enable IF AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data |= IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE;
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Disable SCU IF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
- data |= SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
- if (common_attr->tuner_if_agc_pol)
- data |= SCU_RAM_AGC_KI_INV_IF_POL__M;
- else
- data &= ~SCU_RAM_AGC_KI_INV_IF_POL__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Write value to output pin */
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_IF_IACCU_HI_TGT_MAX__A, agc_settings->output_level, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_AGC_CTRL_OFF:
- /* Disable If AGC DAC */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= (~IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE);
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Disable SCU IF AGC loop */
- rc = (*scu_rr16)(dev_addr, SCU_RAM_AGC_KI__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data &= ~SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
- data |= SCU_RAM_AGC_KI_IF_AGC_DISABLE__M;
- rc = (*scu_wr16)(dev_addr, SCU_RAM_AGC_KI__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EINVAL;
- } /* switch ( agcsettings->ctrl_mode ) */
- /* always set the top to support configurations without if-loop */
- rc = (*scu_wr16) (dev_addr, SCU_RAM_AGC_INGAIN_TGT_MIN__A, agc_settings->top, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* Store if agc settings */
- switch (agc_settings->standard) {
- case DRX_STANDARD_8VSB:
- ext_attr->vsb_if_agc_cfg = *agc_settings;
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- ext_attr->qam_if_agc_cfg = *agc_settings;
- break;
- #endif
- default:
- return -EIO;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_iqm_af ()
- * \brief Configure IQM AF registers
- * \param demod instance of demodulator.
- * \param active
- * \return int.
- */
- static int set_iqm_af(struct drx_demod_instance *demod, bool active)
- {
- u16 data = 0;
- struct i2c_device_addr *dev_addr = NULL;
- int rc;
- dev_addr = demod->my_i2c_dev_addr;
- /* Configure IQM */
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_STDBY__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (!active)
- data &= ((~IQM_AF_STDBY_STDBY_ADC_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_AMP_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_PD_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE) & (~IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE));
- else
- data |= (IQM_AF_STDBY_STDBY_ADC_A2_ACTIVE | IQM_AF_STDBY_STDBY_AMP_A2_ACTIVE | IQM_AF_STDBY_STDBY_PD_A2_ACTIVE | IQM_AF_STDBY_STDBY_TAGC_IF_A2_ACTIVE | IQM_AF_STDBY_STDBY_TAGC_RF_A2_ACTIVE);
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== END 8VSB & QAM COMMON DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== 8VSB DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn int power_down_vsb ()
- * \brief Powr down QAM related blocks.
- * \param demod instance of demodulator.
- * \param channel pointer to channel data.
- * \return int.
- */
- static int power_down_vsb(struct drx_demod_instance *demod, bool primary)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxjscu_cmd cmd_scu = { /* command */ 0,
- /* parameter_len */ 0,
- /* result_len */ 0,
- /* *parameter */ NULL,
- /* *result */ NULL
- };
- struct drx_cfg_mpeg_output cfg_mpeg_output;
- int rc;
- u16 cmd_result = 0;
- /*
- STOP demodulator
- reset of FEC and VSB HW
- */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB |
- SCU_RAM_COMMAND_CMD_DEMOD_STOP;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* stop all comm_exec */
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (primary) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_iqm_af(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- cfg_mpeg_output.enable_mpeg_output = false;
- rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_vsb_leak_n_gain ()
- * \brief Set ATSC demod.
- * \param demod instance of demodulator.
- * \return int.
- */
- static int set_vsb_leak_n_gain(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- int rc;
- static const u8 vsb_ffe_leak_gain_ram0[] = {
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO1 */
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO2 */
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO3 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO4 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO5 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO6 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO7 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO8 */
- DRXJ_16TO8(0xf), /* FFETRAINLKRATIO9 */
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO10 */
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO11 */
- DRXJ_16TO8(0x8), /* FFETRAINLKRATIO12 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO1 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO2 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO3 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO4 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO5 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO6 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO7 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO8 */
- DRXJ_16TO8(0x20), /* FFERCA1TRAINLKRATIO9 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO10 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO11 */
- DRXJ_16TO8(0x10), /* FFERCA1TRAINLKRATIO12 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO1 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO2 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO3 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO4 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO5 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO6 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO7 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO8 */
- DRXJ_16TO8(0x20), /* FFERCA1DATALKRATIO9 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO10 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO11 */
- DRXJ_16TO8(0x10), /* FFERCA1DATALKRATIO12 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO1 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO2 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO3 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO4 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO5 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO6 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO7 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO8 */
- DRXJ_16TO8(0x20), /* FFERCA2TRAINLKRATIO9 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO10 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO11 */
- DRXJ_16TO8(0x10), /* FFERCA2TRAINLKRATIO12 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO1 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO2 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO3 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO4 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO5 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO6 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO7 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO8 */
- DRXJ_16TO8(0x20), /* FFERCA2DATALKRATIO9 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO10 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO11 */
- DRXJ_16TO8(0x10), /* FFERCA2DATALKRATIO12 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO1 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO2 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO3 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO4 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO5 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO6 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO7 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO8 */
- DRXJ_16TO8(0x0e), /* FFEDDM1TRAINLKRATIO9 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO10 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO11 */
- DRXJ_16TO8(0x07), /* FFEDDM1TRAINLKRATIO12 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO1 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO2 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO3 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO4 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO5 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO6 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO7 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO8 */
- DRXJ_16TO8(0x0e), /* FFEDDM1DATALKRATIO9 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO10 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO11 */
- DRXJ_16TO8(0x07), /* FFEDDM1DATALKRATIO12 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO1 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO2 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO3 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO4 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO5 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO6 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO7 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO8 */
- DRXJ_16TO8(0x0c), /* FFEDDM2TRAINLKRATIO9 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO10 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO11 */
- DRXJ_16TO8(0x06), /* FFEDDM2TRAINLKRATIO12 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO1 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO2 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO3 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO4 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO5 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO6 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO7 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO8 */
- DRXJ_16TO8(0x0c), /* FFEDDM2DATALKRATIO9 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO10 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO11 */
- DRXJ_16TO8(0x06), /* FFEDDM2DATALKRATIO12 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN1 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN2 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN3 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN4 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN5 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN6 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN7 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN8 */
- DRXJ_16TO8(0x4040), /* FIRTRAINGAIN9 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN10 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN11 */
- DRXJ_16TO8(0x2020), /* FIRTRAINGAIN12 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN1 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN2 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN3 */
- DRXJ_16TO8(0x1010), /* FIRRCA1GAIN4 */
- DRXJ_16TO8(0x1010), /* FIRRCA1GAIN5 */
- DRXJ_16TO8(0x1010), /* FIRRCA1GAIN6 */
- DRXJ_16TO8(0x1010), /* FIRRCA1GAIN7 */
- DRXJ_16TO8(0x1010) /* FIRRCA1GAIN8 */
- };
- static const u8 vsb_ffe_leak_gain_ram1[] = {
- DRXJ_16TO8(0x1010), /* FIRRCA1GAIN9 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN10 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN11 */
- DRXJ_16TO8(0x0808), /* FIRRCA1GAIN12 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN1 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN2 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN3 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN4 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN5 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN6 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN7 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN8 */
- DRXJ_16TO8(0x1010), /* FIRRCA2GAIN9 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN10 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN11 */
- DRXJ_16TO8(0x0808), /* FIRRCA2GAIN12 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN1 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN2 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN3 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN4 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN5 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN6 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN7 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN8 */
- DRXJ_16TO8(0x0606), /* FIRDDM1GAIN9 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN10 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN11 */
- DRXJ_16TO8(0x0303), /* FIRDDM1GAIN12 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN1 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN2 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN3 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN4 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN5 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN6 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN7 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN8 */
- DRXJ_16TO8(0x0505), /* FIRDDM2GAIN9 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN10 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN11 */
- DRXJ_16TO8(0x0303), /* FIRDDM2GAIN12 */
- DRXJ_16TO8(0x001f), /* DFETRAINLKRATIO */
- DRXJ_16TO8(0x01ff), /* DFERCA1TRAINLKRATIO */
- DRXJ_16TO8(0x01ff), /* DFERCA1DATALKRATIO */
- DRXJ_16TO8(0x004f), /* DFERCA2TRAINLKRATIO */
- DRXJ_16TO8(0x004f), /* DFERCA2DATALKRATIO */
- DRXJ_16TO8(0x01ff), /* DFEDDM1TRAINLKRATIO */
- DRXJ_16TO8(0x01ff), /* DFEDDM1DATALKRATIO */
- DRXJ_16TO8(0x0352), /* DFEDDM2TRAINLKRATIO */
- DRXJ_16TO8(0x0352), /* DFEDDM2DATALKRATIO */
- DRXJ_16TO8(0x0000), /* DFETRAINGAIN */
- DRXJ_16TO8(0x2020), /* DFERCA1GAIN */
- DRXJ_16TO8(0x1010), /* DFERCA2GAIN */
- DRXJ_16TO8(0x1818), /* DFEDDM1GAIN */
- DRXJ_16TO8(0x1212) /* DFEDDM2GAIN */
- };
- dev_addr = demod->my_i2c_dev_addr;
- rc = drxdap_fasi_write_block(dev_addr, VSB_SYSCTRL_RAM0_FFETRAINLKRATIO1__A, sizeof(vsb_ffe_leak_gain_ram0), ((u8 *)vsb_ffe_leak_gain_ram0), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, VSB_SYSCTRL_RAM1_FIRRCA1GAIN9__A, sizeof(vsb_ffe_leak_gain_ram1), ((u8 *)vsb_ffe_leak_gain_ram1), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_vsb()
- * \brief Set 8VSB demod.
- * \param demod instance of demodulator.
- * \return int.
- *
- */
- static int set_vsb(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- int rc;
- struct drx_common_attr *common_attr = NULL;
- struct drxjscu_cmd cmd_scu;
- struct drxj_data *ext_attr = NULL;
- u16 cmd_result = 0;
- u16 cmd_param = 0;
- static const u8 vsb_taps_re[] = {
- DRXJ_16TO8(-2), /* re0 */
- DRXJ_16TO8(4), /* re1 */
- DRXJ_16TO8(1), /* re2 */
- DRXJ_16TO8(-4), /* re3 */
- DRXJ_16TO8(1), /* re4 */
- DRXJ_16TO8(4), /* re5 */
- DRXJ_16TO8(-3), /* re6 */
- DRXJ_16TO8(-3), /* re7 */
- DRXJ_16TO8(6), /* re8 */
- DRXJ_16TO8(1), /* re9 */
- DRXJ_16TO8(-9), /* re10 */
- DRXJ_16TO8(3), /* re11 */
- DRXJ_16TO8(12), /* re12 */
- DRXJ_16TO8(-9), /* re13 */
- DRXJ_16TO8(-15), /* re14 */
- DRXJ_16TO8(17), /* re15 */
- DRXJ_16TO8(19), /* re16 */
- DRXJ_16TO8(-29), /* re17 */
- DRXJ_16TO8(-22), /* re18 */
- DRXJ_16TO8(45), /* re19 */
- DRXJ_16TO8(25), /* re20 */
- DRXJ_16TO8(-70), /* re21 */
- DRXJ_16TO8(-28), /* re22 */
- DRXJ_16TO8(111), /* re23 */
- DRXJ_16TO8(30), /* re24 */
- DRXJ_16TO8(-201), /* re25 */
- DRXJ_16TO8(-31), /* re26 */
- DRXJ_16TO8(629) /* re27 */
- };
- dev_addr = demod->my_i2c_dev_addr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* stop all comm_exec */
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* reset demodulator */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
- | SCU_RAM_COMMAND_CMD_DEMOD_RESET;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_DCF_BYPASS__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_ADJ_SEL__A, IQM_FS_ADJ_SEL_B_VSB, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_ADJ_SEL__A, IQM_RC_ADJ_SEL_B_VSB, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->iqm_rc_rate_ofs = 0x00AD0D79;
- rc = drxdap_fasi_write_reg32(dev_addr, IQM_RC_RATE_OFS_LO__A, ext_attr->iqm_rc_rate_ofs, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CFAGC_GAINSHIFT__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1TRK__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_CROUT_ENA__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_STRETCH__A, 28, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_ACTIVE__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SYMMETRIC__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_MIDTAP__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_OUT_ENA__A, IQM_CF_OUT_ENA_VSB__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE__A, 1393, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(vsb_taps_re), ((u8 *)vsb_taps_re), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(vsb_taps_re), ((u8 *)vsb_taps_re), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BNTHRESH__A, 330, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* set higher threshold */
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CLPLASTNUM__A, 90, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* burst detection on */
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_RCA1__A, 0x0042, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* drop thresholds by 1 dB */
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_RCA2__A, 0x0053, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* drop thresholds by 2 dB */
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_EQCTRL__A, 0x1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* cma on */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_GPIO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* GPIO */
- /* Initialize the FEC Subsystem */
- rc = drxj_dap_write_reg16(dev_addr, FEC_TOP_ANNEX__A, FEC_TOP_ANNEX_D, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- {
- u16 fec_oc_snc_mode = 0;
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_MODE__A, &fec_oc_snc_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* output data even when not locked */
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_MODE__A, fec_oc_snc_mode | FEC_OC_SNC_MODE_UNLOCK_ENABLE__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* set clip */
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_LEN__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_TH__A, 470, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SNS_LEN__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_PT__A, 0xD4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no transparent, no A&C framing; parity is set in mpegoutput */
- {
- u16 fec_oc_reg_mode = 0;
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_MODE__A, &fec_oc_reg_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_MODE__A, fec_oc_reg_mode & (~(FEC_OC_MODE_TRANSPARENT__M | FEC_OC_MODE_CLEAR__M | FEC_OC_MODE_RETAIN_FRAMING__M)), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_DI_TIMEOUT_LO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* timeout counter for restarting */
- rc = drxj_dap_write_reg16(dev_addr, FEC_DI_TIMEOUT_HI__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MODE__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* bypass disabled */
- /* initialize RS packet error measurement parameters */
- rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PERIOD__A, FEC_RS_MEASUREMENT_PERIOD, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PRESCALE__A, FEC_RS_MEASUREMENT_PRESCALE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* init measurement period of MER/SER */
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_MEASUREMENT_PERIOD__A, VSB_TOP_MEASUREMENT_PERIOD, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_reg32(dev_addr, SCU_RAM_FEC_ACCUM_CW_CORRECTED_LO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_MEAS_COUNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CKGN1TRK__A, 128, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* B-Input to ADC, PGA+filter in standby */
- if (!ext_attr->has_lna) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AMUX__A, 0x02, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* turn on IQMAF. It has to be in front of setAgc**() */
- rc = set_iqm_af(demod, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = adc_synchronization(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = init_agc(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_agc_if(demod, &(ext_attr->vsb_if_agc_cfg), false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_agc_rf(demod, &(ext_attr->vsb_rf_agc_cfg), false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- {
- /* TODO fix this, store a struct drxj_cfg_afe_gain structure in struct drxj_data instead
- of only the gain */
- struct drxj_cfg_afe_gain vsb_pga_cfg = { DRX_STANDARD_8VSB, 0 };
- vsb_pga_cfg.gain = ext_attr->vsb_pga_cfg;
- rc = ctrl_set_cfg_afe_gain(demod, &vsb_pga_cfg);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = ctrl_set_cfg_pre_saw(demod, &(ext_attr->vsb_pre_saw_cfg));
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Mpeg output has to be in front of FEC active */
- rc = set_mpegtei_handling(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = bit_reverse_mpeg_output(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_mpeg_start_width(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- {
- /* TODO: move to set_standard after hardware reset value problem is solved */
- /* Configure initial MPEG output */
- struct drx_cfg_mpeg_output cfg_mpeg_output;
- memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
- cfg_mpeg_output.enable_mpeg_output = true;
- rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* TBD: what parameters should be set */
- cmd_param = 0x00; /* Default mode AGC on, etc */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
- | SCU_RAM_COMMAND_CMD_DEMOD_SET_PARAM;
- cmd_scu.parameter_len = 1;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = &cmd_param;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BEAGC_GAINSHIFT__A, 0x0004, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SNRTH_PT__A, 0x00D2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_SYSSMTRNCTRL__A, VSB_TOP_SYSSMTRNCTRL__PRE | VSB_TOP_SYSSMTRNCTRL_NCOTIMEOUTCNTEN__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_BEDETCTRL__A, 0x142, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_LBAGCREFLVL__A, 640, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1ACQ__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN1TRK__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_TOP_CYGN2TRK__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* start demodulator */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB
- | SCU_RAM_COMMAND_CMD_DEMOD_START;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, VSB_COMM_EXEC__A, VSB_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn static short get_vsb_post_rs_pck_err(struct i2c_device_addr *dev_addr, u16 *PckErrs)
- * \brief Get the values of packet error in 8VSB mode
- * \return Error code
- */
- static int get_vsb_post_rs_pck_err(struct i2c_device_addr *dev_addr,
- u32 *pck_errs, u32 *pck_count)
- {
- int rc;
- u16 data = 0;
- u16 period = 0;
- u16 prescale = 0;
- u16 packet_errors_mant = 0;
- u16 packet_errors_exp = 0;
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_FAILURES__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- packet_errors_mant = data & FEC_RS_NR_FAILURES_FIXED_MANT__M;
- packet_errors_exp = (data & FEC_RS_NR_FAILURES_EXP__M)
- >> FEC_RS_NR_FAILURES_EXP__B;
- period = FEC_RS_MEASUREMENT_PERIOD;
- prescale = FEC_RS_MEASUREMENT_PRESCALE;
- /* packet error rate = (error packet number) per second */
- /* 77.3 us is time for per packet */
- if (period * prescale == 0) {
- pr_err("error: period and/or prescale is zero!\n");
- return -EIO;
- }
- *pck_errs = packet_errors_mant * (1 << packet_errors_exp);
- *pck_count = period * prescale * 77;
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn static short GetVSBBer(struct i2c_device_addr *dev_addr, u32 *ber)
- * \brief Get the values of ber in VSB mode
- * \return Error code
- */
- static int get_vs_bpost_viterbi_ber(struct i2c_device_addr *dev_addr,
- u32 *ber, u32 *cnt)
- {
- int rc;
- u16 data = 0;
- u16 period = 0;
- u16 prescale = 0;
- u16 bit_errors_mant = 0;
- u16 bit_errors_exp = 0;
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_BIT_ERRORS__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- period = FEC_RS_MEASUREMENT_PERIOD;
- prescale = FEC_RS_MEASUREMENT_PRESCALE;
- bit_errors_mant = data & FEC_RS_NR_BIT_ERRORS_FIXED_MANT__M;
- bit_errors_exp = (data & FEC_RS_NR_BIT_ERRORS_EXP__M)
- >> FEC_RS_NR_BIT_ERRORS_EXP__B;
- *cnt = period * prescale * 207 * ((bit_errors_exp > 2) ? 1 : 8);
- if (((bit_errors_mant << bit_errors_exp) >> 3) > 68700)
- *ber = (*cnt) * 26570;
- else {
- if (period * prescale == 0) {
- pr_err("error: period and/or prescale is zero!\n");
- return -EIO;
- }
- *ber = bit_errors_mant << ((bit_errors_exp > 2) ?
- (bit_errors_exp - 3) : bit_errors_exp);
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn static short get_vs_bpre_viterbi_ber(struct i2c_device_addr *dev_addr, u32 *ber)
- * \brief Get the values of ber in VSB mode
- * \return Error code
- */
- static int get_vs_bpre_viterbi_ber(struct i2c_device_addr *dev_addr,
- u32 *ber, u32 *cnt)
- {
- u16 data = 0;
- int rc;
- rc = drxj_dap_read_reg16(dev_addr, VSB_TOP_NR_SYM_ERRS__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- return -EIO;
- }
- *ber = data;
- *cnt = VSB_TOP_MEASUREMENT_PERIOD * SYMBOLS_PER_SEGMENT;
- return 0;
- }
- /*
- * \fn static int get_vsbmer(struct i2c_device_addr *dev_addr, u16 *mer)
- * \brief Get the values of MER
- * \return Error code
- */
- static int get_vsbmer(struct i2c_device_addr *dev_addr, u16 *mer)
- {
- int rc;
- u16 data_hi = 0;
- rc = drxj_dap_read_reg16(dev_addr, VSB_TOP_ERR_ENERGY_H__A, &data_hi, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- *mer =
- (u16) (log1_times100(21504) - log1_times100((data_hi << 6) / 52));
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== END 8VSB DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== QAM DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- * \fn int power_down_qam ()
- * \brief Powr down QAM related blocks.
- * \param demod instance of demodulator.
- * \param channel pointer to channel data.
- * \return int.
- */
- static int power_down_qam(struct drx_demod_instance *demod, bool primary)
- {
- struct drxjscu_cmd cmd_scu = { /* command */ 0,
- /* parameter_len */ 0,
- /* result_len */ 0,
- /* *parameter */ NULL,
- /* *result */ NULL
- };
- int rc;
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drx_cfg_mpeg_output cfg_mpeg_output;
- struct drx_common_attr *common_attr = demod->my_common_attr;
- u16 cmd_result = 0;
- /*
- STOP demodulator
- resets IQM, QAM and FEC HW blocks
- */
- /* stop all comm_exec */
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_STOP;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (primary) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_iqm_af(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
- cfg_mpeg_output.enable_mpeg_output = false;
- rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam_measurement ()
- * \brief Setup of the QAM Measuremnt intervals for signal quality
- * \param demod instance of demod.
- * \param constellation current constellation.
- * \return int.
- *
- * NOTE:
- * Take into account that for certain settings the errorcounters can overflow.
- * The implementation does not check this.
- *
- * TODO: overriding the ext_attr->fec_bits_desired by constellation dependent
- * constants to get a measurement period of approx. 1 sec. Remove fec_bits_desired
- * field ?
- *
- */
- #ifndef DRXJ_VSB_ONLY
- static int
- set_qam_measurement(struct drx_demod_instance *demod,
- enum drx_modulation constellation, u32 symbol_rate)
- {
- struct i2c_device_addr *dev_addr = NULL; /* device address for I2C writes */
- struct drxj_data *ext_attr = NULL; /* Global data container for DRXJ specific data */
- int rc;
- u32 fec_bits_desired = 0; /* BER accounting period */
- u16 fec_rs_plen = 0; /* defines RS BER measurement period */
- u16 fec_rs_prescale = 0; /* ReedSolomon Measurement Prescale */
- u32 fec_rs_period = 0; /* Value for corresponding I2C register */
- u32 fec_rs_bit_cnt = 0; /* Actual precise amount of bits */
- u32 fec_oc_snc_fail_period = 0; /* Value for corresponding I2C register */
- u32 qam_vd_period = 0; /* Value for corresponding I2C register */
- u32 qam_vd_bit_cnt = 0; /* Actual precise amount of bits */
- u16 fec_vd_plen = 0; /* no of trellis symbols: VD SER measur period */
- u16 qam_vd_prescale = 0; /* Viterbi Measurement Prescale */
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- fec_bits_desired = ext_attr->fec_bits_desired;
- fec_rs_prescale = ext_attr->fec_rs_prescale;
- switch (constellation) {
- case DRX_CONSTELLATION_QAM16:
- fec_bits_desired = 4 * symbol_rate;
- break;
- case DRX_CONSTELLATION_QAM32:
- fec_bits_desired = 5 * symbol_rate;
- break;
- case DRX_CONSTELLATION_QAM64:
- fec_bits_desired = 6 * symbol_rate;
- break;
- case DRX_CONSTELLATION_QAM128:
- fec_bits_desired = 7 * symbol_rate;
- break;
- case DRX_CONSTELLATION_QAM256:
- fec_bits_desired = 8 * symbol_rate;
- break;
- default:
- return -EINVAL;
- }
- /* Parameters for Reed-Solomon Decoder */
- /* fecrs_period = (int)ceil(FEC_BITS_DESIRED/(fecrs_prescale*plen)) */
- /* rs_bit_cnt = fecrs_period*fecrs_prescale*plen */
- /* result is within 32 bit arithmetic -> */
- /* no need for mult or frac functions */
- /* TODO: use constant instead of calculation and remove the fec_rs_plen in ext_attr */
- switch (ext_attr->standard) {
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- fec_rs_plen = 204 * 8;
- break;
- case DRX_STANDARD_ITU_B:
- fec_rs_plen = 128 * 7;
- break;
- default:
- return -EINVAL;
- }
- ext_attr->fec_rs_plen = fec_rs_plen; /* for getSigQual */
- fec_rs_bit_cnt = fec_rs_prescale * fec_rs_plen; /* temp storage */
- if (fec_rs_bit_cnt == 0) {
- pr_err("error: fec_rs_bit_cnt is zero!\n");
- return -EIO;
- }
- fec_rs_period = fec_bits_desired / fec_rs_bit_cnt + 1; /* ceil */
- if (ext_attr->standard != DRX_STANDARD_ITU_B)
- fec_oc_snc_fail_period = fec_rs_period;
- /* limit to max 16 bit value (I2C register width) if needed */
- if (fec_rs_period > 0xFFFF)
- fec_rs_period = 0xFFFF;
- /* write corresponding registers */
- switch (ext_attr->standard) {
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_C:
- break;
- case DRX_STANDARD_ITU_B:
- switch (constellation) {
- case DRX_CONSTELLATION_QAM64:
- fec_rs_period = 31581;
- fec_oc_snc_fail_period = 17932;
- break;
- case DRX_CONSTELLATION_QAM256:
- fec_rs_period = 45446;
- fec_oc_snc_fail_period = 25805;
- break;
- default:
- return -EINVAL;
- }
- break;
- default:
- return -EINVAL;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_OC_SNC_FAIL_PERIOD__A, (u16)fec_oc_snc_fail_period, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PERIOD__A, (u16)fec_rs_period, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_RS_MEASUREMENT_PRESCALE__A, fec_rs_prescale, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->fec_rs_period = (u16) fec_rs_period;
- ext_attr->fec_rs_prescale = fec_rs_prescale;
- rc = drxdap_fasi_write_reg32(dev_addr, SCU_RAM_FEC_ACCUM_CW_CORRECTED_LO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_MEAS_COUNT__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_FEC_ACCUM_PKT_FAILURES__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- /* Parameters for Viterbi Decoder */
- /* qamvd_period = (int)ceil(FEC_BITS_DESIRED/ */
- /* (qamvd_prescale*plen*(qam_constellation+1))) */
- /* vd_bit_cnt = qamvd_period*qamvd_prescale*plen */
- /* result is within 32 bit arithmetic -> */
- /* no need for mult or frac functions */
- /* a(8 bit) * b(8 bit) = 16 bit result => mult32 not needed */
- fec_vd_plen = ext_attr->fec_vd_plen;
- qam_vd_prescale = ext_attr->qam_vd_prescale;
- qam_vd_bit_cnt = qam_vd_prescale * fec_vd_plen; /* temp storage */
- switch (constellation) {
- case DRX_CONSTELLATION_QAM64:
- /* a(16 bit) * b(4 bit) = 20 bit result => mult32 not needed */
- qam_vd_period =
- qam_vd_bit_cnt * (QAM_TOP_CONSTELLATION_QAM64 + 1)
- * (QAM_TOP_CONSTELLATION_QAM64 + 1);
- break;
- case DRX_CONSTELLATION_QAM256:
- /* a(16 bit) * b(5 bit) = 21 bit result => mult32 not needed */
- qam_vd_period =
- qam_vd_bit_cnt * (QAM_TOP_CONSTELLATION_QAM256 + 1)
- * (QAM_TOP_CONSTELLATION_QAM256 + 1);
- break;
- default:
- return -EINVAL;
- }
- if (qam_vd_period == 0) {
- pr_err("error: qam_vd_period is zero!\n");
- return -EIO;
- }
- qam_vd_period = fec_bits_desired / qam_vd_period;
- /* limit to max 16 bit value (I2C register width) if needed */
- if (qam_vd_period > 0xFFFF)
- qam_vd_period = 0xFFFF;
- /* a(16 bit) * b(16 bit) = 32 bit result => mult32 not needed */
- qam_vd_bit_cnt *= qam_vd_period;
- rc = drxj_dap_write_reg16(dev_addr, QAM_VD_MEASUREMENT_PERIOD__A, (u16)qam_vd_period, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_VD_MEASUREMENT_PRESCALE__A, qam_vd_prescale, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->qam_vd_period = (u16) qam_vd_period;
- ext_attr->qam_vd_prescale = qam_vd_prescale;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam16 ()
- * \brief QAM16 specific setup
- * \param demod instance of demod.
- * \return int.
- */
- static int set_qam16(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- static const u8 qam_dq_qual_fun[] = {
- DRXJ_16TO8(2), /* fun0 */
- DRXJ_16TO8(2), /* fun1 */
- DRXJ_16TO8(2), /* fun2 */
- DRXJ_16TO8(2), /* fun3 */
- DRXJ_16TO8(3), /* fun4 */
- DRXJ_16TO8(3), /* fun5 */
- };
- static const u8 qam_eq_cma_rad[] = {
- DRXJ_16TO8(13517), /* RAD0 */
- DRXJ_16TO8(13517), /* RAD1 */
- DRXJ_16TO8(13517), /* RAD2 */
- DRXJ_16TO8(13517), /* RAD3 */
- DRXJ_16TO8(13517), /* RAD4 */
- DRXJ_16TO8(13517), /* RAD5 */
- };
- rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 140, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 120, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 230, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 95, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 105, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 56, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 220, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 25, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 6, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-24), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-65), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-127), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 20, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 240, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 40960, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam32 ()
- * \brief QAM32 specific setup
- * \param demod instance of demod.
- * \return int.
- */
- static int set_qam32(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- static const u8 qam_dq_qual_fun[] = {
- DRXJ_16TO8(3), /* fun0 */
- DRXJ_16TO8(3), /* fun1 */
- DRXJ_16TO8(3), /* fun2 */
- DRXJ_16TO8(3), /* fun3 */
- DRXJ_16TO8(4), /* fun4 */
- DRXJ_16TO8(4), /* fun5 */
- };
- static const u8 qam_eq_cma_rad[] = {
- DRXJ_16TO8(6707), /* RAD0 */
- DRXJ_16TO8(6707), /* RAD1 */
- DRXJ_16TO8(6707), /* RAD2 */
- DRXJ_16TO8(6707), /* RAD3 */
- DRXJ_16TO8(6707), /* RAD4 */
- DRXJ_16TO8(6707), /* RAD5 */
- };
- rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 90, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 170, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 56, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 140, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, (u16)(-16), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-26), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-56), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-86), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 20, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 176, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 20480, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam64 ()
- * \brief QAM64 specific setup
- * \param demod instance of demod.
- * \return int.
- */
- static int set_qam64(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- static const u8 qam_dq_qual_fun[] = {
- /* this is hw reset value. no necessary to re-write */
- DRXJ_16TO8(4), /* fun0 */
- DRXJ_16TO8(4), /* fun1 */
- DRXJ_16TO8(4), /* fun2 */
- DRXJ_16TO8(4), /* fun3 */
- DRXJ_16TO8(6), /* fun4 */
- DRXJ_16TO8(6), /* fun5 */
- };
- static const u8 qam_eq_cma_rad[] = {
- DRXJ_16TO8(13336), /* RAD0 */
- DRXJ_16TO8(12618), /* RAD1 */
- DRXJ_16TO8(11988), /* RAD2 */
- DRXJ_16TO8(13809), /* RAD3 */
- DRXJ_16TO8(13809), /* RAD4 */
- DRXJ_16TO8(15609), /* RAD5 */
- };
- rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 105, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 195, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 84, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 141, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 7, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-15), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, (u16)(-45), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-80), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 30, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 48, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 160, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 43008, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam128 ()
- * \brief QAM128 specific setup
- * \param demod: instance of demod.
- * \return int.
- */
- static int set_qam128(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- static const u8 qam_dq_qual_fun[] = {
- DRXJ_16TO8(6), /* fun0 */
- DRXJ_16TO8(6), /* fun1 */
- DRXJ_16TO8(6), /* fun2 */
- DRXJ_16TO8(6), /* fun3 */
- DRXJ_16TO8(9), /* fun4 */
- DRXJ_16TO8(9), /* fun5 */
- };
- static const u8 qam_eq_cma_rad[] = {
- DRXJ_16TO8(6164), /* RAD0 */
- DRXJ_16TO8(6598), /* RAD1 */
- DRXJ_16TO8(6394), /* RAD2 */
- DRXJ_16TO8(6409), /* RAD3 */
- DRXJ_16TO8(6656), /* RAD4 */
- DRXJ_16TO8(7238), /* RAD5 */
- };
- rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 140, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 65, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, (u16)(-1), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-23), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 20, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 32, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 144, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 20992, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int set_qam256 ()
- * \brief QAM256 specific setup
- * \param demod: instance of demod.
- * \return int.
- */
- static int set_qam256(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- static const u8 qam_dq_qual_fun[] = {
- DRXJ_16TO8(8), /* fun0 */
- DRXJ_16TO8(8), /* fun1 */
- DRXJ_16TO8(8), /* fun2 */
- DRXJ_16TO8(8), /* fun3 */
- DRXJ_16TO8(12), /* fun4 */
- DRXJ_16TO8(12), /* fun5 */
- };
- static const u8 qam_eq_cma_rad[] = {
- DRXJ_16TO8(12345), /* RAD0 */
- DRXJ_16TO8(12345), /* RAD1 */
- DRXJ_16TO8(13626), /* RAD2 */
- DRXJ_16TO8(12931), /* RAD3 */
- DRXJ_16TO8(14719), /* RAD4 */
- DRXJ_16TO8(15356), /* RAD5 */
- };
- rc = drxdap_fasi_write_block(dev_addr, QAM_DQ_QUAL_FUN0__A, sizeof(qam_dq_qual_fun), ((u8 *)qam_dq_qual_fun), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, SCU_RAM_QAM_EQ_CMA_RAD0__A, sizeof(qam_eq_cma_rad), ((u8 *)qam_eq_cma_rad), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RTH__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FTH__A, 60, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_PTH__A, 100, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_QTH__A, 150, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_CTH__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MTH__A, 110, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RATE_LIM__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_FREQ_LIM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_COUNT_LIM__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_MEDIAN_AV_MULT__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_RADIUS_AV_LIMIT__A, 74, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET1__A, 18, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET2__A, 13, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET3__A, 7, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET4__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_LCAVG_OFFSET5__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_FINE__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CA_COARSE__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_MEDIUM__A, 50, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CP_COARSE__A, 255, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_FINE__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_MEDIUM__A, 25, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CI_COARSE__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_MEDIUM__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EP_COARSE__A, 24, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_FINE__A, 12, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_MEDIUM__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_EI_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_FINE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_MEDIUM__A, 48, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF_COARSE__A, 80, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_FINE__A, 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_MEDIUM__A, 15, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_LC_CF1_COARSE__A, 16, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_SL_SIG_POWER__A, 43520, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- #define QAM_SET_OP_ALL 0x1
- #define QAM_SET_OP_CONSTELLATION 0x2
- #define QAM_SET_OP_SPECTRUM 0X4
- /*
- * \fn int set_qam ()
- * \brief Set QAM demod.
- * \param demod: instance of demod.
- * \param channel: pointer to channel data.
- * \return int.
- */
- static int
- set_qam(struct drx_demod_instance *demod,
- struct drx_channel *channel, s32 tuner_freq_offset, u32 op)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- struct drx_common_attr *common_attr = NULL;
- int rc;
- u32 adc_frequency = 0;
- u32 iqm_rc_rate = 0;
- u16 cmd_result = 0;
- u16 lc_symbol_freq = 0;
- u16 iqm_rc_stretch = 0;
- u16 set_env_parameters = 0;
- u16 set_param_parameters[2] = { 0 };
- struct drxjscu_cmd cmd_scu = { /* command */ 0,
- /* parameter_len */ 0,
- /* result_len */ 0,
- /* parameter */ NULL,
- /* result */ NULL
- };
- static const u8 qam_a_taps[] = {
- DRXJ_16TO8(-1), /* re0 */
- DRXJ_16TO8(1), /* re1 */
- DRXJ_16TO8(1), /* re2 */
- DRXJ_16TO8(-1), /* re3 */
- DRXJ_16TO8(-1), /* re4 */
- DRXJ_16TO8(2), /* re5 */
- DRXJ_16TO8(1), /* re6 */
- DRXJ_16TO8(-2), /* re7 */
- DRXJ_16TO8(0), /* re8 */
- DRXJ_16TO8(3), /* re9 */
- DRXJ_16TO8(-1), /* re10 */
- DRXJ_16TO8(-3), /* re11 */
- DRXJ_16TO8(4), /* re12 */
- DRXJ_16TO8(1), /* re13 */
- DRXJ_16TO8(-8), /* re14 */
- DRXJ_16TO8(4), /* re15 */
- DRXJ_16TO8(13), /* re16 */
- DRXJ_16TO8(-13), /* re17 */
- DRXJ_16TO8(-19), /* re18 */
- DRXJ_16TO8(28), /* re19 */
- DRXJ_16TO8(25), /* re20 */
- DRXJ_16TO8(-53), /* re21 */
- DRXJ_16TO8(-31), /* re22 */
- DRXJ_16TO8(96), /* re23 */
- DRXJ_16TO8(37), /* re24 */
- DRXJ_16TO8(-190), /* re25 */
- DRXJ_16TO8(-40), /* re26 */
- DRXJ_16TO8(619) /* re27 */
- };
- static const u8 qam_b64_taps[] = {
- DRXJ_16TO8(0), /* re0 */
- DRXJ_16TO8(-2), /* re1 */
- DRXJ_16TO8(1), /* re2 */
- DRXJ_16TO8(2), /* re3 */
- DRXJ_16TO8(-2), /* re4 */
- DRXJ_16TO8(0), /* re5 */
- DRXJ_16TO8(4), /* re6 */
- DRXJ_16TO8(-2), /* re7 */
- DRXJ_16TO8(-4), /* re8 */
- DRXJ_16TO8(4), /* re9 */
- DRXJ_16TO8(3), /* re10 */
- DRXJ_16TO8(-6), /* re11 */
- DRXJ_16TO8(0), /* re12 */
- DRXJ_16TO8(6), /* re13 */
- DRXJ_16TO8(-5), /* re14 */
- DRXJ_16TO8(-3), /* re15 */
- DRXJ_16TO8(11), /* re16 */
- DRXJ_16TO8(-4), /* re17 */
- DRXJ_16TO8(-19), /* re18 */
- DRXJ_16TO8(19), /* re19 */
- DRXJ_16TO8(28), /* re20 */
- DRXJ_16TO8(-45), /* re21 */
- DRXJ_16TO8(-36), /* re22 */
- DRXJ_16TO8(90), /* re23 */
- DRXJ_16TO8(42), /* re24 */
- DRXJ_16TO8(-185), /* re25 */
- DRXJ_16TO8(-46), /* re26 */
- DRXJ_16TO8(614) /* re27 */
- };
- static const u8 qam_b256_taps[] = {
- DRXJ_16TO8(-2), /* re0 */
- DRXJ_16TO8(4), /* re1 */
- DRXJ_16TO8(1), /* re2 */
- DRXJ_16TO8(-4), /* re3 */
- DRXJ_16TO8(0), /* re4 */
- DRXJ_16TO8(4), /* re5 */
- DRXJ_16TO8(-2), /* re6 */
- DRXJ_16TO8(-4), /* re7 */
- DRXJ_16TO8(5), /* re8 */
- DRXJ_16TO8(2), /* re9 */
- DRXJ_16TO8(-8), /* re10 */
- DRXJ_16TO8(2), /* re11 */
- DRXJ_16TO8(11), /* re12 */
- DRXJ_16TO8(-8), /* re13 */
- DRXJ_16TO8(-15), /* re14 */
- DRXJ_16TO8(16), /* re15 */
- DRXJ_16TO8(19), /* re16 */
- DRXJ_16TO8(-27), /* re17 */
- DRXJ_16TO8(-22), /* re18 */
- DRXJ_16TO8(44), /* re19 */
- DRXJ_16TO8(26), /* re20 */
- DRXJ_16TO8(-69), /* re21 */
- DRXJ_16TO8(-28), /* re22 */
- DRXJ_16TO8(110), /* re23 */
- DRXJ_16TO8(31), /* re24 */
- DRXJ_16TO8(-201), /* re25 */
- DRXJ_16TO8(-32), /* re26 */
- DRXJ_16TO8(628) /* re27 */
- };
- static const u8 qam_c_taps[] = {
- DRXJ_16TO8(-3), /* re0 */
- DRXJ_16TO8(3), /* re1 */
- DRXJ_16TO8(2), /* re2 */
- DRXJ_16TO8(-4), /* re3 */
- DRXJ_16TO8(0), /* re4 */
- DRXJ_16TO8(4), /* re5 */
- DRXJ_16TO8(-1), /* re6 */
- DRXJ_16TO8(-4), /* re7 */
- DRXJ_16TO8(3), /* re8 */
- DRXJ_16TO8(3), /* re9 */
- DRXJ_16TO8(-5), /* re10 */
- DRXJ_16TO8(0), /* re11 */
- DRXJ_16TO8(9), /* re12 */
- DRXJ_16TO8(-4), /* re13 */
- DRXJ_16TO8(-12), /* re14 */
- DRXJ_16TO8(10), /* re15 */
- DRXJ_16TO8(16), /* re16 */
- DRXJ_16TO8(-21), /* re17 */
- DRXJ_16TO8(-20), /* re18 */
- DRXJ_16TO8(37), /* re19 */
- DRXJ_16TO8(25), /* re20 */
- DRXJ_16TO8(-62), /* re21 */
- DRXJ_16TO8(-28), /* re22 */
- DRXJ_16TO8(105), /* re23 */
- DRXJ_16TO8(31), /* re24 */
- DRXJ_16TO8(-197), /* re25 */
- DRXJ_16TO8(-33), /* re26 */
- DRXJ_16TO8(626) /* re27 */
- };
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
- if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM256:
- iqm_rc_rate = 0x00AE3562;
- lc_symbol_freq =
- QAM_LC_SYMBOL_FREQ_FREQ_QAM_B_256;
- channel->symbolrate = 5360537;
- iqm_rc_stretch = IQM_RC_STRETCH_QAM_B_256;
- break;
- case DRX_CONSTELLATION_QAM64:
- iqm_rc_rate = 0x00C05A0E;
- lc_symbol_freq = 409;
- channel->symbolrate = 5056941;
- iqm_rc_stretch = IQM_RC_STRETCH_QAM_B_64;
- break;
- default:
- return -EINVAL;
- }
- } else {
- adc_frequency = (common_attr->sys_clock_freq * 1000) / 3;
- if (channel->symbolrate == 0) {
- pr_err("error: channel symbolrate is zero!\n");
- return -EIO;
- }
- iqm_rc_rate =
- (adc_frequency / channel->symbolrate) * (1 << 21) +
- (frac28
- ((adc_frequency % channel->symbolrate),
- channel->symbolrate) >> 7) - (1 << 23);
- lc_symbol_freq =
- (u16) (frac28
- (channel->symbolrate +
- (adc_frequency >> 13),
- adc_frequency) >> 16);
- if (lc_symbol_freq > 511)
- lc_symbol_freq = 511;
- iqm_rc_stretch = 21;
- }
- if (ext_attr->standard == DRX_STANDARD_ITU_A) {
- set_env_parameters = QAM_TOP_ANNEX_A; /* annex */
- set_param_parameters[0] = channel->constellation; /* constellation */
- set_param_parameters[1] = DRX_INTERLEAVEMODE_I12_J17; /* interleave mode */
- } else if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- set_env_parameters = QAM_TOP_ANNEX_B; /* annex */
- set_param_parameters[0] = channel->constellation; /* constellation */
- set_param_parameters[1] = channel->interleavemode; /* interleave mode */
- } else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
- set_env_parameters = QAM_TOP_ANNEX_C; /* annex */
- set_param_parameters[0] = channel->constellation; /* constellation */
- set_param_parameters[1] = DRX_INTERLEAVEMODE_I12_J17; /* interleave mode */
- } else {
- return -EINVAL;
- }
- }
- if (op & QAM_SET_OP_ALL) {
- /*
- STEP 1: reset demodulator
- resets IQM, QAM and FEC HW blocks
- resets SCU variables
- */
- /* stop all comm_exec */
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_RESET;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
- /*
- STEP 2: configure demodulator
- -set env
- -set params (resets IQM,QAM,FEC HW; initializes some SCU variables )
- */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_SET_ENV;
- cmd_scu.parameter_len = 1;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = &set_env_parameters;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_SET_PARAM;
- cmd_scu.parameter_len = 2;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = set_param_parameters;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* set symbol rate */
- rc = drxdap_fasi_write_reg32(dev_addr, IQM_RC_RATE_OFS_LO__A, iqm_rc_rate, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->iqm_rc_rate_ofs = iqm_rc_rate;
- rc = set_qam_measurement(demod, channel->constellation, channel->symbolrate);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* STEP 3: enable the system in a mode where the ADC provides valid signal
- setup constellation independent registers */
- /* from qam_cmd.py script (qam_driver_b) */
- /* TODO: remove re-writes of HW reset values */
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_SPECTRUM)) {
- rc = set_frequency(demod, channel, tuner_freq_offset);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_SYMBOL_FREQ__A, lc_symbol_freq, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_STRETCH__A, iqm_rc_stretch, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- if (op & QAM_SET_OP_ALL) {
- if (!ext_attr->has_lna) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_AMUX__A, 0x02, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SYMMETRIC__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_MIDTAP__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_OUT_ENA__A, IQM_CF_OUT_ENA_QAM__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_WR_RSV_0__A, 0x5f, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* scu temporary shut down agc */
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SYNC_SEL__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_LEN__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_CLP_TH__A, 448, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_SNS_LEN__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PDREF__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_STDBY__A, 0x10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PGA_GAIN__A, 11, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, IQM_CF_SCALE_SH__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_TIMEOUT__A, QAM_SY_TIMEOUT__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, QAM_SY_SYNC_LWM__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, QAM_SY_SYNC_AWM__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, QAM_SY_SYNC_HWM__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- } else {
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM16:
- case DRX_CONSTELLATION_QAM64:
- case DRX_CONSTELLATION_QAM256:
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, 0x03, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, 0x04, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, QAM_SY_SYNC_HWM__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- break;
- case DRX_CONSTELLATION_QAM32:
- case DRX_CONSTELLATION_QAM128:
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_LWM__A, 0x03, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_AWM__A, 0x05, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_SY_SYNC_HWM__A, 0x06, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EIO;
- } /* switch */
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_MODE__A, QAM_LC_MODE__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /*! reset default val ! */
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_RATE_LIMIT__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_LPF_FACTORP__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_LPF_FACTORI__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_MODE__A, 7, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB0__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB1__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB2__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB3__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB4__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB5__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB6__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB8__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB9__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB10__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB12__A, 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB15__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB16__A, 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB20__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_QUAL_TAB25__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_ADJ_SEL__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_ADJ_SEL__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_ADJ_SEL__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_POW_MEAS_LEN__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_GPIO__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* No more resets of the IQM, current standard correctly set =>
- now AGCs can be configured. */
- /* turn on IQMAF. It has to be in front of setAgc**() */
- rc = set_iqm_af(demod, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = adc_synchronization(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = init_agc(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_agc_if(demod, &(ext_attr->qam_if_agc_cfg), false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_agc_rf(demod, &(ext_attr->qam_rf_agc_cfg), false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- {
- /* TODO fix this, store a struct drxj_cfg_afe_gain structure in struct drxj_data instead
- of only the gain */
- struct drxj_cfg_afe_gain qam_pga_cfg = { DRX_STANDARD_ITU_B, 0 };
- qam_pga_cfg.gain = ext_attr->qam_pga_cfg;
- rc = ctrl_set_cfg_afe_gain(demod, &qam_pga_cfg);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = ctrl_set_cfg_pre_saw(demod, &(ext_attr->qam_pre_saw_cfg));
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
- if (ext_attr->standard == DRX_STANDARD_ITU_A) {
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_a_taps), ((u8 *)qam_a_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_a_taps), ((u8 *)qam_a_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM64:
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_b64_taps), ((u8 *)qam_b64_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_b64_taps), ((u8 *)qam_b64_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_QAM256:
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_b256_taps), ((u8 *)qam_b256_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_b256_taps), ((u8 *)qam_b256_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EIO;
- }
- } else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_RE0__A, sizeof(qam_c_taps), ((u8 *)qam_c_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxdap_fasi_write_block(dev_addr, IQM_CF_TAP_IM0__A, sizeof(qam_c_taps), ((u8 *)qam_c_taps), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* SETP 4: constellation specific setup */
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM16:
- rc = set_qam16(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_QAM32:
- rc = set_qam32(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_QAM64:
- rc = set_qam64(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_QAM128:
- rc = set_qam128(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_QAM256:
- rc = set_qam256(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- return -EIO;
- } /* switch */
- }
- if ((op & QAM_SET_OP_ALL)) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_SCALE_SH__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Mpeg output has to be in front of FEC active */
- rc = set_mpegtei_handling(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = bit_reverse_mpeg_output(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_mpeg_start_width(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- {
- /* TODO: move to set_standard after hardware reset value problem is solved */
- /* Configure initial MPEG output */
- struct drx_cfg_mpeg_output cfg_mpeg_output;
- memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
- cfg_mpeg_output.enable_mpeg_output = true;
- rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- }
- if ((op & QAM_SET_OP_ALL) || (op & QAM_SET_OP_CONSTELLATION)) {
- /* STEP 5: start QAM demodulator (starts FEC, QAM and IQM HW) */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_START;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_COMM_EXEC__A, QAM_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, FEC_COMM_EXEC__A, FEC_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- static int ctrl_get_qam_sig_quality(struct drx_demod_instance *demod);
- static int qam_flip_spec(struct drx_demod_instance *demod, struct drx_channel *channel)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxj_data *ext_attr = demod->my_ext_attr;
- int rc;
- u32 iqm_fs_rate_ofs = 0;
- u32 iqm_fs_rate_lo = 0;
- u16 qam_ctl_ena = 0;
- u16 data = 0;
- u16 equ_mode = 0;
- u16 fsm_state = 0;
- int i = 0;
- int ofsofs = 0;
- /* Silence the controlling of lc, equ, and the acquisition state machine */
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, &qam_ctl_ena, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, qam_ctl_ena & ~(SCU_RAM_QAM_CTL_ENA_ACQ__M | SCU_RAM_QAM_CTL_ENA_EQU__M | SCU_RAM_QAM_CTL_ENA_LC__M), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* freeze the frequency control loop */
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CF__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CF1__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_atomic_read_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, &iqm_fs_rate_ofs, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_atomic_read_reg32(dev_addr, IQM_FS_RATE_LO__A, &iqm_fs_rate_lo, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ofsofs = iqm_fs_rate_lo - iqm_fs_rate_ofs;
- iqm_fs_rate_ofs = ~iqm_fs_rate_ofs + 1;
- iqm_fs_rate_ofs -= 2 * ofsofs;
- /* freeze dq/fq updating */
- rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_MODE__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- data = (data & 0xfff9);
- rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* lc_cp / _ci / _ca */
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_CI__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_LC_EP__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_LA_FACTOR__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* flip the spec */
- rc = drxdap_fasi_write_reg32(dev_addr, IQM_FS_RATE_OFS_LO__A, iqm_fs_rate_ofs, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->iqm_fs_rate_ofs = iqm_fs_rate_ofs;
- ext_attr->pos_image = (ext_attr->pos_image) ? false : true;
- /* freeze dq/fq updating */
- rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_MODE__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- equ_mode = data;
- data = (data & 0xfff9);
- rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- for (i = 0; i < 28; i++) {
- rc = drxj_dap_read_reg16(dev_addr, QAM_DQ_TAP_IM_EL0__A + (2 * i), &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_TAP_IM_EL0__A + (2 * i), -data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- for (i = 0; i < 24; i++) {
- rc = drxj_dap_read_reg16(dev_addr, QAM_FQ_TAP_IM_EL0__A + (2 * i), &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_TAP_IM_EL0__A + (2 * i), -data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- data = equ_mode;
- rc = drxj_dap_write_reg16(dev_addr, QAM_DQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, QAM_FQ_MODE__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_FSM_STATE_TGT__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- i = 0;
- while ((fsm_state != 4) && (i++ < 100)) {
- rc = drxj_dap_read_reg16(dev_addr, SCU_RAM_QAM_FSM_STATE__A, &fsm_state, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_QAM_CTL_ENA__A, (qam_ctl_ena | 0x0016), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- #define NO_LOCK 0x0
- #define DEMOD_LOCKED 0x1
- #define SYNC_FLIPPED 0x2
- #define SPEC_MIRRORED 0x4
- /*
- * \fn int qam64auto ()
- * \brief auto do sync pattern switching and mirroring.
- * \param demod: instance of demod.
- * \param channel: pointer to channel data.
- * \param tuner_freq_offset: tuner frequency offset.
- * \param lock_status: pointer to lock status.
- * \return int.
- */
- static int
- qam64auto(struct drx_demod_instance *demod,
- struct drx_channel *channel,
- s32 tuner_freq_offset, enum drx_lock_status *lock_status)
- {
- struct drxj_data *ext_attr = demod->my_ext_attr;
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drx39xxj_state *state = dev_addr->user_data;
- struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
- int rc;
- u32 lck_state = NO_LOCK;
- u32 start_time = 0;
- u32 d_locked_time = 0;
- u32 timeout_ofs = 0;
- u16 data = 0;
- /* external attributes for storing acquired channel constellation */
- *lock_status = DRX_NOT_LOCKED;
- start_time = jiffies_to_msecs(jiffies);
- lck_state = NO_LOCK;
- do {
- rc = ctrl_lock_status(demod, lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch (lck_state) {
- case NO_LOCK:
- if (*lock_status == DRXJ_DEMOD_LOCK) {
- rc = ctrl_get_qam_sig_quality(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (p->cnr.stat[0].svalue > 20800) {
- lck_state = DEMOD_LOCKED;
- /* some delay to see if fec_lock possible TODO find the right value */
- timeout_ofs += DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME; /* see something, waiting longer */
- d_locked_time = jiffies_to_msecs(jiffies);
- }
- }
- break;
- case DEMOD_LOCKED:
- if ((*lock_status == DRXJ_DEMOD_LOCK) && /* still demod_lock in 150ms */
- ((jiffies_to_msecs(jiffies) - d_locked_time) >
- DRXJ_QAM_FEC_LOCK_WAITTIME)) {
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data | 0x1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- lck_state = SYNC_FLIPPED;
- msleep(10);
- }
- break;
- case SYNC_FLIPPED:
- if (*lock_status == DRXJ_DEMOD_LOCK) {
- if (channel->mirror == DRX_MIRROR_AUTO) {
- /* flip sync pattern back */
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data & 0xFFFE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* flip spectrum */
- ext_attr->mirror = DRX_MIRROR_YES;
- rc = qam_flip_spec(demod, channel);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- lck_state = SPEC_MIRRORED;
- /* reset timer TODO: still need 500ms? */
- start_time = d_locked_time =
- jiffies_to_msecs(jiffies);
- timeout_ofs = 0;
- } else { /* no need to wait lock */
- start_time =
- jiffies_to_msecs(jiffies) -
- DRXJ_QAM_MAX_WAITTIME - timeout_ofs;
- }
- }
- break;
- case SPEC_MIRRORED:
- if ((*lock_status == DRXJ_DEMOD_LOCK) && /* still demod_lock in 150ms */
- ((jiffies_to_msecs(jiffies) - d_locked_time) >
- DRXJ_QAM_FEC_LOCK_WAITTIME)) {
- rc = ctrl_get_qam_sig_quality(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (p->cnr.stat[0].svalue > 20800) {
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr, QAM_SY_TIMEOUT__A, data | 0x1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no need to wait lock */
- start_time =
- jiffies_to_msecs(jiffies) -
- DRXJ_QAM_MAX_WAITTIME - timeout_ofs;
- }
- }
- break;
- default:
- break;
- }
- msleep(10);
- } while
- ((*lock_status != DRX_LOCKED) &&
- (*lock_status != DRX_NEVER_LOCK) &&
- ((jiffies_to_msecs(jiffies) - start_time) <
- (DRXJ_QAM_MAX_WAITTIME + timeout_ofs))
- );
- /* Returning control to apllication ... */
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int qam256auto ()
- * \brief auto do sync pattern switching and mirroring.
- * \param demod: instance of demod.
- * \param channel: pointer to channel data.
- * \param tuner_freq_offset: tuner frequency offset.
- * \param lock_status: pointer to lock status.
- * \return int.
- */
- static int
- qam256auto(struct drx_demod_instance *demod,
- struct drx_channel *channel,
- s32 tuner_freq_offset, enum drx_lock_status *lock_status)
- {
- struct drxj_data *ext_attr = demod->my_ext_attr;
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drx39xxj_state *state = dev_addr->user_data;
- struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
- int rc;
- u32 lck_state = NO_LOCK;
- u32 start_time = 0;
- u32 d_locked_time = 0;
- u32 timeout_ofs = DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME;
- /* external attributes for storing acquired channel constellation */
- *lock_status = DRX_NOT_LOCKED;
- start_time = jiffies_to_msecs(jiffies);
- lck_state = NO_LOCK;
- do {
- rc = ctrl_lock_status(demod, lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- switch (lck_state) {
- case NO_LOCK:
- if (*lock_status == DRXJ_DEMOD_LOCK) {
- rc = ctrl_get_qam_sig_quality(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (p->cnr.stat[0].svalue > 26800) {
- lck_state = DEMOD_LOCKED;
- timeout_ofs += DRXJ_QAM_DEMOD_LOCK_EXT_WAITTIME; /* see something, wait longer */
- d_locked_time = jiffies_to_msecs(jiffies);
- }
- }
- break;
- case DEMOD_LOCKED:
- if (*lock_status == DRXJ_DEMOD_LOCK) {
- if ((channel->mirror == DRX_MIRROR_AUTO) &&
- ((jiffies_to_msecs(jiffies) - d_locked_time) >
- DRXJ_QAM_FEC_LOCK_WAITTIME)) {
- ext_attr->mirror = DRX_MIRROR_YES;
- rc = qam_flip_spec(demod, channel);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- lck_state = SPEC_MIRRORED;
- /* reset timer TODO: still need 300ms? */
- start_time = jiffies_to_msecs(jiffies);
- timeout_ofs = -DRXJ_QAM_MAX_WAITTIME / 2;
- }
- }
- break;
- case SPEC_MIRRORED:
- break;
- default:
- break;
- }
- msleep(10);
- } while
- ((*lock_status < DRX_LOCKED) &&
- (*lock_status != DRX_NEVER_LOCK) &&
- ((jiffies_to_msecs(jiffies) - start_time) <
- (DRXJ_QAM_MAX_WAITTIME + timeout_ofs)));
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_qam_channel ()
- * \brief Set QAM channel according to the requested constellation.
- * \param demod: instance of demod.
- * \param channel: pointer to channel data.
- * \return int.
- */
- static int
- set_qam_channel(struct drx_demod_instance *demod,
- struct drx_channel *channel, s32 tuner_freq_offset)
- {
- struct drxj_data *ext_attr = NULL;
- int rc;
- enum drx_lock_status lock_status = DRX_NOT_LOCKED;
- bool auto_flag = false;
- /* external attributes for storing acquired channel constellation */
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* set QAM channel constellation */
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM16:
- case DRX_CONSTELLATION_QAM32:
- case DRX_CONSTELLATION_QAM128:
- return -EINVAL;
- case DRX_CONSTELLATION_QAM64:
- case DRX_CONSTELLATION_QAM256:
- if (ext_attr->standard != DRX_STANDARD_ITU_B)
- return -EINVAL;
- ext_attr->constellation = channel->constellation;
- if (channel->mirror == DRX_MIRROR_AUTO)
- ext_attr->mirror = DRX_MIRROR_NO;
- else
- ext_attr->mirror = channel->mirror;
- rc = set_qam(demod, channel, tuner_freq_offset, QAM_SET_OP_ALL);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (channel->constellation == DRX_CONSTELLATION_QAM64)
- rc = qam64auto(demod, channel, tuner_freq_offset,
- &lock_status);
- else
- rc = qam256auto(demod, channel, tuner_freq_offset,
- &lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_CONSTELLATION_AUTO: /* for channel scan */
- if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- u16 qam_ctl_ena = 0;
- auto_flag = true;
- /* try to lock default QAM constellation: QAM256 */
- channel->constellation = DRX_CONSTELLATION_QAM256;
- ext_attr->constellation = DRX_CONSTELLATION_QAM256;
- if (channel->mirror == DRX_MIRROR_AUTO)
- ext_attr->mirror = DRX_MIRROR_NO;
- else
- ext_attr->mirror = channel->mirror;
- rc = set_qam(demod, channel, tuner_freq_offset,
- QAM_SET_OP_ALL);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = qam256auto(demod, channel, tuner_freq_offset,
- &lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (lock_status >= DRX_LOCKED) {
- channel->constellation = DRX_CONSTELLATION_AUTO;
- break;
- }
- /* QAM254 not locked. Try QAM64 constellation */
- channel->constellation = DRX_CONSTELLATION_QAM64;
- ext_attr->constellation = DRX_CONSTELLATION_QAM64;
- if (channel->mirror == DRX_MIRROR_AUTO)
- ext_attr->mirror = DRX_MIRROR_NO;
- else
- ext_attr->mirror = channel->mirror;
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- &qam_ctl_ena, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- qam_ctl_ena & ~SCU_RAM_QAM_CTL_ENA_ACQ__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_FSM_STATE_TGT__A,
- 0x2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* force to rate hunting */
- rc = set_qam(demod, channel, tuner_freq_offset,
- QAM_SET_OP_CONSTELLATION);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- qam_ctl_ena, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = qam64auto(demod, channel, tuner_freq_offset,
- &lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- channel->constellation = DRX_CONSTELLATION_AUTO;
- } else if (ext_attr->standard == DRX_STANDARD_ITU_C) {
- u16 qam_ctl_ena = 0;
- channel->constellation = DRX_CONSTELLATION_QAM64;
- ext_attr->constellation = DRX_CONSTELLATION_QAM64;
- auto_flag = true;
- if (channel->mirror == DRX_MIRROR_AUTO)
- ext_attr->mirror = DRX_MIRROR_NO;
- else
- ext_attr->mirror = channel->mirror;
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- &qam_ctl_ena, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- qam_ctl_ena & ~SCU_RAM_QAM_CTL_ENA_ACQ__M, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_FSM_STATE_TGT__A,
- 0x2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* force to rate hunting */
- rc = set_qam(demod, channel, tuner_freq_offset,
- QAM_SET_OP_CONSTELLATION);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(demod->my_i2c_dev_addr,
- SCU_RAM_QAM_CTL_ENA__A,
- qam_ctl_ena, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = qam64auto(demod, channel, tuner_freq_offset,
- &lock_status);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- channel->constellation = DRX_CONSTELLATION_AUTO;
- } else {
- return -EINVAL;
- }
- break;
- default:
- return -EINVAL;
- }
- return 0;
- rw_error:
- /* restore starting value */
- if (auto_flag)
- channel->constellation = DRX_CONSTELLATION_AUTO;
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn static short get_qamrs_err_count(struct i2c_device_addr *dev_addr)
- * \brief Get RS error count in QAM mode (used for post RS BER calculation)
- * \return Error code
- *
- * precondition: measurement period & measurement prescale must be set
- *
- */
- static int
- get_qamrs_err_count(struct i2c_device_addr *dev_addr,
- struct drxjrs_errors *rs_errors)
- {
- int rc;
- u16 nr_bit_errors = 0,
- nr_symbol_errors = 0,
- nr_packet_errors = 0, nr_failures = 0, nr_snc_par_fail_count = 0;
- /* check arguments */
- if (dev_addr == NULL)
- return -EINVAL;
- /* all reported errors are received in the */
- /* most recently finished measurment period */
- /* no of pre RS bit errors */
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_BIT_ERRORS__A, &nr_bit_errors, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no of symbol errors */
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_SYMBOL_ERRORS__A, &nr_symbol_errors, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no of packet errors */
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_PACKET_ERRORS__A, &nr_packet_errors, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no of failures to decode */
- rc = drxj_dap_read_reg16(dev_addr, FEC_RS_NR_FAILURES__A, &nr_failures, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* no of post RS bit erros */
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_FAIL_COUNT__A, &nr_snc_par_fail_count, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* TODO: NOTE */
- /* These register values are fetched in non-atomic fashion */
- /* It is possible that the read values contain unrelated information */
- rs_errors->nr_bit_errors = nr_bit_errors & FEC_RS_NR_BIT_ERRORS__M;
- rs_errors->nr_symbol_errors = nr_symbol_errors & FEC_RS_NR_SYMBOL_ERRORS__M;
- rs_errors->nr_packet_errors = nr_packet_errors & FEC_RS_NR_PACKET_ERRORS__M;
- rs_errors->nr_failures = nr_failures & FEC_RS_NR_FAILURES__M;
- rs_errors->nr_snc_par_fail_count =
- nr_snc_par_fail_count & FEC_OC_SNC_FAIL_COUNT__M;
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int get_sig_strength()
- * \brief Retrieve signal strength for VSB and QAM.
- * \param demod Pointer to demod instance
- * \param u16-t Pointer to signal strength data; range 0, .. , 100.
- * \return int.
- * \retval 0 sig_strength contains valid data.
- * \retval -EINVAL sig_strength is NULL.
- * \retval -EIO Erroneous data, sig_strength contains invalid data.
- */
- #define DRXJ_AGC_TOP 0x2800
- #define DRXJ_AGC_SNS 0x1600
- #define DRXJ_RFAGC_MAX 0x3fff
- #define DRXJ_RFAGC_MIN 0x800
- static int get_sig_strength(struct drx_demod_instance *demod, u16 *sig_strength)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- u16 rf_gain = 0;
- u16 if_gain = 0;
- u16 if_agc_sns = 0;
- u16 if_agc_top = 0;
- u16 rf_agc_max = 0;
- u16 rf_agc_min = 0;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_AGC_IF__A, &if_gain, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if_gain &= IQM_AF_AGC_IF__M;
- rc = drxj_dap_read_reg16(dev_addr, IQM_AF_AGC_RF__A, &rf_gain, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rf_gain &= IQM_AF_AGC_RF__M;
- if_agc_sns = DRXJ_AGC_SNS;
- if_agc_top = DRXJ_AGC_TOP;
- rf_agc_max = DRXJ_RFAGC_MAX;
- rf_agc_min = DRXJ_RFAGC_MIN;
- if (if_gain > if_agc_top) {
- if (rf_gain > rf_agc_max)
- *sig_strength = 100;
- else if (rf_gain > rf_agc_min) {
- if (rf_agc_max == rf_agc_min) {
- pr_err("error: rf_agc_max == rf_agc_min\n");
- return -EIO;
- }
- *sig_strength =
- 75 + 25 * (rf_gain - rf_agc_min) / (rf_agc_max -
- rf_agc_min);
- } else
- *sig_strength = 75;
- } else if (if_gain > if_agc_sns) {
- if (if_agc_top == if_agc_sns) {
- pr_err("error: if_agc_top == if_agc_sns\n");
- return -EIO;
- }
- *sig_strength =
- 20 + 55 * (if_gain - if_agc_sns) / (if_agc_top - if_agc_sns);
- } else {
- if (!if_agc_sns) {
- pr_err("error: if_agc_sns is zero!\n");
- return -EIO;
- }
- *sig_strength = (20 * if_gain / if_agc_sns);
- }
- if (*sig_strength <= 7)
- *sig_strength = 0;
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int ctrl_get_qam_sig_quality()
- * \brief Retrieve QAM signal quality from device.
- * \param devmod Pointer to demodulator instance.
- * \param sig_quality Pointer to signal quality data.
- * \return int.
- * \retval 0 sig_quality contains valid data.
- * \retval -EINVAL sig_quality is NULL.
- * \retval -EIO Erroneous data, sig_quality contains invalid data.
- * Pre-condition: Device must be started and in lock.
- */
- static int
- ctrl_get_qam_sig_quality(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxj_data *ext_attr = demod->my_ext_attr;
- struct drx39xxj_state *state = dev_addr->user_data;
- struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
- struct drxjrs_errors measuredrs_errors = { 0, 0, 0, 0, 0 };
- enum drx_modulation constellation = ext_attr->constellation;
- int rc;
- u32 pre_bit_err_rs = 0; /* pre RedSolomon Bit Error Rate */
- u32 post_bit_err_rs = 0; /* post RedSolomon Bit Error Rate */
- u32 pkt_errs = 0; /* no of packet errors in RS */
- u16 qam_sl_err_power = 0; /* accumulated error between raw and sliced symbols */
- u16 qsym_err_vd = 0; /* quadrature symbol errors in QAM_VD */
- u16 fec_oc_period = 0; /* SNC sync failure measurement period */
- u16 fec_rs_prescale = 0; /* ReedSolomon Measurement Prescale */
- u16 fec_rs_period = 0; /* Value for corresponding I2C register */
- /* calculation constants */
- u32 rs_bit_cnt = 0; /* RedSolomon Bit Count */
- u32 qam_sl_sig_power = 0; /* used for MER, depends of QAM constellation */
- /* intermediate results */
- u32 e = 0; /* exponent value used for QAM BER/SER */
- u32 m = 0; /* mantisa value used for QAM BER/SER */
- u32 ber_cnt = 0; /* BER count */
- /* signal quality info */
- u32 qam_sl_mer = 0; /* QAM MER */
- u32 qam_pre_rs_ber = 0; /* Pre RedSolomon BER */
- u32 qam_post_rs_ber = 0; /* Post RedSolomon BER */
- u32 qam_vd_ser = 0; /* ViterbiDecoder SER */
- u16 qam_vd_prescale = 0; /* Viterbi Measurement Prescale */
- u16 qam_vd_period = 0; /* Viterbi Measurement period */
- u32 vd_bit_cnt = 0; /* ViterbiDecoder Bit Count */
- p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- /* read the physical registers */
- /* Get the RS error data */
- rc = get_qamrs_err_count(dev_addr, &measuredrs_errors);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* get the register value needed for MER */
- rc = drxj_dap_read_reg16(dev_addr, QAM_SL_ERR_POWER__A, &qam_sl_err_power, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* get the register value needed for post RS BER */
- rc = drxj_dap_read_reg16(dev_addr, FEC_OC_SNC_FAIL_PERIOD__A, &fec_oc_period, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* get constants needed for signal quality calculation */
- fec_rs_period = ext_attr->fec_rs_period;
- fec_rs_prescale = ext_attr->fec_rs_prescale;
- rs_bit_cnt = fec_rs_period * fec_rs_prescale * ext_attr->fec_rs_plen;
- qam_vd_period = ext_attr->qam_vd_period;
- qam_vd_prescale = ext_attr->qam_vd_prescale;
- vd_bit_cnt = qam_vd_period * qam_vd_prescale * ext_attr->fec_vd_plen;
- /* DRXJ_QAM_SL_SIG_POWER_QAMxxx * 4 */
- switch (constellation) {
- case DRX_CONSTELLATION_QAM16:
- qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM16 << 2;
- break;
- case DRX_CONSTELLATION_QAM32:
- qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM32 << 2;
- break;
- case DRX_CONSTELLATION_QAM64:
- qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM64 << 2;
- break;
- case DRX_CONSTELLATION_QAM128:
- qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM128 << 2;
- break;
- case DRX_CONSTELLATION_QAM256:
- qam_sl_sig_power = DRXJ_QAM_SL_SIG_POWER_QAM256 << 2;
- break;
- default:
- return -EIO;
- }
- /* ------------------------------ */
- /* MER Calculation */
- /* ------------------------------ */
- /* MER is good if it is above 27.5 for QAM256 or 21.5 for QAM64 */
- /* 10.0*log10(qam_sl_sig_power * 4.0 / qam_sl_err_power); */
- if (qam_sl_err_power == 0)
- qam_sl_mer = 0;
- else
- qam_sl_mer = log1_times100(qam_sl_sig_power) - log1_times100((u32)qam_sl_err_power);
- /* ----------------------------------------- */
- /* Pre Viterbi Symbol Error Rate Calculation */
- /* ----------------------------------------- */
- /* pre viterbi SER is good if it is below 0.025 */
- /* get the register value */
- /* no of quadrature symbol errors */
- rc = drxj_dap_read_reg16(dev_addr, QAM_VD_NR_QSYM_ERRORS__A, &qsym_err_vd, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Extract the Exponent and the Mantisa */
- /* of number of quadrature symbol errors */
- e = (qsym_err_vd & QAM_VD_NR_QSYM_ERRORS_EXP__M) >>
- QAM_VD_NR_QSYM_ERRORS_EXP__B;
- m = (qsym_err_vd & QAM_VD_NR_SYMBOL_ERRORS_FIXED_MANT__M) >>
- QAM_VD_NR_SYMBOL_ERRORS_FIXED_MANT__B;
- if ((m << e) >> 3 > 549752)
- qam_vd_ser = 500000 * vd_bit_cnt * ((e > 2) ? 1 : 8) / 8;
- else
- qam_vd_ser = m << ((e > 2) ? (e - 3) : e);
- /* --------------------------------------- */
- /* pre and post RedSolomon BER Calculation */
- /* --------------------------------------- */
- /* pre RS BER is good if it is below 3.5e-4 */
- /* get the register values */
- pre_bit_err_rs = (u32) measuredrs_errors.nr_bit_errors;
- pkt_errs = post_bit_err_rs = (u32) measuredrs_errors.nr_snc_par_fail_count;
- /* Extract the Exponent and the Mantisa of the */
- /* pre Reed-Solomon bit error count */
- e = (pre_bit_err_rs & FEC_RS_NR_BIT_ERRORS_EXP__M) >>
- FEC_RS_NR_BIT_ERRORS_EXP__B;
- m = (pre_bit_err_rs & FEC_RS_NR_BIT_ERRORS_FIXED_MANT__M) >>
- FEC_RS_NR_BIT_ERRORS_FIXED_MANT__B;
- ber_cnt = m << e;
- /*qam_pre_rs_ber = frac_times1e6( ber_cnt, rs_bit_cnt ); */
- if (m > (rs_bit_cnt >> (e + 1)) || (rs_bit_cnt >> e) == 0)
- qam_pre_rs_ber = 500000 * rs_bit_cnt >> e;
- else
- qam_pre_rs_ber = ber_cnt;
- /* post RS BER = 1000000* (11.17 * FEC_OC_SNC_FAIL_COUNT__A) / */
- /* (1504.0 * FEC_OC_SNC_FAIL_PERIOD__A) */
- /*
- => c = (1000000*100*11.17)/1504 =
- post RS BER = (( c* FEC_OC_SNC_FAIL_COUNT__A) /
- (100 * FEC_OC_SNC_FAIL_PERIOD__A)
- *100 and /100 is for more precision.
- => (20 bits * 12 bits) /(16 bits * 7 bits) => safe in 32 bits computation
- Precision errors still possible.
- */
- if (!fec_oc_period) {
- qam_post_rs_ber = 0xFFFFFFFF;
- } else {
- e = post_bit_err_rs * 742686;
- m = fec_oc_period * 100;
- qam_post_rs_ber = e / m;
- }
- /* fill signal quality data structure */
- p->pre_bit_count.stat[0].scale = FE_SCALE_COUNTER;
- p->post_bit_count.stat[0].scale = FE_SCALE_COUNTER;
- p->pre_bit_error.stat[0].scale = FE_SCALE_COUNTER;
- p->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
- p->block_error.stat[0].scale = FE_SCALE_COUNTER;
- p->cnr.stat[0].scale = FE_SCALE_DECIBEL;
- p->cnr.stat[0].svalue = ((u16) qam_sl_mer) * 100;
- if (ext_attr->standard == DRX_STANDARD_ITU_B) {
- p->pre_bit_error.stat[0].uvalue += qam_vd_ser;
- p->pre_bit_count.stat[0].uvalue += vd_bit_cnt * ((e > 2) ? 1 : 8) / 8;
- } else {
- p->pre_bit_error.stat[0].uvalue += qam_pre_rs_ber;
- p->pre_bit_count.stat[0].uvalue += rs_bit_cnt >> e;
- }
- p->post_bit_error.stat[0].uvalue += qam_post_rs_ber;
- p->post_bit_count.stat[0].uvalue += rs_bit_cnt >> e;
- p->block_error.stat[0].uvalue += pkt_errs;
- #ifdef DRXJ_SIGNAL_ACCUM_ERR
- rc = get_acc_pkt_err(demod, &sig_quality->packet_error);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- #endif
- return 0;
- rw_error:
- p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- return rc;
- }
- #endif /* #ifndef DRXJ_VSB_ONLY */
- /*============================================================================*/
- /*== END QAM DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*============================================================================*/
- /*== ATV DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*============================================================================*/
- /*
- Implementation notes.
- NTSC/FM AGCs
- Four AGCs are used for NTSC:
- (1) RF (used to attenuate the input signal in case of to much power)
- (2) IF (used to attenuate the input signal in case of to much power)
- (3) Video AGC (used to amplify the output signal in case input to low)
- (4) SIF AGC (used to amplify the output signal in case input to low)
- Video AGC is coupled to RF and IF. SIF AGC is not coupled. It is assumed
- that the coupling between Video AGC and the RF and IF AGCs also works in
- favor of the SIF AGC.
- Three AGCs are used for FM:
- (1) RF (used to attenuate the input signal in case of to much power)
- (2) IF (used to attenuate the input signal in case of to much power)
- (3) SIF AGC (used to amplify the output signal in case input to low)
- The SIF AGC is now coupled to the RF/IF AGCs.
- The SIF AGC is needed for both SIF ouput and the internal SIF signal to
- the AUD block.
- RF and IF AGCs DACs are part of AFE, Video and SIF AGC DACs are part of
- the ATV block. The AGC control algorithms are all implemented in
- microcode.
- ATV SETTINGS
- (Shadow settings will not be used for now, they will be implemented
- later on because of the schedule)
- Several HW/SCU "settings" can be used for ATV. The standard selection
- will reset most of these settings. To avoid that the end user apllication
- has to perform these settings each time the ATV or FM standards is
- selected the driver will shadow these settings. This enables the end user
- to perform the settings only once after a drx_open(). The driver must
- write the shadow settings to HW/SCU incase:
- ( setstandard FM/ATV) ||
- ( settings have changed && FM/ATV standard is active)
- The shadow settings will be stored in the device specific data container.
- A set of flags will be defined to flag changes in shadow settings.
- A routine will be implemented to write all changed shadow settings to
- HW/SCU.
- The "settings" will consist of: AGC settings, filter settings etc.
- Disadvantage of use of shadow settings:
- Direct changes in HW/SCU registers will not be reflected in the
- shadow settings and these changes will be overwritten during a next
- update. This can happen during evaluation. This will not be a problem
- for normal customer usage.
- */
- /* -------------------------------------------------------------------------- */
- /*
- * \fn int power_down_atv ()
- * \brief Power down ATV.
- * \param demod instance of demodulator
- * \param standard either NTSC or FM (sub strandard for ATV )
- * \return int.
- *
- * Stops and thus resets ATV and IQM block
- * SIF and CVBS ADC are powered down
- * Calls audio power down
- */
- static int
- power_down_atv(struct drx_demod_instance *demod, enum drx_standard standard, bool primary)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxjscu_cmd cmd_scu = { /* command */ 0,
- /* parameter_len */ 0,
- /* result_len */ 0,
- /* *parameter */ NULL,
- /* *result */ NULL
- };
- int rc;
- u16 cmd_result = 0;
- /* ATV NTSC */
- /* Stop ATV SCU (will reset ATV and IQM hardware */
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_ATV |
- SCU_RAM_COMMAND_CMD_DEMOD_STOP;
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 1;
- cmd_scu.parameter = NULL;
- cmd_scu.result = &cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Disable ATV outputs (ATV reset enables CVBS, undo this) */
- rc = drxj_dap_write_reg16(dev_addr, ATV_TOP_STDBY__A, (ATV_TOP_STDBY_SIF_STDBY_STANDBY & (~ATV_TOP_STDBY_CVBS_STDBY_A2_ACTIVE)), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ATV_COMM_EXEC__A, ATV_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (primary) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_COMM_EXEC__A, IQM_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_iqm_af(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } else {
- rc = drxj_dap_write_reg16(dev_addr, IQM_FS_COMM_EXEC__A, IQM_FS_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_FD_COMM_EXEC__A, IQM_FD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RC_COMM_EXEC__A, IQM_RC_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_RT_COMM_EXEC__A, IQM_RT_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, IQM_CF_COMM_EXEC__A, IQM_CF_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = power_down_aud(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \brief Power up AUD.
- * \param demod instance of demodulator
- * \return int.
- *
- */
- static int power_down_aud(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- int rc;
- dev_addr = (struct i2c_device_addr *)demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- rc = drxj_dap_write_reg16(dev_addr, AUD_COMM_EXEC__A, AUD_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->aud_data.audio_is_active = false;
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int set_orx_nsu_aox()
- * \brief Configure OrxNsuAox for OOB
- * \param demod instance of demodulator.
- * \param active
- * \return int.
- */
- static int set_orx_nsu_aox(struct drx_demod_instance *demod, bool active)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- u16 data = 0;
- /* Configure NSU_AOX */
- rc = drxj_dap_read_reg16(dev_addr, ORX_NSU_AOX_STDBY_W__A, &data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (!active)
- data &= ((~ORX_NSU_AOX_STDBY_W_STDBYADC_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYAMP_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYBIAS_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYPLL_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYPD_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYTAGC_IF_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYTAGC_RF_A2_ON) & (~ORX_NSU_AOX_STDBY_W_STDBYFLT_A2_ON));
- else
- data |= (ORX_NSU_AOX_STDBY_W_STDBYADC_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYAMP_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYBIAS_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYPLL_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYPD_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYTAGC_IF_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYTAGC_RF_A2_ON | ORX_NSU_AOX_STDBY_W_STDBYFLT_A2_ON);
- rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_STDBY_W__A, data, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*
- * \fn int ctrl_set_oob()
- * \brief Set OOB channel to be used.
- * \param demod instance of demodulator
- * \param oob_param OOB parameters for channel setting.
- * \frequency should be in KHz
- * \return int.
- *
- * Accepts only. Returns error otherwise.
- * Demapper value is written after scu_command START
- * because START command causes COMM_EXEC transition
- * from 0 to 1 which causes all registers to be
- * overwritten with initial value
- *
- */
- /* Nyquist filter impulse response */
- #define IMPULSE_COSINE_ALPHA_0_3 {-3, -4, -1, 6, 10, 7, -5, -20, -25, -10, 29, 79, 123, 140} /*sqrt raised-cosine filter with alpha=0.3 */
- #define IMPULSE_COSINE_ALPHA_0_5 { 2, 0, -2, -2, 2, 5, 2, -10, -20, -14, 20, 74, 125, 145} /*sqrt raised-cosine filter with alpha=0.5 */
- #define IMPULSE_COSINE_ALPHA_RO_0_5 { 0, 0, 1, 2, 3, 0, -7, -15, -16, 0, 34, 77, 114, 128} /*full raised-cosine filter with alpha=0.5 (receiver only) */
- /* Coefficients for the nyquist fitler (total: 27 taps) */
- #define NYQFILTERLEN 27
- static int ctrl_set_oob(struct drx_demod_instance *demod, struct drxoob *oob_param)
- {
- int rc;
- s32 freq = 0; /* KHz */
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- u16 i = 0;
- bool mirror_freq_spect_oob = false;
- u16 trk_filter_value = 0;
- struct drxjscu_cmd scu_cmd;
- u16 set_param_parameters[3];
- u16 cmd_result[2] = { 0, 0 };
- s16 nyquist_coeffs[4][(NYQFILTERLEN + 1) / 2] = {
- IMPULSE_COSINE_ALPHA_0_3, /* Target Mode 0 */
- IMPULSE_COSINE_ALPHA_0_3, /* Target Mode 1 */
- IMPULSE_COSINE_ALPHA_0_5, /* Target Mode 2 */
- IMPULSE_COSINE_ALPHA_RO_0_5 /* Target Mode 3 */
- };
- u8 mode_val[4] = { 2, 2, 0, 1 };
- u8 pfi_coeffs[4][6] = {
- {DRXJ_16TO8(-92), DRXJ_16TO8(-108), DRXJ_16TO8(100)}, /* TARGET_MODE = 0: PFI_A = -23/32; PFI_B = -54/32; PFI_C = 25/32; fg = 0.5 MHz (Att=26dB) */
- {DRXJ_16TO8(-64), DRXJ_16TO8(-80), DRXJ_16TO8(80)}, /* TARGET_MODE = 1: PFI_A = -16/32; PFI_B = -40/32; PFI_C = 20/32; fg = 1.0 MHz (Att=28dB) */
- {DRXJ_16TO8(-80), DRXJ_16TO8(-98), DRXJ_16TO8(92)}, /* TARGET_MODE = 2, 3: PFI_A = -20/32; PFI_B = -49/32; PFI_C = 23/32; fg = 0.8 MHz (Att=25dB) */
- {DRXJ_16TO8(-80), DRXJ_16TO8(-98), DRXJ_16TO8(92)} /* TARGET_MODE = 2, 3: PFI_A = -20/32; PFI_B = -49/32; PFI_C = 23/32; fg = 0.8 MHz (Att=25dB) */
- };
- u16 mode_index;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- mirror_freq_spect_oob = ext_attr->mirror_freq_spect_oob;
- /* Check parameters */
- if (oob_param == NULL) {
- /* power off oob module */
- scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
- | SCU_RAM_COMMAND_CMD_DEMOD_STOP;
- scu_cmd.parameter_len = 0;
- scu_cmd.result_len = 1;
- scu_cmd.result = cmd_result;
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_orx_nsu_aox(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->oob_power_on = false;
- return 0;
- }
- freq = oob_param->frequency;
- if ((freq < 70000) || (freq > 130000))
- return -EIO;
- freq = (freq - 50000) / 50;
- {
- u16 index = 0;
- u16 remainder = 0;
- u16 *trk_filtercfg = ext_attr->oob_trk_filter_cfg;
- index = (u16) ((freq - 400) / 200);
- remainder = (u16) ((freq - 400) % 200);
- trk_filter_value =
- trk_filtercfg[index] - (trk_filtercfg[index] -
- trk_filtercfg[index +
- 1]) / 10 * remainder /
- 20;
- }
- /********/
- /* Stop */
- /********/
- rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
- | SCU_RAM_COMMAND_CMD_DEMOD_STOP;
- scu_cmd.parameter_len = 0;
- scu_cmd.result_len = 1;
- scu_cmd.result = cmd_result;
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /********/
- /* Reset */
- /********/
- scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
- | SCU_RAM_COMMAND_CMD_DEMOD_RESET;
- scu_cmd.parameter_len = 0;
- scu_cmd.result_len = 1;
- scu_cmd.result = cmd_result;
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /**********/
- /* SET_ENV */
- /**********/
- /* set frequency, spectrum inversion and data rate */
- scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
- | SCU_RAM_COMMAND_CMD_DEMOD_SET_ENV;
- scu_cmd.parameter_len = 3;
- /* 1-data rate;2-frequency */
- switch (oob_param->standard) {
- case DRX_OOB_MODE_A:
- if (
- /* signal is transmitted inverted */
- ((oob_param->spectrum_inverted == true) &&
- /* and tuner is not mirroring the signal */
- (!mirror_freq_spect_oob)) |
- /* or */
- /* signal is transmitted noninverted */
- ((oob_param->spectrum_inverted == false) &&
- /* and tuner is mirroring the signal */
- (mirror_freq_spect_oob))
- )
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_2048KBPS_INVSPEC;
- else
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_2048KBPS_REGSPEC;
- break;
- case DRX_OOB_MODE_B_GRADE_A:
- if (
- /* signal is transmitted inverted */
- ((oob_param->spectrum_inverted == true) &&
- /* and tuner is not mirroring the signal */
- (!mirror_freq_spect_oob)) |
- /* or */
- /* signal is transmitted noninverted */
- ((oob_param->spectrum_inverted == false) &&
- /* and tuner is mirroring the signal */
- (mirror_freq_spect_oob))
- )
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_1544KBPS_INVSPEC;
- else
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_1544KBPS_REGSPEC;
- break;
- case DRX_OOB_MODE_B_GRADE_B:
- default:
- if (
- /* signal is transmitted inverted */
- ((oob_param->spectrum_inverted == true) &&
- /* and tuner is not mirroring the signal */
- (!mirror_freq_spect_oob)) |
- /* or */
- /* signal is transmitted noninverted */
- ((oob_param->spectrum_inverted == false) &&
- /* and tuner is mirroring the signal */
- (mirror_freq_spect_oob))
- )
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_3088KBPS_INVSPEC;
- else
- set_param_parameters[0] =
- SCU_RAM_ORX_RF_RX_DATA_RATE_3088KBPS_REGSPEC;
- break;
- }
- set_param_parameters[1] = (u16) (freq & 0xFFFF);
- set_param_parameters[2] = trk_filter_value;
- scu_cmd.parameter = set_param_parameters;
- scu_cmd.result_len = 1;
- scu_cmd.result = cmd_result;
- mode_index = mode_val[(set_param_parameters[0] & 0xC0) >> 6];
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0xFABA, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* Write magic word to enable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_OOB_CRX_CFG__A, OOB_CRX_DRIVE_STRENGTH << SIO_PDR_OOB_CRX_CFG_DRIVE__B | 0x03 << SIO_PDR_OOB_CRX_CFG_MODE__B, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_PDR_OOB_DRX_CFG__A, OOB_DRX_DRIVE_STRENGTH << SIO_PDR_OOB_DRX_CFG_DRIVE__B | 0x03 << SIO_PDR_OOB_DRX_CFG_MODE__B, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_TOP_COMM_KEY__A, 0x0000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- } /* Write magic word to disable pdr reg write */
- rc = drxj_dap_write_reg16(dev_addr, ORX_TOP_COMM_KEY__A, 0, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_AAG_LEN_W__A, 16000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_AAG_THR_W__A, 40, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* ddc */
- rc = drxj_dap_write_reg16(dev_addr, ORX_DDC_OFO_SET_W__A, ORX_DDC_OFO_SET_W__PRE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* nsu */
- rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_LOPOW_W__A, ext_attr->oob_lo_pow, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* initialization for target mode */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TARGET_MODE__A, SCU_RAM_ORX_TARGET_MODE_2048KBPS_SQRT, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FREQ_GAIN_CORR__A, SCU_RAM_ORX_FREQ_GAIN_CORR_2048KBPS, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Reset bits for timing and freq. recovery */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_CPH__A, 0x0001, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_CTI__A, 0x0002, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_KRN__A, 0x0004, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_RST_KRP__A, 0x0008, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* AGN_LOCK = {2048>>3, -2048, 8, -8, 0, 1}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_TH__A, 2048 >> 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_ONLOCK_TTH__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_UNLOCK_TTH__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_AGN_LOCK_MASK__A, 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* DGN_LOCK = {10, -2048, 8, -8, 0, 1<<1}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_TH__A, 10, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_ONLOCK_TTH__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_UNLOCK_TTH__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_DGN_LOCK_MASK__A, 1 << 1, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* FRQ_LOCK = {15,-2048, 8, -8, 0, 1<<2}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_TH__A, 17, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_ONLOCK_TTH__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_UNLOCK_TTH__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_FRQ_LOCK_MASK__A, 1 << 2, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* PHA_LOCK = {5000, -2048, 8, -8, 0, 1<<3}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_TH__A, 3000, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_ONLOCK_TTH__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_UNLOCK_TTH__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_PHA_LOCK_MASK__A, 1 << 3, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* TIM_LOCK = {300, -2048, 8, -8, 0, 1<<4}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_TH__A, 400, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_ONLOCK_TTH__A, 8, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_UNLOCK_TTH__A, (u16)(-8), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_TIM_LOCK_MASK__A, 1 << 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* EQU_LOCK = {20, -2048, 8, -8, 0, 1<<5}; */
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_TH__A, 20, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_TOTH__A, (u16)(-2048), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_ONLOCK_TTH__A, 4, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_UNLOCK_TTH__A, (u16)(-4), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_ORX_EQU_LOCK_MASK__A, 1 << 5, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* PRE-Filter coefficients (PFI) */
- rc = drxdap_fasi_write_block(dev_addr, ORX_FWP_PFI_A_W__A, sizeof(pfi_coeffs[mode_index]), ((u8 *)pfi_coeffs[mode_index]), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_TOP_MDE_W__A, mode_index, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* NYQUIST-Filter coefficients (NYQ) */
- for (i = 0; i < (NYQFILTERLEN + 1) / 2; i++) {
- rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_ADR_W__A, i, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_COF_RW__A, nyquist_coeffs[mode_index][i], 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_FWP_NYQ_ADR_W__A, 31, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_COMM_EXEC__A, ORX_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /********/
- /* Start */
- /********/
- scu_cmd.command = SCU_RAM_COMMAND_STANDARD_OOB
- | SCU_RAM_COMMAND_CMD_DEMOD_START;
- scu_cmd.parameter_len = 0;
- scu_cmd.result_len = 1;
- scu_cmd.result = cmd_result;
- rc = scu_command(dev_addr, &scu_cmd);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_orx_nsu_aox(demod, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, ORX_NSU_AOX_STHR_W__A, ext_attr->oob_pre_saw, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->oob_power_on = true;
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== END OOB DATAPATH FUNCTIONS ==*/
- /*============================================================================*/
- /*=============================================================================
- ===== MC command related functions ==========================================
- ===========================================================================*/
- /*=============================================================================
- ===== ctrl_set_channel() ==========================================================
- ===========================================================================*/
- /*
- * \fn int ctrl_set_channel()
- * \brief Select a new transmission channel.
- * \param demod instance of demod.
- * \param channel Pointer to channel data.
- * \return int.
- *
- * In case the tuner module is not used and in case of NTSC/FM the pogrammer
- * must tune the tuner to the centre frequency of the NTSC/FM channel.
- *
- */
- static int
- ctrl_set_channel(struct drx_demod_instance *demod, struct drx_channel *channel)
- {
- int rc;
- s32 tuner_freq_offset = 0;
- struct drxj_data *ext_attr = NULL;
- struct i2c_device_addr *dev_addr = NULL;
- enum drx_standard standard = DRX_STANDARD_UNKNOWN;
- #ifndef DRXJ_VSB_ONLY
- u32 min_symbol_rate = 0;
- u32 max_symbol_rate = 0;
- int bandwidth_temp = 0;
- int bandwidth = 0;
- #endif
- /*== check arguments ======================================================*/
- if ((demod == NULL) || (channel == NULL))
- return -EINVAL;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- standard = ext_attr->standard;
- /* check valid standards */
- switch (standard) {
- case DRX_STANDARD_8VSB:
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- #endif /* DRXJ_VSB_ONLY */
- break;
- case DRX_STANDARD_UNKNOWN:
- default:
- return -EINVAL;
- }
- /* check bandwidth QAM annex B, NTSC and 8VSB */
- if ((standard == DRX_STANDARD_ITU_B) ||
- (standard == DRX_STANDARD_8VSB) ||
- (standard == DRX_STANDARD_NTSC)) {
- switch (channel->bandwidth) {
- case DRX_BANDWIDTH_6MHZ:
- case DRX_BANDWIDTH_UNKNOWN: /* fall through */
- channel->bandwidth = DRX_BANDWIDTH_6MHZ;
- break;
- case DRX_BANDWIDTH_8MHZ: /* fall through */
- case DRX_BANDWIDTH_7MHZ: /* fall through */
- default:
- return -EINVAL;
- }
- }
- /* For QAM annex A and annex C:
- -check symbolrate and constellation
- -derive bandwidth from symbolrate (input bandwidth is ignored)
- */
- #ifndef DRXJ_VSB_ONLY
- if ((standard == DRX_STANDARD_ITU_A) ||
- (standard == DRX_STANDARD_ITU_C)) {
- struct drxuio_cfg uio_cfg = { DRX_UIO1, DRX_UIO_MODE_FIRMWARE_SAW };
- int bw_rolloff_factor = 0;
- bw_rolloff_factor = (standard == DRX_STANDARD_ITU_A) ? 115 : 113;
- min_symbol_rate = DRXJ_QAM_SYMBOLRATE_MIN;
- max_symbol_rate = DRXJ_QAM_SYMBOLRATE_MAX;
- /* config SMA_TX pin to SAW switch mode */
- rc = ctrl_set_uio_cfg(demod, &uio_cfg);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (channel->symbolrate < min_symbol_rate ||
- channel->symbolrate > max_symbol_rate) {
- return -EINVAL;
- }
- switch (channel->constellation) {
- case DRX_CONSTELLATION_QAM16: /* fall through */
- case DRX_CONSTELLATION_QAM32: /* fall through */
- case DRX_CONSTELLATION_QAM64: /* fall through */
- case DRX_CONSTELLATION_QAM128: /* fall through */
- case DRX_CONSTELLATION_QAM256:
- bandwidth_temp = channel->symbolrate * bw_rolloff_factor;
- bandwidth = bandwidth_temp / 100;
- if ((bandwidth_temp % 100) >= 50)
- bandwidth++;
- if (bandwidth <= 6100000) {
- channel->bandwidth = DRX_BANDWIDTH_6MHZ;
- } else if ((bandwidth > 6100000)
- && (bandwidth <= 7100000)) {
- channel->bandwidth = DRX_BANDWIDTH_7MHZ;
- } else if (bandwidth > 7100000) {
- channel->bandwidth = DRX_BANDWIDTH_8MHZ;
- }
- break;
- default:
- return -EINVAL;
- }
- }
- /* For QAM annex B:
- -check constellation
- */
- if (standard == DRX_STANDARD_ITU_B) {
- switch (channel->constellation) {
- case DRX_CONSTELLATION_AUTO:
- case DRX_CONSTELLATION_QAM256:
- case DRX_CONSTELLATION_QAM64:
- break;
- default:
- return -EINVAL;
- }
- switch (channel->interleavemode) {
- case DRX_INTERLEAVEMODE_I128_J1:
- case DRX_INTERLEAVEMODE_I128_J1_V2:
- case DRX_INTERLEAVEMODE_I128_J2:
- case DRX_INTERLEAVEMODE_I64_J2:
- case DRX_INTERLEAVEMODE_I128_J3:
- case DRX_INTERLEAVEMODE_I32_J4:
- case DRX_INTERLEAVEMODE_I128_J4:
- case DRX_INTERLEAVEMODE_I16_J8:
- case DRX_INTERLEAVEMODE_I128_J5:
- case DRX_INTERLEAVEMODE_I8_J16:
- case DRX_INTERLEAVEMODE_I128_J6:
- case DRX_INTERLEAVEMODE_I128_J7:
- case DRX_INTERLEAVEMODE_I128_J8:
- case DRX_INTERLEAVEMODE_I12_J17:
- case DRX_INTERLEAVEMODE_I5_J4:
- case DRX_INTERLEAVEMODE_B52_M240:
- case DRX_INTERLEAVEMODE_B52_M720:
- case DRX_INTERLEAVEMODE_UNKNOWN:
- case DRX_INTERLEAVEMODE_AUTO:
- break;
- default:
- return -EINVAL;
- }
- }
- if ((ext_attr->uio_sma_tx_mode) == DRX_UIO_MODE_FIRMWARE_SAW) {
- /* SAW SW, user UIO is used for switchable SAW */
- struct drxuio_data uio1 = { DRX_UIO1, false };
- switch (channel->bandwidth) {
- case DRX_BANDWIDTH_8MHZ:
- uio1.value = true;
- break;
- case DRX_BANDWIDTH_7MHZ:
- uio1.value = false;
- break;
- case DRX_BANDWIDTH_6MHZ:
- uio1.value = false;
- break;
- case DRX_BANDWIDTH_UNKNOWN:
- default:
- return -EINVAL;
- }
- rc = ctrl_uio_write(demod, &uio1);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- #endif /* DRXJ_VSB_ONLY */
- rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- tuner_freq_offset = 0;
- /*== Setup demod for specific standard ====================================*/
- switch (standard) {
- case DRX_STANDARD_8VSB:
- if (channel->mirror == DRX_MIRROR_AUTO)
- ext_attr->mirror = DRX_MIRROR_NO;
- else
- ext_attr->mirror = channel->mirror;
- rc = set_vsb(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_frequency(demod, channel, tuner_freq_offset);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- rc = set_qam_channel(demod, channel, tuner_freq_offset);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- #endif
- case DRX_STANDARD_UNKNOWN:
- default:
- return -EIO;
- }
- /* flag the packet error counter reset */
- ext_attr->reset_pkt_err_acc = true;
- return 0;
- rw_error:
- return rc;
- }
- /*=============================================================================
- ===== SigQuality() ==========================================================
- ===========================================================================*/
- /*
- * \fn int ctrl_sig_quality()
- * \brief Retrieve signal quality form device.
- * \param devmod Pointer to demodulator instance.
- * \param sig_quality Pointer to signal quality data.
- * \return int.
- * \retval 0 sig_quality contains valid data.
- * \retval -EINVAL sig_quality is NULL.
- * \retval -EIO Erroneous data, sig_quality contains invalid data.
- */
- static int
- ctrl_sig_quality(struct drx_demod_instance *demod,
- enum drx_lock_status lock_status)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- struct drxj_data *ext_attr = demod->my_ext_attr;
- struct drx39xxj_state *state = dev_addr->user_data;
- struct dtv_frontend_properties *p = &state->frontend.dtv_property_cache;
- enum drx_standard standard = ext_attr->standard;
- int rc;
- u32 ber, cnt, err, pkt;
- u16 mer, strength = 0;
- rc = get_sig_strength(demod, &strength);
- if (rc < 0) {
- pr_err("error getting signal strength %d\n", rc);
- p->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- p->strength.stat[0].scale = FE_SCALE_RELATIVE;
- p->strength.stat[0].uvalue = 65535UL * strength/ 100;
- }
- switch (standard) {
- case DRX_STANDARD_8VSB:
- #ifdef DRXJ_SIGNAL_ACCUM_ERR
- rc = get_acc_pkt_err(demod, &pkt);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- #endif
- if (lock_status != DRXJ_DEMOD_LOCK && lock_status != DRX_LOCKED) {
- p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- rc = get_vsb_post_rs_pck_err(dev_addr, &err, &pkt);
- if (rc != 0) {
- pr_err("error %d getting UCB\n", rc);
- p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- p->block_error.stat[0].scale = FE_SCALE_COUNTER;
- p->block_error.stat[0].uvalue += err;
- p->block_count.stat[0].scale = FE_SCALE_COUNTER;
- p->block_count.stat[0].uvalue += pkt;
- }
- /* PostViterbi is compute in steps of 10^(-6) */
- rc = get_vs_bpre_viterbi_ber(dev_addr, &ber, &cnt);
- if (rc != 0) {
- pr_err("error %d getting pre-ber\n", rc);
- p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- p->pre_bit_error.stat[0].scale = FE_SCALE_COUNTER;
- p->pre_bit_error.stat[0].uvalue += ber;
- p->pre_bit_count.stat[0].scale = FE_SCALE_COUNTER;
- p->pre_bit_count.stat[0].uvalue += cnt;
- }
- rc = get_vs_bpost_viterbi_ber(dev_addr, &ber, &cnt);
- if (rc != 0) {
- pr_err("error %d getting post-ber\n", rc);
- p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- p->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
- p->post_bit_error.stat[0].uvalue += ber;
- p->post_bit_count.stat[0].scale = FE_SCALE_COUNTER;
- p->post_bit_count.stat[0].uvalue += cnt;
- }
- rc = get_vsbmer(dev_addr, &mer);
- if (rc != 0) {
- pr_err("error %d getting MER\n", rc);
- p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- } else {
- p->cnr.stat[0].svalue = mer * 100;
- p->cnr.stat[0].scale = FE_SCALE_DECIBEL;
- }
- }
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- rc = ctrl_get_qam_sig_quality(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- #endif
- default:
- return -EIO;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int ctrl_lock_status()
- * \brief Retrieve lock status .
- * \param dev_addr Pointer to demodulator device address.
- * \param lock_stat Pointer to lock status structure.
- * \return int.
- *
- */
- static int
- ctrl_lock_status(struct drx_demod_instance *demod, enum drx_lock_status *lock_stat)
- {
- enum drx_standard standard = DRX_STANDARD_UNKNOWN;
- struct drxj_data *ext_attr = NULL;
- struct i2c_device_addr *dev_addr = NULL;
- struct drxjscu_cmd cmd_scu = { /* command */ 0,
- /* parameter_len */ 0,
- /* result_len */ 0,
- /* *parameter */ NULL,
- /* *result */ NULL
- };
- int rc;
- u16 cmd_result[2] = { 0, 0 };
- u16 demod_lock = SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_DEMOD_LOCKED;
- /* check arguments */
- if ((demod == NULL) || (lock_stat == NULL))
- return -EINVAL;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- standard = ext_attr->standard;
- *lock_stat = DRX_NOT_LOCKED;
- /* define the SCU command code */
- switch (standard) {
- case DRX_STANDARD_8VSB:
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_VSB |
- SCU_RAM_COMMAND_CMD_DEMOD_GET_LOCK;
- demod_lock |= 0x6;
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- cmd_scu.command = SCU_RAM_COMMAND_STANDARD_QAM |
- SCU_RAM_COMMAND_CMD_DEMOD_GET_LOCK;
- break;
- #endif
- case DRX_STANDARD_UNKNOWN: /* fallthrough */
- default:
- return -EIO;
- }
- /* define the SCU command parameters and execute the command */
- cmd_scu.parameter_len = 0;
- cmd_scu.result_len = 2;
- cmd_scu.parameter = NULL;
- cmd_scu.result = cmd_result;
- rc = scu_command(dev_addr, &cmd_scu);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* set the lock status */
- if (cmd_scu.result[1] < demod_lock) {
- /* 0x0000 NOT LOCKED */
- *lock_stat = DRX_NOT_LOCKED;
- } else if (cmd_scu.result[1] < SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_LOCKED) {
- *lock_stat = DRXJ_DEMOD_LOCK;
- } else if (cmd_scu.result[1] <
- SCU_RAM_PARAM_1_RES_DEMOD_GET_LOCK_NEVER_LOCK) {
- /* 0x8000 DEMOD + FEC LOCKED (system lock) */
- *lock_stat = DRX_LOCKED;
- } else {
- /* 0xC000 NEVER LOCKED */
- /* (system will never be able to lock to the signal) */
- *lock_stat = DRX_NEVER_LOCK;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int ctrl_set_standard()
- * \brief Set modulation standard to be used.
- * \param standard Modulation standard.
- * \return int.
- *
- * Setup stuff for the desired demodulation standard.
- * Disable and power down the previous selected demodulation standard
- *
- */
- static int
- ctrl_set_standard(struct drx_demod_instance *demod, enum drx_standard *standard)
- {
- struct drxj_data *ext_attr = NULL;
- int rc;
- enum drx_standard prev_standard;
- /* check arguments */
- if ((standard == NULL) || (demod == NULL))
- return -EINVAL;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- prev_standard = ext_attr->standard;
- /*
- Stop and power down previous standard
- */
- switch (prev_standard) {
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- rc = power_down_qam(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- #endif
- case DRX_STANDARD_8VSB:
- rc = power_down_vsb(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_STANDARD_UNKNOWN:
- /* Do nothing */
- break;
- case DRX_STANDARD_AUTO: /* fallthrough */
- default:
- return -EINVAL;
- }
- /*
- Initialize channel independent registers
- Power up new standard
- */
- ext_attr->standard = *standard;
- switch (*standard) {
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- do {
- u16 dummy;
- rc = drxj_dap_read_reg16(demod->my_i2c_dev_addr, SCU_RAM_VERSION_HI__A, &dummy, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- } while (0);
- break;
- #endif
- case DRX_STANDARD_8VSB:
- rc = set_vsb_leak_n_gain(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- default:
- ext_attr->standard = DRX_STANDARD_UNKNOWN;
- return -EINVAL;
- break;
- }
- return 0;
- rw_error:
- /* Don't know what the standard is now ... try again */
- ext_attr->standard = DRX_STANDARD_UNKNOWN;
- return rc;
- }
- /*============================================================================*/
- static void drxj_reset_mode(struct drxj_data *ext_attr)
- {
- /* Initialize default AFE configuration for QAM */
- if (ext_attr->has_lna) {
- /* IF AGC off, PGA active */
- #ifndef DRXJ_VSB_ONLY
- ext_attr->qam_if_agc_cfg.standard = DRX_STANDARD_ITU_B;
- ext_attr->qam_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_OFF;
- ext_attr->qam_pga_cfg = 140 + (11 * 13);
- #endif
- ext_attr->vsb_if_agc_cfg.standard = DRX_STANDARD_8VSB;
- ext_attr->vsb_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_OFF;
- ext_attr->vsb_pga_cfg = 140 + (11 * 13);
- } else {
- /* IF AGC on, PGA not active */
- #ifndef DRXJ_VSB_ONLY
- ext_attr->qam_if_agc_cfg.standard = DRX_STANDARD_ITU_B;
- ext_attr->qam_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
- ext_attr->qam_if_agc_cfg.min_output_level = 0;
- ext_attr->qam_if_agc_cfg.max_output_level = 0x7FFF;
- ext_attr->qam_if_agc_cfg.speed = 3;
- ext_attr->qam_if_agc_cfg.top = 1297;
- ext_attr->qam_pga_cfg = 140;
- #endif
- ext_attr->vsb_if_agc_cfg.standard = DRX_STANDARD_8VSB;
- ext_attr->vsb_if_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
- ext_attr->vsb_if_agc_cfg.min_output_level = 0;
- ext_attr->vsb_if_agc_cfg.max_output_level = 0x7FFF;
- ext_attr->vsb_if_agc_cfg.speed = 3;
- ext_attr->vsb_if_agc_cfg.top = 1024;
- ext_attr->vsb_pga_cfg = 140;
- }
- /* TODO: remove min_output_level and max_output_level for both QAM and VSB after */
- /* mc has not used them */
- #ifndef DRXJ_VSB_ONLY
- ext_attr->qam_rf_agc_cfg.standard = DRX_STANDARD_ITU_B;
- ext_attr->qam_rf_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
- ext_attr->qam_rf_agc_cfg.min_output_level = 0;
- ext_attr->qam_rf_agc_cfg.max_output_level = 0x7FFF;
- ext_attr->qam_rf_agc_cfg.speed = 3;
- ext_attr->qam_rf_agc_cfg.top = 9500;
- ext_attr->qam_rf_agc_cfg.cut_off_current = 4000;
- ext_attr->qam_pre_saw_cfg.standard = DRX_STANDARD_ITU_B;
- ext_attr->qam_pre_saw_cfg.reference = 0x07;
- ext_attr->qam_pre_saw_cfg.use_pre_saw = true;
- #endif
- /* Initialize default AFE configuration for VSB */
- ext_attr->vsb_rf_agc_cfg.standard = DRX_STANDARD_8VSB;
- ext_attr->vsb_rf_agc_cfg.ctrl_mode = DRX_AGC_CTRL_AUTO;
- ext_attr->vsb_rf_agc_cfg.min_output_level = 0;
- ext_attr->vsb_rf_agc_cfg.max_output_level = 0x7FFF;
- ext_attr->vsb_rf_agc_cfg.speed = 3;
- ext_attr->vsb_rf_agc_cfg.top = 9500;
- ext_attr->vsb_rf_agc_cfg.cut_off_current = 4000;
- ext_attr->vsb_pre_saw_cfg.standard = DRX_STANDARD_8VSB;
- ext_attr->vsb_pre_saw_cfg.reference = 0x07;
- ext_attr->vsb_pre_saw_cfg.use_pre_saw = true;
- }
- /*
- * \fn int ctrl_power_mode()
- * \brief Set the power mode of the device to the specified power mode
- * \param demod Pointer to demodulator instance.
- * \param mode Pointer to new power mode.
- * \return int.
- * \retval 0 Success
- * \retval -EIO I2C error or other failure
- * \retval -EINVAL Invalid mode argument.
- *
- *
- */
- static int
- ctrl_power_mode(struct drx_demod_instance *demod, enum drx_power_mode *mode)
- {
- struct drx_common_attr *common_attr = (struct drx_common_attr *) NULL;
- struct drxj_data *ext_attr = (struct drxj_data *) NULL;
- struct i2c_device_addr *dev_addr = (struct i2c_device_addr *)NULL;
- int rc;
- u16 sio_cc_pwd_mode = 0;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- dev_addr = demod->my_i2c_dev_addr;
- /* Check arguments */
- if (mode == NULL)
- return -EINVAL;
- /* If already in requested power mode, do nothing */
- if (common_attr->current_power_mode == *mode)
- return 0;
- switch (*mode) {
- case DRX_POWER_UP:
- case DRXJ_POWER_DOWN_MAIN_PATH:
- sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_NONE;
- break;
- case DRXJ_POWER_DOWN_CORE:
- sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_CLOCK;
- break;
- case DRXJ_POWER_DOWN_PLL:
- sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_PLL;
- break;
- case DRX_POWER_DOWN:
- sio_cc_pwd_mode = SIO_CC_PWD_MODE_LEVEL_OSC;
- break;
- default:
- /* Unknow sleep mode */
- return -EINVAL;
- break;
- }
- /* Check if device needs to be powered up */
- if ((common_attr->current_power_mode != DRX_POWER_UP)) {
- rc = power_up_device(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- if ((*mode == DRX_POWER_UP)) {
- /* Restore analog & pin configuration */
- /* Initialize default AFE configuration for VSB */
- drxj_reset_mode(ext_attr);
- } else {
- /* Power down to requested mode */
- /* Backup some register settings */
- /* Set pins with possible pull-ups connected to them in input mode */
- /* Analog power down */
- /* ADC power down */
- /* Power down device */
- /* stop all comm_exec */
- /*
- Stop and power down previous standard
- */
- switch (ext_attr->standard) {
- case DRX_STANDARD_ITU_A:
- case DRX_STANDARD_ITU_B:
- case DRX_STANDARD_ITU_C:
- rc = power_down_qam(demod, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_STANDARD_8VSB:
- rc = power_down_vsb(demod, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_STANDARD_PAL_SECAM_BG: /* fallthrough */
- case DRX_STANDARD_PAL_SECAM_DK: /* fallthrough */
- case DRX_STANDARD_PAL_SECAM_I: /* fallthrough */
- case DRX_STANDARD_PAL_SECAM_L: /* fallthrough */
- case DRX_STANDARD_PAL_SECAM_LP: /* fallthrough */
- case DRX_STANDARD_NTSC: /* fallthrough */
- case DRX_STANDARD_FM:
- rc = power_down_atv(demod, ext_attr->standard, true);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- break;
- case DRX_STANDARD_UNKNOWN:
- /* Do nothing */
- break;
- case DRX_STANDARD_AUTO: /* fallthrough */
- default:
- return -EIO;
- }
- ext_attr->standard = DRX_STANDARD_UNKNOWN;
- }
- if (*mode != DRXJ_POWER_DOWN_MAIN_PATH) {
- rc = drxj_dap_write_reg16(dev_addr, SIO_CC_PWD_MODE__A, sio_cc_pwd_mode, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_CC_UPDATE__A, SIO_CC_UPDATE_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if ((*mode != DRX_POWER_UP)) {
- /* Initialize HI, wakeup key especially before put IC to sleep */
- rc = init_hi(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- ext_attr->hi_cfg_ctrl |= SIO_HI_RA_RAM_PAR_5_CFG_SLEEP_ZZZ;
- rc = hi_cfg_command(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- }
- common_attr->current_power_mode = *mode;
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*== CTRL Set/Get Config related functions ===================================*/
- /*============================================================================*/
- /*
- * \fn int ctrl_set_cfg_pre_saw()
- * \brief Set Pre-saw reference.
- * \param demod demod instance
- * \param u16 *
- * \return int.
- *
- * Check arguments
- * Dispatch handling to standard specific function.
- *
- */
- static int
- ctrl_set_cfg_pre_saw(struct drx_demod_instance *demod, struct drxj_cfg_pre_saw *pre_saw)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- int rc;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- /* check arguments */
- if ((pre_saw == NULL) || (pre_saw->reference > IQM_AF_PDREF__M)
- ) {
- return -EINVAL;
- }
- /* Only if standard is currently active */
- if ((ext_attr->standard == pre_saw->standard) ||
- (DRXJ_ISQAMSTD(ext_attr->standard) &&
- DRXJ_ISQAMSTD(pre_saw->standard)) ||
- (DRXJ_ISATVSTD(ext_attr->standard) &&
- DRXJ_ISATVSTD(pre_saw->standard))) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PDREF__A, pre_saw->reference, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* Store pre-saw settings */
- switch (pre_saw->standard) {
- case DRX_STANDARD_8VSB:
- ext_attr->vsb_pre_saw_cfg = *pre_saw;
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- ext_attr->qam_pre_saw_cfg = *pre_saw;
- break;
- #endif
- default:
- return -EINVAL;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn int ctrl_set_cfg_afe_gain()
- * \brief Set AFE Gain.
- * \param demod demod instance
- * \param u16 *
- * \return int.
- *
- * Check arguments
- * Dispatch handling to standard specific function.
- *
- */
- static int
- ctrl_set_cfg_afe_gain(struct drx_demod_instance *demod, struct drxj_cfg_afe_gain *afe_gain)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- int rc;
- u8 gain = 0;
- /* check arguments */
- if (afe_gain == NULL)
- return -EINVAL;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- switch (afe_gain->standard) {
- case DRX_STANDARD_8VSB: /* fallthrough */
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- #endif
- /* Do nothing */
- break;
- default:
- return -EINVAL;
- }
- /* TODO PGA gain is also written by microcode (at least by QAM and VSB)
- So I (PJ) think interface requires choice between auto, user mode */
- if (afe_gain->gain >= 329)
- gain = 15;
- else if (afe_gain->gain <= 147)
- gain = 0;
- else
- gain = (afe_gain->gain - 140 + 6) / 13;
- /* Only if standard is currently active */
- if (ext_attr->standard == afe_gain->standard) {
- rc = drxj_dap_write_reg16(dev_addr, IQM_AF_PGA_GAIN__A, gain, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- }
- /* Store AFE Gain settings */
- switch (afe_gain->standard) {
- case DRX_STANDARD_8VSB:
- ext_attr->vsb_pga_cfg = gain * 13 + 140;
- break;
- #ifndef DRXJ_VSB_ONLY
- case DRX_STANDARD_ITU_A: /* fallthrough */
- case DRX_STANDARD_ITU_B: /* fallthrough */
- case DRX_STANDARD_ITU_C:
- ext_attr->qam_pga_cfg = gain * 13 + 140;
- break;
- #endif
- default:
- return -EIO;
- }
- return 0;
- rw_error:
- return rc;
- }
- /*============================================================================*/
- /*=============================================================================
- ===== EXPORTED FUNCTIONS ====================================================*/
- static int drx_ctrl_u_code(struct drx_demod_instance *demod,
- struct drxu_code_info *mc_info,
- enum drxu_code_action action);
- static int drxj_set_lna_state(struct drx_demod_instance *demod, bool state);
- /*
- * \fn drxj_open()
- * \brief Open the demod instance, configure device, configure drxdriver
- * \return Status_t Return status.
- *
- * drxj_open() can be called with a NULL ucode image => no ucode upload.
- * This means that drxj_open() must NOT contain SCU commands or, in general,
- * rely on SCU or AUD ucode to be present.
- *
- */
- static int drxj_open(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = NULL;
- struct drxj_data *ext_attr = NULL;
- struct drx_common_attr *common_attr = NULL;
- u32 driver_version = 0;
- struct drxu_code_info ucode_info;
- struct drx_cfg_mpeg_output cfg_mpeg_output;
- int rc;
- enum drx_power_mode power_mode = DRX_POWER_UP;
- if ((demod == NULL) ||
- (demod->my_common_attr == NULL) ||
- (demod->my_ext_attr == NULL) ||
- (demod->my_i2c_dev_addr == NULL) ||
- (demod->my_common_attr->is_opened)) {
- return -EINVAL;
- }
- /* Check arguments */
- if (demod->my_ext_attr == NULL)
- return -EINVAL;
- dev_addr = demod->my_i2c_dev_addr;
- ext_attr = (struct drxj_data *) demod->my_ext_attr;
- common_attr = (struct drx_common_attr *) demod->my_common_attr;
- rc = ctrl_power_mode(demod, &power_mode);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- if (power_mode != DRX_POWER_UP) {
- rc = -EINVAL;
- pr_err("failed to powerup device\n");
- goto rw_error;
- }
- /* has to be in front of setIqmAf and setOrxNsuAox */
- rc = get_device_capabilities(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /*
- * Soft reset of sys- and osc-clockdomain
- *
- * HACK: On windows, it writes a 0x07 here, instead of just 0x03.
- * As we didn't load the firmware here yet, we should do the same.
- * Btw, this is coherent with DRX-K, where we send reset codes
- * for modulation (OFTM, in DRX-k), SYS and OSC clock domains.
- */
- rc = drxj_dap_write_reg16(dev_addr, SIO_CC_SOFT_RST__A, (0x04 | SIO_CC_SOFT_RST_SYS__M | SIO_CC_SOFT_RST_OSC__M), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SIO_CC_UPDATE__A, SIO_CC_UPDATE_KEY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- msleep(1);
- /* TODO first make sure that everything keeps working before enabling this */
- /* PowerDownAnalogBlocks() */
- rc = drxj_dap_write_reg16(dev_addr, ATV_TOP_STDBY__A, (~ATV_TOP_STDBY_CVBS_STDBY_A2_ACTIVE) | ATV_TOP_STDBY_SIF_STDBY_STANDBY, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_iqm_af(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = set_orx_nsu_aox(demod, false);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = init_hi(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* disable mpegoutput pins */
- memcpy(&cfg_mpeg_output, &common_attr->mpeg_cfg, sizeof(cfg_mpeg_output));
- cfg_mpeg_output.enable_mpeg_output = false;
- rc = ctrl_set_cfg_mpeg_output(demod, &cfg_mpeg_output);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Stop AUD Inform SetAudio it will need to do all setting */
- rc = power_down_aud(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Stop SCU */
- rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_STOP, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Upload microcode */
- if (common_attr->microcode_file != NULL) {
- /* Dirty trick to use common ucode upload & verify,
- pretend device is already open */
- common_attr->is_opened = true;
- ucode_info.mc_file = common_attr->microcode_file;
- if (DRX_ISPOWERDOWNMODE(demod->my_common_attr->current_power_mode)) {
- pr_err("Should powerup before loading the firmware.");
- return -EINVAL;
- }
- rc = drx_ctrl_u_code(demod, &ucode_info, UCODE_UPLOAD);
- if (rc != 0) {
- pr_err("error %d while uploading the firmware\n", rc);
- goto rw_error;
- }
- if (common_attr->verify_microcode == true) {
- rc = drx_ctrl_u_code(demod, &ucode_info, UCODE_VERIFY);
- if (rc != 0) {
- pr_err("error %d while verifying the firmware\n",
- rc);
- goto rw_error;
- }
- }
- common_attr->is_opened = false;
- }
- /* Run SCU for a little while to initialize microcode version numbers */
- rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Initialize scan timeout */
- common_attr->scan_demod_lock_timeout = DRXJ_SCAN_TIMEOUT;
- common_attr->scan_desired_lock = DRX_LOCKED;
- drxj_reset_mode(ext_attr);
- ext_attr->standard = DRX_STANDARD_UNKNOWN;
- rc = smart_ant_init(demod);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* Stamp driver version number in SCU data RAM in BCD code
- Done to enable field application engineers to retrieve drxdriver version
- via I2C from SCU RAM
- */
- driver_version = (VERSION_MAJOR / 100) % 10;
- driver_version <<= 4;
- driver_version += (VERSION_MAJOR / 10) % 10;
- driver_version <<= 4;
- driver_version += (VERSION_MAJOR % 10);
- driver_version <<= 4;
- driver_version += (VERSION_MINOR % 10);
- driver_version <<= 4;
- driver_version += (VERSION_PATCH / 1000) % 10;
- driver_version <<= 4;
- driver_version += (VERSION_PATCH / 100) % 10;
- driver_version <<= 4;
- driver_version += (VERSION_PATCH / 10) % 10;
- driver_version <<= 4;
- driver_version += (VERSION_PATCH % 10);
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_DRIVER_VER_HI__A, (u16)(driver_version >> 16), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_RAM_DRIVER_VER_LO__A, (u16)(driver_version & 0xFFFF), 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = ctrl_set_oob(demod, NULL);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- /* refresh the audio data structure with default */
- ext_attr->aud_data = drxj_default_aud_data_g;
- demod->my_common_attr->is_opened = true;
- drxj_set_lna_state(demod, false);
- return 0;
- rw_error:
- common_attr->is_opened = false;
- return rc;
- }
- /*============================================================================*/
- /*
- * \fn drxj_close()
- * \brief Close the demod instance, power down the device
- * \return Status_t Return status.
- *
- */
- static int drxj_close(struct drx_demod_instance *demod)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- enum drx_power_mode power_mode = DRX_POWER_UP;
- if ((demod->my_common_attr == NULL) ||
- (demod->my_ext_attr == NULL) ||
- (demod->my_i2c_dev_addr == NULL) ||
- (!demod->my_common_attr->is_opened)) {
- return -EINVAL;
- }
- /* power up */
- rc = ctrl_power_mode(demod, &power_mode);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- rc = drxj_dap_write_reg16(dev_addr, SCU_COMM_EXEC__A, SCU_COMM_EXEC_ACTIVE, 0);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- power_mode = DRX_POWER_DOWN;
- rc = ctrl_power_mode(demod, &power_mode);
- if (rc != 0) {
- pr_err("error %d\n", rc);
- goto rw_error;
- }
- DRX_ATTR_ISOPENED(demod) = false;
- return 0;
- rw_error:
- DRX_ATTR_ISOPENED(demod) = false;
- return rc;
- }
- /*
- * Microcode related functions
- */
- /*
- * drx_u_code_compute_crc - Compute CRC of block of microcode data.
- * @block_data: Pointer to microcode data.
- * @nr_words: Size of microcode block (number of 16 bits words).
- *
- * returns The computed CRC residue.
- */
- static u16 drx_u_code_compute_crc(u8 *block_data, u16 nr_words)
- {
- u16 i = 0;
- u16 j = 0;
- u32 crc_word = 0;
- u32 carry = 0;
- while (i < nr_words) {
- crc_word |= (u32)be16_to_cpu(*(__be16 *)(block_data));
- for (j = 0; j < 16; j++) {
- crc_word <<= 1;
- if (carry != 0)
- crc_word ^= 0x80050000UL;
- carry = crc_word & 0x80000000UL;
- }
- i++;
- block_data += (sizeof(u16));
- }
- return (u16)(crc_word >> 16);
- }
- /*
- * drx_check_firmware - checks if the loaded firmware is valid
- *
- * @demod: demod structure
- * @mc_data: pointer to the start of the firmware
- * @size: firmware size
- */
- static int drx_check_firmware(struct drx_demod_instance *demod, u8 *mc_data,
- unsigned size)
- {
- struct drxu_code_block_hdr block_hdr;
- int i;
- unsigned count = 2 * sizeof(u16);
- u32 mc_dev_type, mc_version, mc_base_version;
- u16 mc_nr_of_blks = be16_to_cpu(*(__be16 *)(mc_data + sizeof(u16)));
- /*
- * Scan microcode blocks first for version info
- * and firmware check
- */
- /* Clear version block */
- DRX_ATTR_MCRECORD(demod).aux_type = 0;
- DRX_ATTR_MCRECORD(demod).mc_dev_type = 0;
- DRX_ATTR_MCRECORD(demod).mc_version = 0;
- DRX_ATTR_MCRECORD(demod).mc_base_version = 0;
- for (i = 0; i < mc_nr_of_blks; i++) {
- if (count + 3 * sizeof(u16) + sizeof(u32) > size)
- goto eof;
- /* Process block header */
- block_hdr.addr = be32_to_cpu(*(__be32 *)(mc_data + count));
- count += sizeof(u32);
- block_hdr.size = be16_to_cpu(*(__be16 *)(mc_data + count));
- count += sizeof(u16);
- block_hdr.flags = be16_to_cpu(*(__be16 *)(mc_data + count));
- count += sizeof(u16);
- block_hdr.CRC = be16_to_cpu(*(__be16 *)(mc_data + count));
- count += sizeof(u16);
- pr_debug("%u: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
- count, block_hdr.addr, block_hdr.size, block_hdr.flags,
- block_hdr.CRC);
- if (block_hdr.flags & 0x8) {
- u8 *auxblk = ((void *)mc_data) + block_hdr.addr;
- u16 auxtype;
- if (block_hdr.addr + sizeof(u16) > size)
- goto eof;
- auxtype = be16_to_cpu(*(__be16 *)(auxblk));
- /* Aux block. Check type */
- if (DRX_ISMCVERTYPE(auxtype)) {
- if (block_hdr.addr + 2 * sizeof(u16) + 2 * sizeof (u32) > size)
- goto eof;
- auxblk += sizeof(u16);
- mc_dev_type = be32_to_cpu(*(__be32 *)(auxblk));
- auxblk += sizeof(u32);
- mc_version = be32_to_cpu(*(__be32 *)(auxblk));
- auxblk += sizeof(u32);
- mc_base_version = be32_to_cpu(*(__be32 *)(auxblk));
- DRX_ATTR_MCRECORD(demod).aux_type = auxtype;
- DRX_ATTR_MCRECORD(demod).mc_dev_type = mc_dev_type;
- DRX_ATTR_MCRECORD(demod).mc_version = mc_version;
- DRX_ATTR_MCRECORD(demod).mc_base_version = mc_base_version;
- pr_info("Firmware dev %x, ver %x, base ver %x\n",
- mc_dev_type, mc_version, mc_base_version);
- }
- } else if (count + block_hdr.size * sizeof(u16) > size)
- goto eof;
- count += block_hdr.size * sizeof(u16);
- }
- return 0;
- eof:
- pr_err("Firmware is truncated at pos %u/%u\n", count, size);
- return -EINVAL;
- }
- /*
- * drx_ctrl_u_code - Handle microcode upload or verify.
- * @dev_addr: Address of device.
- * @mc_info: Pointer to information about microcode data.
- * @action: Either UCODE_UPLOAD or UCODE_VERIFY
- *
- * This function returns:
- * 0:
- * - In case of UCODE_UPLOAD: code is successfully uploaded.
- * - In case of UCODE_VERIFY: image on device is equal to
- * image provided to this control function.
- * -EIO:
- * - In case of UCODE_UPLOAD: I2C error.
- * - In case of UCODE_VERIFY: I2C error or image on device
- * is not equal to image provided to this control function.
- * -EINVAL:
- * - Invalid arguments.
- * - Provided image is corrupt
- */
- static int drx_ctrl_u_code(struct drx_demod_instance *demod,
- struct drxu_code_info *mc_info,
- enum drxu_code_action action)
- {
- struct i2c_device_addr *dev_addr = demod->my_i2c_dev_addr;
- int rc;
- u16 i = 0;
- u16 mc_nr_of_blks = 0;
- u16 mc_magic_word = 0;
- const u8 *mc_data_init = NULL;
- u8 *mc_data = NULL;
- unsigned size;
- char *mc_file;
- /* Check arguments */
- if (!mc_info || !mc_info->mc_file)
- return -EINVAL;
- mc_file = mc_info->mc_file;
- if (!demod->firmware) {
- const struct firmware *fw = NULL;
- rc = request_firmware(&fw, mc_file, demod->i2c->dev.parent);
- if (rc < 0) {
- pr_err("Couldn't read firmware %s\n", mc_file);
- return rc;
- }
- demod->firmware = fw;
- if (demod->firmware->size < 2 * sizeof(u16)) {
- rc = -EINVAL;
- pr_err("Firmware is too short!\n");
- goto release;
- }
- pr_info("Firmware %s, size %zu\n",
- mc_file, demod->firmware->size);
- }
- mc_data_init = demod->firmware->data;
- size = demod->firmware->size;
- mc_data = (void *)mc_data_init;
- /* Check data */
- mc_magic_word = be16_to_cpu(*(__be16 *)(mc_data));
- mc_data += sizeof(u16);
- mc_nr_of_blks = be16_to_cpu(*(__be16 *)(mc_data));
- mc_data += sizeof(u16);
- if ((mc_magic_word != DRX_UCODE_MAGIC_WORD) || (mc_nr_of_blks == 0)) {
- rc = -EINVAL;
- pr_err("Firmware magic word doesn't match\n");
- goto release;
- }
- if (action == UCODE_UPLOAD) {
- rc = drx_check_firmware(demod, (u8 *)mc_data_init, size);
- if (rc)
- goto release;
- pr_info("Uploading firmware %s\n", mc_file);
- } else {
- pr_info("Verifying if firmware upload was ok.\n");
- }
- /* Process microcode blocks */
- for (i = 0; i < mc_nr_of_blks; i++) {
- struct drxu_code_block_hdr block_hdr;
- u16 mc_block_nr_bytes = 0;
- /* Process block header */
- block_hdr.addr = be32_to_cpu(*(__be32 *)(mc_data));
- mc_data += sizeof(u32);
- block_hdr.size = be16_to_cpu(*(__be16 *)(mc_data));
- mc_data += sizeof(u16);
- block_hdr.flags = be16_to_cpu(*(__be16 *)(mc_data));
- mc_data += sizeof(u16);
- block_hdr.CRC = be16_to_cpu(*(__be16 *)(mc_data));
- mc_data += sizeof(u16);
- pr_debug("%u: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
- (unsigned)(mc_data - mc_data_init), block_hdr.addr,
- block_hdr.size, block_hdr.flags, block_hdr.CRC);
- /* Check block header on:
- - data larger than 64Kb
- - if CRC enabled check CRC
- */
- if ((block_hdr.size > 0x7FFF) ||
- (((block_hdr.flags & DRX_UCODE_CRC_FLAG) != 0) &&
- (block_hdr.CRC != drx_u_code_compute_crc(mc_data, block_hdr.size)))
- ) {
- /* Wrong data ! */
- rc = -EINVAL;
- pr_err("firmware CRC is wrong\n");
- goto release;
- }
- if (!block_hdr.size)
- continue;
- mc_block_nr_bytes = block_hdr.size * ((u16) sizeof(u16));
- /* Perform the desired action */
- switch (action) {
- case UCODE_UPLOAD: /* Upload microcode */
- if (drxdap_fasi_write_block(dev_addr,
- block_hdr.addr,
- mc_block_nr_bytes,
- mc_data, 0x0000)) {
- rc = -EIO;
- pr_err("error writing firmware at pos %u\n",
- (unsigned)(mc_data - mc_data_init));
- goto release;
- }
- break;
- case UCODE_VERIFY: { /* Verify uploaded microcode */
- int result = 0;
- u8 mc_data_buffer[DRX_UCODE_MAX_BUF_SIZE];
- u32 bytes_to_comp = 0;
- u32 bytes_left = mc_block_nr_bytes;
- u32 curr_addr = block_hdr.addr;
- u8 *curr_ptr = mc_data;
- while (bytes_left != 0) {
- if (bytes_left > DRX_UCODE_MAX_BUF_SIZE)
- bytes_to_comp = DRX_UCODE_MAX_BUF_SIZE;
- else
- bytes_to_comp = bytes_left;
- if (drxdap_fasi_read_block(dev_addr,
- curr_addr,
- (u16)bytes_to_comp,
- (u8 *)mc_data_buffer,
- 0x0000)) {
- pr_err("error reading firmware at pos %u\n",
- (unsigned)(mc_data - mc_data_init));
- return -EIO;
- }
- result = memcmp(curr_ptr, mc_data_buffer,
- bytes_to_comp);
- if (result) {
- pr_err("error verifying firmware at pos %u\n",
- (unsigned)(mc_data - mc_data_init));
- return -EIO;
- }
- curr_addr += ((dr_xaddr_t)(bytes_to_comp / 2));
- curr_ptr =&(curr_ptr[bytes_to_comp]);
- bytes_left -=((u32) bytes_to_comp);
- }
- break;
- }
- default:
- return -EINVAL;
- break;
- }
- mc_data += mc_block_nr_bytes;
- }
- return 0;
- release:
- release_firmware(demod->firmware);
- demod->firmware = NULL;
- return rc;
- }
- /* caller is expected to check if lna is supported before enabling */
- static int drxj_set_lna_state(struct drx_demod_instance *demod, bool state)
- {
- struct drxuio_cfg uio_cfg;
- struct drxuio_data uio_data;
- int result;
- uio_cfg.uio = DRX_UIO1;
- uio_cfg.mode = DRX_UIO_MODE_READWRITE;
- /* Configure user-I/O #3: enable read/write */
- result = ctrl_set_uio_cfg(demod, &uio_cfg);
- if (result) {
- pr_err("Failed to setup LNA GPIO!\n");
- return result;
- }
- uio_data.uio = DRX_UIO1;
- uio_data.value = state;
- result = ctrl_uio_write(demod, &uio_data);
- if (result != 0) {
- pr_err("Failed to %sable LNA!\n",
- state ? "en" : "dis");
- return result;
- }
- return 0;
- }
- /*
- * The Linux DVB Driver for Micronas DRX39xx family (drx3933j)
- *
- * Written by Devin Heitmueller <devin.heitmueller@kernellabs.com>
- */
- static int drx39xxj_set_powerstate(struct dvb_frontend *fe, int enable)
- {
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- int result;
- enum drx_power_mode power_mode;
- if (enable)
- power_mode = DRX_POWER_UP;
- else
- power_mode = DRX_POWER_DOWN;
- result = ctrl_power_mode(demod, &power_mode);
- if (result != 0) {
- pr_err("Power state change failed\n");
- return 0;
- }
- return 0;
- }
- static int drx39xxj_read_status(struct dvb_frontend *fe, enum fe_status *status)
- {
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- int result;
- enum drx_lock_status lock_status;
- *status = 0;
- result = ctrl_lock_status(demod, &lock_status);
- if (result != 0) {
- pr_err("drx39xxj: could not get lock status!\n");
- *status = 0;
- }
- switch (lock_status) {
- case DRX_NEVER_LOCK:
- *status = 0;
- pr_err("drx says NEVER_LOCK\n");
- break;
- case DRX_NOT_LOCKED:
- *status = 0;
- break;
- case DRX_LOCK_STATE_1:
- case DRX_LOCK_STATE_2:
- case DRX_LOCK_STATE_3:
- case DRX_LOCK_STATE_4:
- case DRX_LOCK_STATE_5:
- case DRX_LOCK_STATE_6:
- case DRX_LOCK_STATE_7:
- case DRX_LOCK_STATE_8:
- case DRX_LOCK_STATE_9:
- *status = FE_HAS_SIGNAL
- | FE_HAS_CARRIER | FE_HAS_VITERBI | FE_HAS_SYNC;
- break;
- case DRX_LOCKED:
- *status = FE_HAS_SIGNAL
- | FE_HAS_CARRIER
- | FE_HAS_VITERBI | FE_HAS_SYNC | FE_HAS_LOCK;
- break;
- default:
- pr_err("Lock state unknown %d\n", lock_status);
- }
- ctrl_sig_quality(demod, lock_status);
- return 0;
- }
- static int drx39xxj_read_ber(struct dvb_frontend *fe, u32 *ber)
- {
- struct dtv_frontend_properties *p = &fe->dtv_property_cache;
- if (p->pre_bit_error.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
- *ber = 0;
- return 0;
- }
- if (!p->pre_bit_count.stat[0].uvalue) {
- if (!p->pre_bit_error.stat[0].uvalue)
- *ber = 0;
- else
- *ber = 1000000;
- } else {
- *ber = frac_times1e6(p->pre_bit_error.stat[0].uvalue,
- p->pre_bit_count.stat[0].uvalue);
- }
- return 0;
- }
- static int drx39xxj_read_signal_strength(struct dvb_frontend *fe,
- u16 *strength)
- {
- struct dtv_frontend_properties *p = &fe->dtv_property_cache;
- if (p->strength.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
- *strength = 0;
- return 0;
- }
- *strength = p->strength.stat[0].uvalue;
- return 0;
- }
- static int drx39xxj_read_snr(struct dvb_frontend *fe, u16 *snr)
- {
- struct dtv_frontend_properties *p = &fe->dtv_property_cache;
- u64 tmp64;
- if (p->cnr.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
- *snr = 0;
- return 0;
- }
- tmp64 = p->cnr.stat[0].svalue;
- do_div(tmp64, 10);
- *snr = tmp64;
- return 0;
- }
- static int drx39xxj_read_ucblocks(struct dvb_frontend *fe, u32 *ucb)
- {
- struct dtv_frontend_properties *p = &fe->dtv_property_cache;
- if (p->block_error.stat[0].scale == FE_SCALE_NOT_AVAILABLE) {
- *ucb = 0;
- return 0;
- }
- *ucb = p->block_error.stat[0].uvalue;
- return 0;
- }
- static int drx39xxj_set_frontend(struct dvb_frontend *fe)
- {
- #ifdef DJH_DEBUG
- int i;
- #endif
- struct dtv_frontend_properties *p = &fe->dtv_property_cache;
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- enum drx_standard standard = DRX_STANDARD_8VSB;
- struct drx_channel channel;
- int result;
- static const struct drx_channel def_channel = {
- /* frequency */ 0,
- /* bandwidth */ DRX_BANDWIDTH_6MHZ,
- /* mirror */ DRX_MIRROR_NO,
- /* constellation */ DRX_CONSTELLATION_AUTO,
- /* hierarchy */ DRX_HIERARCHY_UNKNOWN,
- /* priority */ DRX_PRIORITY_UNKNOWN,
- /* coderate */ DRX_CODERATE_UNKNOWN,
- /* guard */ DRX_GUARD_UNKNOWN,
- /* fftmode */ DRX_FFTMODE_UNKNOWN,
- /* classification */ DRX_CLASSIFICATION_AUTO,
- /* symbolrate */ 5057000,
- /* interleavemode */ DRX_INTERLEAVEMODE_UNKNOWN,
- /* ldpc */ DRX_LDPC_UNKNOWN,
- /* carrier */ DRX_CARRIER_UNKNOWN,
- /* frame mode */ DRX_FRAMEMODE_UNKNOWN
- };
- u32 constellation = DRX_CONSTELLATION_AUTO;
- /* Bring the demod out of sleep */
- drx39xxj_set_powerstate(fe, 1);
- if (fe->ops.tuner_ops.set_params) {
- u32 int_freq;
- if (fe->ops.i2c_gate_ctrl)
- fe->ops.i2c_gate_ctrl(fe, 1);
- /* Set tuner to desired frequency and standard */
- fe->ops.tuner_ops.set_params(fe);
- /* Use the tuner's IF */
- if (fe->ops.tuner_ops.get_if_frequency) {
- fe->ops.tuner_ops.get_if_frequency(fe, &int_freq);
- demod->my_common_attr->intermediate_freq = int_freq / 1000;
- }
- if (fe->ops.i2c_gate_ctrl)
- fe->ops.i2c_gate_ctrl(fe, 0);
- }
- switch (p->delivery_system) {
- case SYS_ATSC:
- standard = DRX_STANDARD_8VSB;
- break;
- case SYS_DVBC_ANNEX_B:
- standard = DRX_STANDARD_ITU_B;
- switch (p->modulation) {
- case QAM_64:
- constellation = DRX_CONSTELLATION_QAM64;
- break;
- case QAM_256:
- constellation = DRX_CONSTELLATION_QAM256;
- break;
- default:
- constellation = DRX_CONSTELLATION_AUTO;
- break;
- }
- break;
- default:
- return -EINVAL;
- }
- /* Set the standard (will be powered up if necessary */
- result = ctrl_set_standard(demod, &standard);
- if (result != 0) {
- pr_err("Failed to set standard! result=%02x\n",
- result);
- return -EINVAL;
- }
- /* set channel parameters */
- channel = def_channel;
- channel.frequency = p->frequency / 1000;
- channel.bandwidth = DRX_BANDWIDTH_6MHZ;
- channel.constellation = constellation;
- /* program channel */
- result = ctrl_set_channel(demod, &channel);
- if (result != 0) {
- pr_err("Failed to set channel!\n");
- return -EINVAL;
- }
- /* Just for giggles, let's shut off the LNA again.... */
- drxj_set_lna_state(demod, false);
- /* After set_frontend, except for strength, stats aren't available */
- p->strength.stat[0].scale = FE_SCALE_RELATIVE;
- return 0;
- }
- static int drx39xxj_sleep(struct dvb_frontend *fe)
- {
- /* power-down the demodulator */
- return drx39xxj_set_powerstate(fe, 0);
- }
- static int drx39xxj_i2c_gate_ctrl(struct dvb_frontend *fe, int enable)
- {
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- bool i2c_gate_state;
- int result;
- #ifdef DJH_DEBUG
- pr_debug("i2c gate call: enable=%d state=%d\n", enable,
- state->i2c_gate_open);
- #endif
- if (enable)
- i2c_gate_state = true;
- else
- i2c_gate_state = false;
- if (state->i2c_gate_open == enable) {
- /* We're already in the desired state */
- return 0;
- }
- result = ctrl_i2c_bridge(demod, &i2c_gate_state);
- if (result != 0) {
- pr_err("drx39xxj: could not open i2c gate [%d]\n",
- result);
- dump_stack();
- } else {
- state->i2c_gate_open = enable;
- }
- return 0;
- }
- static int drx39xxj_init(struct dvb_frontend *fe)
- {
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- int rc = 0;
- if (fe->exit == DVB_FE_DEVICE_RESUME) {
- /* so drxj_open() does what it needs to do */
- demod->my_common_attr->is_opened = false;
- rc = drxj_open(demod);
- if (rc != 0)
- pr_err("drx39xxj_init(): DRX open failed rc=%d!\n", rc);
- } else
- drx39xxj_set_powerstate(fe, 1);
- return rc;
- }
- static int drx39xxj_set_lna(struct dvb_frontend *fe)
- {
- struct dtv_frontend_properties *c = &fe->dtv_property_cache;
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- struct drxj_data *ext_attr = demod->my_ext_attr;
- if (c->lna) {
- if (!ext_attr->has_lna) {
- pr_err("LNA is not supported on this device!\n");
- return -EINVAL;
- }
- }
- return drxj_set_lna_state(demod, c->lna);
- }
- static int drx39xxj_get_tune_settings(struct dvb_frontend *fe,
- struct dvb_frontend_tune_settings *tune)
- {
- tune->min_delay_ms = 1000;
- return 0;
- }
- static void drx39xxj_release(struct dvb_frontend *fe)
- {
- struct drx39xxj_state *state = fe->demodulator_priv;
- struct drx_demod_instance *demod = state->demod;
- /* if device is removed don't access it */
- if (fe->exit != DVB_FE_DEVICE_REMOVED)
- drxj_close(demod);
- kfree(demod->my_ext_attr);
- kfree(demod->my_common_attr);
- kfree(demod->my_i2c_dev_addr);
- release_firmware(demod->firmware);
- kfree(demod);
- kfree(state);
- }
- static const struct dvb_frontend_ops drx39xxj_ops;
- struct dvb_frontend *drx39xxj_attach(struct i2c_adapter *i2c)
- {
- struct drx39xxj_state *state = NULL;
- struct i2c_device_addr *demod_addr = NULL;
- struct drx_common_attr *demod_comm_attr = NULL;
- struct drxj_data *demod_ext_attr = NULL;
- struct drx_demod_instance *demod = NULL;
- struct dtv_frontend_properties *p;
- int result;
- /* allocate memory for the internal state */
- state = kzalloc(sizeof(struct drx39xxj_state), GFP_KERNEL);
- if (state == NULL)
- goto error;
- demod = kmalloc(sizeof(struct drx_demod_instance), GFP_KERNEL);
- if (demod == NULL)
- goto error;
- demod_addr = kmemdup(&drxj_default_addr_g,
- sizeof(struct i2c_device_addr), GFP_KERNEL);
- if (demod_addr == NULL)
- goto error;
- demod_comm_attr = kmemdup(&drxj_default_comm_attr_g,
- sizeof(struct drx_common_attr), GFP_KERNEL);
- if (demod_comm_attr == NULL)
- goto error;
- demod_ext_attr = kmemdup(&drxj_data_g, sizeof(struct drxj_data),
- GFP_KERNEL);
- if (demod_ext_attr == NULL)
- goto error;
- /* setup the state */
- state->i2c = i2c;
- state->demod = demod;
- /* setup the demod data */
- memcpy(demod, &drxj_default_demod_g, sizeof(struct drx_demod_instance));
- demod->my_i2c_dev_addr = demod_addr;
- demod->my_common_attr = demod_comm_attr;
- demod->my_i2c_dev_addr->user_data = state;
- demod->my_common_attr->microcode_file = DRX39XX_MAIN_FIRMWARE;
- demod->my_common_attr->verify_microcode = true;
- demod->my_common_attr->intermediate_freq = 5000;
- demod->my_common_attr->current_power_mode = DRX_POWER_DOWN;
- demod->my_ext_attr = demod_ext_attr;
- ((struct drxj_data *)demod_ext_attr)->uio_sma_tx_mode = DRX_UIO_MODE_READWRITE;
- demod->i2c = i2c;
- result = drxj_open(demod);
- if (result != 0) {
- pr_err("DRX open failed! Aborting\n");
- goto error;
- }
- /* create dvb_frontend */
- memcpy(&state->frontend.ops, &drx39xxj_ops,
- sizeof(struct dvb_frontend_ops));
- state->frontend.demodulator_priv = state;
- /* Initialize stats - needed for DVBv5 stats to work */
- p = &state->frontend.dtv_property_cache;
- p->strength.len = 1;
- p->pre_bit_count.len = 1;
- p->pre_bit_error.len = 1;
- p->post_bit_count.len = 1;
- p->post_bit_error.len = 1;
- p->block_count.len = 1;
- p->block_error.len = 1;
- p->cnr.len = 1;
- p->strength.stat[0].scale = FE_SCALE_RELATIVE;
- p->pre_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->pre_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->block_count.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- p->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
- return &state->frontend;
- error:
- kfree(demod_ext_attr);
- kfree(demod_comm_attr);
- kfree(demod_addr);
- kfree(demod);
- kfree(state);
- return NULL;
- }
- EXPORT_SYMBOL(drx39xxj_attach);
- static const struct dvb_frontend_ops drx39xxj_ops = {
- .delsys = { SYS_ATSC, SYS_DVBC_ANNEX_B },
- .info = {
- .name = "Micronas DRX39xxj family Frontend",
- .frequency_stepsize = 62500,
- .frequency_min = 51000000,
- .frequency_max = 858000000,
- .caps = FE_CAN_QAM_64 | FE_CAN_QAM_256 | FE_CAN_8VSB
- },
- .init = drx39xxj_init,
- .i2c_gate_ctrl = drx39xxj_i2c_gate_ctrl,
- .sleep = drx39xxj_sleep,
- .set_frontend = drx39xxj_set_frontend,
- .get_tune_settings = drx39xxj_get_tune_settings,
- .read_status = drx39xxj_read_status,
- .read_ber = drx39xxj_read_ber,
- .read_signal_strength = drx39xxj_read_signal_strength,
- .read_snr = drx39xxj_read_snr,
- .read_ucblocks = drx39xxj_read_ucblocks,
- .release = drx39xxj_release,
- .set_lna = drx39xxj_set_lna,
- };
- MODULE_DESCRIPTION("Micronas DRX39xxj Frontend");
- MODULE_AUTHOR("Devin Heitmueller");
- MODULE_LICENSE("GPL");
- MODULE_FIRMWARE(DRX39XX_MAIN_FIRMWARE);
|