q6asm.c 295 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868786978707871787278737874787578767877787878797880788178827883788478857886788778887889789078917892789378947895789678977898789979007901790279037904790579067907790879097910791179127913791479157916791779187919792079217922792379247925792679277928792979307931793279337934793579367937793879397940794179427943794479457946794779487949795079517952795379547955795679577958795979607961796279637964796579667967796879697970797179727973797479757976797779787979798079817982798379847985798679877988798979907991799279937994799579967997799879998000800180028003800480058006800780088009801080118012801380148015801680178018801980208021802280238024802580268027802880298030803180328033803480358036803780388039804080418042804380448045804680478048804980508051805280538054805580568057805880598060806180628063806480658066806780688069807080718072807380748075807680778078807980808081808280838084808580868087808880898090809180928093809480958096809780988099810081018102810381048105810681078108810981108111811281138114811581168117811881198120812181228123812481258126812781288129813081318132813381348135813681378138813981408141814281438144814581468147814881498150815181528153815481558156815781588159816081618162816381648165816681678168816981708171817281738174817581768177817881798180818181828183818481858186818781888189819081918192819381948195819681978198819982008201820282038204820582068207820882098210821182128213821482158216821782188219822082218222822382248225822682278228822982308231823282338234823582368237823882398240824182428243824482458246824782488249825082518252825382548255825682578258825982608261826282638264826582668267826882698270827182728273827482758276827782788279828082818282828382848285828682878288828982908291829282938294829582968297829882998300830183028303830483058306830783088309831083118312831383148315831683178318831983208321832283238324832583268327832883298330833183328333833483358336833783388339834083418342834383448345834683478348834983508351835283538354835583568357835883598360836183628363836483658366836783688369837083718372837383748375837683778378837983808381838283838384838583868387838883898390839183928393839483958396839783988399840084018402840384048405840684078408840984108411841284138414841584168417841884198420842184228423842484258426842784288429843084318432843384348435843684378438843984408441844284438444844584468447844884498450845184528453845484558456845784588459846084618462846384648465846684678468846984708471847284738474847584768477847884798480848184828483848484858486848784888489849084918492849384948495849684978498849985008501850285038504850585068507850885098510851185128513851485158516851785188519852085218522852385248525852685278528852985308531853285338534853585368537853885398540854185428543854485458546854785488549855085518552855385548555855685578558855985608561856285638564856585668567856885698570857185728573857485758576857785788579858085818582858385848585858685878588858985908591859285938594859585968597859885998600860186028603860486058606860786088609861086118612861386148615861686178618861986208621862286238624862586268627862886298630863186328633863486358636863786388639864086418642864386448645864686478648864986508651865286538654865586568657865886598660866186628663866486658666866786688669867086718672867386748675867686778678867986808681868286838684868586868687868886898690869186928693869486958696869786988699870087018702870387048705870687078708870987108711871287138714871587168717871887198720872187228723872487258726872787288729873087318732873387348735873687378738873987408741874287438744874587468747874887498750875187528753875487558756875787588759876087618762876387648765876687678768876987708771877287738774877587768777877887798780878187828783878487858786878787888789879087918792879387948795879687978798879988008801880288038804880588068807880888098810881188128813881488158816881788188819882088218822882388248825882688278828882988308831883288338834883588368837883888398840884188428843884488458846884788488849885088518852885388548855885688578858885988608861886288638864886588668867886888698870887188728873887488758876887788788879888088818882888388848885888688878888888988908891889288938894889588968897889888998900890189028903890489058906890789088909891089118912891389148915891689178918891989208921892289238924892589268927892889298930893189328933893489358936893789388939894089418942894389448945894689478948894989508951895289538954895589568957895889598960896189628963896489658966896789688969897089718972897389748975897689778978897989808981898289838984898589868987898889898990899189928993899489958996899789988999900090019002900390049005900690079008900990109011901290139014901590169017901890199020902190229023902490259026902790289029903090319032903390349035903690379038903990409041904290439044904590469047904890499050905190529053905490559056905790589059906090619062906390649065906690679068906990709071907290739074907590769077907890799080908190829083908490859086908790889089909090919092909390949095909690979098909991009101910291039104910591069107910891099110911191129113911491159116911791189119912091219122912391249125912691279128912991309131913291339134913591369137913891399140914191429143914491459146914791489149915091519152915391549155915691579158915991609161916291639164916591669167916891699170917191729173917491759176917791789179918091819182918391849185918691879188918991909191919291939194919591969197919891999200920192029203920492059206920792089209921092119212921392149215921692179218921992209221922292239224922592269227922892299230923192329233923492359236923792389239924092419242924392449245924692479248924992509251925292539254925592569257925892599260926192629263926492659266926792689269927092719272927392749275927692779278927992809281928292839284928592869287928892899290929192929293929492959296929792989299930093019302930393049305930693079308930993109311931293139314931593169317931893199320932193229323932493259326932793289329933093319332933393349335933693379338933993409341934293439344934593469347934893499350935193529353935493559356935793589359936093619362936393649365936693679368936993709371937293739374937593769377937893799380938193829383938493859386938793889389939093919392939393949395939693979398939994009401940294039404940594069407940894099410941194129413941494159416941794189419942094219422942394249425942694279428942994309431943294339434943594369437943894399440944194429443944494459446944794489449945094519452945394549455945694579458945994609461946294639464946594669467946894699470947194729473947494759476947794789479948094819482948394849485948694879488948994909491949294939494949594969497949894999500950195029503950495059506950795089509951095119512951395149515951695179518951995209521952295239524952595269527952895299530953195329533953495359536953795389539954095419542954395449545954695479548954995509551955295539554955595569557955895599560956195629563956495659566956795689569957095719572957395749575957695779578957995809581958295839584958595869587958895899590959195929593959495959596959795989599960096019602960396049605960696079608960996109611961296139614961596169617961896199620962196229623962496259626962796289629963096319632963396349635963696379638963996409641964296439644964596469647964896499650965196529653965496559656965796589659966096619662966396649665966696679668966996709671967296739674967596769677967896799680968196829683968496859686968796889689969096919692969396949695969696979698969997009701970297039704970597069707970897099710971197129713971497159716971797189719972097219722972397249725972697279728972997309731973297339734973597369737973897399740974197429743974497459746974797489749975097519752975397549755975697579758975997609761976297639764976597669767976897699770977197729773977497759776977797789779978097819782978397849785978697879788978997909791979297939794979597969797979897999800980198029803980498059806980798089809981098119812981398149815981698179818981998209821982298239824982598269827982898299830983198329833983498359836983798389839984098419842984398449845984698479848984998509851985298539854985598569857985898599860986198629863986498659866986798689869987098719872987398749875987698779878987998809881988298839884988598869887988898899890989198929893989498959896989798989899990099019902990399049905990699079908990999109911991299139914991599169917991899199920992199229923992499259926992799289929993099319932993399349935993699379938993999409941994299439944994599469947994899499950995199529953995499559956995799589959996099619962996399649965996699679968996999709971997299739974997599769977997899799980998199829983998499859986998799889989999099919992999399949995999699979998999910000100011000210003100041000510006100071000810009100101001110012100131001410015100161001710018100191002010021100221002310024100251002610027100281002910030100311003210033100341003510036100371003810039100401004110042100431004410045100461004710048100491005010051100521005310054100551005610057100581005910060100611006210063100641006510066100671006810069100701007110072100731007410075100761007710078100791008010081100821008310084100851008610087100881008910090100911009210093100941009510096100971009810099101001010110102101031010410105101061010710108101091011010111101121011310114101151011610117101181011910120101211012210123101241012510126101271012810129101301013110132101331013410135101361013710138101391014010141101421014310144101451014610147101481014910150101511015210153101541015510156101571015810159101601016110162101631016410165101661016710168101691017010171101721017310174101751017610177101781017910180101811018210183101841018510186101871018810189101901019110192101931019410195101961019710198101991020010201102021020310204102051020610207102081020910210102111021210213102141021510216102171021810219102201022110222102231022410225102261022710228102291023010231102321023310234102351023610237102381023910240102411024210243102441024510246102471024810249102501025110252102531025410255102561025710258102591026010261102621026310264102651026610267102681026910270102711027210273102741027510276102771027810279102801028110282102831028410285102861028710288102891029010291102921029310294102951029610297102981029910300103011030210303103041030510306103071030810309103101031110312103131031410315103161031710318103191032010321103221032310324103251032610327103281032910330103311033210333103341033510336103371033810339103401034110342103431034410345103461034710348103491035010351103521035310354103551035610357103581035910360103611036210363103641036510366103671036810369103701037110372103731037410375103761037710378103791038010381103821038310384103851038610387103881038910390103911039210393103941039510396103971039810399104001040110402104031040410405104061040710408104091041010411104121041310414104151041610417104181041910420104211042210423104241042510426104271042810429104301043110432104331043410435104361043710438104391044010441104421044310444104451044610447104481044910450104511045210453104541045510456104571045810459104601046110462104631046410465104661046710468104691047010471104721047310474104751047610477104781047910480104811048210483104841048510486104871048810489104901049110492104931049410495104961049710498104991050010501105021050310504105051050610507105081050910510105111051210513105141051510516105171051810519105201052110522105231052410525105261052710528105291053010531105321053310534105351053610537105381053910540105411054210543105441054510546105471054810549105501055110552105531055410555105561055710558105591056010561105621056310564105651056610567105681056910570105711057210573105741057510576105771057810579105801058110582105831058410585105861058710588105891059010591105921059310594105951059610597105981059910600106011060210603106041060510606106071060810609106101061110612106131061410615106161061710618106191062010621106221062310624106251062610627106281062910630106311063210633106341063510636106371063810639106401064110642106431064410645106461064710648106491065010651106521065310654106551065610657106581065910660106611066210663106641066510666106671066810669106701067110672106731067410675106761067710678106791068010681106821068310684106851068610687106881068910690106911069210693106941069510696106971069810699107001070110702107031070410705107061070710708107091071010711107121071310714107151071610717107181071910720107211072210723107241072510726107271072810729107301073110732107331073410735107361073710738107391074010741107421074310744107451074610747107481074910750107511075210753107541075510756107571075810759107601076110762107631076410765107661076710768107691077010771107721077310774107751077610777107781077910780107811078210783107841078510786107871078810789107901079110792107931079410795107961079710798107991080010801108021080310804108051080610807108081080910810108111081210813108141081510816108171081810819108201082110822108231082410825108261082710828108291083010831108321083310834108351083610837108381083910840108411084210843108441084510846108471084810849108501085110852108531085410855108561085710858108591086010861108621086310864108651086610867108681086910870108711087210873108741087510876108771087810879108801088110882108831088410885108861088710888108891089010891108921089310894108951089610897108981089910900109011090210903109041090510906109071090810909109101091110912109131091410915109161091710918109191092010921109221092310924109251092610927109281092910930109311093210933109341093510936109371093810939
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2012-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/fs.h>
  6. #include <linux/mutex.h>
  7. #include <linux/wait.h>
  8. #include <linux/miscdevice.h>
  9. #include <linux/uaccess.h>
  10. #include <linux/sched.h>
  11. #include <linux/dma-mapping.h>
  12. #include <linux/miscdevice.h>
  13. #include <linux/delay.h>
  14. #include <linux/slab.h>
  15. #include <linux/debugfs.h>
  16. #include <linux/time.h>
  17. #include <linux/atomic.h>
  18. #include <linux/mm.h>
  19. #include <asm/ioctls.h>
  20. #include <linux/memory.h>
  21. #include <sound/compress_params.h>
  22. #include <dsp/msm_audio_ion.h>
  23. #include <dsp/apr_audio-v2.h>
  24. #include <dsp/audio_cal_utils.h>
  25. #include <dsp/q6asm-v2.h>
  26. #include <dsp/q6audio-v2.h>
  27. #include <dsp/q6common.h>
  28. #include <dsp/q6core.h>
  29. #include "adsp_err.h"
  30. #define TIMEOUT_MS 1000
  31. #define TRUE 0x01
  32. #define FALSE 0x00
  33. #define SESSION_MAX 8
  34. #define ENC_FRAMES_PER_BUFFER 0x01
  35. enum {
  36. ASM_TOPOLOGY_CAL = 0,
  37. ASM_CUSTOM_TOP_CAL,
  38. ASM_AUDSTRM_CAL,
  39. ASM_RTAC_APR_CAL,
  40. ASM_MAX_CAL_TYPES
  41. };
  42. union asm_token_struct {
  43. struct {
  44. u8 stream_id;
  45. u8 session_id;
  46. u8 buf_index;
  47. u8 flags;
  48. } _token;
  49. u32 token;
  50. } __packed;
  51. enum {
  52. ASM_DIRECTION_OFFSET,
  53. ASM_CMD_NO_WAIT_OFFSET,
  54. /*
  55. * Offset is limited to 7 because flags is stored in u8
  56. * field in asm_token_structure defined above. The offset
  57. * starts from 0.
  58. */
  59. ASM_MAX_OFFSET = 7,
  60. };
  61. enum {
  62. WAIT_CMD,
  63. NO_WAIT_CMD
  64. };
  65. #define ASM_SET_BIT(n, x) (n |= 1 << x)
  66. #define ASM_TEST_BIT(n, x) ((n >> x) & 1)
  67. /* TODO, combine them together */
  68. static DEFINE_MUTEX(session_lock);
  69. struct asm_mmap {
  70. atomic_t ref_cnt;
  71. void *apr;
  72. };
  73. static struct asm_mmap this_mmap;
  74. struct audio_session {
  75. struct audio_client *ac;
  76. spinlock_t session_lock;
  77. struct mutex mutex_lock_per_session;
  78. };
  79. /* session id: 0 reserved */
  80. static struct audio_session session[ASM_ACTIVE_STREAMS_ALLOWED + 1];
  81. struct asm_buffer_node {
  82. struct list_head list;
  83. phys_addr_t buf_phys_addr;
  84. uint32_t mmap_hdl;
  85. };
  86. static int32_t q6asm_srvc_callback(struct apr_client_data *data, void *priv);
  87. static int32_t q6asm_callback(struct apr_client_data *data, void *priv);
  88. static void q6asm_add_hdr(struct audio_client *ac, struct apr_hdr *hdr,
  89. uint32_t pkt_size, uint32_t cmd_flg);
  90. static void q6asm_add_hdr_custom_topology(struct audio_client *ac,
  91. struct apr_hdr *hdr,
  92. uint32_t pkt_size);
  93. static void q6asm_add_hdr_async(struct audio_client *ac, struct apr_hdr *hdr,
  94. uint32_t pkt_size, uint32_t cmd_flg);
  95. static int q6asm_memory_map_regions(struct audio_client *ac, int dir,
  96. uint32_t bufsz, uint32_t bufcnt,
  97. bool is_contiguous);
  98. static int q6asm_memory_unmap_regions(struct audio_client *ac, int dir);
  99. static void q6asm_reset_buf_state(struct audio_client *ac);
  100. static int q6asm_map_channels(u8 *channel_mapping, uint32_t channels,
  101. bool use_back_flavor);
  102. void *q6asm_mmap_apr_reg(void);
  103. static int q6asm_is_valid_session(struct apr_client_data *data, void *priv);
  104. static int q6asm_get_asm_topology_apptype(struct q6asm_cal_info *cal_info);
  105. /* for ASM custom topology */
  106. static struct cal_type_data *cal_data[ASM_MAX_CAL_TYPES];
  107. static struct audio_buffer common_buf[2];
  108. static struct audio_client common_client;
  109. static int set_custom_topology;
  110. static int topology_map_handle;
  111. struct generic_get_data_ {
  112. int valid;
  113. int is_inband;
  114. int size_in_ints;
  115. int ints[];
  116. };
  117. static struct generic_get_data_ *generic_get_data;
  118. #ifdef CONFIG_DEBUG_FS
  119. #define OUT_BUFFER_SIZE 56
  120. #define IN_BUFFER_SIZE 24
  121. static struct timeval out_cold_tv;
  122. static struct timeval out_warm_tv;
  123. static struct timeval out_cont_tv;
  124. static struct timeval in_cont_tv;
  125. static long out_enable_flag;
  126. static long in_enable_flag;
  127. static struct dentry *out_dentry;
  128. static struct dentry *in_dentry;
  129. static int in_cont_index;
  130. /*This var is used to keep track of first write done for cold output latency */
  131. static int out_cold_index;
  132. static char *out_buffer;
  133. static char *in_buffer;
  134. static uint32_t adsp_reg_event_opcode[] = {
  135. ASM_STREAM_CMD_REGISTER_PP_EVENTS,
  136. ASM_STREAM_CMD_REGISTER_ENCDEC_EVENTS,
  137. ASM_STREAM_CMD_REGISTER_IEC_61937_FMT_UPDATE };
  138. static uint32_t adsp_raise_event_opcode[] = {
  139. ASM_STREAM_PP_EVENT,
  140. ASM_STREAM_CMD_ENCDEC_EVENTS,
  141. ASM_IEC_61937_MEDIA_FMT_EVENT };
  142. static int is_adsp_reg_event(uint32_t cmd)
  143. {
  144. int i;
  145. for (i = 0; i < ARRAY_SIZE(adsp_reg_event_opcode); i++) {
  146. if (cmd == adsp_reg_event_opcode[i])
  147. return i;
  148. }
  149. return -EINVAL;
  150. }
  151. static int is_adsp_raise_event(uint32_t cmd)
  152. {
  153. int i;
  154. for (i = 0; i < ARRAY_SIZE(adsp_raise_event_opcode); i++) {
  155. if (cmd == adsp_raise_event_opcode[i])
  156. return i;
  157. }
  158. return -EINVAL;
  159. }
  160. static inline void q6asm_set_flag_in_token(union asm_token_struct *asm_token,
  161. int flag, int flag_offset)
  162. {
  163. if (flag)
  164. ASM_SET_BIT(asm_token->_token.flags, flag_offset);
  165. }
  166. static inline int q6asm_get_flag_from_token(union asm_token_struct *asm_token,
  167. int flag_offset)
  168. {
  169. return ASM_TEST_BIT(asm_token->_token.flags, flag_offset);
  170. }
  171. static inline void q6asm_update_token(u32 *token, u8 session_id, u8 stream_id,
  172. u8 buf_index, u8 dir, u8 nowait_flag)
  173. {
  174. union asm_token_struct asm_token;
  175. asm_token.token = 0;
  176. asm_token._token.session_id = session_id;
  177. asm_token._token.stream_id = stream_id;
  178. asm_token._token.buf_index = buf_index;
  179. q6asm_set_flag_in_token(&asm_token, dir, ASM_DIRECTION_OFFSET);
  180. q6asm_set_flag_in_token(&asm_token, nowait_flag,
  181. ASM_CMD_NO_WAIT_OFFSET);
  182. *token = asm_token.token;
  183. }
  184. static inline uint32_t q6asm_get_pcm_format_id(uint32_t media_format_block_ver)
  185. {
  186. uint32_t pcm_format_id;
  187. switch (media_format_block_ver) {
  188. case PCM_MEDIA_FORMAT_V5:
  189. pcm_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V5;
  190. break;
  191. case PCM_MEDIA_FORMAT_V4:
  192. pcm_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V4;
  193. break;
  194. case PCM_MEDIA_FORMAT_V3:
  195. pcm_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V3;
  196. break;
  197. case PCM_MEDIA_FORMAT_V2:
  198. default:
  199. pcm_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V2;
  200. break;
  201. }
  202. return pcm_format_id;
  203. }
  204. /*
  205. * q6asm_get_buf_index_from_token:
  206. * Retrieve buffer index from token.
  207. *
  208. * @token: token value sent to ASM service on q6.
  209. * Returns buffer index in the read/write commands.
  210. */
  211. uint8_t q6asm_get_buf_index_from_token(uint32_t token)
  212. {
  213. union asm_token_struct asm_token;
  214. asm_token.token = token;
  215. return asm_token._token.buf_index;
  216. }
  217. EXPORT_SYMBOL(q6asm_get_buf_index_from_token);
  218. /*
  219. * q6asm_get_stream_id_from_token:
  220. * Retrieve stream id from token.
  221. *
  222. * @token: token value sent to ASM service on q6.
  223. * Returns stream id.
  224. */
  225. uint8_t q6asm_get_stream_id_from_token(uint32_t token)
  226. {
  227. union asm_token_struct asm_token;
  228. asm_token.token = token;
  229. return asm_token._token.stream_id;
  230. }
  231. EXPORT_SYMBOL(q6asm_get_stream_id_from_token);
  232. static int audio_output_latency_dbgfs_open(struct inode *inode,
  233. struct file *file)
  234. {
  235. file->private_data = inode->i_private;
  236. return 0;
  237. }
  238. static ssize_t audio_output_latency_dbgfs_read(struct file *file,
  239. char __user *buf, size_t count, loff_t *ppos)
  240. {
  241. if (out_buffer == NULL) {
  242. pr_err("%s: out_buffer is null\n", __func__);
  243. return 0;
  244. }
  245. if (count < OUT_BUFFER_SIZE) {
  246. pr_err("%s: read size %d exceeds buf size %zd\n", __func__,
  247. OUT_BUFFER_SIZE, count);
  248. return 0;
  249. }
  250. snprintf(out_buffer, OUT_BUFFER_SIZE, "%ld,%ld,%ld,%ld,%ld,%ld,",
  251. out_cold_tv.tv_sec, out_cold_tv.tv_usec, out_warm_tv.tv_sec,
  252. out_warm_tv.tv_usec, out_cont_tv.tv_sec, out_cont_tv.tv_usec);
  253. return simple_read_from_buffer(buf, OUT_BUFFER_SIZE, ppos,
  254. out_buffer, OUT_BUFFER_SIZE);
  255. }
  256. static ssize_t audio_output_latency_dbgfs_write(struct file *file,
  257. const char __user *buf, size_t count, loff_t *ppos)
  258. {
  259. char *temp;
  260. if (count > 2*sizeof(char)) {
  261. pr_err("%s: err count is more %zd\n", __func__, count);
  262. return -EINVAL;
  263. }
  264. temp = kmalloc(2*sizeof(char), GFP_KERNEL);
  265. out_cold_index = 0;
  266. if (temp) {
  267. if (copy_from_user(temp, buf, 2*sizeof(char))) {
  268. pr_err("%s: copy from user failed for size %zd\n",
  269. __func__, 2*sizeof(char));
  270. kfree(temp);
  271. return -EFAULT;
  272. }
  273. if (!kstrtol(temp, 10, &out_enable_flag)) {
  274. kfree(temp);
  275. return count;
  276. }
  277. kfree(temp);
  278. }
  279. return -EINVAL;
  280. }
  281. static const struct file_operations audio_output_latency_debug_fops = {
  282. .open = audio_output_latency_dbgfs_open,
  283. .read = audio_output_latency_dbgfs_read,
  284. .write = audio_output_latency_dbgfs_write
  285. };
  286. static int audio_input_latency_dbgfs_open(struct inode *inode,
  287. struct file *file)
  288. {
  289. file->private_data = inode->i_private;
  290. return 0;
  291. }
  292. static ssize_t audio_input_latency_dbgfs_read(struct file *file,
  293. char __user *buf, size_t count, loff_t *ppos)
  294. {
  295. if (in_buffer == NULL) {
  296. pr_err("%s: in_buffer is null\n", __func__);
  297. return 0;
  298. }
  299. if (count < IN_BUFFER_SIZE) {
  300. pr_err("%s: read size %d exceeds buf size %zd\n", __func__,
  301. IN_BUFFER_SIZE, count);
  302. return 0;
  303. }
  304. snprintf(in_buffer, IN_BUFFER_SIZE, "%ld,%ld,",
  305. in_cont_tv.tv_sec, in_cont_tv.tv_usec);
  306. return simple_read_from_buffer(buf, IN_BUFFER_SIZE, ppos,
  307. in_buffer, IN_BUFFER_SIZE);
  308. }
  309. static ssize_t audio_input_latency_dbgfs_write(struct file *file,
  310. const char __user *buf, size_t count, loff_t *ppos)
  311. {
  312. char *temp;
  313. if (count > 2*sizeof(char)) {
  314. pr_err("%s: err count is more %zd\n", __func__, count);
  315. return -EINVAL;
  316. }
  317. temp = kmalloc(2*sizeof(char), GFP_KERNEL);
  318. if (temp) {
  319. if (copy_from_user(temp, buf, 2*sizeof(char))) {
  320. pr_err("%s: copy from user failed for size %zd\n",
  321. __func__, 2*sizeof(char));
  322. kfree(temp);
  323. return -EFAULT;
  324. }
  325. if (!kstrtol(temp, 10, &in_enable_flag)) {
  326. kfree(temp);
  327. return count;
  328. }
  329. kfree(temp);
  330. }
  331. return -EINVAL;
  332. }
  333. static const struct file_operations audio_input_latency_debug_fops = {
  334. .open = audio_input_latency_dbgfs_open,
  335. .read = audio_input_latency_dbgfs_read,
  336. .write = audio_input_latency_dbgfs_write
  337. };
  338. static void config_debug_fs_write_cb(void)
  339. {
  340. if (out_enable_flag) {
  341. /* For first Write done log the time and reset
  342. * out_cold_index
  343. */
  344. if (out_cold_index != 1) {
  345. do_gettimeofday(&out_cold_tv);
  346. pr_debug("COLD: apr_send_pkt at %ld sec %ld microsec\n",
  347. out_cold_tv.tv_sec,
  348. out_cold_tv.tv_usec);
  349. out_cold_index = 1;
  350. }
  351. pr_debug("%s: out_enable_flag %ld\n",
  352. __func__, out_enable_flag);
  353. }
  354. }
  355. static void config_debug_fs_read_cb(void)
  356. {
  357. if (in_enable_flag) {
  358. /* when in_cont_index == 7, DSP would be
  359. * writing into the 8th 512 byte buffer and this
  360. * timestamp is tapped here.Once done it then writes
  361. * to 9th 512 byte buffer.These two buffers(8th, 9th)
  362. * reach the test application in 5th iteration and that
  363. * timestamp is tapped at user level. The difference
  364. * of these two timestamps gives us the time between
  365. * the time at which dsp started filling the sample
  366. * required and when it reached the test application.
  367. * Hence continuous input latency
  368. */
  369. if (in_cont_index == 7) {
  370. do_gettimeofday(&in_cont_tv);
  371. pr_info("%s: read buffer at %ld sec %ld microsec\n",
  372. __func__,
  373. in_cont_tv.tv_sec, in_cont_tv.tv_usec);
  374. }
  375. in_cont_index++;
  376. }
  377. }
  378. static void config_debug_fs_reset_index(void)
  379. {
  380. in_cont_index = 0;
  381. }
  382. static void config_debug_fs_run(void)
  383. {
  384. if (out_enable_flag) {
  385. do_gettimeofday(&out_cold_tv);
  386. pr_debug("%s: COLD apr_send_pkt at %ld sec %ld microsec\n",
  387. __func__, out_cold_tv.tv_sec, out_cold_tv.tv_usec);
  388. }
  389. }
  390. static void config_debug_fs_write(struct audio_buffer *ab)
  391. {
  392. if (out_enable_flag) {
  393. char zero_pattern[2] = {0x00, 0x00};
  394. /* If First two byte is non zero and last two byte
  395. * is zero then it is warm output pattern
  396. */
  397. if ((strcmp(((char *)ab->data), zero_pattern)) &&
  398. (!strcmp(((char *)ab->data + 2), zero_pattern))) {
  399. do_gettimeofday(&out_warm_tv);
  400. pr_debug("%s: WARM:apr_send_pkt at %ld sec %ld microsec\n",
  401. __func__,
  402. out_warm_tv.tv_sec,
  403. out_warm_tv.tv_usec);
  404. pr_debug("%s: Warm Pattern Matched\n", __func__);
  405. }
  406. /* If First two byte is zero and last two byte is
  407. * non zero then it is cont output pattern
  408. */
  409. else if ((!strcmp(((char *)ab->data), zero_pattern))
  410. && (strcmp(((char *)ab->data + 2), zero_pattern))) {
  411. do_gettimeofday(&out_cont_tv);
  412. pr_debug("%s: CONT:apr_send_pkt at %ld sec %ld microsec\n",
  413. __func__,
  414. out_cont_tv.tv_sec,
  415. out_cont_tv.tv_usec);
  416. pr_debug("%s: Cont Pattern Matched\n", __func__);
  417. }
  418. }
  419. }
  420. static void config_debug_fs_init(void)
  421. {
  422. out_buffer = kzalloc(OUT_BUFFER_SIZE, GFP_KERNEL);
  423. if (out_buffer == NULL)
  424. goto outbuf_fail;
  425. in_buffer = kzalloc(IN_BUFFER_SIZE, GFP_KERNEL);
  426. if (in_buffer == NULL)
  427. goto inbuf_fail;
  428. out_dentry = debugfs_create_file("audio_out_latency_measurement_node",
  429. 0664,
  430. NULL, NULL, &audio_output_latency_debug_fops);
  431. if (IS_ERR(out_dentry)) {
  432. pr_err("%s: debugfs_create_file failed\n", __func__);
  433. goto file_fail;
  434. }
  435. in_dentry = debugfs_create_file("audio_in_latency_measurement_node",
  436. 0664,
  437. NULL, NULL, &audio_input_latency_debug_fops);
  438. if (IS_ERR(in_dentry)) {
  439. pr_err("%s: debugfs_create_file failed\n", __func__);
  440. goto file_fail;
  441. }
  442. return;
  443. file_fail:
  444. kfree(in_buffer);
  445. inbuf_fail:
  446. kfree(out_buffer);
  447. outbuf_fail:
  448. in_buffer = NULL;
  449. out_buffer = NULL;
  450. }
  451. #else
  452. static void config_debug_fs_write(struct audio_buffer *ab)
  453. {
  454. }
  455. static void config_debug_fs_run(void)
  456. {
  457. }
  458. static void config_debug_fs_reset_index(void)
  459. {
  460. }
  461. static void config_debug_fs_read_cb(void)
  462. {
  463. }
  464. static void config_debug_fs_write_cb(void)
  465. {
  466. }
  467. static void config_debug_fs_init(void)
  468. {
  469. }
  470. #endif
  471. int q6asm_mmap_apr_dereg(void)
  472. {
  473. int c;
  474. c = atomic_sub_return(1, &this_mmap.ref_cnt);
  475. if (c == 0) {
  476. apr_deregister(this_mmap.apr);
  477. common_client.mmap_apr = NULL;
  478. pr_debug("%s: APR De-Register common port\n", __func__);
  479. } else if (c < 0) {
  480. pr_err("%s: APR Common Port Already Closed %d\n",
  481. __func__, c);
  482. atomic_set(&this_mmap.ref_cnt, 0);
  483. }
  484. return 0;
  485. }
  486. static int q6asm_session_alloc(struct audio_client *ac)
  487. {
  488. int n;
  489. for (n = 1; n <= ASM_ACTIVE_STREAMS_ALLOWED; n++) {
  490. if (!(session[n].ac)) {
  491. session[n].ac = ac;
  492. return n;
  493. }
  494. }
  495. pr_err("%s: session not available\n", __func__);
  496. return -ENOMEM;
  497. }
  498. static int q6asm_get_session_id_from_audio_client(struct audio_client *ac)
  499. {
  500. int n;
  501. for (n = 1; n <= ASM_ACTIVE_STREAMS_ALLOWED; n++) {
  502. if (session[n].ac == ac)
  503. return n;
  504. }
  505. pr_debug("%s: cannot find matching audio client. ac = %pK\n",
  506. __func__, ac);
  507. return 0;
  508. }
  509. static bool q6asm_is_valid_audio_client(struct audio_client *ac)
  510. {
  511. return q6asm_get_session_id_from_audio_client(ac) ? 1 : 0;
  512. }
  513. static void q6asm_session_free(struct audio_client *ac)
  514. {
  515. int session_id;
  516. unsigned long flags = 0;
  517. pr_debug("%s: sessionid[%d]\n", __func__, ac->session);
  518. session_id = ac->session;
  519. mutex_lock(&session[session_id].mutex_lock_per_session);
  520. rtac_remove_popp_from_adm_devices(ac->session);
  521. spin_lock_irqsave(&(session[session_id].session_lock), flags);
  522. session[ac->session].ac = NULL;
  523. ac->session = 0;
  524. ac->perf_mode = LEGACY_PCM_MODE;
  525. ac->fptr_cache_ops = NULL;
  526. ac->cb = NULL;
  527. ac->priv = NULL;
  528. kfree(ac);
  529. ac = NULL;
  530. spin_unlock_irqrestore(&(session[session_id].session_lock), flags);
  531. mutex_unlock(&session[session_id].mutex_lock_per_session);
  532. }
  533. static uint32_t q6asm_get_next_buf(struct audio_client *ac,
  534. uint32_t curr_buf, uint32_t max_buf_cnt)
  535. {
  536. dev_vdbg(ac->dev, "%s: curr_buf = %d, max_buf_cnt = %d\n",
  537. __func__, curr_buf, max_buf_cnt);
  538. curr_buf += 1;
  539. return (curr_buf >= max_buf_cnt) ? 0 : curr_buf;
  540. }
  541. static int q6asm_map_cal_memory(int32_t cal_type,
  542. struct cal_block_data *cal_block)
  543. {
  544. int result = 0;
  545. struct asm_buffer_node *buf_node = NULL;
  546. struct list_head *ptr, *next;
  547. if (cal_block == NULL) {
  548. pr_err("%s: cal_block is NULL!\n",
  549. __func__);
  550. goto done;
  551. }
  552. if (cal_block->cal_data.paddr == 0) {
  553. pr_debug("%s: No address to map!\n",
  554. __func__);
  555. goto done;
  556. }
  557. common_client.mmap_apr = q6asm_mmap_apr_reg();
  558. if (common_client.mmap_apr == NULL) {
  559. pr_err("%s: q6asm_mmap_apr_reg failed\n",
  560. __func__);
  561. result = -EPERM;
  562. goto done;
  563. }
  564. common_client.apr = common_client.mmap_apr;
  565. if (cal_block->map_data.map_size == 0) {
  566. pr_debug("%s: map size is 0!\n",
  567. __func__);
  568. goto done;
  569. }
  570. /* Use second asm buf to map memory */
  571. if (common_client.port[IN].buf == NULL) {
  572. pr_err("%s: common buf is NULL\n",
  573. __func__);
  574. result = -EINVAL;
  575. goto done;
  576. }
  577. common_client.port[IN].buf->phys = cal_block->cal_data.paddr;
  578. result = q6asm_memory_map_regions(&common_client,
  579. IN, cal_block->map_data.map_size, 1, 1);
  580. if (result < 0) {
  581. pr_err("%s: mmap did not work! size = %zd result %d\n",
  582. __func__,
  583. cal_block->map_data.map_size, result);
  584. pr_debug("%s: mmap did not work! addr = 0x%pK, size = %zd\n",
  585. __func__,
  586. &cal_block->cal_data.paddr,
  587. cal_block->map_data.map_size);
  588. goto done;
  589. }
  590. list_for_each_safe(ptr, next,
  591. &common_client.port[IN].mem_map_handle) {
  592. buf_node = list_entry(ptr, struct asm_buffer_node,
  593. list);
  594. if (buf_node->buf_phys_addr == cal_block->cal_data.paddr) {
  595. cal_block->map_data.q6map_handle = buf_node->mmap_hdl;
  596. break;
  597. }
  598. }
  599. done:
  600. return result;
  601. }
  602. static int remap_cal_data(int32_t cal_type, struct cal_block_data *cal_block)
  603. {
  604. int ret = 0;
  605. if (cal_block->map_data.dma_buf == NULL) {
  606. pr_err("%s: No ION allocation for cal type %d!\n",
  607. __func__, cal_type);
  608. ret = -EINVAL;
  609. goto done;
  610. }
  611. if ((cal_block->map_data.map_size > 0) &&
  612. (cal_block->map_data.q6map_handle == 0)) {
  613. ret = q6asm_map_cal_memory(cal_type, cal_block);
  614. if (ret < 0) {
  615. pr_err("%s: mmap did not work! size = %zd ret %d\n",
  616. __func__, cal_block->map_data.map_size, ret);
  617. goto done;
  618. }
  619. }
  620. done:
  621. return ret;
  622. }
  623. static int q6asm_unmap_cal_memory(int32_t cal_type,
  624. struct cal_block_data *cal_block)
  625. {
  626. int result = 0;
  627. int result2 = 0;
  628. if (cal_block == NULL) {
  629. pr_err("%s: cal_block is NULL!\n",
  630. __func__);
  631. result = -EINVAL;
  632. goto done;
  633. }
  634. if (cal_block->map_data.q6map_handle == 0) {
  635. pr_debug("%s: No address to unmap!\n",
  636. __func__);
  637. result = -EINVAL;
  638. goto done;
  639. }
  640. if (common_client.mmap_apr == NULL) {
  641. common_client.mmap_apr = q6asm_mmap_apr_reg();
  642. if (common_client.mmap_apr == NULL) {
  643. pr_err("%s: q6asm_mmap_apr_reg failed\n",
  644. __func__);
  645. result = -EPERM;
  646. goto done;
  647. }
  648. }
  649. result2 = q6asm_memory_unmap_regions(&common_client, IN);
  650. if (result2 < 0) {
  651. pr_err("%s: unmap failed, err %d\n",
  652. __func__, result2);
  653. result = result2;
  654. }
  655. cal_block->map_data.q6map_handle = 0;
  656. done:
  657. return result;
  658. }
  659. int q6asm_unmap_cal_data(int cal_type, struct cal_block_data *cal_block)
  660. {
  661. int ret = 0;
  662. if ((cal_block->map_data.map_size > 0) &&
  663. (cal_block->map_data.q6map_handle != 0)) {
  664. ret = q6asm_unmap_cal_memory(cal_type, cal_block);
  665. if (ret < 0) {
  666. pr_err("%s: unmap did not work! size = %zd ret %d\n",
  667. __func__, cal_block->map_data.map_size, ret);
  668. goto done;
  669. }
  670. }
  671. done:
  672. return ret;
  673. }
  674. int send_asm_custom_topology(struct audio_client *ac)
  675. {
  676. struct cal_block_data *cal_block = NULL;
  677. struct cmd_set_topologies asm_top;
  678. int result = 0;
  679. int result1 = 0;
  680. if (cal_data[ASM_CUSTOM_TOP_CAL] == NULL)
  681. goto done;
  682. mutex_lock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  683. if (!set_custom_topology)
  684. goto unlock;
  685. set_custom_topology = 0;
  686. cal_block = cal_utils_get_only_cal_block(cal_data[ASM_CUSTOM_TOP_CAL]);
  687. if (cal_block == NULL || cal_utils_is_cal_stale(cal_block))
  688. goto unlock;
  689. if (cal_block->cal_data.size == 0) {
  690. pr_debug("%s: No cal to send!\n", __func__);
  691. goto unlock;
  692. }
  693. pr_debug("%s: Sending cal_index %d\n", __func__, ASM_CUSTOM_TOP_CAL);
  694. result = remap_cal_data(ASM_CUST_TOPOLOGY_CAL_TYPE, cal_block);
  695. if (result) {
  696. pr_err("%s: Remap_cal_data failed for cal %d!\n",
  697. __func__, ASM_CUSTOM_TOP_CAL);
  698. goto unlock;
  699. }
  700. q6asm_add_hdr_custom_topology(ac, &asm_top.hdr, sizeof(asm_top));
  701. atomic_set(&ac->mem_state, -1);
  702. asm_top.hdr.opcode = ASM_CMD_ADD_TOPOLOGIES;
  703. asm_top.payload_addr_lsw = lower_32_bits(cal_block->cal_data.paddr);
  704. asm_top.payload_addr_msw = msm_audio_populate_upper_32_bits(
  705. cal_block->cal_data.paddr);
  706. asm_top.mem_map_handle = cal_block->map_data.q6map_handle;
  707. asm_top.payload_size = cal_block->cal_data.size;
  708. pr_debug("%s: Sending ASM_CMD_ADD_TOPOLOGIES payload = %pK, size = %d, map handle = 0x%x\n",
  709. __func__, &cal_block->cal_data.paddr,
  710. asm_top.payload_size, asm_top.mem_map_handle);
  711. result = apr_send_pkt(ac->apr, (uint32_t *) &asm_top);
  712. if (result < 0) {
  713. pr_err("%s: Set topologies failed result %d\n",
  714. __func__, result);
  715. pr_debug("%s: Set topologies failed payload = 0x%pK\n",
  716. __func__, &cal_block->cal_data.paddr);
  717. goto unmap;
  718. }
  719. result = wait_event_timeout(ac->mem_wait,
  720. (atomic_read(&ac->mem_state) >= 0),
  721. msecs_to_jiffies(TIMEOUT_MS));
  722. if (!result) {
  723. pr_err("%s: Set topologies failed timeout\n", __func__);
  724. pr_debug("%s: Set topologies failed after timedout payload = 0x%pK\n",
  725. __func__, &cal_block->cal_data.paddr);
  726. result = -ETIMEDOUT;
  727. goto unmap;
  728. }
  729. if (atomic_read(&ac->mem_state) > 0) {
  730. pr_err("%s: DSP returned error[%s]\n",
  731. __func__, adsp_err_get_err_str(
  732. atomic_read(&ac->mem_state)));
  733. result = adsp_err_get_lnx_err_code(
  734. atomic_read(&ac->mem_state));
  735. goto unmap;
  736. }
  737. unmap:
  738. result1 = q6asm_unmap_cal_memory(ASM_CUST_TOPOLOGY_CAL_TYPE,
  739. cal_block);
  740. if (result1 < 0) {
  741. result = result1;
  742. pr_debug("%s: unmap cal failed! %d\n", __func__, result);
  743. }
  744. unlock:
  745. mutex_unlock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  746. done:
  747. return result;
  748. }
  749. int q6asm_map_rtac_block(struct rtac_cal_block_data *cal_block)
  750. {
  751. int result = 0;
  752. struct asm_buffer_node *buf_node = NULL;
  753. struct list_head *ptr, *next;
  754. pr_debug("%s:\n", __func__);
  755. if (cal_block == NULL) {
  756. pr_err("%s: cal_block is NULL!\n",
  757. __func__);
  758. result = -EINVAL;
  759. goto done;
  760. }
  761. if (cal_block->cal_data.paddr == 0) {
  762. pr_debug("%s: No address to map!\n",
  763. __func__);
  764. result = -EINVAL;
  765. goto done;
  766. }
  767. if (common_client.mmap_apr == NULL) {
  768. common_client.mmap_apr = q6asm_mmap_apr_reg();
  769. if (common_client.mmap_apr == NULL) {
  770. pr_err("%s: q6asm_mmap_apr_reg failed\n",
  771. __func__);
  772. result = -EPERM;
  773. goto done;
  774. }
  775. }
  776. if (cal_block->map_data.map_size == 0) {
  777. pr_debug("%s: map size is 0!\n",
  778. __func__);
  779. result = -EINVAL;
  780. goto done;
  781. }
  782. /* Use second asm buf to map memory */
  783. if (common_client.port[OUT].buf == NULL) {
  784. pr_err("%s: common buf is NULL\n",
  785. __func__);
  786. result = -EINVAL;
  787. goto done;
  788. }
  789. common_client.port[OUT].buf->phys = cal_block->cal_data.paddr;
  790. result = q6asm_memory_map_regions(&common_client,
  791. OUT, cal_block->map_data.map_size, 1, 1);
  792. if (result < 0) {
  793. pr_err("%s: mmap did not work! size = %d result %d\n",
  794. __func__,
  795. cal_block->map_data.map_size, result);
  796. pr_debug("%s: mmap did not work! addr = 0x%pK, size = %d\n",
  797. __func__,
  798. &cal_block->cal_data.paddr,
  799. cal_block->map_data.map_size);
  800. goto done;
  801. }
  802. list_for_each_safe(ptr, next,
  803. &common_client.port[OUT].mem_map_handle) {
  804. buf_node = list_entry(ptr, struct asm_buffer_node,
  805. list);
  806. if (buf_node->buf_phys_addr == cal_block->cal_data.paddr) {
  807. cal_block->map_data.map_handle = buf_node->mmap_hdl;
  808. break;
  809. }
  810. }
  811. done:
  812. return result;
  813. }
  814. int q6asm_unmap_rtac_block(uint32_t *mem_map_handle)
  815. {
  816. int result = 0;
  817. int result2 = 0;
  818. pr_debug("%s:\n", __func__);
  819. if (mem_map_handle == NULL) {
  820. pr_debug("%s: Map handle is NULL, nothing to unmap\n",
  821. __func__);
  822. goto done;
  823. }
  824. if (*mem_map_handle == 0) {
  825. pr_debug("%s: Map handle is 0, nothing to unmap\n",
  826. __func__);
  827. goto done;
  828. }
  829. if (common_client.mmap_apr == NULL) {
  830. common_client.mmap_apr = q6asm_mmap_apr_reg();
  831. if (common_client.mmap_apr == NULL) {
  832. pr_err("%s: q6asm_mmap_apr_reg failed\n",
  833. __func__);
  834. result = -EPERM;
  835. goto done;
  836. }
  837. }
  838. result2 = q6asm_memory_unmap_regions(&common_client, OUT);
  839. if (result2 < 0) {
  840. pr_err("%s: unmap failed, err %d\n",
  841. __func__, result2);
  842. result = result2;
  843. } else {
  844. *mem_map_handle = 0;
  845. }
  846. result2 = q6asm_mmap_apr_dereg();
  847. if (result2 < 0) {
  848. pr_err("%s: q6asm_mmap_apr_dereg failed, err %d\n",
  849. __func__, result2);
  850. result = result2;
  851. }
  852. done:
  853. return result;
  854. }
  855. int q6asm_audio_client_buf_free(unsigned int dir,
  856. struct audio_client *ac)
  857. {
  858. struct audio_port_data *port;
  859. int cnt = 0;
  860. int rc = 0;
  861. pr_debug("%s: Session id %d\n", __func__, ac->session);
  862. mutex_lock(&ac->cmd_lock);
  863. if (ac->io_mode & SYNC_IO_MODE) {
  864. port = &ac->port[dir];
  865. if (!port->buf) {
  866. pr_err("%s: buf NULL\n", __func__);
  867. mutex_unlock(&ac->cmd_lock);
  868. return 0;
  869. }
  870. cnt = port->max_buf_cnt - 1;
  871. if (cnt >= 0) {
  872. rc = q6asm_memory_unmap_regions(ac, dir);
  873. if (rc < 0)
  874. pr_err("%s: Memory_unmap_regions failed %d\n",
  875. __func__, rc);
  876. }
  877. while (cnt >= 0) {
  878. if (port->buf[cnt].data) {
  879. if (!rc || atomic_read(&ac->reset))
  880. msm_audio_ion_free(
  881. port->buf[cnt].dma_buf);
  882. port->buf[cnt].dma_buf = NULL;
  883. port->buf[cnt].data = NULL;
  884. port->buf[cnt].phys = 0;
  885. --(port->max_buf_cnt);
  886. }
  887. --cnt;
  888. }
  889. kfree(port->buf);
  890. port->buf = NULL;
  891. }
  892. mutex_unlock(&ac->cmd_lock);
  893. return 0;
  894. }
  895. /**
  896. * q6asm_audio_client_buf_free_contiguous -
  897. * frees the memory buffers for ASM
  898. *
  899. * @dir: RX or TX direction
  900. * @ac: audio client handle
  901. *
  902. * Returns 0 on success or error on failure
  903. */
  904. int q6asm_audio_client_buf_free_contiguous(unsigned int dir,
  905. struct audio_client *ac)
  906. {
  907. struct audio_port_data *port;
  908. int cnt = 0;
  909. int rc = 0;
  910. pr_debug("%s: Session id %d\n", __func__, ac->session);
  911. mutex_lock(&ac->cmd_lock);
  912. port = &ac->port[dir];
  913. if (!port->buf) {
  914. mutex_unlock(&ac->cmd_lock);
  915. return 0;
  916. }
  917. cnt = port->max_buf_cnt - 1;
  918. if (cnt >= 0) {
  919. rc = q6asm_memory_unmap(ac, port->buf[0].phys, dir);
  920. if (rc < 0)
  921. pr_err("%s: Memory_unmap_regions failed %d\n",
  922. __func__, rc);
  923. }
  924. if (port->buf[0].data) {
  925. pr_debug("%s: data[%pK], phys[%pK], dma_buf[%pK]\n",
  926. __func__,
  927. port->buf[0].data,
  928. &port->buf[0].phys,
  929. port->buf[0].dma_buf);
  930. if (!rc || atomic_read(&ac->reset))
  931. msm_audio_ion_free(port->buf[0].dma_buf);
  932. port->buf[0].dma_buf = NULL;
  933. }
  934. while (cnt >= 0) {
  935. port->buf[cnt].data = NULL;
  936. port->buf[cnt].phys = 0;
  937. cnt--;
  938. }
  939. port->max_buf_cnt = 0;
  940. kfree(port->buf);
  941. port->buf = NULL;
  942. mutex_unlock(&ac->cmd_lock);
  943. return 0;
  944. }
  945. EXPORT_SYMBOL(q6asm_audio_client_buf_free_contiguous);
  946. /**
  947. * q6asm_audio_client_free -
  948. * frees the audio client for ASM
  949. *
  950. * @ac: audio client handle
  951. *
  952. */
  953. void q6asm_audio_client_free(struct audio_client *ac)
  954. {
  955. int loopcnt;
  956. struct audio_port_data *port;
  957. if (!ac) {
  958. pr_err("%s: ac %pK\n", __func__, ac);
  959. return;
  960. }
  961. if (!ac->session) {
  962. pr_err("%s: ac session invalid\n", __func__);
  963. return;
  964. }
  965. mutex_lock(&session_lock);
  966. pr_debug("%s: Session id %d\n", __func__, ac->session);
  967. if (ac->io_mode & SYNC_IO_MODE) {
  968. for (loopcnt = 0; loopcnt <= OUT; loopcnt++) {
  969. port = &ac->port[loopcnt];
  970. if (!port->buf)
  971. continue;
  972. pr_debug("%s: loopcnt = %d\n",
  973. __func__, loopcnt);
  974. q6asm_audio_client_buf_free(loopcnt, ac);
  975. }
  976. }
  977. rtac_set_asm_handle(ac->session, NULL);
  978. apr_deregister(ac->apr2);
  979. apr_deregister(ac->apr);
  980. q6asm_mmap_apr_dereg();
  981. ac->apr2 = NULL;
  982. ac->apr = NULL;
  983. ac->mmap_apr = NULL;
  984. q6asm_session_free(ac);
  985. pr_debug("%s: APR De-Register\n", __func__);
  986. /*done:*/
  987. mutex_unlock(&session_lock);
  988. }
  989. EXPORT_SYMBOL(q6asm_audio_client_free);
  990. /**
  991. * q6asm_set_io_mode -
  992. * Update IO mode for ASM
  993. *
  994. * @ac: audio client handle
  995. * @mode1: IO mode to update
  996. *
  997. * Returns 0 on success or error on failure
  998. */
  999. int q6asm_set_io_mode(struct audio_client *ac, uint32_t mode1)
  1000. {
  1001. uint32_t mode;
  1002. int ret = 0;
  1003. if (ac == NULL) {
  1004. pr_err("%s: APR handle NULL\n", __func__);
  1005. return -EINVAL;
  1006. }
  1007. ac->io_mode &= 0xFF00;
  1008. mode = (mode1 & 0xF);
  1009. pr_debug("%s: ac->mode after anding with FF00:0x%x,\n",
  1010. __func__, ac->io_mode);
  1011. if ((mode == ASYNC_IO_MODE) || (mode == SYNC_IO_MODE)) {
  1012. ac->io_mode |= mode1;
  1013. pr_debug("%s: Set Mode to 0x%x\n", __func__, ac->io_mode);
  1014. } else {
  1015. pr_err("%s: Not an valid IO Mode:%d\n", __func__, ac->io_mode);
  1016. ret = -EINVAL;
  1017. }
  1018. return ret;
  1019. }
  1020. EXPORT_SYMBOL(q6asm_set_io_mode);
  1021. void *q6asm_mmap_apr_reg(void)
  1022. {
  1023. if ((atomic_read(&this_mmap.ref_cnt) == 0) ||
  1024. (this_mmap.apr == NULL)) {
  1025. this_mmap.apr = apr_register("ADSP", "ASM",
  1026. (apr_fn)q6asm_srvc_callback,
  1027. 0x0FFFFFFFF, &this_mmap);
  1028. if (this_mmap.apr == NULL) {
  1029. pr_debug("%s: Unable to register APR ASM common port\n",
  1030. __func__);
  1031. goto fail;
  1032. }
  1033. }
  1034. atomic_inc(&this_mmap.ref_cnt);
  1035. return this_mmap.apr;
  1036. fail:
  1037. return NULL;
  1038. }
  1039. /**
  1040. * q6asm_send_stream_cmd -
  1041. * command to send for ASM stream
  1042. *
  1043. * @ac: audio client handle
  1044. * @data: event data
  1045. *
  1046. * Returns 0 on success or error on failure
  1047. */
  1048. int q6asm_send_stream_cmd(struct audio_client *ac,
  1049. struct msm_adsp_event_data *data)
  1050. {
  1051. char *asm_params = NULL;
  1052. struct apr_hdr hdr;
  1053. int rc;
  1054. uint32_t sz = 0;
  1055. uint64_t actual_sz = 0;
  1056. int session_id = 0;
  1057. if (!data || !ac) {
  1058. pr_err("%s: %s is NULL\n", __func__,
  1059. (!data) ? "data" : "ac");
  1060. rc = -EINVAL;
  1061. goto done;
  1062. }
  1063. session_id = q6asm_get_session_id_from_audio_client(ac);
  1064. if (!session_id) {
  1065. rc = -EINVAL;
  1066. goto done;
  1067. }
  1068. if (data->event_type >= ARRAY_SIZE(adsp_reg_event_opcode)) {
  1069. pr_err("%s: event %u out of boundary of array size of (%lu)\n",
  1070. __func__, data->event_type,
  1071. (long)ARRAY_SIZE(adsp_reg_event_opcode));
  1072. rc = -EINVAL;
  1073. goto done;
  1074. }
  1075. actual_sz = sizeof(struct apr_hdr) + data->payload_len;
  1076. if (actual_sz > U32_MAX) {
  1077. pr_err("%s: payload size 0x%X exceeds limit\n",
  1078. __func__, data->payload_len);
  1079. rc = -EINVAL;
  1080. goto done;
  1081. }
  1082. sz = (uint32_t)actual_sz;
  1083. asm_params = kzalloc(sz, GFP_KERNEL);
  1084. if (!asm_params) {
  1085. rc = -ENOMEM;
  1086. goto done;
  1087. }
  1088. mutex_lock(&session[session_id].mutex_lock_per_session);
  1089. if (!q6asm_is_valid_audio_client(ac)) {
  1090. rc = -EINVAL;
  1091. goto fail_send_param;
  1092. }
  1093. q6asm_add_hdr_async(ac, &hdr, sz, TRUE);
  1094. atomic_set(&ac->cmd_state_pp, -1);
  1095. hdr.opcode = adsp_reg_event_opcode[data->event_type];
  1096. memcpy(asm_params, &hdr, sizeof(struct apr_hdr));
  1097. memcpy(asm_params + sizeof(struct apr_hdr),
  1098. data->payload, data->payload_len);
  1099. rc = apr_send_pkt(ac->apr, (uint32_t *) asm_params);
  1100. if (rc < 0) {
  1101. pr_err("%s: stream event cmd apr pkt failed\n", __func__);
  1102. rc = -EINVAL;
  1103. goto fail_send_param;
  1104. }
  1105. rc = wait_event_timeout(ac->cmd_wait,
  1106. (atomic_read(&ac->cmd_state_pp) >= 0),
  1107. msecs_to_jiffies(TIMEOUT_MS));
  1108. if (!rc) {
  1109. pr_err("%s: timeout for stream event cmd resp\n", __func__);
  1110. rc = -ETIMEDOUT;
  1111. goto fail_send_param;
  1112. }
  1113. if (atomic_read(&ac->cmd_state_pp) > 0) {
  1114. pr_err("%s: DSP returned error[%s] for stream event cmd\n",
  1115. __func__, adsp_err_get_err_str(
  1116. atomic_read(&ac->cmd_state_pp)));
  1117. rc = adsp_err_get_lnx_err_code(
  1118. atomic_read(&ac->cmd_state_pp));
  1119. goto fail_send_param;
  1120. }
  1121. rc = 0;
  1122. fail_send_param:
  1123. mutex_unlock(&session[session_id].mutex_lock_per_session);
  1124. kfree(asm_params);
  1125. done:
  1126. return rc;
  1127. }
  1128. EXPORT_SYMBOL(q6asm_send_stream_cmd);
  1129. /**
  1130. * q6asm_audio_client_alloc -
  1131. * Alloc audio client for ASM
  1132. *
  1133. * @cb: callback fn
  1134. * @priv: private data
  1135. *
  1136. * Returns ac pointer on success or NULL on failure
  1137. */
  1138. struct audio_client *q6asm_audio_client_alloc(app_cb cb, void *priv)
  1139. {
  1140. struct audio_client *ac;
  1141. int n;
  1142. int lcnt = 0;
  1143. int rc = 0;
  1144. ac = kzalloc(sizeof(struct audio_client), GFP_KERNEL);
  1145. if (!ac)
  1146. return NULL;
  1147. mutex_lock(&session_lock);
  1148. n = q6asm_session_alloc(ac);
  1149. if (n <= 0) {
  1150. pr_err("%s: ASM Session alloc fail n=%d\n", __func__, n);
  1151. mutex_unlock(&session_lock);
  1152. kfree(ac);
  1153. goto fail_session;
  1154. }
  1155. ac->session = n;
  1156. ac->cb = cb;
  1157. ac->path_delay = UINT_MAX;
  1158. ac->priv = priv;
  1159. ac->io_mode = SYNC_IO_MODE;
  1160. ac->perf_mode = LEGACY_PCM_MODE;
  1161. ac->fptr_cache_ops = NULL;
  1162. /* DSP expects stream id from 1 */
  1163. ac->stream_id = 1;
  1164. ac->apr = apr_register("ADSP", "ASM",
  1165. (apr_fn)q6asm_callback,
  1166. ((ac->session) << 8 | 0x0001),
  1167. ac);
  1168. if (ac->apr == NULL) {
  1169. pr_err("%s: Registration with APR failed\n", __func__);
  1170. mutex_unlock(&session_lock);
  1171. goto fail_apr1;
  1172. }
  1173. ac->apr2 = apr_register("ADSP", "ASM",
  1174. (apr_fn)q6asm_callback,
  1175. ((ac->session) << 8 | 0x0002),
  1176. ac);
  1177. if (ac->apr2 == NULL) {
  1178. pr_err("%s: Registration with APR-2 failed\n", __func__);
  1179. mutex_unlock(&session_lock);
  1180. goto fail_apr2;
  1181. }
  1182. rtac_set_asm_handle(n, ac->apr);
  1183. pr_debug("%s: Registering the common port with APR\n", __func__);
  1184. ac->mmap_apr = q6asm_mmap_apr_reg();
  1185. if (ac->mmap_apr == NULL) {
  1186. mutex_unlock(&session_lock);
  1187. goto fail_mmap;
  1188. }
  1189. init_waitqueue_head(&ac->cmd_wait);
  1190. init_waitqueue_head(&ac->time_wait);
  1191. init_waitqueue_head(&ac->mem_wait);
  1192. atomic_set(&ac->time_flag, 1);
  1193. atomic_set(&ac->reset, 0);
  1194. INIT_LIST_HEAD(&ac->port[0].mem_map_handle);
  1195. INIT_LIST_HEAD(&ac->port[1].mem_map_handle);
  1196. pr_debug("%s: mem_map_handle list init'ed\n", __func__);
  1197. mutex_init(&ac->cmd_lock);
  1198. for (lcnt = 0; lcnt <= OUT; lcnt++) {
  1199. mutex_init(&ac->port[lcnt].lock);
  1200. spin_lock_init(&ac->port[lcnt].dsp_lock);
  1201. }
  1202. atomic_set(&ac->cmd_state, 0);
  1203. atomic_set(&ac->cmd_state_pp, 0);
  1204. atomic_set(&ac->mem_state, 0);
  1205. rc = send_asm_custom_topology(ac);
  1206. if (rc < 0) {
  1207. mutex_unlock(&session_lock);
  1208. goto fail_mmap;
  1209. }
  1210. pr_debug("%s: session[%d]\n", __func__, ac->session);
  1211. mutex_unlock(&session_lock);
  1212. return ac;
  1213. fail_mmap:
  1214. apr_deregister(ac->apr2);
  1215. fail_apr2:
  1216. apr_deregister(ac->apr);
  1217. fail_apr1:
  1218. q6asm_session_free(ac);
  1219. fail_session:
  1220. return NULL;
  1221. }
  1222. EXPORT_SYMBOL(q6asm_audio_client_alloc);
  1223. /**
  1224. * q6asm_get_audio_client -
  1225. * Retrieve audio client for ASM
  1226. *
  1227. * @session_id: ASM session id
  1228. *
  1229. * Returns valid pointer on success or NULL on failure
  1230. */
  1231. struct audio_client *q6asm_get_audio_client(int session_id)
  1232. {
  1233. if (session_id == ASM_CONTROL_SESSION)
  1234. return &common_client;
  1235. if ((session_id <= 0) || (session_id > ASM_ACTIVE_STREAMS_ALLOWED)) {
  1236. pr_err("%s: invalid session: %d\n", __func__, session_id);
  1237. goto err;
  1238. }
  1239. if (!(session[session_id].ac)) {
  1240. pr_err("%s: session not active: %d\n", __func__, session_id);
  1241. goto err;
  1242. }
  1243. return session[session_id].ac;
  1244. err:
  1245. return NULL;
  1246. }
  1247. EXPORT_SYMBOL(q6asm_get_audio_client);
  1248. /**
  1249. * q6asm_audio_client_buf_alloc -
  1250. * Allocs memory from ION for ASM
  1251. *
  1252. * @dir: RX or TX direction
  1253. * @ac: Audio client handle
  1254. * @bufsz: size of each buffer
  1255. * @bufcnt: number of buffers to alloc
  1256. *
  1257. * Returns 0 on success or error on failure
  1258. */
  1259. int q6asm_audio_client_buf_alloc(unsigned int dir,
  1260. struct audio_client *ac,
  1261. unsigned int bufsz,
  1262. uint32_t bufcnt)
  1263. {
  1264. int cnt = 0;
  1265. int rc = 0;
  1266. struct audio_buffer *buf;
  1267. size_t len;
  1268. if (!(ac) || !(bufsz) || ((dir != IN) && (dir != OUT))) {
  1269. pr_err("%s: ac %pK bufsz %d dir %d\n", __func__, ac, bufsz,
  1270. dir);
  1271. return -EINVAL;
  1272. }
  1273. pr_debug("%s: session[%d]bufsz[%d]bufcnt[%d]\n", __func__, ac->session,
  1274. bufsz, bufcnt);
  1275. if (ac->session <= 0 || ac->session > 8) {
  1276. pr_err("%s: Session ID is invalid, session = %d\n", __func__,
  1277. ac->session);
  1278. goto fail;
  1279. }
  1280. if (ac->io_mode & SYNC_IO_MODE) {
  1281. if (ac->port[dir].buf) {
  1282. pr_debug("%s: buffer already allocated\n", __func__);
  1283. return 0;
  1284. }
  1285. mutex_lock(&ac->cmd_lock);
  1286. if (bufcnt > (U32_MAX/sizeof(struct audio_buffer))) {
  1287. pr_err("%s: Buffer size overflows", __func__);
  1288. mutex_unlock(&ac->cmd_lock);
  1289. goto fail;
  1290. }
  1291. buf = kzalloc(((sizeof(struct audio_buffer))*bufcnt),
  1292. GFP_KERNEL);
  1293. if (!buf) {
  1294. mutex_unlock(&ac->cmd_lock);
  1295. goto fail;
  1296. }
  1297. ac->port[dir].buf = buf;
  1298. while (cnt < bufcnt) {
  1299. if (bufsz > 0) {
  1300. if (!buf[cnt].data) {
  1301. rc = msm_audio_ion_alloc(
  1302. &buf[cnt].dma_buf,
  1303. bufsz,
  1304. &buf[cnt].phys,
  1305. &len,
  1306. &buf[cnt].data);
  1307. if (rc) {
  1308. pr_err("%s: ION Get Physical for AUDIO failed, rc = %d\n",
  1309. __func__, rc);
  1310. mutex_unlock(&ac->cmd_lock);
  1311. goto fail;
  1312. }
  1313. buf[cnt].used = 1;
  1314. buf[cnt].size = bufsz;
  1315. buf[cnt].actual_size = bufsz;
  1316. pr_debug("%s: data[%pK]phys[%pK][%pK]\n",
  1317. __func__,
  1318. buf[cnt].data,
  1319. &buf[cnt].phys,
  1320. &buf[cnt].phys);
  1321. cnt++;
  1322. }
  1323. }
  1324. }
  1325. ac->port[dir].max_buf_cnt = cnt;
  1326. mutex_unlock(&ac->cmd_lock);
  1327. rc = q6asm_memory_map_regions(ac, dir, bufsz, cnt, 0);
  1328. if (rc < 0) {
  1329. pr_err("%s: CMD Memory_map_regions failed %d for size %d\n",
  1330. __func__, rc, bufsz);
  1331. goto fail;
  1332. }
  1333. }
  1334. return 0;
  1335. fail:
  1336. q6asm_audio_client_buf_free(dir, ac);
  1337. return -EINVAL;
  1338. }
  1339. EXPORT_SYMBOL(q6asm_audio_client_buf_alloc);
  1340. /**
  1341. * q6asm_audio_client_buf_alloc_contiguous -
  1342. * Alloc contiguous memory from ION for ASM
  1343. *
  1344. * @dir: RX or TX direction
  1345. * @ac: Audio client handle
  1346. * @bufsz: size of each buffer
  1347. * @bufcnt: number of buffers to alloc
  1348. *
  1349. * Returns 0 on success or error on failure
  1350. */
  1351. int q6asm_audio_client_buf_alloc_contiguous(unsigned int dir,
  1352. struct audio_client *ac,
  1353. unsigned int bufsz,
  1354. unsigned int bufcnt)
  1355. {
  1356. int cnt = 0;
  1357. int rc = 0;
  1358. struct audio_buffer *buf;
  1359. size_t len;
  1360. int bytes_to_alloc;
  1361. if (!(ac) || ((dir != IN) && (dir != OUT))) {
  1362. pr_err("%s: ac %pK dir %d\n", __func__, ac, dir);
  1363. return -EINVAL;
  1364. }
  1365. pr_debug("%s: session[%d]bufsz[%d]bufcnt[%d]\n",
  1366. __func__, ac->session,
  1367. bufsz, bufcnt);
  1368. if (ac->session <= 0 || ac->session > 8) {
  1369. pr_err("%s: Session ID is invalid, session = %d\n", __func__,
  1370. ac->session);
  1371. goto fail;
  1372. }
  1373. if (ac->port[dir].buf) {
  1374. pr_err("%s: buffer already allocated\n", __func__);
  1375. return 0;
  1376. }
  1377. mutex_lock(&ac->cmd_lock);
  1378. buf = kzalloc(((sizeof(struct audio_buffer))*bufcnt),
  1379. GFP_KERNEL);
  1380. if (!buf) {
  1381. pr_err("%s: buffer allocation failed\n", __func__);
  1382. mutex_unlock(&ac->cmd_lock);
  1383. goto fail;
  1384. }
  1385. ac->port[dir].buf = buf;
  1386. /* check for integer overflow */
  1387. if ((bufcnt > 0) && ((INT_MAX / bufcnt) < bufsz)) {
  1388. pr_err("%s: integer overflow\n", __func__);
  1389. mutex_unlock(&ac->cmd_lock);
  1390. goto fail;
  1391. }
  1392. bytes_to_alloc = bufsz * bufcnt;
  1393. /* The size to allocate should be multiple of 4K bytes */
  1394. bytes_to_alloc = PAGE_ALIGN(bytes_to_alloc);
  1395. rc = msm_audio_ion_alloc(&buf[0].dma_buf,
  1396. bytes_to_alloc,
  1397. &buf[0].phys, &len,
  1398. &buf[0].data);
  1399. if (rc) {
  1400. pr_err("%s: Audio ION alloc is failed, rc = %d\n",
  1401. __func__, rc);
  1402. mutex_unlock(&ac->cmd_lock);
  1403. goto fail;
  1404. }
  1405. buf[0].used = dir ^ 1;
  1406. buf[0].size = bufsz;
  1407. buf[0].actual_size = bufsz;
  1408. cnt = 1;
  1409. while (cnt < bufcnt) {
  1410. if (bufsz > 0) {
  1411. buf[cnt].data = buf[0].data + (cnt * bufsz);
  1412. buf[cnt].phys = buf[0].phys + (cnt * bufsz);
  1413. if (!buf[cnt].data) {
  1414. pr_err("%s: Buf alloc failed\n",
  1415. __func__);
  1416. mutex_unlock(&ac->cmd_lock);
  1417. goto fail;
  1418. }
  1419. buf[cnt].used = dir ^ 1;
  1420. buf[cnt].size = bufsz;
  1421. buf[cnt].actual_size = bufsz;
  1422. pr_debug("%s: data[%pK]phys[%pK][%pK]\n",
  1423. __func__,
  1424. buf[cnt].data,
  1425. &buf[cnt].phys,
  1426. &buf[cnt].phys);
  1427. }
  1428. cnt++;
  1429. }
  1430. ac->port[dir].max_buf_cnt = cnt;
  1431. mutex_unlock(&ac->cmd_lock);
  1432. rc = q6asm_memory_map_regions(ac, dir, bufsz, cnt, 1);
  1433. if (rc < 0) {
  1434. pr_err("%s: CMD Memory_map_regions failed %d for size %d\n",
  1435. __func__, rc, bufsz);
  1436. goto fail;
  1437. }
  1438. return 0;
  1439. fail:
  1440. q6asm_audio_client_buf_free_contiguous(dir, ac);
  1441. return -EINVAL;
  1442. }
  1443. EXPORT_SYMBOL(q6asm_audio_client_buf_alloc_contiguous);
  1444. static int32_t q6asm_srvc_callback(struct apr_client_data *data, void *priv)
  1445. {
  1446. uint32_t dir = 0;
  1447. uint32_t i = IN;
  1448. uint32_t *payload;
  1449. unsigned long dsp_flags = 0;
  1450. unsigned long flags = 0;
  1451. struct asm_buffer_node *buf_node = NULL;
  1452. struct list_head *ptr, *next;
  1453. union asm_token_struct asm_token;
  1454. struct audio_client *ac = NULL;
  1455. struct audio_port_data *port;
  1456. int session_id;
  1457. if (!data) {
  1458. pr_err("%s: Invalid CB\n", __func__);
  1459. return 0;
  1460. }
  1461. payload = data->payload;
  1462. if (data->opcode == RESET_EVENTS) {
  1463. pr_debug("%s: Reset event is received: %d %d apr[%pK]\n",
  1464. __func__,
  1465. data->reset_event,
  1466. data->reset_proc,
  1467. this_mmap.apr);
  1468. atomic_set(&this_mmap.ref_cnt, 0);
  1469. apr_reset(this_mmap.apr);
  1470. this_mmap.apr = NULL;
  1471. for (; i <= OUT; i++) {
  1472. list_for_each_safe(ptr, next,
  1473. &common_client.port[i].mem_map_handle) {
  1474. buf_node = list_entry(ptr,
  1475. struct asm_buffer_node,
  1476. list);
  1477. if (buf_node->buf_phys_addr ==
  1478. common_client.port[i].buf->phys) {
  1479. list_del(&buf_node->list);
  1480. kfree(buf_node);
  1481. }
  1482. }
  1483. pr_debug("%s: Clearing custom topology\n", __func__);
  1484. }
  1485. cal_utils_clear_cal_block_q6maps(ASM_MAX_CAL_TYPES, cal_data);
  1486. common_client.mmap_apr = NULL;
  1487. mutex_lock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  1488. set_custom_topology = 1;
  1489. mutex_unlock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  1490. topology_map_handle = 0;
  1491. rtac_clear_mapping(ASM_RTAC_CAL);
  1492. return 0;
  1493. }
  1494. asm_token.token = data->token;
  1495. session_id = asm_token._token.session_id;
  1496. if ((session_id > 0 && session_id <= ASM_ACTIVE_STREAMS_ALLOWED))
  1497. spin_lock_irqsave(&(session[session_id].session_lock), flags);
  1498. ac = q6asm_get_audio_client(session_id);
  1499. dir = q6asm_get_flag_from_token(&asm_token, ASM_DIRECTION_OFFSET);
  1500. if (!ac) {
  1501. pr_debug("%s: session[%d] already freed\n",
  1502. __func__, session_id);
  1503. if ((session_id > 0 &&
  1504. session_id <= ASM_ACTIVE_STREAMS_ALLOWED))
  1505. spin_unlock_irqrestore(
  1506. &(session[session_id].session_lock), flags);
  1507. return 0;
  1508. }
  1509. if (data->payload_size > sizeof(int)) {
  1510. pr_debug("%s:ptr0[0x%x]ptr1[0x%x]opcode[0x%x] token[0x%x]payload_s[%d] src[%d] dest[%d]sid[%d]dir[%d]\n",
  1511. __func__, payload[0], payload[1], data->opcode,
  1512. data->token, data->payload_size, data->src_port,
  1513. data->dest_port, asm_token._token.session_id, dir);
  1514. pr_debug("%s:Payload = [0x%x] status[0x%x]\n",
  1515. __func__, payload[0], payload[1]);
  1516. } else if (data->payload_size == sizeof(int)) {
  1517. pr_debug("%s:ptr0[0x%x]opcode[0x%x] token[0x%x]payload_s[%d] src[%d] dest[%d]sid[%d]dir[%d]\n",
  1518. __func__, payload[0], data->opcode,
  1519. data->token, data->payload_size, data->src_port,
  1520. data->dest_port, asm_token._token.session_id, dir);
  1521. pr_debug("%s:Payload = [0x%x]\n",
  1522. __func__, payload[0]);
  1523. }
  1524. if (data->opcode == APR_BASIC_RSP_RESULT) {
  1525. switch (payload[0]) {
  1526. case ASM_CMD_SHARED_MEM_MAP_REGIONS:
  1527. case ASM_CMD_SHARED_MEM_UNMAP_REGIONS:
  1528. case ASM_CMD_ADD_TOPOLOGIES:
  1529. if (payload[1] != 0) {
  1530. pr_err("%s: cmd = 0x%x returned error = 0x%x sid:%d\n",
  1531. __func__, payload[0], payload[1],
  1532. asm_token._token.session_id);
  1533. if (payload[0] ==
  1534. ASM_CMD_SHARED_MEM_UNMAP_REGIONS)
  1535. atomic_set(&ac->unmap_cb_success, 0);
  1536. atomic_set(&ac->mem_state, payload[1]);
  1537. wake_up(&ac->mem_wait);
  1538. } else {
  1539. if (payload[0] ==
  1540. ASM_CMD_SHARED_MEM_UNMAP_REGIONS)
  1541. atomic_set(&ac->unmap_cb_success, 1);
  1542. }
  1543. if (atomic_cmpxchg(&ac->mem_state, -1, 0) == -1)
  1544. wake_up(&ac->mem_wait);
  1545. dev_vdbg(ac->dev, "%s: Payload = [0x%x] status[0x%x]\n",
  1546. __func__, payload[0], payload[1]);
  1547. break;
  1548. default:
  1549. pr_debug("%s: command[0x%x] not expecting rsp\n",
  1550. __func__, payload[0]);
  1551. break;
  1552. }
  1553. if ((session_id > 0 &&
  1554. session_id <= ASM_ACTIVE_STREAMS_ALLOWED))
  1555. spin_unlock_irqrestore(
  1556. &(session[session_id].session_lock), flags);
  1557. return 0;
  1558. }
  1559. port = &ac->port[dir];
  1560. switch (data->opcode) {
  1561. case ASM_CMDRSP_SHARED_MEM_MAP_REGIONS:{
  1562. pr_debug("%s:PL#0[0x%x] dir=0x%x s_id=0x%x\n",
  1563. __func__, payload[0], dir, asm_token._token.session_id);
  1564. spin_lock_irqsave(&port->dsp_lock, dsp_flags);
  1565. if (atomic_cmpxchg(&ac->mem_state, -1, 0) == -1) {
  1566. ac->port[dir].tmp_hdl = payload[0];
  1567. wake_up(&ac->mem_wait);
  1568. }
  1569. spin_unlock_irqrestore(&port->dsp_lock, dsp_flags);
  1570. break;
  1571. }
  1572. case ASM_CMD_SHARED_MEM_UNMAP_REGIONS:{
  1573. pr_debug("%s: PL#0[0x%x]PL#1 [0x%x]\n",
  1574. __func__, payload[0], payload[1]);
  1575. spin_lock_irqsave(&port->dsp_lock, dsp_flags);
  1576. if (atomic_cmpxchg(&ac->mem_state, -1, 0) == -1)
  1577. wake_up(&ac->mem_wait);
  1578. spin_unlock_irqrestore(&port->dsp_lock, dsp_flags);
  1579. break;
  1580. }
  1581. default:
  1582. pr_debug("%s: command[0x%x]success [0x%x]\n",
  1583. __func__, payload[0], payload[1]);
  1584. }
  1585. if (ac->cb)
  1586. ac->cb(data->opcode, data->token,
  1587. data->payload, ac->priv);
  1588. if ((session_id > 0 && session_id <= ASM_ACTIVE_STREAMS_ALLOWED))
  1589. spin_unlock_irqrestore(
  1590. &(session[session_id].session_lock), flags);
  1591. return 0;
  1592. }
  1593. static void q6asm_process_mtmx_get_param_rsp(struct audio_client *ac,
  1594. struct asm_mtmx_strtr_get_params_cmdrsp *cmdrsp)
  1595. {
  1596. struct asm_session_mtmx_strtr_param_session_time_v3_t *time;
  1597. if (cmdrsp->err_code) {
  1598. dev_err_ratelimited(ac->dev,
  1599. "%s: err=%x, mod_id=%x, param_id=%x\n",
  1600. __func__, cmdrsp->err_code,
  1601. cmdrsp->param_info.module_id,
  1602. cmdrsp->param_info.param_id);
  1603. return;
  1604. }
  1605. dev_dbg_ratelimited(ac->dev,
  1606. "%s: mod_id=%x, param_id=%x\n", __func__,
  1607. cmdrsp->param_info.module_id,
  1608. cmdrsp->param_info.param_id);
  1609. switch (cmdrsp->param_info.module_id) {
  1610. case ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC:
  1611. switch (cmdrsp->param_info.param_id) {
  1612. case ASM_SESSION_MTMX_STRTR_PARAM_SESSION_TIME_V3:
  1613. time = &cmdrsp->param_data.session_time;
  1614. dev_vdbg(ac->dev, "%s: GET_TIME_V3, time_lsw=%x, time_msw=%x\n",
  1615. __func__, time->session_time_lsw,
  1616. time->session_time_msw);
  1617. ac->time_stamp = (uint64_t)(((uint64_t)
  1618. time->session_time_msw << 32) |
  1619. time->session_time_lsw);
  1620. if (time->flags &
  1621. ASM_SESSION_MTMX_STRTR_PARAM_STIME_TSTMP_FLG_BMASK)
  1622. dev_warn_ratelimited(ac->dev,
  1623. "%s: recv inval tstmp\n",
  1624. __func__);
  1625. if (atomic_cmpxchg(&ac->time_flag, 1, 0))
  1626. wake_up(&ac->time_wait);
  1627. break;
  1628. default:
  1629. dev_err(ac->dev, "%s: unexpected param_id %x\n",
  1630. __func__, cmdrsp->param_info.param_id);
  1631. break;
  1632. }
  1633. break;
  1634. default:
  1635. dev_err(ac->dev, "%s: unexpected mod_id %x\n", __func__,
  1636. cmdrsp->param_info.module_id);
  1637. break;
  1638. }
  1639. }
  1640. static int32_t q6asm_callback(struct apr_client_data *data, void *priv)
  1641. {
  1642. int i = 0;
  1643. struct audio_client *ac = (struct audio_client *)priv;
  1644. unsigned long dsp_flags = 0;
  1645. uint32_t *payload;
  1646. uint32_t wakeup_flag = 1;
  1647. int32_t ret = 0;
  1648. union asm_token_struct asm_token;
  1649. uint8_t buf_index;
  1650. struct msm_adsp_event_data *pp_event_package = NULL;
  1651. uint32_t payload_size = 0;
  1652. unsigned long flags = 0;
  1653. int session_id;
  1654. if (ac == NULL) {
  1655. pr_err("%s: ac NULL\n", __func__);
  1656. return -EINVAL;
  1657. }
  1658. if (data == NULL) {
  1659. pr_err("%s: data NULL\n", __func__);
  1660. return -EINVAL;
  1661. }
  1662. session_id = q6asm_get_session_id_from_audio_client(ac);
  1663. if (session_id <= 0 || session_id > ASM_ACTIVE_STREAMS_ALLOWED) {
  1664. pr_err("%s: Session ID is invalid, session = %d\n", __func__,
  1665. session_id);
  1666. return -EINVAL;
  1667. }
  1668. spin_lock_irqsave(&(session[session_id].session_lock), flags);
  1669. if (!q6asm_is_valid_audio_client(ac)) {
  1670. pr_err("%s: audio client pointer is invalid, ac = %pK\n",
  1671. __func__, ac);
  1672. spin_unlock_irqrestore(
  1673. &(session[session_id].session_lock), flags);
  1674. return -EINVAL;
  1675. }
  1676. payload = data->payload;
  1677. asm_token.token = data->token;
  1678. if (q6asm_get_flag_from_token(&asm_token, ASM_CMD_NO_WAIT_OFFSET)) {
  1679. pr_debug("%s: No wait command opcode[0x%x] cmd_opcode:%x\n",
  1680. __func__, data->opcode, payload ? payload[0] : 0);
  1681. wakeup_flag = 0;
  1682. }
  1683. if (data->opcode == RESET_EVENTS) {
  1684. atomic_set(&ac->reset, 1);
  1685. if (ac->apr == NULL) {
  1686. ac->apr = ac->apr2;
  1687. ac->apr2 = NULL;
  1688. }
  1689. pr_debug("%s: Reset event is received: %d %d apr[%pK]\n",
  1690. __func__,
  1691. data->reset_event, data->reset_proc, ac->apr);
  1692. if (ac->cb)
  1693. ac->cb(data->opcode, data->token,
  1694. (uint32_t *)data->payload, ac->priv);
  1695. apr_reset(ac->apr);
  1696. ac->apr = NULL;
  1697. atomic_set(&ac->time_flag, 0);
  1698. atomic_set(&ac->cmd_state, 0);
  1699. atomic_set(&ac->mem_state, 0);
  1700. atomic_set(&ac->cmd_state_pp, 0);
  1701. wake_up(&ac->time_wait);
  1702. wake_up(&ac->cmd_wait);
  1703. wake_up(&ac->mem_wait);
  1704. spin_unlock_irqrestore(
  1705. &(session[session_id].session_lock), flags);
  1706. return 0;
  1707. }
  1708. dev_vdbg(ac->dev, "%s: session[%d]opcode[0x%x] token[0x%x]payload_size[%d] src[%d] dest[%d]\n",
  1709. __func__,
  1710. ac->session, data->opcode,
  1711. data->token, data->payload_size, data->src_port,
  1712. data->dest_port);
  1713. if ((data->opcode != ASM_DATA_EVENT_RENDERED_EOS) &&
  1714. (data->opcode != ASM_DATA_EVENT_EOS) &&
  1715. (data->opcode != ASM_SESSION_EVENT_RX_UNDERFLOW)) {
  1716. if (payload == NULL) {
  1717. pr_err("%s: payload is null\n", __func__);
  1718. spin_unlock_irqrestore(
  1719. &(session[session_id].session_lock), flags);
  1720. return -EINVAL;
  1721. }
  1722. dev_vdbg(ac->dev, "%s: Payload = [0x%x] status[0x%x] opcode 0x%x\n",
  1723. __func__, payload[0], payload[1], data->opcode);
  1724. }
  1725. if (data->opcode == APR_BASIC_RSP_RESULT) {
  1726. switch (payload[0]) {
  1727. case ASM_STREAM_CMD_SET_PP_PARAMS_V2:
  1728. case ASM_STREAM_CMD_SET_PP_PARAMS_V3:
  1729. if (rtac_make_asm_callback(ac->session, payload,
  1730. data->payload_size))
  1731. break;
  1732. case ASM_SESSION_CMD_PAUSE:
  1733. case ASM_SESSION_CMD_SUSPEND:
  1734. case ASM_DATA_CMD_EOS:
  1735. case ASM_STREAM_CMD_CLOSE:
  1736. case ASM_STREAM_CMD_FLUSH:
  1737. case ASM_SESSION_CMD_RUN_V2:
  1738. case ASM_SESSION_CMD_REGISTER_FORX_OVERFLOW_EVENTS:
  1739. case ASM_STREAM_CMD_FLUSH_READBUFS:
  1740. pr_debug("%s: session %d opcode 0x%x token 0x%x Payload = [0x%x] src %d dest %d\n",
  1741. __func__, ac->session, data->opcode, data->token,
  1742. payload[0], data->src_port, data->dest_port);
  1743. ret = q6asm_is_valid_session(data, priv);
  1744. if (ret != 0) {
  1745. pr_err("%s: session invalid %d\n", __func__, ret);
  1746. spin_unlock_irqrestore(
  1747. &(session[session_id].session_lock), flags);
  1748. return ret;
  1749. }
  1750. case ASM_SESSION_CMD_SET_MTMX_STRTR_PARAMS_V2:
  1751. case ASM_STREAM_CMD_OPEN_READ_V3:
  1752. case ASM_STREAM_CMD_OPEN_WRITE_V3:
  1753. case ASM_STREAM_CMD_OPEN_PULL_MODE_WRITE:
  1754. case ASM_STREAM_CMD_OPEN_PUSH_MODE_READ:
  1755. case ASM_STREAM_CMD_OPEN_READWRITE_V2:
  1756. case ASM_STREAM_CMD_OPEN_LOOPBACK_V2:
  1757. case ASM_STREAM_CMD_OPEN_TRANSCODE_LOOPBACK:
  1758. case ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2:
  1759. case ASM_DATA_CMD_IEC_60958_MEDIA_FMT:
  1760. case ASM_STREAM_CMD_SET_ENCDEC_PARAM:
  1761. case ASM_STREAM_CMD_SET_ENCDEC_PARAM_V2:
  1762. case ASM_STREAM_CMD_REGISTER_ENCDEC_EVENTS:
  1763. case ASM_STREAM_CMD_REGISTER_IEC_61937_FMT_UPDATE:
  1764. case ASM_DATA_CMD_REMOVE_INITIAL_SILENCE:
  1765. case ASM_DATA_CMD_REMOVE_TRAILING_SILENCE:
  1766. case ASM_SESSION_CMD_REGISTER_FOR_RX_UNDERFLOW_EVENTS:
  1767. case ASM_STREAM_CMD_OPEN_READ_COMPRESSED:
  1768. case ASM_STREAM_CMD_OPEN_WRITE_COMPRESSED:
  1769. pr_debug("%s: session %d opcode 0x%x token 0x%x Payload = [0x%x] stat 0x%x src %d dest %d\n",
  1770. __func__, ac->session,
  1771. data->opcode, data->token,
  1772. payload[0], payload[1],
  1773. data->src_port, data->dest_port);
  1774. if (payload[1] != 0) {
  1775. pr_err("%s: cmd = 0x%x returned error = 0x%x\n",
  1776. __func__, payload[0], payload[1]);
  1777. if (wakeup_flag) {
  1778. if ((is_adsp_reg_event(payload[0]) >=
  1779. 0) ||
  1780. (payload[0] ==
  1781. ASM_STREAM_CMD_SET_PP_PARAMS_V2) ||
  1782. (payload[0] ==
  1783. ASM_STREAM_CMD_SET_PP_PARAMS_V3))
  1784. atomic_set(&ac->cmd_state_pp,
  1785. payload[1]);
  1786. else
  1787. atomic_set(&ac->cmd_state,
  1788. payload[1]);
  1789. wake_up(&ac->cmd_wait);
  1790. }
  1791. spin_unlock_irqrestore(
  1792. &(session[session_id].session_lock),
  1793. flags);
  1794. return 0;
  1795. }
  1796. if ((is_adsp_reg_event(payload[0]) >= 0) ||
  1797. (payload[0] == ASM_STREAM_CMD_SET_PP_PARAMS_V2) ||
  1798. (payload[0] == ASM_STREAM_CMD_SET_PP_PARAMS_V3)) {
  1799. if (atomic_read(&ac->cmd_state_pp) &&
  1800. wakeup_flag) {
  1801. atomic_set(&ac->cmd_state_pp, 0);
  1802. wake_up(&ac->cmd_wait);
  1803. }
  1804. } else {
  1805. if (atomic_read(&ac->cmd_state) &&
  1806. wakeup_flag) {
  1807. atomic_set(&ac->cmd_state, 0);
  1808. wake_up(&ac->cmd_wait);
  1809. }
  1810. }
  1811. if (ac->cb)
  1812. ac->cb(data->opcode, data->token,
  1813. (uint32_t *)data->payload, ac->priv);
  1814. break;
  1815. case ASM_CMD_ADD_TOPOLOGIES:
  1816. pr_debug("%s:Payload = [0x%x]stat[0x%x]\n",
  1817. __func__, payload[0], payload[1]);
  1818. if (payload[1] != 0) {
  1819. pr_err("%s: cmd = 0x%x returned error = 0x%x\n",
  1820. __func__, payload[0], payload[1]);
  1821. if (wakeup_flag) {
  1822. atomic_set(&ac->mem_state, payload[1]);
  1823. wake_up(&ac->mem_wait);
  1824. }
  1825. spin_unlock_irqrestore(
  1826. &(session[session_id].session_lock),
  1827. flags);
  1828. return 0;
  1829. }
  1830. if (atomic_read(&ac->mem_state) && wakeup_flag) {
  1831. atomic_set(&ac->mem_state, 0);
  1832. wake_up(&ac->mem_wait);
  1833. }
  1834. if (ac->cb)
  1835. ac->cb(data->opcode, data->token,
  1836. (uint32_t *)data->payload, ac->priv);
  1837. break;
  1838. case ASM_DATA_EVENT_WATERMARK: {
  1839. pr_debug("%s: Watermark opcode[0x%x] status[0x%x]",
  1840. __func__, payload[0], payload[1]);
  1841. break;
  1842. }
  1843. case ASM_STREAM_CMD_GET_PP_PARAMS_V2:
  1844. case ASM_STREAM_CMD_GET_PP_PARAMS_V3:
  1845. pr_debug("%s: ASM_STREAM_CMD_GET_PP_PARAMS session %d opcode 0x%x token 0x%x src %d dest %d\n",
  1846. __func__, ac->session, data->opcode,
  1847. data->token, data->src_port, data->dest_port);
  1848. /* Should only come here if there is an APR */
  1849. /* error or malformed APR packet. Otherwise */
  1850. /* response will be returned as */
  1851. /* ASM_STREAM_CMDRSP_GET_PP_PARAMS_V2 */
  1852. if (payload[1] != 0) {
  1853. pr_err("%s: ASM get param error = %d, resuming\n",
  1854. __func__, payload[1]);
  1855. rtac_make_asm_callback(ac->session, payload,
  1856. data->payload_size);
  1857. }
  1858. break;
  1859. case ASM_STREAM_CMD_REGISTER_PP_EVENTS:
  1860. pr_debug("%s: ASM_STREAM_CMD_REGISTER_PP_EVENTS session %d opcode 0x%x token 0x%x src %d dest %d\n",
  1861. __func__, ac->session,
  1862. data->opcode, data->token,
  1863. data->src_port, data->dest_port);
  1864. if (payload[1] != 0)
  1865. pr_err("%s: ASM get param error = %d, resuming\n",
  1866. __func__, payload[1]);
  1867. atomic_set(&ac->cmd_state_pp, payload[1]);
  1868. wake_up(&ac->cmd_wait);
  1869. break;
  1870. default:
  1871. pr_debug("%s: command[0x%x] not expecting rsp\n",
  1872. __func__, payload[0]);
  1873. break;
  1874. }
  1875. spin_unlock_irqrestore(
  1876. &(session[session_id].session_lock), flags);
  1877. return 0;
  1878. }
  1879. switch (data->opcode) {
  1880. case ASM_DATA_EVENT_WRITE_DONE_V2:{
  1881. struct audio_port_data *port = &ac->port[IN];
  1882. dev_vdbg(ac->dev, "%s: Rxed opcode[0x%x] status[0x%x] token[%d]",
  1883. __func__, payload[0], payload[1],
  1884. data->token);
  1885. if (ac->io_mode & SYNC_IO_MODE) {
  1886. if (port->buf == NULL) {
  1887. pr_err("%s: Unexpected Write Done\n",
  1888. __func__);
  1889. spin_unlock_irqrestore(
  1890. &(session[session_id].session_lock),
  1891. flags);
  1892. return -EINVAL;
  1893. }
  1894. spin_lock_irqsave(&port->dsp_lock, dsp_flags);
  1895. buf_index = asm_token._token.buf_index;
  1896. if (lower_32_bits(port->buf[buf_index].phys) !=
  1897. payload[0] ||
  1898. msm_audio_populate_upper_32_bits(
  1899. port->buf[buf_index].phys) != payload[1]) {
  1900. pr_debug("%s: Expected addr %pK\n",
  1901. __func__, &port->buf[buf_index].phys);
  1902. pr_err("%s: rxedl[0x%x] rxedu [0x%x]\n",
  1903. __func__, payload[0], payload[1]);
  1904. spin_unlock_irqrestore(&port->dsp_lock,
  1905. dsp_flags);
  1906. spin_unlock_irqrestore(
  1907. &(session[session_id].session_lock),
  1908. flags);
  1909. return -EINVAL;
  1910. }
  1911. port->buf[buf_index].used = 1;
  1912. spin_unlock_irqrestore(&port->dsp_lock, dsp_flags);
  1913. config_debug_fs_write_cb();
  1914. for (i = 0; i < port->max_buf_cnt; i++)
  1915. dev_vdbg(ac->dev, "%s %d\n",
  1916. __func__, port->buf[i].used);
  1917. }
  1918. break;
  1919. }
  1920. case ASM_STREAM_CMDRSP_GET_PP_PARAMS_V2:
  1921. case ASM_STREAM_CMDRSP_GET_PP_PARAMS_V3:
  1922. pr_debug("%s: ASM_STREAM_CMDRSP_GET_PP_PARAMS session %d opcode 0x%x token 0x%x src %d dest %d\n",
  1923. __func__, ac->session, data->opcode, data->token,
  1924. data->src_port, data->dest_port);
  1925. if (payload[0] != 0) {
  1926. pr_err("%s: ASM_STREAM_CMDRSP_GET_PP_PARAMS returned error = 0x%x\n",
  1927. __func__, payload[0]);
  1928. } else if (generic_get_data) {
  1929. generic_get_data->valid = 1;
  1930. if (generic_get_data->is_inband) {
  1931. pr_debug("%s: payload[1] = 0x%x, payload[2]=0x%x, payload[3]=0x%x\n",
  1932. __func__, payload[1], payload[2], payload[3]);
  1933. generic_get_data->size_in_ints = payload[3]>>2;
  1934. for (i = 0; i < payload[3]>>2; i++) {
  1935. generic_get_data->ints[i] =
  1936. payload[4+i];
  1937. pr_debug("%s: ASM callback val %i = %i\n",
  1938. __func__, i, payload[4+i]);
  1939. }
  1940. pr_debug("%s: callback size in ints = %i\n",
  1941. __func__,
  1942. generic_get_data->size_in_ints);
  1943. }
  1944. if (atomic_read(&ac->cmd_state) && wakeup_flag) {
  1945. atomic_set(&ac->cmd_state, 0);
  1946. wake_up(&ac->cmd_wait);
  1947. }
  1948. break;
  1949. }
  1950. rtac_make_asm_callback(ac->session, payload,
  1951. data->payload_size);
  1952. break;
  1953. case ASM_DATA_EVENT_READ_DONE_V2:{
  1954. struct audio_port_data *port = &ac->port[OUT];
  1955. config_debug_fs_read_cb();
  1956. dev_vdbg(ac->dev, "%s: ReadDone: status=%d buff_add=0x%x act_size=%d offset=%d\n",
  1957. __func__, payload[READDONE_IDX_STATUS],
  1958. payload[READDONE_IDX_BUFADD_LSW],
  1959. payload[READDONE_IDX_SIZE],
  1960. payload[READDONE_IDX_OFFSET]);
  1961. dev_vdbg(ac->dev, "%s: ReadDone:msw_ts=%d lsw_ts=%d memmap_hdl=0x%x flags=%d id=%d num=%d\n",
  1962. __func__, payload[READDONE_IDX_MSW_TS],
  1963. payload[READDONE_IDX_LSW_TS],
  1964. payload[READDONE_IDX_MEMMAP_HDL],
  1965. payload[READDONE_IDX_FLAGS],
  1966. payload[READDONE_IDX_SEQ_ID],
  1967. payload[READDONE_IDX_NUMFRAMES]);
  1968. if (ac->io_mode & SYNC_IO_MODE) {
  1969. if (port->buf == NULL) {
  1970. pr_err("%s: Unexpected Read Done\n", __func__);
  1971. spin_unlock_irqrestore(
  1972. &(session[session_id].session_lock),
  1973. flags);
  1974. return -EINVAL;
  1975. }
  1976. spin_lock_irqsave(&port->dsp_lock, dsp_flags);
  1977. buf_index = asm_token._token.buf_index;
  1978. port->buf[buf_index].used = 0;
  1979. if (lower_32_bits(port->buf[buf_index].phys) !=
  1980. payload[READDONE_IDX_BUFADD_LSW] ||
  1981. msm_audio_populate_upper_32_bits(
  1982. port->buf[buf_index].phys) !=
  1983. payload[READDONE_IDX_BUFADD_MSW]) {
  1984. dev_vdbg(ac->dev, "%s: Expected addr %pK\n",
  1985. __func__, &port->buf[buf_index].phys);
  1986. pr_err("%s: rxedl[0x%x] rxedu[0x%x]\n",
  1987. __func__,
  1988. payload[READDONE_IDX_BUFADD_LSW],
  1989. payload[READDONE_IDX_BUFADD_MSW]);
  1990. spin_unlock_irqrestore(&port->dsp_lock,
  1991. dsp_flags);
  1992. break;
  1993. }
  1994. port->buf[buf_index].actual_size =
  1995. payload[READDONE_IDX_SIZE];
  1996. spin_unlock_irqrestore(&port->dsp_lock, dsp_flags);
  1997. }
  1998. break;
  1999. }
  2000. case ASM_DATA_EVENT_EOS:
  2001. case ASM_DATA_EVENT_RENDERED_EOS:
  2002. pr_debug("%s: EOS ACK received: rxed session %d opcode 0x%x token 0x%x src %d dest %d\n",
  2003. __func__, ac->session,
  2004. data->opcode, data->token,
  2005. data->src_port, data->dest_port);
  2006. break;
  2007. case ASM_SESSION_EVENTX_OVERFLOW:
  2008. pr_debug("%s: ASM_SESSION_EVENTX_OVERFLOW session %d opcode 0x%x token 0x%x src %d dest %d\n",
  2009. __func__, ac->session,
  2010. data->opcode, data->token,
  2011. data->src_port, data->dest_port);
  2012. break;
  2013. case ASM_SESSION_EVENT_RX_UNDERFLOW:
  2014. pr_debug("%s: ASM_SESSION_EVENT_RX_UNDERFLOW session %d opcode 0x%x token 0x%x src %d dest %d\n",
  2015. __func__, ac->session,
  2016. data->opcode, data->token,
  2017. data->src_port, data->dest_port);
  2018. break;
  2019. case ASM_SESSION_CMDRSP_GET_SESSIONTIME_V3:
  2020. dev_vdbg(ac->dev, "%s: ASM_SESSION_CMDRSP_GET_SESSIONTIME_V3, payload[0] = %d, payload[1] = %d, payload[2] = %d\n",
  2021. __func__,
  2022. payload[0], payload[1], payload[2]);
  2023. ac->time_stamp = (uint64_t)(((uint64_t)payload[2] << 32) |
  2024. payload[1]);
  2025. if (atomic_cmpxchg(&ac->time_flag, 1, 0))
  2026. wake_up(&ac->time_wait);
  2027. break;
  2028. case ASM_DATA_EVENT_SR_CM_CHANGE_NOTIFY:
  2029. case ASM_DATA_EVENT_ENC_SR_CM_CHANGE_NOTIFY:
  2030. pr_debug("%s: ASM_DATA_EVENT_SR_CM_CHANGE_NOTIFY session %d opcode 0x%x token 0x%x src %d dest %d\n",
  2031. __func__, ac->session,
  2032. data->opcode, data->token,
  2033. data->src_port, data->dest_port);
  2034. pr_debug("%s: ASM_DATA_EVENT_SR_CM_CHANGE_NOTIFY, payload[0] = %d, payload[1] = %d, payload[2] = %d, payload[3] = %d\n",
  2035. __func__,
  2036. payload[0], payload[1], payload[2],
  2037. payload[3]);
  2038. break;
  2039. case ASM_SESSION_CMDRSP_GET_MTMX_STRTR_PARAMS_V2:
  2040. q6asm_process_mtmx_get_param_rsp(ac, (void *) payload);
  2041. break;
  2042. case ASM_STREAM_PP_EVENT:
  2043. case ASM_STREAM_CMD_ENCDEC_EVENTS:
  2044. case ASM_STREAM_CMD_REGISTER_IEC_61937_FMT_UPDATE:
  2045. pr_debug("%s: ASM_STREAM_EVENT payload[0][0x%x] payload[1][0x%x]",
  2046. __func__, payload[0], payload[1]);
  2047. i = is_adsp_raise_event(data->opcode);
  2048. if (i < 0) {
  2049. spin_unlock_irqrestore(
  2050. &(session[session_id].session_lock), flags);
  2051. return 0;
  2052. }
  2053. /* repack payload for asm_stream_pp_event
  2054. * package is composed of event type + size + actual payload
  2055. */
  2056. payload_size = data->payload_size;
  2057. if (payload_size > UINT_MAX - sizeof(struct msm_adsp_event_data)) {
  2058. pr_err("%s: payload size = %d exceeds limit.\n",
  2059. __func__, payload_size);
  2060. spin_unlock(&(session[session_id].session_lock));
  2061. return -EINVAL;
  2062. }
  2063. pp_event_package = kzalloc(payload_size
  2064. + sizeof(struct msm_adsp_event_data),
  2065. GFP_ATOMIC);
  2066. if (!pp_event_package) {
  2067. spin_unlock_irqrestore(
  2068. &(session[session_id].session_lock), flags);
  2069. return -ENOMEM;
  2070. }
  2071. pp_event_package->event_type = i;
  2072. pp_event_package->payload_len = payload_size;
  2073. memcpy((void *)pp_event_package->payload,
  2074. data->payload, payload_size);
  2075. ac->cb(data->opcode, data->token,
  2076. (void *)pp_event_package, ac->priv);
  2077. kfree(pp_event_package);
  2078. spin_unlock_irqrestore(
  2079. &(session[session_id].session_lock), flags);
  2080. return 0;
  2081. case ASM_SESSION_CMDRSP_ADJUST_SESSION_CLOCK_V2:
  2082. pr_debug("%s: ASM_SESSION_CMDRSP_ADJUST_SESSION_CLOCK_V2 sesion %d status 0x%x msw %u lsw %u\n",
  2083. __func__, ac->session, payload[0], payload[2],
  2084. payload[1]);
  2085. wake_up(&ac->cmd_wait);
  2086. break;
  2087. case ASM_SESSION_CMDRSP_GET_PATH_DELAY_V2:
  2088. pr_debug("%s: ASM_SESSION_CMDRSP_GET_PATH_DELAY_V2 session %d status 0x%x msw %u lsw %u\n",
  2089. __func__, ac->session, payload[0], payload[2],
  2090. payload[1]);
  2091. if (payload[0] == 0) {
  2092. atomic_set(&ac->cmd_state, 0);
  2093. /* ignore msw, as a delay that large shouldn't happen */
  2094. ac->path_delay = payload[1];
  2095. } else {
  2096. atomic_set(&ac->cmd_state, payload[0]);
  2097. ac->path_delay = UINT_MAX;
  2098. }
  2099. wake_up(&ac->cmd_wait);
  2100. break;
  2101. }
  2102. if (ac->cb)
  2103. ac->cb(data->opcode, data->token,
  2104. data->payload, ac->priv);
  2105. spin_unlock_irqrestore(
  2106. &(session[session_id].session_lock), flags);
  2107. return 0;
  2108. }
  2109. /**
  2110. * q6asm_is_cpu_buf_avail -
  2111. * retrieve next CPU buf avail
  2112. *
  2113. * @dir: RX or TX direction
  2114. * @ac: Audio client handle
  2115. * @size: size pointer to be updated with size of buffer
  2116. * @index: index pointer to be updated with
  2117. * CPU buffer index available
  2118. *
  2119. * Returns buffer pointer on success or NULL on failure
  2120. */
  2121. void *q6asm_is_cpu_buf_avail(int dir, struct audio_client *ac, uint32_t *size,
  2122. uint32_t *index)
  2123. {
  2124. void *data;
  2125. unsigned char idx;
  2126. struct audio_port_data *port;
  2127. if (!ac || ((dir != IN) && (dir != OUT))) {
  2128. pr_err("%s: ac %pK dir %d\n", __func__, ac, dir);
  2129. return NULL;
  2130. }
  2131. if (ac->io_mode & SYNC_IO_MODE) {
  2132. port = &ac->port[dir];
  2133. mutex_lock(&port->lock);
  2134. idx = port->cpu_buf;
  2135. if (port->buf == NULL) {
  2136. pr_err("%s: Buffer pointer null\n", __func__);
  2137. mutex_unlock(&port->lock);
  2138. return NULL;
  2139. }
  2140. /* dir 0: used = 0 means buf in use
  2141. * dir 1: used = 1 means buf in use
  2142. */
  2143. if (port->buf[idx].used == dir) {
  2144. /* To make it more robust, we could loop and get the
  2145. * next avail buf, its risky though
  2146. */
  2147. pr_err("%s: Next buf idx[0x%x] not available, dir[%d]\n",
  2148. __func__, idx, dir);
  2149. mutex_unlock(&port->lock);
  2150. return NULL;
  2151. }
  2152. *size = port->buf[idx].actual_size;
  2153. *index = port->cpu_buf;
  2154. data = port->buf[idx].data;
  2155. dev_vdbg(ac->dev, "%s: session[%d]index[%d] data[%pK]size[%d]\n",
  2156. __func__,
  2157. ac->session,
  2158. port->cpu_buf,
  2159. data, *size);
  2160. /* By default increase the cpu_buf cnt
  2161. * user accesses this function,increase cpu
  2162. * buf(to avoid another api)
  2163. */
  2164. port->buf[idx].used = dir;
  2165. port->cpu_buf = q6asm_get_next_buf(ac, port->cpu_buf,
  2166. port->max_buf_cnt);
  2167. mutex_unlock(&port->lock);
  2168. return data;
  2169. }
  2170. return NULL;
  2171. }
  2172. EXPORT_SYMBOL(q6asm_is_cpu_buf_avail);
  2173. /**
  2174. * q6asm_cpu_buf_release -
  2175. * releases cpu buffer for ASM
  2176. *
  2177. * @dir: RX or TX direction
  2178. * @ac: Audio client handle
  2179. *
  2180. * Returns 0 on success or error on failure
  2181. */
  2182. int q6asm_cpu_buf_release(int dir, struct audio_client *ac)
  2183. {
  2184. struct audio_port_data *port;
  2185. int ret = 0;
  2186. int idx;
  2187. if (!ac || ((dir != IN) && (dir != OUT))) {
  2188. pr_err("%s: ac %pK dir %d\n", __func__, ac, dir);
  2189. ret = -EINVAL;
  2190. goto exit;
  2191. }
  2192. if (ac->io_mode & SYNC_IO_MODE) {
  2193. port = &ac->port[dir];
  2194. mutex_lock(&port->lock);
  2195. idx = port->cpu_buf;
  2196. if (port->cpu_buf == 0) {
  2197. port->cpu_buf = port->max_buf_cnt - 1;
  2198. } else if (port->cpu_buf < port->max_buf_cnt) {
  2199. port->cpu_buf = port->cpu_buf - 1;
  2200. } else {
  2201. pr_err("%s: buffer index(%d) out of range\n",
  2202. __func__, port->cpu_buf);
  2203. ret = -EINVAL;
  2204. mutex_unlock(&port->lock);
  2205. goto exit;
  2206. }
  2207. port->buf[port->cpu_buf].used = dir ^ 1;
  2208. mutex_unlock(&port->lock);
  2209. }
  2210. exit:
  2211. return ret;
  2212. }
  2213. EXPORT_SYMBOL(q6asm_cpu_buf_release);
  2214. /**
  2215. * q6asm_is_cpu_buf_avail_nolock -
  2216. * retrieve next CPU buf avail without lock acquire
  2217. *
  2218. * @dir: RX or TX direction
  2219. * @ac: Audio client handle
  2220. * @size: size pointer to be updated with size of buffer
  2221. * @index: index pointer to be updated with
  2222. * CPU buffer index available
  2223. *
  2224. * Returns buffer pointer on success or NULL on failure
  2225. */
  2226. void *q6asm_is_cpu_buf_avail_nolock(int dir, struct audio_client *ac,
  2227. uint32_t *size, uint32_t *index)
  2228. {
  2229. void *data;
  2230. unsigned char idx;
  2231. struct audio_port_data *port;
  2232. if (!ac || ((dir != IN) && (dir != OUT))) {
  2233. pr_err("%s: ac %pK dir %d\n", __func__, ac, dir);
  2234. return NULL;
  2235. }
  2236. port = &ac->port[dir];
  2237. idx = port->cpu_buf;
  2238. if (port->buf == NULL) {
  2239. pr_err("%s: Buffer pointer null\n", __func__);
  2240. return NULL;
  2241. }
  2242. /*
  2243. * dir 0: used = 0 means buf in use
  2244. * dir 1: used = 1 means buf in use
  2245. */
  2246. if (port->buf[idx].used == dir) {
  2247. /*
  2248. * To make it more robust, we could loop and get the
  2249. * next avail buf, its risky though
  2250. */
  2251. pr_err("%s: Next buf idx[0x%x] not available, dir[%d]\n",
  2252. __func__, idx, dir);
  2253. return NULL;
  2254. }
  2255. *size = port->buf[idx].actual_size;
  2256. *index = port->cpu_buf;
  2257. data = port->buf[idx].data;
  2258. dev_vdbg(ac->dev, "%s: session[%d]index[%d] data[%pK]size[%d]\n",
  2259. __func__, ac->session, port->cpu_buf,
  2260. data, *size);
  2261. /*
  2262. * By default increase the cpu_buf cnt
  2263. * user accesses this function,increase cpu
  2264. * buf(to avoid another api)
  2265. */
  2266. port->buf[idx].used = dir;
  2267. port->cpu_buf = q6asm_get_next_buf(ac, port->cpu_buf,
  2268. port->max_buf_cnt);
  2269. return data;
  2270. }
  2271. EXPORT_SYMBOL(q6asm_is_cpu_buf_avail_nolock);
  2272. int q6asm_is_dsp_buf_avail(int dir, struct audio_client *ac)
  2273. {
  2274. int ret = -1;
  2275. struct audio_port_data *port;
  2276. uint32_t idx;
  2277. if (!ac || (dir != OUT)) {
  2278. pr_err("%s: ac %pK dir %d\n", __func__, ac, dir);
  2279. return ret;
  2280. }
  2281. if (ac->io_mode & SYNC_IO_MODE) {
  2282. port = &ac->port[dir];
  2283. mutex_lock(&port->lock);
  2284. idx = port->dsp_buf;
  2285. if (port->buf[idx].used == (dir ^ 1)) {
  2286. /* To make it more robust, we could loop and get the
  2287. * next avail buf, its risky though
  2288. */
  2289. pr_err("%s: Next buf idx[0x%x] not available, dir[%d]\n",
  2290. __func__, idx, dir);
  2291. mutex_unlock(&port->lock);
  2292. return ret;
  2293. }
  2294. dev_vdbg(ac->dev, "%s: session[%d]dsp_buf=%d cpu_buf=%d\n",
  2295. __func__,
  2296. ac->session, port->dsp_buf, port->cpu_buf);
  2297. ret = ((port->dsp_buf != port->cpu_buf) ? 0 : -1);
  2298. mutex_unlock(&port->lock);
  2299. }
  2300. return ret;
  2301. }
  2302. static void __q6asm_add_hdr(struct audio_client *ac, struct apr_hdr *hdr,
  2303. uint32_t pkt_size, uint32_t cmd_flg, uint32_t stream_id)
  2304. {
  2305. unsigned long flags = 0;
  2306. dev_vdbg(ac->dev, "%s: pkt_size=%d cmd_flg=%d session=%d stream_id=%d\n",
  2307. __func__, pkt_size, cmd_flg, ac->session, stream_id);
  2308. mutex_lock(&ac->cmd_lock);
  2309. spin_lock_irqsave(&(session[ac->session].session_lock), flags);
  2310. if (ac->apr == NULL) {
  2311. pr_err("%s: AC APR handle NULL", __func__);
  2312. spin_unlock_irqrestore(
  2313. &(session[ac->session].session_lock), flags);
  2314. mutex_unlock(&ac->cmd_lock);
  2315. return;
  2316. }
  2317. hdr->hdr_field = APR_HDR_FIELD(APR_MSG_TYPE_SEQ_CMD,
  2318. APR_HDR_LEN(sizeof(struct apr_hdr)),
  2319. APR_PKT_VER);
  2320. hdr->src_svc = ((struct apr_svc *)ac->apr)->id;
  2321. hdr->src_domain = APR_DOMAIN_APPS;
  2322. hdr->dest_svc = APR_SVC_ASM;
  2323. hdr->dest_domain = APR_DOMAIN_ADSP;
  2324. hdr->src_port = ((ac->session << 8) & 0xFF00) | (stream_id);
  2325. hdr->dest_port = ((ac->session << 8) & 0xFF00) | (stream_id);
  2326. if (cmd_flg)
  2327. q6asm_update_token(&hdr->token,
  2328. ac->session,
  2329. 0, /* Stream ID is NA */
  2330. 0, /* Buffer index is NA */
  2331. 0, /* Direction flag is NA */
  2332. WAIT_CMD);
  2333. hdr->pkt_size = pkt_size;
  2334. spin_unlock_irqrestore(
  2335. &(session[ac->session].session_lock), flags);
  2336. mutex_unlock(&ac->cmd_lock);
  2337. }
  2338. static void q6asm_add_hdr(struct audio_client *ac, struct apr_hdr *hdr,
  2339. uint32_t pkt_size, uint32_t cmd_flg)
  2340. {
  2341. __q6asm_add_hdr(ac, hdr, pkt_size, cmd_flg, ac->stream_id);
  2342. }
  2343. static void q6asm_stream_add_hdr(struct audio_client *ac, struct apr_hdr *hdr,
  2344. uint32_t pkt_size, uint32_t cmd_flg, int32_t stream_id)
  2345. {
  2346. __q6asm_add_hdr(ac, hdr, pkt_size, cmd_flg, stream_id);
  2347. }
  2348. static void __q6asm_add_hdr_async(struct audio_client *ac, struct apr_hdr *hdr,
  2349. uint32_t pkt_size, uint32_t cmd_flg,
  2350. uint32_t stream_id, u8 no_wait_flag)
  2351. {
  2352. dev_vdbg(ac->dev, "%s: pkt_size = %d, cmd_flg = %d, session = %d stream_id=%d\n",
  2353. __func__, pkt_size, cmd_flg, ac->session, stream_id);
  2354. hdr->hdr_field = APR_HDR_FIELD(APR_MSG_TYPE_SEQ_CMD,
  2355. APR_HDR_LEN(sizeof(struct apr_hdr)),
  2356. APR_PKT_VER);
  2357. if (ac->apr == NULL) {
  2358. pr_err("%s: AC APR is NULL", __func__);
  2359. return;
  2360. }
  2361. hdr->src_svc = ((struct apr_svc *)ac->apr)->id;
  2362. hdr->src_domain = APR_DOMAIN_APPS;
  2363. hdr->dest_svc = APR_SVC_ASM;
  2364. hdr->dest_domain = APR_DOMAIN_ADSP;
  2365. hdr->src_port = ((ac->session << 8) & 0xFF00) | (stream_id);
  2366. hdr->dest_port = ((ac->session << 8) & 0xFF00) | (stream_id);
  2367. if (cmd_flg) {
  2368. q6asm_update_token(&hdr->token,
  2369. ac->session,
  2370. 0, /* Stream ID is NA */
  2371. 0, /* Buffer index is NA */
  2372. 0, /* Direction flag is NA */
  2373. no_wait_flag);
  2374. }
  2375. hdr->pkt_size = pkt_size;
  2376. }
  2377. static void q6asm_add_hdr_async(struct audio_client *ac, struct apr_hdr *hdr,
  2378. uint32_t pkt_size, uint32_t cmd_flg)
  2379. {
  2380. __q6asm_add_hdr_async(ac, hdr, pkt_size, cmd_flg,
  2381. ac->stream_id, WAIT_CMD);
  2382. }
  2383. static void q6asm_stream_add_hdr_async(struct audio_client *ac,
  2384. struct apr_hdr *hdr, uint32_t pkt_size,
  2385. uint32_t cmd_flg, int32_t stream_id)
  2386. {
  2387. __q6asm_add_hdr_async(ac, hdr, pkt_size, cmd_flg,
  2388. stream_id, NO_WAIT_CMD);
  2389. }
  2390. static void q6asm_add_hdr_custom_topology(struct audio_client *ac,
  2391. struct apr_hdr *hdr,
  2392. uint32_t pkt_size)
  2393. {
  2394. pr_debug("%s: pkt_size=%d session=%d\n",
  2395. __func__, pkt_size, ac->session);
  2396. if (ac->apr == NULL) {
  2397. pr_err("%s: AC APR handle NULL\n", __func__);
  2398. return;
  2399. }
  2400. mutex_lock(&ac->cmd_lock);
  2401. hdr->hdr_field = APR_HDR_FIELD(APR_MSG_TYPE_SEQ_CMD,
  2402. APR_HDR_LEN(sizeof(struct apr_hdr)),
  2403. APR_PKT_VER);
  2404. hdr->src_svc = ((struct apr_svc *)ac->apr)->id;
  2405. hdr->src_domain = APR_DOMAIN_APPS;
  2406. hdr->dest_svc = APR_SVC_ASM;
  2407. hdr->dest_domain = APR_DOMAIN_ADSP;
  2408. hdr->src_port = ((ac->session << 8) & 0xFF00) | 0x01;
  2409. hdr->dest_port = 0;
  2410. q6asm_update_token(&hdr->token,
  2411. ac->session,
  2412. 0, /* Stream ID is NA */
  2413. 0, /* Buffer index is NA */
  2414. 0, /* Direction flag is NA */
  2415. WAIT_CMD);
  2416. hdr->pkt_size = pkt_size;
  2417. mutex_unlock(&ac->cmd_lock);
  2418. }
  2419. static void q6asm_add_mmaphdr(struct audio_client *ac, struct apr_hdr *hdr,
  2420. u32 pkt_size, int dir)
  2421. {
  2422. pr_debug("%s: pkt size=%d\n",
  2423. __func__, pkt_size);
  2424. hdr->hdr_field = APR_HDR_FIELD(APR_MSG_TYPE_SEQ_CMD,
  2425. APR_HDR_LEN(APR_HDR_SIZE), APR_PKT_VER);
  2426. hdr->src_port = 0;
  2427. hdr->dest_port = 0;
  2428. q6asm_update_token(&hdr->token,
  2429. ac->session,
  2430. 0, /* Stream ID is NA */
  2431. 0, /* Buffer index is NA */
  2432. dir,
  2433. WAIT_CMD);
  2434. hdr->pkt_size = pkt_size;
  2435. }
  2436. /**
  2437. * q6asm_set_pp_params
  2438. * command to set ASM parameter data
  2439. * send memory mapping header for out of band case
  2440. * send pre-packed parameter data for in band case
  2441. *
  2442. * @ac: audio client handle
  2443. * @mem_hdr: memory mapping header
  2444. * @param_data: pre-packed parameter payload
  2445. * @param_size: size of pre-packed parameter data
  2446. *
  2447. * Returns 0 on success or error on failure
  2448. */
  2449. int q6asm_set_pp_params(struct audio_client *ac,
  2450. struct mem_mapping_hdr *mem_hdr, u8 *param_data,
  2451. u32 param_size)
  2452. {
  2453. struct asm_stream_cmd_set_pp_params *asm_set_param = NULL;
  2454. int pkt_size = 0;
  2455. int ret = 0;
  2456. int session_id = 0;
  2457. if (ac == NULL) {
  2458. pr_err("%s: Audio Client is NULL\n", __func__);
  2459. return -EINVAL;
  2460. } else if (ac->apr == NULL) {
  2461. pr_err("%s: APR pointer is NULL\n", __func__);
  2462. return -EINVAL;
  2463. }
  2464. session_id = q6asm_get_session_id_from_audio_client(ac);
  2465. if (!session_id)
  2466. return -EINVAL;
  2467. pkt_size = sizeof(struct asm_stream_cmd_set_pp_params);
  2468. /* Add param size to packet size when sending in-band only */
  2469. if (param_data != NULL)
  2470. pkt_size += param_size;
  2471. asm_set_param = kzalloc(pkt_size, GFP_KERNEL);
  2472. if (!asm_set_param)
  2473. return -ENOMEM;
  2474. mutex_lock(&session[session_id].mutex_lock_per_session);
  2475. if (!q6asm_is_valid_audio_client(ac)) {
  2476. ret = -EINVAL;
  2477. goto done;
  2478. }
  2479. if (ac->apr == NULL) {
  2480. pr_err("%s: AC APR handle NULL\n", __func__);
  2481. ret = -EINVAL;
  2482. goto done;
  2483. }
  2484. q6asm_add_hdr_async(ac, &asm_set_param->apr_hdr, pkt_size, TRUE);
  2485. /* With pre-packed data, only the opcode differs from V2 and V3. */
  2486. if (q6common_is_instance_id_supported())
  2487. asm_set_param->apr_hdr.opcode = ASM_STREAM_CMD_SET_PP_PARAMS_V3;
  2488. else
  2489. asm_set_param->apr_hdr.opcode = ASM_STREAM_CMD_SET_PP_PARAMS_V2;
  2490. asm_set_param->payload_size = param_size;
  2491. if (mem_hdr != NULL) {
  2492. /* Out of band case */
  2493. asm_set_param->mem_hdr = *mem_hdr;
  2494. } else if (param_data != NULL) {
  2495. /*
  2496. * In band case. Parameter data must be pre-packed with its
  2497. * header before calling this function. Use
  2498. * q6common_pack_pp_params to pack parameter data and header
  2499. * correctly.
  2500. */
  2501. memcpy(&asm_set_param->param_data, param_data, param_size);
  2502. } else {
  2503. pr_err("%s: Received NULL pointers for both mem header and param data\n",
  2504. __func__);
  2505. ret = -EINVAL;
  2506. goto done;
  2507. }
  2508. atomic_set(&ac->cmd_state_pp, -1);
  2509. ret = apr_send_pkt(ac->apr, (uint32_t *)asm_set_param);
  2510. if (ret < 0) {
  2511. pr_err("%s: apr send failed rc %d\n", __func__, ret);
  2512. ret = -EINVAL;
  2513. goto done;
  2514. }
  2515. ret = wait_event_timeout(ac->cmd_wait,
  2516. atomic_read(&ac->cmd_state_pp) >= 0,
  2517. msecs_to_jiffies(TIMEOUT_MS));
  2518. if (!ret) {
  2519. pr_err("%s: timeout sending apr pkt\n", __func__);
  2520. ret = -ETIMEDOUT;
  2521. goto done;
  2522. }
  2523. if (atomic_read(&ac->cmd_state_pp) > 0) {
  2524. pr_err("%s: DSP returned error[%s]\n", __func__,
  2525. adsp_err_get_err_str(atomic_read(&ac->cmd_state_pp)));
  2526. ret = adsp_err_get_lnx_err_code(atomic_read(&ac->cmd_state_pp));
  2527. goto done;
  2528. }
  2529. ret = 0;
  2530. done:
  2531. mutex_unlock(&session[session_id].mutex_lock_per_session);
  2532. kfree(asm_set_param);
  2533. return ret;
  2534. }
  2535. EXPORT_SYMBOL(q6asm_set_pp_params);
  2536. /**
  2537. * q6asm_pack_and_set_pp_param_in_band
  2538. * command to pack and set parameter data for in band case
  2539. *
  2540. * @ac: audio client handle
  2541. * @param_hdr: parameter header
  2542. * @param_data: parameter data
  2543. *
  2544. * Returns 0 on success or error on failure
  2545. */
  2546. int q6asm_pack_and_set_pp_param_in_band(struct audio_client *ac,
  2547. struct param_hdr_v3 param_hdr,
  2548. u8 *param_data)
  2549. {
  2550. u8 *packed_data = NULL;
  2551. u32 packed_size = sizeof(union param_hdrs) + param_hdr.param_size;
  2552. int ret = 0;
  2553. if (ac == NULL) {
  2554. pr_err("%s: Audio Client is NULL\n", __func__);
  2555. return -EINVAL;
  2556. }
  2557. packed_data = kzalloc(packed_size, GFP_KERNEL);
  2558. if (packed_data == NULL)
  2559. return -ENOMEM;
  2560. ret = q6common_pack_pp_params(packed_data, &param_hdr, param_data,
  2561. &packed_size);
  2562. if (ret) {
  2563. pr_err("%s: Failed to pack params, error %d\n", __func__, ret);
  2564. goto done;
  2565. }
  2566. ret = q6asm_set_pp_params(ac, NULL, packed_data, packed_size);
  2567. done:
  2568. kfree(packed_data);
  2569. return ret;
  2570. }
  2571. EXPORT_SYMBOL(q6asm_pack_and_set_pp_param_in_band);
  2572. /**
  2573. * q6asm_set_soft_volume_module_instance_ids
  2574. * command to set module and instance ids for soft volume
  2575. *
  2576. * @instance: soft volume instance
  2577. * @param_hdr: parameter header
  2578. *
  2579. * Returns 0 on success or error on failure
  2580. */
  2581. int q6asm_set_soft_volume_module_instance_ids(int instance,
  2582. struct param_hdr_v3 *param_hdr)
  2583. {
  2584. if (param_hdr == NULL) {
  2585. pr_err("%s: Param header is NULL\n", __func__);
  2586. return -EINVAL;
  2587. }
  2588. switch (instance) {
  2589. case SOFT_VOLUME_INSTANCE_2:
  2590. param_hdr->module_id = ASM_MODULE_ID_VOL_CTRL2;
  2591. param_hdr->instance_id = INSTANCE_ID_0;
  2592. return 0;
  2593. case SOFT_VOLUME_INSTANCE_1:
  2594. param_hdr->module_id = ASM_MODULE_ID_VOL_CTRL;
  2595. param_hdr->instance_id = INSTANCE_ID_0;
  2596. return 0;
  2597. default:
  2598. pr_err("%s: Invalid instance %d\n", __func__, instance);
  2599. return -EINVAL;
  2600. }
  2601. }
  2602. EXPORT_SYMBOL(q6asm_set_soft_volume_module_instance_ids);
  2603. /**
  2604. * q6asm_open_read_compressed -
  2605. * command to open ASM in compressed read mode
  2606. *
  2607. * @ac: Audio client handle
  2608. * @format: capture format for ASM
  2609. * @passthrough_flag: flag to indicate passthrough option
  2610. *
  2611. * Returns 0 on success or error on failure
  2612. */
  2613. int q6asm_open_read_compressed(struct audio_client *ac, uint32_t format,
  2614. uint32_t passthrough_flag)
  2615. {
  2616. int rc = 0;
  2617. struct asm_stream_cmd_open_read_compressed open;
  2618. if (ac == NULL) {
  2619. pr_err("%s: ac[%pK] NULL\n", __func__, ac);
  2620. rc = -EINVAL;
  2621. goto fail_cmd;
  2622. }
  2623. if (ac->apr == NULL) {
  2624. pr_err("%s: APR handle[%pK] NULL\n", __func__, ac->apr);
  2625. rc = -EINVAL;
  2626. goto fail_cmd;
  2627. }
  2628. pr_debug("%s: session[%d] wr_format[0x%x]\n", __func__, ac->session,
  2629. format);
  2630. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  2631. open.hdr.opcode = ASM_STREAM_CMD_OPEN_READ_COMPRESSED;
  2632. atomic_set(&ac->cmd_state, -1);
  2633. /*
  2634. * Below flag indicates whether DSP shall keep IEC61937 packing or
  2635. * unpack to raw compressed format
  2636. */
  2637. if (format == FORMAT_IEC61937) {
  2638. open.mode_flags = 0x1;
  2639. pr_debug("%s: Flag 1 IEC61937 output\n", __func__);
  2640. } else {
  2641. open.mode_flags = 0;
  2642. open.frames_per_buf = 1;
  2643. pr_debug("%s: Flag 0 RAW_COMPR output\n", __func__);
  2644. }
  2645. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  2646. if (rc < 0) {
  2647. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  2648. __func__, open.hdr.opcode, rc);
  2649. rc = -EINVAL;
  2650. goto fail_cmd;
  2651. }
  2652. rc = wait_event_timeout(ac->cmd_wait,
  2653. (atomic_read(&ac->cmd_state) >= 0),
  2654. msecs_to_jiffies(TIMEOUT_MS));
  2655. if (!rc) {
  2656. pr_err("%s: timeout. waited for OPEN_READ_COMPR rc[%d]\n",
  2657. __func__, rc);
  2658. rc = -ETIMEDOUT;
  2659. goto fail_cmd;
  2660. }
  2661. if (atomic_read(&ac->cmd_state) > 0) {
  2662. pr_err("%s: DSP returned error[%s]\n",
  2663. __func__, adsp_err_get_err_str(
  2664. atomic_read(&ac->cmd_state)));
  2665. rc = adsp_err_get_lnx_err_code(
  2666. atomic_read(&ac->cmd_state));
  2667. goto fail_cmd;
  2668. }
  2669. return 0;
  2670. fail_cmd:
  2671. return rc;
  2672. }
  2673. EXPORT_SYMBOL(q6asm_open_read_compressed);
  2674. static int __q6asm_open_read(struct audio_client *ac,
  2675. uint32_t format, uint16_t bits_per_sample,
  2676. uint32_t pcm_format_block_ver,
  2677. bool ts_mode, uint32_t enc_cfg_id)
  2678. {
  2679. int rc = 0x00;
  2680. struct asm_stream_cmd_open_read_v3 open;
  2681. struct q6asm_cal_info cal_info;
  2682. config_debug_fs_reset_index();
  2683. if (ac == NULL) {
  2684. pr_err("%s: APR handle NULL\n", __func__);
  2685. return -EINVAL;
  2686. }
  2687. if (ac->apr == NULL) {
  2688. pr_err("%s: AC APR handle NULL\n", __func__);
  2689. return -EINVAL;
  2690. }
  2691. pr_debug("%s: session[%d]\n", __func__, ac->session);
  2692. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  2693. atomic_set(&ac->cmd_state, -1);
  2694. open.hdr.opcode = ASM_STREAM_CMD_OPEN_READ_V3;
  2695. /* Stream prio : High, provide meta info with encoded frames */
  2696. open.src_endpointype = ASM_END_POINT_DEVICE_MATRIX;
  2697. rc = q6asm_get_asm_topology_apptype(&cal_info);
  2698. open.preprocopo_id = cal_info.topology_id;
  2699. open.bits_per_sample = bits_per_sample;
  2700. open.mode_flags = 0x0;
  2701. ac->topology = open.preprocopo_id;
  2702. ac->app_type = cal_info.app_type;
  2703. if (ac->perf_mode == LOW_LATENCY_PCM_MODE) {
  2704. open.mode_flags |= ASM_LOW_LATENCY_TX_STREAM_SESSION <<
  2705. ASM_SHIFT_STREAM_PERF_MODE_FLAG_IN_OPEN_READ;
  2706. } else {
  2707. open.mode_flags |= ASM_LEGACY_STREAM_SESSION <<
  2708. ASM_SHIFT_STREAM_PERF_MODE_FLAG_IN_OPEN_READ;
  2709. }
  2710. switch (format) {
  2711. case FORMAT_LINEAR_PCM:
  2712. open.mode_flags |= 0x00;
  2713. open.enc_cfg_id = q6asm_get_pcm_format_id(pcm_format_block_ver);
  2714. if (ts_mode)
  2715. open.mode_flags |= ABSOLUTE_TIMESTAMP_ENABLE;
  2716. break;
  2717. case FORMAT_MPEG4_AAC:
  2718. open.mode_flags |= BUFFER_META_ENABLE;
  2719. open.enc_cfg_id = ASM_MEDIA_FMT_AAC_V2;
  2720. break;
  2721. case FORMAT_G711_ALAW_FS:
  2722. open.mode_flags |= BUFFER_META_ENABLE;
  2723. open.enc_cfg_id = ASM_MEDIA_FMT_G711_ALAW_FS;
  2724. break;
  2725. case FORMAT_G711_MLAW_FS:
  2726. open.mode_flags |= BUFFER_META_ENABLE;
  2727. open.enc_cfg_id = ASM_MEDIA_FMT_G711_MLAW_FS;
  2728. break;
  2729. case FORMAT_V13K:
  2730. open.mode_flags |= BUFFER_META_ENABLE;
  2731. open.enc_cfg_id = ASM_MEDIA_FMT_V13K_FS;
  2732. break;
  2733. case FORMAT_EVRC:
  2734. open.mode_flags |= BUFFER_META_ENABLE;
  2735. open.enc_cfg_id = ASM_MEDIA_FMT_EVRC_FS;
  2736. break;
  2737. case FORMAT_AMRNB:
  2738. open.mode_flags |= BUFFER_META_ENABLE;
  2739. open.enc_cfg_id = ASM_MEDIA_FMT_AMRNB_FS;
  2740. break;
  2741. case FORMAT_AMRWB:
  2742. open.mode_flags |= BUFFER_META_ENABLE;
  2743. open.enc_cfg_id = ASM_MEDIA_FMT_AMRWB_FS;
  2744. break;
  2745. case FORMAT_BESPOKE:
  2746. open.mode_flags |= BUFFER_META_ENABLE;
  2747. open.enc_cfg_id = enc_cfg_id;
  2748. if (ts_mode)
  2749. open.mode_flags |= ABSOLUTE_TIMESTAMP_ENABLE;
  2750. break;
  2751. default:
  2752. pr_err("%s: Invalid format 0x%x\n",
  2753. __func__, format);
  2754. rc = -EINVAL;
  2755. goto fail_cmd;
  2756. }
  2757. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  2758. if (rc < 0) {
  2759. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  2760. __func__, open.hdr.opcode, rc);
  2761. rc = -EINVAL;
  2762. goto fail_cmd;
  2763. }
  2764. rc = wait_event_timeout(ac->cmd_wait,
  2765. (atomic_read(&ac->cmd_state) >= 0),
  2766. msecs_to_jiffies(TIMEOUT_MS));
  2767. if (!rc) {
  2768. pr_err("%s: timeout. waited for open read\n",
  2769. __func__);
  2770. rc = -ETIMEDOUT;
  2771. goto fail_cmd;
  2772. }
  2773. if (atomic_read(&ac->cmd_state) > 0) {
  2774. pr_err("%s: DSP returned error[%s]\n",
  2775. __func__, adsp_err_get_err_str(
  2776. atomic_read(&ac->cmd_state)));
  2777. rc = adsp_err_get_lnx_err_code(
  2778. atomic_read(&ac->cmd_state));
  2779. goto fail_cmd;
  2780. }
  2781. ac->io_mode |= TUN_READ_IO_MODE;
  2782. return 0;
  2783. fail_cmd:
  2784. return rc;
  2785. }
  2786. /**
  2787. * q6asm_open_read -
  2788. * command to open ASM in read mode
  2789. *
  2790. * @ac: Audio client handle
  2791. * @format: capture format for ASM
  2792. *
  2793. * Returns 0 on success or error on failure
  2794. */
  2795. int q6asm_open_read(struct audio_client *ac,
  2796. uint32_t format)
  2797. {
  2798. return __q6asm_open_read(ac, format, 16,
  2799. PCM_MEDIA_FORMAT_V2 /*media fmt block ver*/,
  2800. false/*ts_mode*/, ENC_CFG_ID_NONE);
  2801. }
  2802. EXPORT_SYMBOL(q6asm_open_read);
  2803. int q6asm_open_read_v2(struct audio_client *ac, uint32_t format,
  2804. uint16_t bits_per_sample)
  2805. {
  2806. return __q6asm_open_read(ac, format, bits_per_sample,
  2807. PCM_MEDIA_FORMAT_V2 /*media fmt block ver*/,
  2808. false/*ts_mode*/, ENC_CFG_ID_NONE);
  2809. }
  2810. /*
  2811. * asm_open_read_v3 - Opens audio capture session
  2812. *
  2813. * @ac: Client session handle
  2814. * @format: encoder format
  2815. * @bits_per_sample: bit width of capture session
  2816. */
  2817. int q6asm_open_read_v3(struct audio_client *ac, uint32_t format,
  2818. uint16_t bits_per_sample)
  2819. {
  2820. return __q6asm_open_read(ac, format, bits_per_sample,
  2821. PCM_MEDIA_FORMAT_V3/*media fmt block ver*/,
  2822. false/*ts_mode*/, ENC_CFG_ID_NONE);
  2823. }
  2824. EXPORT_SYMBOL(q6asm_open_read_v3);
  2825. /*
  2826. * asm_open_read_v4 - Opens audio capture session
  2827. *
  2828. * @ac: Client session handle
  2829. * @format: encoder format
  2830. * @bits_per_sample: bit width of capture session
  2831. * @ts_mode: timestamp mode
  2832. */
  2833. int q6asm_open_read_v4(struct audio_client *ac, uint32_t format,
  2834. uint16_t bits_per_sample, bool ts_mode,
  2835. uint32_t enc_cfg_id)
  2836. {
  2837. return __q6asm_open_read(ac, format, bits_per_sample,
  2838. PCM_MEDIA_FORMAT_V4 /*media fmt block ver*/,
  2839. ts_mode, enc_cfg_id);
  2840. }
  2841. EXPORT_SYMBOL(q6asm_open_read_v4);
  2842. /*
  2843. * asm_open_read_v5 - Opens audio capture session
  2844. *
  2845. * @ac: Client session handle
  2846. * @format: encoder format
  2847. * @bits_per_sample: bit width of capture session
  2848. * @ts_mode: timestamp mode
  2849. */
  2850. int q6asm_open_read_v5(struct audio_client *ac, uint32_t format,
  2851. uint16_t bits_per_sample, bool ts_mode, uint32_t enc_cfg_id)
  2852. {
  2853. return __q6asm_open_read(ac, format, bits_per_sample,
  2854. PCM_MEDIA_FORMAT_V5 /*media fmt block ver*/,
  2855. ts_mode, enc_cfg_id);
  2856. }
  2857. EXPORT_SYMBOL(q6asm_open_read_v5);
  2858. /**
  2859. * q6asm_open_write_compressed -
  2860. * command to open ASM in compressed write mode
  2861. *
  2862. * @ac: Audio client handle
  2863. * @format: playback format for ASM
  2864. * @passthrough_flag: flag to indicate passthrough option
  2865. *
  2866. * Returns 0 on success or error on failure
  2867. */
  2868. int q6asm_open_write_compressed(struct audio_client *ac, uint32_t format,
  2869. uint32_t passthrough_flag)
  2870. {
  2871. int rc = 0;
  2872. struct asm_stream_cmd_open_write_compressed open;
  2873. if (ac == NULL) {
  2874. pr_err("%s: ac[%pK] NULL\n", __func__, ac);
  2875. rc = -EINVAL;
  2876. goto fail_cmd;
  2877. }
  2878. if (ac->apr == NULL) {
  2879. pr_err("%s: APR handle[%pK] NULL\n", __func__, ac->apr);
  2880. rc = -EINVAL;
  2881. goto fail_cmd;
  2882. }
  2883. pr_debug("%s: session[%d] wr_format[0x%x]", __func__, ac->session,
  2884. format);
  2885. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  2886. open.hdr.opcode = ASM_STREAM_CMD_OPEN_WRITE_COMPRESSED;
  2887. atomic_set(&ac->cmd_state, -1);
  2888. switch (format) {
  2889. case FORMAT_AC3:
  2890. open.fmt_id = ASM_MEDIA_FMT_AC3;
  2891. break;
  2892. case FORMAT_EAC3:
  2893. open.fmt_id = ASM_MEDIA_FMT_EAC3;
  2894. break;
  2895. case FORMAT_DTS:
  2896. open.fmt_id = ASM_MEDIA_FMT_DTS;
  2897. break;
  2898. case FORMAT_DSD:
  2899. open.fmt_id = ASM_MEDIA_FMT_DSD;
  2900. break;
  2901. case FORMAT_GEN_COMPR:
  2902. open.fmt_id = ASM_MEDIA_FMT_GENERIC_COMPRESSED;
  2903. break;
  2904. case FORMAT_TRUEHD:
  2905. open.fmt_id = ASM_MEDIA_FMT_TRUEHD;
  2906. break;
  2907. case FORMAT_IEC61937:
  2908. open.fmt_id = ASM_MEDIA_FMT_IEC;
  2909. break;
  2910. default:
  2911. pr_err("%s: Invalid format[%d]\n", __func__, format);
  2912. rc = -EINVAL;
  2913. goto fail_cmd;
  2914. }
  2915. /* Below flag indicates the DSP that Compressed audio input
  2916. * stream is not IEC 61937 or IEC 60958 packetizied
  2917. */
  2918. if (passthrough_flag == COMPRESSED_PASSTHROUGH ||
  2919. passthrough_flag == COMPRESSED_PASSTHROUGH_DSD ||
  2920. passthrough_flag == COMPRESSED_PASSTHROUGH_GEN) {
  2921. open.flags = 0x0;
  2922. pr_debug("%s: Flag 0 COMPRESSED_PASSTHROUGH\n", __func__);
  2923. } else if (passthrough_flag == COMPRESSED_PASSTHROUGH_CONVERT) {
  2924. open.flags = 0x8;
  2925. pr_debug("%s: Flag 8 - COMPRESSED_PASSTHROUGH_CONVERT\n",
  2926. __func__);
  2927. } else if (passthrough_flag == COMPRESSED_PASSTHROUGH_IEC61937) {
  2928. open.flags = 0x1;
  2929. pr_debug("%s: Flag 1 - COMPRESSED_PASSTHROUGH_IEC61937\n",
  2930. __func__);
  2931. } else {
  2932. pr_err("%s: Invalid passthrough type[%d]\n",
  2933. __func__, passthrough_flag);
  2934. rc = -EINVAL;
  2935. goto fail_cmd;
  2936. }
  2937. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  2938. if (rc < 0) {
  2939. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  2940. __func__, open.hdr.opcode, rc);
  2941. rc = -EINVAL;
  2942. goto fail_cmd;
  2943. }
  2944. rc = wait_event_timeout(ac->cmd_wait,
  2945. (atomic_read(&ac->cmd_state) >= 0),
  2946. msecs_to_jiffies(TIMEOUT_MS));
  2947. if (!rc) {
  2948. pr_err("%s: timeout. waited for OPEN_WRITE_COMPR rc[%d]\n",
  2949. __func__, rc);
  2950. rc = -ETIMEDOUT;
  2951. goto fail_cmd;
  2952. }
  2953. if (atomic_read(&ac->cmd_state) > 0) {
  2954. pr_err("%s: DSP returned error[%s]\n",
  2955. __func__, adsp_err_get_err_str(
  2956. atomic_read(&ac->cmd_state)));
  2957. rc = adsp_err_get_lnx_err_code(
  2958. atomic_read(&ac->cmd_state));
  2959. goto fail_cmd;
  2960. }
  2961. return 0;
  2962. fail_cmd:
  2963. return rc;
  2964. }
  2965. EXPORT_SYMBOL(q6asm_open_write_compressed);
  2966. static int __q6asm_open_write(struct audio_client *ac, uint32_t format,
  2967. uint16_t bits_per_sample, uint32_t stream_id,
  2968. bool is_gapless_mode,
  2969. uint32_t pcm_format_block_ver)
  2970. {
  2971. int rc = 0x00;
  2972. struct asm_stream_cmd_open_write_v3 open;
  2973. struct q6asm_cal_info cal_info;
  2974. if (ac == NULL) {
  2975. pr_err("%s: APR handle NULL\n", __func__);
  2976. return -EINVAL;
  2977. }
  2978. if (ac->apr == NULL) {
  2979. pr_err("%s: AC APR handle NULL\n", __func__);
  2980. return -EINVAL;
  2981. }
  2982. dev_vdbg(ac->dev, "%s: session[%d] wr_format[0x%x]\n",
  2983. __func__, ac->session, format);
  2984. q6asm_stream_add_hdr(ac, &open.hdr, sizeof(open), TRUE, stream_id);
  2985. atomic_set(&ac->cmd_state, -1);
  2986. /*
  2987. * Updated the token field with stream/session for compressed playback
  2988. * Platform driver must know the the stream with which the command is
  2989. * associated
  2990. */
  2991. if (ac->io_mode & COMPRESSED_STREAM_IO)
  2992. q6asm_update_token(&open.hdr.token,
  2993. ac->session,
  2994. stream_id,
  2995. 0, /* Buffer index is NA */
  2996. 0, /* Direction flag is NA */
  2997. WAIT_CMD);
  2998. dev_vdbg(ac->dev, "%s: token = 0x%x, stream_id %d, session 0x%x\n",
  2999. __func__, open.hdr.token, stream_id, ac->session);
  3000. open.hdr.opcode = ASM_STREAM_CMD_OPEN_WRITE_V3;
  3001. open.mode_flags = 0x00;
  3002. if (ac->perf_mode == ULL_POST_PROCESSING_PCM_MODE)
  3003. open.mode_flags |= ASM_ULL_POST_PROCESSING_STREAM_SESSION;
  3004. else if (ac->perf_mode == ULTRA_LOW_LATENCY_PCM_MODE)
  3005. open.mode_flags |= ASM_ULTRA_LOW_LATENCY_STREAM_SESSION;
  3006. else if (ac->perf_mode == LOW_LATENCY_PCM_MODE)
  3007. open.mode_flags |= ASM_LOW_LATENCY_STREAM_SESSION;
  3008. else {
  3009. open.mode_flags |= ASM_LEGACY_STREAM_SESSION;
  3010. if (is_gapless_mode)
  3011. open.mode_flags |= 1 << ASM_SHIFT_GAPLESS_MODE_FLAG;
  3012. }
  3013. /* source endpoint : matrix */
  3014. open.sink_endpointype = ASM_END_POINT_DEVICE_MATRIX;
  3015. open.bits_per_sample = bits_per_sample;
  3016. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3017. open.postprocopo_id = cal_info.topology_id;
  3018. if (ac->perf_mode != LEGACY_PCM_MODE)
  3019. open.postprocopo_id = ASM_STREAM_POSTPROCOPO_ID_NONE;
  3020. pr_debug("%s: perf_mode %d asm_topology 0x%x bps %d\n", __func__,
  3021. ac->perf_mode, open.postprocopo_id, open.bits_per_sample);
  3022. /*
  3023. * For Gapless playback it will use the same session for next stream,
  3024. * So use the same topology
  3025. */
  3026. if (!ac->topology) {
  3027. ac->topology = open.postprocopo_id;
  3028. ac->app_type = cal_info.app_type;
  3029. }
  3030. switch (format) {
  3031. case FORMAT_LINEAR_PCM:
  3032. open.dec_fmt_id = q6asm_get_pcm_format_id(pcm_format_block_ver);
  3033. break;
  3034. case FORMAT_MPEG4_AAC:
  3035. open.dec_fmt_id = ASM_MEDIA_FMT_AAC_V2;
  3036. break;
  3037. case FORMAT_MPEG4_MULTI_AAC:
  3038. open.dec_fmt_id = ASM_MEDIA_FMT_AAC_V2;
  3039. break;
  3040. case FORMAT_WMA_V9:
  3041. open.dec_fmt_id = ASM_MEDIA_FMT_WMA_V9_V2;
  3042. break;
  3043. case FORMAT_WMA_V10PRO:
  3044. open.dec_fmt_id = ASM_MEDIA_FMT_WMA_V10PRO_V2;
  3045. break;
  3046. case FORMAT_MP3:
  3047. open.dec_fmt_id = ASM_MEDIA_FMT_MP3;
  3048. break;
  3049. case FORMAT_AC3:
  3050. open.dec_fmt_id = ASM_MEDIA_FMT_AC3;
  3051. break;
  3052. case FORMAT_EAC3:
  3053. open.dec_fmt_id = ASM_MEDIA_FMT_EAC3;
  3054. break;
  3055. case FORMAT_MP2:
  3056. open.dec_fmt_id = ASM_MEDIA_FMT_MP2;
  3057. break;
  3058. case FORMAT_FLAC:
  3059. open.dec_fmt_id = ASM_MEDIA_FMT_FLAC;
  3060. break;
  3061. case FORMAT_ALAC:
  3062. open.dec_fmt_id = ASM_MEDIA_FMT_ALAC;
  3063. break;
  3064. case FORMAT_VORBIS:
  3065. open.dec_fmt_id = ASM_MEDIA_FMT_VORBIS;
  3066. break;
  3067. case FORMAT_APE:
  3068. open.dec_fmt_id = ASM_MEDIA_FMT_APE;
  3069. break;
  3070. case FORMAT_DSD:
  3071. open.dec_fmt_id = ASM_MEDIA_FMT_DSD;
  3072. break;
  3073. case FORMAT_APTX:
  3074. open.dec_fmt_id = ASM_MEDIA_FMT_APTX;
  3075. break;
  3076. case FORMAT_GEN_COMPR:
  3077. open.dec_fmt_id = ASM_MEDIA_FMT_GENERIC_COMPRESSED;
  3078. break;
  3079. default:
  3080. pr_err("%s: Invalid format 0x%x\n", __func__, format);
  3081. rc = -EINVAL;
  3082. goto fail_cmd;
  3083. }
  3084. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  3085. if (rc < 0) {
  3086. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  3087. __func__, open.hdr.opcode, rc);
  3088. rc = -EINVAL;
  3089. goto fail_cmd;
  3090. }
  3091. rc = wait_event_timeout(ac->cmd_wait,
  3092. (atomic_read(&ac->cmd_state) >= 0),
  3093. msecs_to_jiffies(TIMEOUT_MS));
  3094. if (!rc) {
  3095. pr_err("%s: timeout. waited for open write\n", __func__);
  3096. rc = -ETIMEDOUT;
  3097. goto fail_cmd;
  3098. }
  3099. if (atomic_read(&ac->cmd_state) > 0) {
  3100. pr_err("%s: DSP returned error[%s]\n",
  3101. __func__, adsp_err_get_err_str(
  3102. atomic_read(&ac->cmd_state)));
  3103. rc = adsp_err_get_lnx_err_code(
  3104. atomic_read(&ac->cmd_state));
  3105. goto fail_cmd;
  3106. }
  3107. ac->io_mode |= TUN_WRITE_IO_MODE;
  3108. return 0;
  3109. fail_cmd:
  3110. return rc;
  3111. }
  3112. int q6asm_open_write(struct audio_client *ac, uint32_t format)
  3113. {
  3114. return __q6asm_open_write(ac, format, 16, ac->stream_id,
  3115. false /*gapless*/,
  3116. PCM_MEDIA_FORMAT_V2 /*pcm_format_block_ver*/);
  3117. }
  3118. EXPORT_SYMBOL(q6asm_open_write);
  3119. int q6asm_open_write_v2(struct audio_client *ac, uint32_t format,
  3120. uint16_t bits_per_sample)
  3121. {
  3122. return __q6asm_open_write(ac, format, bits_per_sample,
  3123. ac->stream_id, false /*gapless*/,
  3124. PCM_MEDIA_FORMAT_V2 /*pcm_format_block_ver*/);
  3125. }
  3126. /*
  3127. * q6asm_open_write_v3 - Opens audio playback session
  3128. *
  3129. * @ac: Client session handle
  3130. * @format: decoder format
  3131. * @bits_per_sample: bit width of playback session
  3132. */
  3133. int q6asm_open_write_v3(struct audio_client *ac, uint32_t format,
  3134. uint16_t bits_per_sample)
  3135. {
  3136. return __q6asm_open_write(ac, format, bits_per_sample,
  3137. ac->stream_id, false /*gapless*/,
  3138. PCM_MEDIA_FORMAT_V3 /*pcm_format_block_ver*/);
  3139. }
  3140. EXPORT_SYMBOL(q6asm_open_write_v3);
  3141. /*
  3142. * q6asm_open_write_v4 - Opens audio playback session
  3143. *
  3144. * @ac: Client session handle
  3145. * @format: decoder format
  3146. * @bits_per_sample: bit width of playback session
  3147. */
  3148. int q6asm_open_write_v4(struct audio_client *ac, uint32_t format,
  3149. uint16_t bits_per_sample)
  3150. {
  3151. return __q6asm_open_write(ac, format, bits_per_sample,
  3152. ac->stream_id, false /*gapless*/,
  3153. PCM_MEDIA_FORMAT_V4 /*pcm_format_block_ver*/);
  3154. }
  3155. EXPORT_SYMBOL(q6asm_open_write_v4);
  3156. int q6asm_stream_open_write_v2(struct audio_client *ac, uint32_t format,
  3157. uint16_t bits_per_sample, int32_t stream_id,
  3158. bool is_gapless_mode)
  3159. {
  3160. return __q6asm_open_write(ac, format, bits_per_sample,
  3161. stream_id, is_gapless_mode,
  3162. PCM_MEDIA_FORMAT_V2 /*pcm_format_block_ver*/);
  3163. }
  3164. /*
  3165. * q6asm_stream_open_write_v3 - Creates audio stream for playback
  3166. *
  3167. * @ac: Client session handle
  3168. * @format: asm playback format
  3169. * @bits_per_sample: bit width of requested stream
  3170. * @stream_id: stream id of stream to be associated with this session
  3171. * @is_gapless_mode: true if gapless mode needs to be enabled
  3172. */
  3173. int q6asm_stream_open_write_v3(struct audio_client *ac, uint32_t format,
  3174. uint16_t bits_per_sample, int32_t stream_id,
  3175. bool is_gapless_mode)
  3176. {
  3177. return __q6asm_open_write(ac, format, bits_per_sample,
  3178. stream_id, is_gapless_mode,
  3179. PCM_MEDIA_FORMAT_V3 /*pcm_format_block_ver*/);
  3180. }
  3181. EXPORT_SYMBOL(q6asm_stream_open_write_v3);
  3182. /*
  3183. * q6asm_open_write_v5 - Opens audio playback session
  3184. *
  3185. * @ac: Client session handle
  3186. * @format: decoder format
  3187. * @bits_per_sample: bit width of playback session
  3188. */
  3189. int q6asm_open_write_v5(struct audio_client *ac, uint32_t format,
  3190. uint16_t bits_per_sample)
  3191. {
  3192. return __q6asm_open_write(ac, format, bits_per_sample,
  3193. ac->stream_id, false /*gapless*/,
  3194. PCM_MEDIA_FORMAT_V5 /*pcm_format_block_ver*/);
  3195. }
  3196. EXPORT_SYMBOL(q6asm_open_write_v5);
  3197. /*
  3198. * q6asm_stream_open_write_v4 - Creates audio stream for playback
  3199. *
  3200. * @ac: Client session handle
  3201. * @format: asm playback format
  3202. * @bits_per_sample: bit width of requested stream
  3203. * @stream_id: stream id of stream to be associated with this session
  3204. * @is_gapless_mode: true if gapless mode needs to be enabled
  3205. */
  3206. int q6asm_stream_open_write_v4(struct audio_client *ac, uint32_t format,
  3207. uint16_t bits_per_sample, int32_t stream_id,
  3208. bool is_gapless_mode)
  3209. {
  3210. return __q6asm_open_write(ac, format, bits_per_sample,
  3211. stream_id, is_gapless_mode,
  3212. PCM_MEDIA_FORMAT_V4 /*pcm_format_block_ver*/);
  3213. }
  3214. EXPORT_SYMBOL(q6asm_stream_open_write_v4);
  3215. /*
  3216. * q6asm_stream_open_write_v5 - Creates audio stream for playback
  3217. *
  3218. * @ac: Client session handle
  3219. * @format: asm playback format
  3220. * @bits_per_sample: bit width of requested stream
  3221. * @stream_id: stream id of stream to be associated with this session
  3222. * @is_gapless_mode: true if gapless mode needs to be enabled
  3223. */
  3224. int q6asm_stream_open_write_v5(struct audio_client *ac, uint32_t format,
  3225. uint16_t bits_per_sample, int32_t stream_id,
  3226. bool is_gapless_mode)
  3227. {
  3228. return __q6asm_open_write(ac, format, bits_per_sample,
  3229. stream_id, is_gapless_mode,
  3230. PCM_MEDIA_FORMAT_V5 /*pcm_format_block_ver*/);
  3231. }
  3232. EXPORT_SYMBOL(q6asm_stream_open_write_v5);
  3233. static int __q6asm_open_read_write(struct audio_client *ac, uint32_t rd_format,
  3234. uint32_t wr_format, bool is_meta_data_mode,
  3235. uint32_t bits_per_sample,
  3236. bool overwrite_topology, int topology)
  3237. {
  3238. int rc = 0x00;
  3239. struct asm_stream_cmd_open_readwrite_v2 open;
  3240. struct q6asm_cal_info cal_info;
  3241. if (ac == NULL) {
  3242. pr_err("%s: APR handle NULL\n", __func__);
  3243. return -EINVAL;
  3244. }
  3245. if (ac->apr == NULL) {
  3246. pr_err("%s: AC APR handle NULL\n", __func__);
  3247. return -EINVAL;
  3248. }
  3249. pr_debug("%s: session[%d]\n", __func__, ac->session);
  3250. pr_debug("%s: wr_format[0x%x]rd_format[0x%x]\n",
  3251. __func__, wr_format, rd_format);
  3252. ac->io_mode |= NT_MODE;
  3253. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  3254. atomic_set(&ac->cmd_state, -1);
  3255. open.hdr.opcode = ASM_STREAM_CMD_OPEN_READWRITE_V2;
  3256. open.mode_flags = is_meta_data_mode ? BUFFER_META_ENABLE : 0;
  3257. open.bits_per_sample = bits_per_sample;
  3258. /* source endpoint : matrix */
  3259. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3260. open.postprocopo_id = cal_info.topology_id;
  3261. open.postprocopo_id = overwrite_topology ?
  3262. topology : open.postprocopo_id;
  3263. ac->topology = open.postprocopo_id;
  3264. ac->app_type = cal_info.app_type;
  3265. switch (wr_format) {
  3266. case FORMAT_LINEAR_PCM:
  3267. case FORMAT_MULTI_CHANNEL_LINEAR_PCM:
  3268. open.dec_fmt_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V2;
  3269. break;
  3270. case FORMAT_MPEG4_AAC:
  3271. open.dec_fmt_id = ASM_MEDIA_FMT_AAC_V2;
  3272. break;
  3273. case FORMAT_MPEG4_MULTI_AAC:
  3274. open.dec_fmt_id = ASM_MEDIA_FMT_AAC_V2;
  3275. break;
  3276. case FORMAT_WMA_V9:
  3277. open.dec_fmt_id = ASM_MEDIA_FMT_WMA_V9_V2;
  3278. break;
  3279. case FORMAT_WMA_V10PRO:
  3280. open.dec_fmt_id = ASM_MEDIA_FMT_WMA_V10PRO_V2;
  3281. break;
  3282. case FORMAT_AMRNB:
  3283. open.dec_fmt_id = ASM_MEDIA_FMT_AMRNB_FS;
  3284. break;
  3285. case FORMAT_AMRWB:
  3286. open.dec_fmt_id = ASM_MEDIA_FMT_AMRWB_FS;
  3287. break;
  3288. case FORMAT_AMR_WB_PLUS:
  3289. open.dec_fmt_id = ASM_MEDIA_FMT_AMR_WB_PLUS_V2;
  3290. break;
  3291. case FORMAT_V13K:
  3292. open.dec_fmt_id = ASM_MEDIA_FMT_V13K_FS;
  3293. break;
  3294. case FORMAT_EVRC:
  3295. open.dec_fmt_id = ASM_MEDIA_FMT_EVRC_FS;
  3296. break;
  3297. case FORMAT_EVRCB:
  3298. open.dec_fmt_id = ASM_MEDIA_FMT_EVRCB_FS;
  3299. break;
  3300. case FORMAT_EVRCWB:
  3301. open.dec_fmt_id = ASM_MEDIA_FMT_EVRCWB_FS;
  3302. break;
  3303. case FORMAT_MP3:
  3304. open.dec_fmt_id = ASM_MEDIA_FMT_MP3;
  3305. break;
  3306. case FORMAT_ALAC:
  3307. open.dec_fmt_id = ASM_MEDIA_FMT_ALAC;
  3308. break;
  3309. case FORMAT_APE:
  3310. open.dec_fmt_id = ASM_MEDIA_FMT_APE;
  3311. break;
  3312. case FORMAT_DSD:
  3313. open.dec_fmt_id = ASM_MEDIA_FMT_DSD;
  3314. break;
  3315. case FORMAT_G711_ALAW_FS:
  3316. open.dec_fmt_id = ASM_MEDIA_FMT_G711_ALAW_FS;
  3317. break;
  3318. case FORMAT_G711_MLAW_FS:
  3319. open.dec_fmt_id = ASM_MEDIA_FMT_G711_MLAW_FS;
  3320. break;
  3321. default:
  3322. pr_err("%s: Invalid format 0x%x\n",
  3323. __func__, wr_format);
  3324. rc = -EINVAL;
  3325. goto fail_cmd;
  3326. }
  3327. switch (rd_format) {
  3328. case FORMAT_LINEAR_PCM:
  3329. case FORMAT_MULTI_CHANNEL_LINEAR_PCM:
  3330. open.enc_cfg_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V2;
  3331. break;
  3332. case FORMAT_MPEG4_AAC:
  3333. open.enc_cfg_id = ASM_MEDIA_FMT_AAC_V2;
  3334. break;
  3335. case FORMAT_G711_ALAW_FS:
  3336. open.enc_cfg_id = ASM_MEDIA_FMT_G711_ALAW_FS;
  3337. break;
  3338. case FORMAT_G711_MLAW_FS:
  3339. open.enc_cfg_id = ASM_MEDIA_FMT_G711_MLAW_FS;
  3340. break;
  3341. case FORMAT_V13K:
  3342. open.enc_cfg_id = ASM_MEDIA_FMT_V13K_FS;
  3343. break;
  3344. case FORMAT_EVRC:
  3345. open.enc_cfg_id = ASM_MEDIA_FMT_EVRC_FS;
  3346. break;
  3347. case FORMAT_AMRNB:
  3348. open.enc_cfg_id = ASM_MEDIA_FMT_AMRNB_FS;
  3349. break;
  3350. case FORMAT_AMRWB:
  3351. open.enc_cfg_id = ASM_MEDIA_FMT_AMRWB_FS;
  3352. break;
  3353. case FORMAT_ALAC:
  3354. open.enc_cfg_id = ASM_MEDIA_FMT_ALAC;
  3355. break;
  3356. case FORMAT_APE:
  3357. open.enc_cfg_id = ASM_MEDIA_FMT_APE;
  3358. break;
  3359. default:
  3360. pr_err("%s: Invalid format 0x%x\n",
  3361. __func__, rd_format);
  3362. rc = -EINVAL;
  3363. goto fail_cmd;
  3364. }
  3365. dev_vdbg(ac->dev, "%s: rdformat[0x%x]wrformat[0x%x]\n", __func__,
  3366. open.enc_cfg_id, open.dec_fmt_id);
  3367. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  3368. if (rc < 0) {
  3369. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  3370. __func__, open.hdr.opcode, rc);
  3371. rc = -EINVAL;
  3372. goto fail_cmd;
  3373. }
  3374. rc = wait_event_timeout(ac->cmd_wait,
  3375. (atomic_read(&ac->cmd_state) >= 0),
  3376. msecs_to_jiffies(TIMEOUT_MS));
  3377. if (!rc) {
  3378. pr_err("%s: timeout. waited for open read-write\n",
  3379. __func__);
  3380. rc = -ETIMEDOUT;
  3381. goto fail_cmd;
  3382. }
  3383. if (atomic_read(&ac->cmd_state) > 0) {
  3384. pr_err("%s: DSP returned error[%s]\n",
  3385. __func__, adsp_err_get_err_str(
  3386. atomic_read(&ac->cmd_state)));
  3387. rc = adsp_err_get_lnx_err_code(
  3388. atomic_read(&ac->cmd_state));
  3389. goto fail_cmd;
  3390. }
  3391. return 0;
  3392. fail_cmd:
  3393. return rc;
  3394. }
  3395. /**
  3396. * q6asm_open_read_write -
  3397. * command to open ASM in read/write mode
  3398. *
  3399. * @ac: Audio client handle
  3400. * @rd_format: capture format for ASM
  3401. * @wr_format: playback format for ASM
  3402. *
  3403. * Returns 0 on success or error on failure
  3404. */
  3405. int q6asm_open_read_write(struct audio_client *ac, uint32_t rd_format,
  3406. uint32_t wr_format)
  3407. {
  3408. return __q6asm_open_read_write(ac, rd_format, wr_format,
  3409. true/*meta data mode*/,
  3410. 16 /*bits_per_sample*/,
  3411. false /*overwrite_topology*/, 0);
  3412. }
  3413. EXPORT_SYMBOL(q6asm_open_read_write);
  3414. /**
  3415. * q6asm_open_read_write_v2 -
  3416. * command to open ASM in bi-directional read/write mode
  3417. *
  3418. * @ac: Audio client handle
  3419. * @rd_format: capture format for ASM
  3420. * @wr_format: playback format for ASM
  3421. * @is_meta_data_mode: mode to indicate if meta data present
  3422. * @bits_per_sample: number of bits per sample
  3423. * @overwrite_topology: topology to be overwritten flag
  3424. * @topology: Topology for ASM
  3425. *
  3426. * Returns 0 on success or error on failure
  3427. */
  3428. int q6asm_open_read_write_v2(struct audio_client *ac, uint32_t rd_format,
  3429. uint32_t wr_format, bool is_meta_data_mode,
  3430. uint32_t bits_per_sample, bool overwrite_topology,
  3431. int topology)
  3432. {
  3433. return __q6asm_open_read_write(ac, rd_format, wr_format,
  3434. is_meta_data_mode, bits_per_sample,
  3435. overwrite_topology, topology);
  3436. }
  3437. EXPORT_SYMBOL(q6asm_open_read_write_v2);
  3438. /**
  3439. * q6asm_open_loopback_v2 -
  3440. * command to open ASM in loopback mode
  3441. *
  3442. * @ac: Audio client handle
  3443. * @bits_per_sample: number of bits per sample
  3444. *
  3445. * Returns 0 on success or error on failure
  3446. */
  3447. int q6asm_open_loopback_v2(struct audio_client *ac, uint16_t bits_per_sample)
  3448. {
  3449. int rc = 0x00;
  3450. struct q6asm_cal_info cal_info;
  3451. if (ac == NULL) {
  3452. pr_err("%s: APR handle NULL\n", __func__);
  3453. return -EINVAL;
  3454. }
  3455. if (ac->apr == NULL) {
  3456. pr_err("%s: AC APR handle NULL\n", __func__);
  3457. return -EINVAL;
  3458. }
  3459. pr_debug("%s: session[%d]\n", __func__, ac->session);
  3460. if (ac->perf_mode == LOW_LATENCY_PCM_MODE) {
  3461. struct asm_stream_cmd_open_transcode_loopback_t open;
  3462. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  3463. atomic_set(&ac->cmd_state, -1);
  3464. open.hdr.opcode = ASM_STREAM_CMD_OPEN_TRANSCODE_LOOPBACK;
  3465. open.mode_flags = 0;
  3466. open.src_endpoint_type = 0;
  3467. open.sink_endpoint_type = 0;
  3468. open.src_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V2;
  3469. open.sink_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V2;
  3470. /* source endpoint : matrix */
  3471. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3472. open.audproc_topo_id = cal_info.topology_id;
  3473. ac->app_type = cal_info.app_type;
  3474. if (ac->perf_mode == LOW_LATENCY_PCM_MODE)
  3475. open.mode_flags |= ASM_LOW_LATENCY_STREAM_SESSION;
  3476. else
  3477. open.mode_flags |= ASM_LEGACY_STREAM_SESSION;
  3478. ac->topology = open.audproc_topo_id;
  3479. open.bits_per_sample = bits_per_sample;
  3480. open.reserved = 0;
  3481. pr_debug("%s: opening a transcode_loopback with mode_flags =[%d] session[%d]\n",
  3482. __func__, open.mode_flags, ac->session);
  3483. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  3484. if (rc < 0) {
  3485. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  3486. __func__, open.hdr.opcode, rc);
  3487. rc = -EINVAL;
  3488. goto fail_cmd;
  3489. }
  3490. } else {/*if(ac->perf_mode == LEGACY_PCM_MODE)*/
  3491. struct asm_stream_cmd_open_loopback_v2 open;
  3492. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  3493. atomic_set(&ac->cmd_state, -1);
  3494. open.hdr.opcode = ASM_STREAM_CMD_OPEN_LOOPBACK_V2;
  3495. open.mode_flags = 0;
  3496. open.src_endpointype = 0;
  3497. open.sink_endpointype = 0;
  3498. /* source endpoint : matrix */
  3499. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3500. open.postprocopo_id = cal_info.topology_id;
  3501. ac->app_type = cal_info.app_type;
  3502. ac->topology = open.postprocopo_id;
  3503. open.bits_per_sample = bits_per_sample;
  3504. open.reserved = 0;
  3505. pr_debug("%s: opening a loopback_v2 with mode_flags =[%d] session[%d]\n",
  3506. __func__, open.mode_flags, ac->session);
  3507. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  3508. if (rc < 0) {
  3509. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  3510. __func__, open.hdr.opcode, rc);
  3511. rc = -EINVAL;
  3512. goto fail_cmd;
  3513. }
  3514. }
  3515. rc = wait_event_timeout(ac->cmd_wait,
  3516. (atomic_read(&ac->cmd_state) >= 0),
  3517. msecs_to_jiffies(TIMEOUT_MS));
  3518. if (!rc) {
  3519. pr_err("%s: timeout. waited for open_loopback\n",
  3520. __func__);
  3521. rc = -ETIMEDOUT;
  3522. goto fail_cmd;
  3523. }
  3524. if (atomic_read(&ac->cmd_state) > 0) {
  3525. pr_err("%s: DSP returned error[%s]\n",
  3526. __func__, adsp_err_get_err_str(
  3527. atomic_read(&ac->cmd_state)));
  3528. rc = adsp_err_get_lnx_err_code(
  3529. atomic_read(&ac->cmd_state));
  3530. goto fail_cmd;
  3531. }
  3532. return 0;
  3533. fail_cmd:
  3534. return rc;
  3535. }
  3536. EXPORT_SYMBOL(q6asm_open_loopback_v2);
  3537. /**
  3538. * q6asm_open_transcode_loopback -
  3539. * command to open ASM in transcode loopback mode
  3540. *
  3541. * @ac: Audio client handle
  3542. * @bits_per_sample: number of bits per sample
  3543. * @source_format: Format of clip
  3544. * @sink_format: end device supported format
  3545. *
  3546. * Returns 0 on success or error on failure
  3547. */
  3548. int q6asm_open_transcode_loopback(struct audio_client *ac,
  3549. uint16_t bits_per_sample,
  3550. uint32_t source_format, uint32_t sink_format)
  3551. {
  3552. int rc = 0x00;
  3553. struct asm_stream_cmd_open_transcode_loopback_t open;
  3554. struct q6asm_cal_info cal_info;
  3555. if (ac == NULL) {
  3556. pr_err("%s: APR handle NULL\n", __func__);
  3557. return -EINVAL;
  3558. }
  3559. if (ac->apr == NULL) {
  3560. pr_err("%s: AC APR handle NULL\n", __func__);
  3561. return -EINVAL;
  3562. }
  3563. pr_debug("%s: session[%d]\n", __func__, ac->session);
  3564. q6asm_add_hdr(ac, &open.hdr, sizeof(open), TRUE);
  3565. atomic_set(&ac->cmd_state, -1);
  3566. open.hdr.opcode = ASM_STREAM_CMD_OPEN_TRANSCODE_LOOPBACK;
  3567. open.mode_flags = 0;
  3568. open.src_endpoint_type = 0;
  3569. open.sink_endpoint_type = 0;
  3570. switch (source_format) {
  3571. case FORMAT_LINEAR_PCM:
  3572. case FORMAT_MULTI_CHANNEL_LINEAR_PCM:
  3573. open.src_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V3;
  3574. break;
  3575. case FORMAT_AC3:
  3576. open.src_format_id = ASM_MEDIA_FMT_AC3;
  3577. break;
  3578. case FORMAT_EAC3:
  3579. open.src_format_id = ASM_MEDIA_FMT_EAC3;
  3580. break;
  3581. default:
  3582. pr_err("%s: Unsupported src fmt [%d]\n",
  3583. __func__, source_format);
  3584. return -EINVAL;
  3585. }
  3586. switch (sink_format) {
  3587. case FORMAT_LINEAR_PCM:
  3588. case FORMAT_MULTI_CHANNEL_LINEAR_PCM:
  3589. open.sink_format_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V3;
  3590. break;
  3591. default:
  3592. pr_err("%s: Unsupported sink fmt [%d]\n",
  3593. __func__, sink_format);
  3594. return -EINVAL;
  3595. }
  3596. /* source endpoint : matrix */
  3597. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3598. open.audproc_topo_id = cal_info.topology_id;
  3599. ac->app_type = cal_info.app_type;
  3600. if (ac->perf_mode == LOW_LATENCY_PCM_MODE)
  3601. open.mode_flags |= ASM_LOW_LATENCY_STREAM_SESSION;
  3602. else
  3603. open.mode_flags |= ASM_LEGACY_STREAM_SESSION;
  3604. ac->topology = open.audproc_topo_id;
  3605. open.bits_per_sample = bits_per_sample;
  3606. open.reserved = 0;
  3607. pr_debug("%s: opening a transcode_loopback with mode_flags =[%d] session[%d]\n",
  3608. __func__, open.mode_flags, ac->session);
  3609. rc = apr_send_pkt(ac->apr, (uint32_t *) &open);
  3610. if (rc < 0) {
  3611. pr_err("%s: open failed op[0x%x]rc[%d]\n",
  3612. __func__, open.hdr.opcode, rc);
  3613. rc = -EINVAL;
  3614. goto fail_cmd;
  3615. }
  3616. rc = wait_event_timeout(ac->cmd_wait,
  3617. (atomic_read(&ac->cmd_state) >= 0),
  3618. msecs_to_jiffies(TIMEOUT_MS));
  3619. if (!rc) {
  3620. pr_err("%s: timeout. waited for open_transcode_loopback\n",
  3621. __func__);
  3622. rc = -ETIMEDOUT;
  3623. goto fail_cmd;
  3624. }
  3625. if (atomic_read(&ac->cmd_state) > 0) {
  3626. pr_err("%s: DSP returned error[%s]\n",
  3627. __func__, adsp_err_get_err_str(
  3628. atomic_read(&ac->cmd_state)));
  3629. rc = adsp_err_get_lnx_err_code(
  3630. atomic_read(&ac->cmd_state));
  3631. goto fail_cmd;
  3632. }
  3633. return 0;
  3634. fail_cmd:
  3635. return rc;
  3636. }
  3637. EXPORT_SYMBOL(q6asm_open_transcode_loopback);
  3638. static
  3639. int q6asm_set_shared_circ_buff(struct audio_client *ac,
  3640. struct asm_stream_cmd_open_shared_io *open,
  3641. int bufsz, int bufcnt,
  3642. int dir)
  3643. {
  3644. struct audio_buffer *buf_circ;
  3645. int bytes_to_alloc, rc;
  3646. size_t len;
  3647. mutex_lock(&ac->cmd_lock);
  3648. if (ac->port[dir].buf) {
  3649. pr_err("%s: Buffer already allocated\n", __func__);
  3650. rc = -EINVAL;
  3651. goto done;
  3652. }
  3653. buf_circ = kzalloc(sizeof(struct audio_buffer), GFP_KERNEL);
  3654. if (!buf_circ) {
  3655. rc = -ENOMEM;
  3656. goto done;
  3657. }
  3658. bytes_to_alloc = bufsz * bufcnt;
  3659. bytes_to_alloc = PAGE_ALIGN(bytes_to_alloc);
  3660. rc = msm_audio_ion_alloc(&buf_circ->dma_buf,
  3661. bytes_to_alloc,
  3662. &buf_circ->phys,
  3663. &len, &buf_circ->data);
  3664. if (rc) {
  3665. pr_err("%s: Audio ION alloc is failed, rc = %d\n", __func__,
  3666. rc);
  3667. kfree(buf_circ);
  3668. goto done;
  3669. }
  3670. ac->port[dir].buf = buf_circ;
  3671. buf_circ->used = dir ^ 1;
  3672. buf_circ->size = bytes_to_alloc;
  3673. buf_circ->actual_size = bytes_to_alloc;
  3674. memset(buf_circ->data, 0, buf_circ->actual_size);
  3675. ac->port[dir].max_buf_cnt = 1;
  3676. open->shared_circ_buf_mem_pool_id = ADSP_MEMORY_MAP_SHMEM8_4K_POOL;
  3677. open->shared_circ_buf_num_regions = 1;
  3678. open->shared_circ_buf_property_flag = 0x00;
  3679. open->shared_circ_buf_start_phy_addr_lsw =
  3680. lower_32_bits(buf_circ->phys);
  3681. open->shared_circ_buf_start_phy_addr_msw =
  3682. msm_audio_populate_upper_32_bits(buf_circ->phys);
  3683. open->shared_circ_buf_size = bufsz * bufcnt;
  3684. open->map_region_circ_buf.shm_addr_lsw = lower_32_bits(buf_circ->phys);
  3685. open->map_region_circ_buf.shm_addr_msw =
  3686. msm_audio_populate_upper_32_bits(buf_circ->phys);
  3687. open->map_region_circ_buf.mem_size_bytes = bytes_to_alloc;
  3688. done:
  3689. mutex_unlock(&ac->cmd_lock);
  3690. return rc;
  3691. }
  3692. static
  3693. int q6asm_set_shared_pos_buff(struct audio_client *ac,
  3694. struct asm_stream_cmd_open_shared_io *open,
  3695. int dir)
  3696. {
  3697. struct audio_buffer *buf_pos = &ac->shared_pos_buf;
  3698. int rc;
  3699. size_t len;
  3700. int bytes_to_alloc = sizeof(struct asm_shared_position_buffer);
  3701. mutex_lock(&ac->cmd_lock);
  3702. bytes_to_alloc = PAGE_ALIGN(bytes_to_alloc);
  3703. rc = msm_audio_ion_alloc(&buf_pos->dma_buf,
  3704. bytes_to_alloc,
  3705. &buf_pos->phys, &len,
  3706. &buf_pos->data);
  3707. if (rc) {
  3708. pr_err("%s: Audio pos buf ION alloc is failed, rc = %d\n",
  3709. __func__, rc);
  3710. goto done;
  3711. }
  3712. buf_pos->used = dir ^ 1;
  3713. buf_pos->size = bytes_to_alloc;
  3714. buf_pos->actual_size = bytes_to_alloc;
  3715. open->shared_pos_buf_mem_pool_id = ADSP_MEMORY_MAP_SHMEM8_4K_POOL;
  3716. open->shared_pos_buf_num_regions = 1;
  3717. open->shared_pos_buf_property_flag = 0x00;
  3718. open->shared_pos_buf_phy_addr_lsw = lower_32_bits(buf_pos->phys);
  3719. open->shared_pos_buf_phy_addr_msw =
  3720. msm_audio_populate_upper_32_bits(buf_pos->phys);
  3721. open->map_region_pos_buf.shm_addr_lsw = lower_32_bits(buf_pos->phys);
  3722. open->map_region_pos_buf.shm_addr_msw =
  3723. msm_audio_populate_upper_32_bits(buf_pos->phys);
  3724. open->map_region_pos_buf.mem_size_bytes = bytes_to_alloc;
  3725. done:
  3726. mutex_unlock(&ac->cmd_lock);
  3727. return rc;
  3728. }
  3729. /*
  3730. * q6asm_open_shared_io: Open an ASM session for pull mode (playback)
  3731. * or push mode (capture).
  3732. * parameters
  3733. * config - session parameters (channels, bits_per_sample, sr)
  3734. * dir - stream direction (IN for playback, OUT for capture)
  3735. * use_default_chmap: true if default channel map to be used
  3736. * channel_map: input channel map
  3737. * returns 0 if successful, error code otherwise
  3738. */
  3739. int q6asm_open_shared_io(struct audio_client *ac,
  3740. struct shared_io_config *config,
  3741. int dir, bool use_default_chmap, u8 *channel_map)
  3742. {
  3743. struct asm_stream_cmd_open_shared_io *open;
  3744. u8 *channel_mapping;
  3745. int i, size_of_open, num_watermarks, bufsz, bufcnt, rc, flags = 0;
  3746. struct q6asm_cal_info cal_info;
  3747. if (!ac || !config)
  3748. return -EINVAL;
  3749. if (!use_default_chmap && (channel_map == NULL)) {
  3750. pr_err("%s: No valid chan map and can't use default\n",
  3751. __func__);
  3752. return -EINVAL;
  3753. }
  3754. bufsz = config->bufsz;
  3755. bufcnt = config->bufcnt;
  3756. num_watermarks = 0;
  3757. ac->config = *config;
  3758. if (ac->session <= 0 || ac->session > SESSION_MAX) {
  3759. pr_err("%s: Session %d is out of bounds\n",
  3760. __func__, ac->session);
  3761. return -EINVAL;
  3762. }
  3763. size_of_open = sizeof(struct asm_stream_cmd_open_shared_io) +
  3764. (sizeof(struct asm_shared_watermark_level) * num_watermarks);
  3765. open = kzalloc(PAGE_ALIGN(size_of_open), GFP_KERNEL);
  3766. if (!open)
  3767. return -ENOMEM;
  3768. q6asm_stream_add_hdr(ac, &open->hdr, size_of_open, TRUE,
  3769. ac->stream_id);
  3770. atomic_set(&ac->cmd_state, 1);
  3771. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x, perf %d\n",
  3772. __func__, open->hdr.token, ac->stream_id, ac->session,
  3773. ac->perf_mode);
  3774. open->hdr.opcode =
  3775. dir == IN ? ASM_STREAM_CMD_OPEN_PULL_MODE_WRITE :
  3776. ASM_STREAM_CMD_OPEN_PUSH_MODE_READ;
  3777. pr_debug("%s perf_mode %d\n", __func__, ac->perf_mode);
  3778. if (dir == IN)
  3779. if (ac->perf_mode == ULL_POST_PROCESSING_PCM_MODE)
  3780. flags = 4 << ASM_SHIFT_STREAM_PERF_FLAG_PULL_MODE_WRITE;
  3781. else if (ac->perf_mode == ULTRA_LOW_LATENCY_PCM_MODE)
  3782. flags = 2 << ASM_SHIFT_STREAM_PERF_FLAG_PULL_MODE_WRITE;
  3783. else if (ac->perf_mode == LOW_LATENCY_PCM_MODE)
  3784. flags = 1 << ASM_SHIFT_STREAM_PERF_FLAG_PULL_MODE_WRITE;
  3785. else
  3786. pr_err("Invalid perf mode for pull write\n");
  3787. else
  3788. if (ac->perf_mode == LOW_LATENCY_PCM_MODE)
  3789. flags = ASM_LOW_LATENCY_TX_STREAM_SESSION <<
  3790. ASM_SHIFT_STREAM_PERF_FLAG_PUSH_MODE_READ;
  3791. else
  3792. pr_err("Invalid perf mode for push read\n");
  3793. if (flags == 0) {
  3794. pr_err("%s: Invalid mode[%d]\n", __func__,
  3795. ac->perf_mode);
  3796. kfree(open);
  3797. return -EINVAL;
  3798. }
  3799. pr_debug("open.mode_flags = 0x%x\n", flags);
  3800. open->mode_flags = flags;
  3801. open->endpoint_type = ASM_END_POINT_DEVICE_MATRIX;
  3802. open->topo_bits_per_sample = config->bits_per_sample;
  3803. rc = q6asm_get_asm_topology_apptype(&cal_info);
  3804. open->topo_id = cal_info.topology_id;
  3805. if (config->format == FORMAT_LINEAR_PCM)
  3806. open->fmt_id = ASM_MEDIA_FMT_MULTI_CHANNEL_PCM_V3;
  3807. else {
  3808. pr_err("%s: Invalid format[%d]\n", __func__, config->format);
  3809. rc = -EINVAL;
  3810. goto done;
  3811. }
  3812. rc = q6asm_set_shared_circ_buff(ac, open, bufsz, bufcnt, dir);
  3813. if (rc)
  3814. goto done;
  3815. ac->port[dir].tmp_hdl = 0;
  3816. rc = q6asm_set_shared_pos_buff(ac, open, dir);
  3817. if (rc)
  3818. goto done;
  3819. /* asm_multi_channel_pcm_fmt_blk_v3 */
  3820. open->fmt.num_channels = config->channels;
  3821. open->fmt.bits_per_sample = config->bits_per_sample;
  3822. open->fmt.sample_rate = config->rate;
  3823. open->fmt.is_signed = 1;
  3824. open->fmt.sample_word_size = config->sample_word_size;
  3825. channel_mapping = open->fmt.channel_mapping;
  3826. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  3827. if (use_default_chmap) {
  3828. rc = q6asm_map_channels(channel_mapping, config->channels,
  3829. false);
  3830. if (rc) {
  3831. pr_err("%s: Map channels failed, ret: %d\n",
  3832. __func__, rc);
  3833. goto done;
  3834. }
  3835. } else {
  3836. memcpy(channel_mapping, channel_map,
  3837. PCM_FORMAT_MAX_NUM_CHANNEL);
  3838. }
  3839. open->num_watermark_levels = num_watermarks;
  3840. for (i = 0; i < num_watermarks; i++) {
  3841. open->watermark[i].watermark_level_bytes = i *
  3842. ((bufsz * bufcnt) / num_watermarks);
  3843. pr_debug("%s: Watermark level set for %i\n",
  3844. __func__,
  3845. open->watermark[i].watermark_level_bytes);
  3846. }
  3847. rc = apr_send_pkt(ac->apr, (uint32_t *) open);
  3848. if (rc < 0) {
  3849. pr_err("%s: Open failed op[0x%x]rc[%d]\n",
  3850. __func__, open->hdr.opcode, rc);
  3851. goto done;
  3852. }
  3853. pr_debug("%s: sent open apr pkt\n", __func__);
  3854. rc = wait_event_timeout(ac->cmd_wait,
  3855. (atomic_read(&ac->cmd_state) <= 0),
  3856. msecs_to_jiffies(TIMEOUT_MS));
  3857. if (!rc) {
  3858. pr_err("%s: Timeout. Waited for open write apr pkt rc[%d]\n",
  3859. __func__, rc);
  3860. rc = -ETIMEDOUT;
  3861. goto done;
  3862. }
  3863. if (atomic_read(&ac->cmd_state) < 0) {
  3864. pr_err("%s: DSP returned error [%d]\n", __func__,
  3865. atomic_read(&ac->cmd_state));
  3866. rc = -EINVAL;
  3867. goto done;
  3868. }
  3869. ac->io_mode |= TUN_WRITE_IO_MODE;
  3870. rc = 0;
  3871. done:
  3872. kfree(open);
  3873. return rc;
  3874. }
  3875. EXPORT_SYMBOL(q6asm_open_shared_io);
  3876. /*
  3877. * q6asm_shared_io_buf: Returns handle to the shared circular buffer being
  3878. * used for pull/push mode.
  3879. * parameters
  3880. * dir - used to identify input/output port
  3881. * returns buffer handle
  3882. */
  3883. struct audio_buffer *q6asm_shared_io_buf(struct audio_client *ac,
  3884. int dir)
  3885. {
  3886. struct audio_port_data *port;
  3887. if (!ac) {
  3888. pr_err("%s: ac is null\n", __func__);
  3889. return NULL;
  3890. }
  3891. port = &ac->port[dir];
  3892. return port->buf;
  3893. }
  3894. EXPORT_SYMBOL(q6asm_shared_io_buf);
  3895. /*
  3896. * q6asm_shared_io_free: Frees memory allocated for a pull/push session
  3897. * parameters
  3898. * dir - port direction
  3899. * returns 0 if successful, error otherwise
  3900. */
  3901. int q6asm_shared_io_free(struct audio_client *ac, int dir)
  3902. {
  3903. struct audio_port_data *port;
  3904. if (!ac) {
  3905. pr_err("%s: audio client is null\n", __func__);
  3906. return -EINVAL;
  3907. }
  3908. port = &ac->port[dir];
  3909. mutex_lock(&ac->cmd_lock);
  3910. if (port->buf && port->buf->data) {
  3911. msm_audio_ion_free(port->buf->dma_buf);
  3912. port->buf->dma_buf = NULL;
  3913. port->max_buf_cnt = 0;
  3914. kfree(port->buf);
  3915. port->buf = NULL;
  3916. }
  3917. if (ac->shared_pos_buf.data) {
  3918. msm_audio_ion_free(ac->shared_pos_buf.dma_buf);
  3919. ac->shared_pos_buf.dma_buf = NULL;
  3920. }
  3921. mutex_unlock(&ac->cmd_lock);
  3922. return 0;
  3923. }
  3924. EXPORT_SYMBOL(q6asm_shared_io_free);
  3925. /*
  3926. * q6asm_get_shared_pos: Returns current read index/write index as observed
  3927. * by the DSP. Note that this is an offset and iterates from [0,BUF_SIZE - 1]
  3928. * parameters - (all output)
  3929. * read_index - offset
  3930. * wall_clk_msw1 - ADSP wallclock msw
  3931. * wall_clk_lsw1 - ADSP wallclock lsw
  3932. * returns 0 if successful, -EAGAIN if DSP failed to update after some
  3933. * retries
  3934. */
  3935. int q6asm_get_shared_pos(struct audio_client *ac, uint32_t *read_index,
  3936. uint32_t *wall_clk_msw1, uint32_t *wall_clk_lsw1)
  3937. {
  3938. struct asm_shared_position_buffer *pos_buf;
  3939. uint32_t frame_cnt1, frame_cnt2;
  3940. int i, j;
  3941. if (!ac) {
  3942. pr_err("%s: audio client is null\n", __func__);
  3943. return -EINVAL;
  3944. }
  3945. pos_buf = ac->shared_pos_buf.data;
  3946. /* always try to get the latest update in the shared pos buffer */
  3947. for (i = 0; i < 2; i++) {
  3948. /* retry until there is an update from DSP */
  3949. for (j = 0; j < 5; j++) {
  3950. frame_cnt1 = pos_buf->frame_counter;
  3951. if (frame_cnt1 != 0)
  3952. break;
  3953. }
  3954. *wall_clk_msw1 = pos_buf->wall_clock_us_msw;
  3955. *wall_clk_lsw1 = pos_buf->wall_clock_us_lsw;
  3956. *read_index = pos_buf->index;
  3957. frame_cnt2 = pos_buf->frame_counter;
  3958. if (frame_cnt1 != frame_cnt2)
  3959. continue;
  3960. return 0;
  3961. }
  3962. pr_err("%s out of tries trying to get a good read, try again\n",
  3963. __func__);
  3964. return -EAGAIN;
  3965. }
  3966. EXPORT_SYMBOL(q6asm_get_shared_pos);
  3967. /**
  3968. * q6asm_run -
  3969. * command to set ASM to run state
  3970. *
  3971. * @ac: Audio client handle
  3972. * @flags: Flags for session
  3973. * @msw_ts: upper 32bits timestamp
  3974. * @lsw_ts: lower 32bits timestamp
  3975. *
  3976. * Returns 0 on success or error on failure
  3977. */
  3978. int q6asm_run(struct audio_client *ac, uint32_t flags,
  3979. uint32_t msw_ts, uint32_t lsw_ts)
  3980. {
  3981. struct asm_session_cmd_run_v2 run;
  3982. int rc;
  3983. if (ac == NULL) {
  3984. pr_err("%s: APR handle NULL\n", __func__);
  3985. return -EINVAL;
  3986. }
  3987. if (ac->apr == NULL) {
  3988. pr_err("%s: AC APR handle NULL\n", __func__);
  3989. return -EINVAL;
  3990. }
  3991. pr_debug("%s: session[%d]\n", __func__, ac->session);
  3992. q6asm_add_hdr(ac, &run.hdr, sizeof(run), TRUE);
  3993. atomic_set(&ac->cmd_state, -1);
  3994. run.hdr.opcode = ASM_SESSION_CMD_RUN_V2;
  3995. run.flags = flags;
  3996. run.time_lsw = lsw_ts;
  3997. run.time_msw = msw_ts;
  3998. config_debug_fs_run();
  3999. rc = apr_send_pkt(ac->apr, (uint32_t *) &run);
  4000. if (rc < 0) {
  4001. pr_err("%s: Commmand run failed[%d]",
  4002. __func__, rc);
  4003. rc = -EINVAL;
  4004. goto fail_cmd;
  4005. }
  4006. rc = wait_event_timeout(ac->cmd_wait,
  4007. (atomic_read(&ac->cmd_state) >= 0),
  4008. msecs_to_jiffies(TIMEOUT_MS));
  4009. if (!rc) {
  4010. pr_err("%s: timeout. waited for run success",
  4011. __func__);
  4012. rc = -ETIMEDOUT;
  4013. goto fail_cmd;
  4014. }
  4015. if (atomic_read(&ac->cmd_state) > 0) {
  4016. pr_err("%s: DSP returned error[%s]\n",
  4017. __func__, adsp_err_get_err_str(
  4018. atomic_read(&ac->cmd_state)));
  4019. rc = adsp_err_get_lnx_err_code(
  4020. atomic_read(&ac->cmd_state));
  4021. goto fail_cmd;
  4022. }
  4023. return 0;
  4024. fail_cmd:
  4025. return rc;
  4026. }
  4027. EXPORT_SYMBOL(q6asm_run);
  4028. static int __q6asm_run_nowait(struct audio_client *ac, uint32_t flags,
  4029. uint32_t msw_ts, uint32_t lsw_ts, uint32_t stream_id)
  4030. {
  4031. struct asm_session_cmd_run_v2 run;
  4032. int rc;
  4033. if (ac == NULL) {
  4034. pr_err("%s: APR handle NULL\n", __func__);
  4035. return -EINVAL;
  4036. }
  4037. if (ac->apr == NULL) {
  4038. pr_err("%s: AC APR handle NULL\n", __func__);
  4039. return -EINVAL;
  4040. }
  4041. pr_debug("%s: session[%d]\n", __func__, ac->session);
  4042. q6asm_stream_add_hdr_async(ac, &run.hdr, sizeof(run), TRUE, stream_id);
  4043. atomic_set(&ac->cmd_state, 1);
  4044. run.hdr.opcode = ASM_SESSION_CMD_RUN_V2;
  4045. run.flags = flags;
  4046. run.time_lsw = lsw_ts;
  4047. run.time_msw = msw_ts;
  4048. rc = apr_send_pkt(ac->apr, (uint32_t *) &run);
  4049. if (rc < 0) {
  4050. pr_err("%s: Commmand run failed[%d]", __func__, rc);
  4051. return -EINVAL;
  4052. }
  4053. return 0;
  4054. }
  4055. /**
  4056. * q6asm_run_nowait -
  4057. * command to set ASM to run state with no wait for ack
  4058. *
  4059. * @ac: Audio client handle
  4060. * @flags: Flags for session
  4061. * @msw_ts: upper 32bits timestamp
  4062. * @lsw_ts: lower 32bits timestamp
  4063. *
  4064. * Returns 0 on success or error on failure
  4065. */
  4066. int q6asm_run_nowait(struct audio_client *ac, uint32_t flags,
  4067. uint32_t msw_ts, uint32_t lsw_ts)
  4068. {
  4069. return __q6asm_run_nowait(ac, flags, msw_ts, lsw_ts, ac->stream_id);
  4070. }
  4071. EXPORT_SYMBOL(q6asm_run_nowait);
  4072. int q6asm_stream_run_nowait(struct audio_client *ac, uint32_t flags,
  4073. uint32_t msw_ts, uint32_t lsw_ts, uint32_t stream_id)
  4074. {
  4075. return __q6asm_run_nowait(ac, flags, msw_ts, lsw_ts, stream_id);
  4076. }
  4077. /**
  4078. * q6asm_enc_cfg_blk_custom -
  4079. * command to set encode cfg block for custom
  4080. *
  4081. * @ac: Audio client handle
  4082. * @sample_rate: Sample rate
  4083. * @channels: number of ASM channels
  4084. * @format: custom format flag
  4085. * @cfg: generic encoder config
  4086. *
  4087. * Returns 0 on success or error on failure
  4088. */
  4089. int q6asm_enc_cfg_blk_custom(struct audio_client *ac,
  4090. uint32_t sample_rate, uint32_t channels,
  4091. uint32_t format, void *cfg)
  4092. {
  4093. struct asm_custom_enc_cfg_t_v2 enc_cfg;
  4094. int rc = 0;
  4095. uint32_t custom_size;
  4096. struct snd_enc_generic *enc_generic = (struct snd_enc_generic *) cfg;
  4097. custom_size = enc_generic->reserved[1];
  4098. pr_debug("%s: session[%d] size[%d] res[2]=[%d] res[3]=[%d]\n",
  4099. __func__, ac->session, custom_size, enc_generic->reserved[2],
  4100. enc_generic->reserved[3]);
  4101. pr_debug("%s: res[4]=[%d] sr[%d] ch[%d] format[%d]\n",
  4102. __func__, enc_generic->reserved[4], sample_rate,
  4103. channels, format);
  4104. memset(&enc_cfg, 0, sizeof(struct asm_custom_enc_cfg_t_v2));
  4105. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4106. atomic_set(&ac->cmd_state, -1);
  4107. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4108. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4109. enc_cfg.encdec.param_size = sizeof(struct asm_custom_enc_cfg_t_v2) -
  4110. sizeof(struct asm_stream_cmd_set_encdec_param);
  4111. enc_cfg.encblk.frames_per_buf = ENC_FRAMES_PER_BUFFER;
  4112. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4113. sizeof(struct asm_enc_cfg_blk_param_v2);
  4114. enc_cfg.num_channels = channels;
  4115. enc_cfg.sample_rate = sample_rate;
  4116. if (q6asm_map_channels(enc_cfg.channel_mapping, channels, false)) {
  4117. pr_err("%s: map channels failed %d\n",
  4118. __func__, channels);
  4119. rc = -EINVAL;
  4120. goto fail_cmd;
  4121. }
  4122. if (format == FORMAT_BESPOKE && custom_size &&
  4123. custom_size <= sizeof(enc_cfg.custom_data)) {
  4124. memcpy(enc_cfg.custom_data, &enc_generic->reserved[2],
  4125. custom_size);
  4126. enc_cfg.custom_size = custom_size;
  4127. }
  4128. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4129. if (rc < 0) {
  4130. pr_err("%s: Comamnd %d failed %d\n",
  4131. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  4132. rc = -EINVAL;
  4133. goto fail_cmd;
  4134. }
  4135. rc = wait_event_timeout(ac->cmd_wait,
  4136. (atomic_read(&ac->cmd_state) >= 0),
  4137. msecs_to_jiffies(TIMEOUT_MS));
  4138. if (!rc) {
  4139. pr_err("%s: timeout. waited for FORMAT_UPDATE\n",
  4140. __func__);
  4141. rc = -ETIMEDOUT;
  4142. goto fail_cmd;
  4143. }
  4144. if (atomic_read(&ac->cmd_state) > 0) {
  4145. pr_err("%s: DSP returned error[%s]\n",
  4146. __func__, adsp_err_get_err_str(
  4147. atomic_read(&ac->cmd_state)));
  4148. rc = adsp_err_get_lnx_err_code(
  4149. atomic_read(&ac->cmd_state));
  4150. goto fail_cmd;
  4151. }
  4152. return 0;
  4153. fail_cmd:
  4154. return rc;
  4155. }
  4156. EXPORT_SYMBOL(q6asm_enc_cfg_blk_custom);
  4157. /**
  4158. * q6asm_enc_cfg_blk_aac -
  4159. * command to set encode cfg block for aac
  4160. *
  4161. * @ac: Audio client handle
  4162. * @frames_per_buf: number of frames per buffer
  4163. * @sample_rate: Sample rate
  4164. * @channels: number of ASM channels
  4165. * @bit_rate: Bit rate info
  4166. * @mode: mode of AAC stream encode
  4167. * @format: aac format flag
  4168. *
  4169. * Returns 0 on success or error on failure
  4170. */
  4171. int q6asm_enc_cfg_blk_aac(struct audio_client *ac,
  4172. uint32_t frames_per_buf,
  4173. uint32_t sample_rate, uint32_t channels,
  4174. uint32_t bit_rate, uint32_t mode, uint32_t format)
  4175. {
  4176. struct asm_aac_enc_cfg_v2 enc_cfg;
  4177. int rc = 0;
  4178. pr_debug("%s: session[%d]frames[%d]SR[%d]ch[%d]bitrate[%d]mode[%d] format[%d]\n",
  4179. __func__, ac->session, frames_per_buf,
  4180. sample_rate, channels, bit_rate, mode, format);
  4181. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4182. atomic_set(&ac->cmd_state, -1);
  4183. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4184. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4185. enc_cfg.encdec.param_size = sizeof(struct asm_aac_enc_cfg_v2) -
  4186. sizeof(struct asm_stream_cmd_set_encdec_param);
  4187. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4188. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4189. sizeof(struct asm_enc_cfg_blk_param_v2);
  4190. enc_cfg.bit_rate = bit_rate;
  4191. enc_cfg.enc_mode = mode;
  4192. enc_cfg.aac_fmt_flag = format;
  4193. enc_cfg.channel_cfg = channels;
  4194. enc_cfg.sample_rate = sample_rate;
  4195. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4196. if (rc < 0) {
  4197. pr_err("%s: Comamnd %d failed %d\n",
  4198. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  4199. rc = -EINVAL;
  4200. goto fail_cmd;
  4201. }
  4202. rc = wait_event_timeout(ac->cmd_wait,
  4203. (atomic_read(&ac->cmd_state) >= 0),
  4204. msecs_to_jiffies(TIMEOUT_MS));
  4205. if (!rc) {
  4206. pr_err("%s: timeout. waited for FORMAT_UPDATE\n",
  4207. __func__);
  4208. rc = -ETIMEDOUT;
  4209. goto fail_cmd;
  4210. }
  4211. if (atomic_read(&ac->cmd_state) > 0) {
  4212. pr_err("%s: DSP returned error[%s]\n",
  4213. __func__, adsp_err_get_err_str(
  4214. atomic_read(&ac->cmd_state)));
  4215. rc = adsp_err_get_lnx_err_code(
  4216. atomic_read(&ac->cmd_state));
  4217. goto fail_cmd;
  4218. }
  4219. return 0;
  4220. fail_cmd:
  4221. return rc;
  4222. }
  4223. EXPORT_SYMBOL(q6asm_enc_cfg_blk_aac);
  4224. /**
  4225. * q6asm_enc_cfg_blk_g711 -
  4226. * command to set encode cfg block for g711
  4227. *
  4228. * @ac: Audio client handle
  4229. * @frames_per_buf: number of frames per buffer
  4230. * @sample_rate: Sample rate
  4231. *
  4232. * Returns 0 on success or error on failure
  4233. */
  4234. int q6asm_enc_cfg_blk_g711(struct audio_client *ac,
  4235. uint32_t frames_per_buf,
  4236. uint32_t sample_rate)
  4237. {
  4238. struct asm_g711_enc_cfg_v2 enc_cfg;
  4239. int rc = 0;
  4240. pr_debug("%s: session[%d]frames[%d]SR[%d]\n",
  4241. __func__, ac->session, frames_per_buf,
  4242. sample_rate);
  4243. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4244. atomic_set(&ac->cmd_state, -1);
  4245. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4246. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4247. enc_cfg.encdec.param_size = sizeof(struct asm_g711_enc_cfg_v2) -
  4248. sizeof(struct asm_stream_cmd_set_encdec_param);
  4249. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4250. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4251. sizeof(struct asm_enc_cfg_blk_param_v2);
  4252. enc_cfg.sample_rate = sample_rate;
  4253. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4254. if (rc < 0) {
  4255. pr_err("%s: Comamnd %d failed %d\n",
  4256. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  4257. rc = -EINVAL;
  4258. goto fail_cmd;
  4259. }
  4260. rc = wait_event_timeout(ac->cmd_wait,
  4261. (atomic_read(&ac->cmd_state) >= 0),
  4262. msecs_to_jiffies(TIMEOUT_MS));
  4263. if (!rc) {
  4264. pr_err("%s: timeout. waited for FORMAT_UPDATE\n",
  4265. __func__);
  4266. rc = -ETIMEDOUT;
  4267. goto fail_cmd;
  4268. }
  4269. if (atomic_read(&ac->cmd_state) > 0) {
  4270. pr_err("%s: DSP returned error[%s]\n",
  4271. __func__, adsp_err_get_err_str(
  4272. atomic_read(&ac->cmd_state)));
  4273. rc = adsp_err_get_lnx_err_code(
  4274. atomic_read(&ac->cmd_state));
  4275. goto fail_cmd;
  4276. }
  4277. return 0;
  4278. fail_cmd:
  4279. return rc;
  4280. }
  4281. EXPORT_SYMBOL(q6asm_enc_cfg_blk_g711);
  4282. /**
  4283. * q6asm_set_encdec_chan_map -
  4284. * command to set encdec channel map
  4285. *
  4286. * @ac: Audio client handle
  4287. * @channels: number of channels
  4288. *
  4289. * Returns 0 on success or error on failure
  4290. */
  4291. int q6asm_set_encdec_chan_map(struct audio_client *ac,
  4292. uint32_t num_channels)
  4293. {
  4294. struct asm_dec_out_chan_map_param chan_map;
  4295. u8 *channel_mapping;
  4296. int rc = 0;
  4297. pr_debug("%s: Session %d, num_channels = %d\n",
  4298. __func__, ac->session, num_channels);
  4299. q6asm_add_hdr(ac, &chan_map.hdr, sizeof(chan_map), TRUE);
  4300. atomic_set(&ac->cmd_state, -1);
  4301. chan_map.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4302. chan_map.encdec.param_id = ASM_PARAM_ID_DEC_OUTPUT_CHAN_MAP;
  4303. chan_map.encdec.param_size = sizeof(struct asm_dec_out_chan_map_param) -
  4304. (sizeof(struct apr_hdr) +
  4305. sizeof(struct asm_stream_cmd_set_encdec_param));
  4306. chan_map.num_channels = num_channels;
  4307. channel_mapping = chan_map.channel_mapping;
  4308. memset(channel_mapping, PCM_CHANNEL_NULL, MAX_CHAN_MAP_CHANNELS);
  4309. if (q6asm_map_channels(channel_mapping, num_channels, false)) {
  4310. pr_err("%s: map channels failed %d\n", __func__, num_channels);
  4311. return -EINVAL;
  4312. }
  4313. rc = apr_send_pkt(ac->apr, (uint32_t *) &chan_map);
  4314. if (rc < 0) {
  4315. pr_err("%s: Command opcode[0x%x]paramid[0x%x] failed %d\n",
  4316. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM,
  4317. ASM_PARAM_ID_DEC_OUTPUT_CHAN_MAP, rc);
  4318. goto fail_cmd;
  4319. }
  4320. rc = wait_event_timeout(ac->cmd_wait,
  4321. (atomic_read(&ac->cmd_state) >= 0),
  4322. msecs_to_jiffies(TIMEOUT_MS));
  4323. if (!rc) {
  4324. pr_err("%s: timeout opcode[0x%x]\n", __func__,
  4325. chan_map.hdr.opcode);
  4326. rc = -ETIMEDOUT;
  4327. goto fail_cmd;
  4328. }
  4329. if (atomic_read(&ac->cmd_state) > 0) {
  4330. pr_err("%s: DSP returned error[%s]\n",
  4331. __func__, adsp_err_get_err_str(
  4332. atomic_read(&ac->cmd_state)));
  4333. rc = adsp_err_get_lnx_err_code(
  4334. atomic_read(&ac->cmd_state));
  4335. goto fail_cmd;
  4336. }
  4337. return 0;
  4338. fail_cmd:
  4339. return rc;
  4340. }
  4341. EXPORT_SYMBOL(q6asm_set_encdec_chan_map);
  4342. /*
  4343. * q6asm_enc_cfg_blk_pcm_v5 - sends encoder configuration parameters
  4344. *
  4345. * @ac: Client session handle
  4346. * @rate: sample rate
  4347. * @channels: number of channels
  4348. * @bits_per_sample: bit width of encoder session
  4349. * @use_default_chmap: true if default channel map to be used
  4350. * @use_back_flavor: to configure back left and right channel
  4351. * @channel_map: input channel map
  4352. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4353. * @endianness: endianness of the pcm data
  4354. * @mode: Mode to provide additional info about the pcm input data
  4355. */
  4356. static int q6asm_enc_cfg_blk_pcm_v5(struct audio_client *ac,
  4357. uint32_t rate, uint32_t channels,
  4358. uint16_t bits_per_sample, bool use_default_chmap,
  4359. bool use_back_flavor, u8 *channel_map,
  4360. uint16_t sample_word_size, uint16_t endianness,
  4361. uint16_t mode)
  4362. {
  4363. struct asm_multi_channel_pcm_enc_cfg_v5 enc_cfg;
  4364. struct asm_enc_cfg_blk_param_v2 enc_fg_blk;
  4365. u8 *channel_mapping;
  4366. u32 frames_per_buf = 0;
  4367. int rc;
  4368. if (!use_default_chmap && (channel_map == NULL)) {
  4369. pr_err("%s: No valid chan map and can't use default\n",
  4370. __func__);
  4371. rc = -EINVAL;
  4372. goto fail_cmd;
  4373. }
  4374. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  4375. ac->session, rate, channels,
  4376. bits_per_sample, sample_word_size);
  4377. memset(&enc_cfg, 0, sizeof(enc_cfg));
  4378. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4379. atomic_set(&ac->cmd_state, -1);
  4380. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4381. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4382. enc_cfg.encdec.param_size = sizeof(enc_cfg) - sizeof(enc_cfg.hdr) -
  4383. sizeof(enc_cfg.encdec);
  4384. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4385. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4386. sizeof(enc_fg_blk);
  4387. enc_cfg.num_channels = channels;
  4388. enc_cfg.bits_per_sample = bits_per_sample;
  4389. enc_cfg.sample_rate = rate;
  4390. enc_cfg.is_signed = 1;
  4391. enc_cfg.sample_word_size = sample_word_size;
  4392. enc_cfg.endianness = endianness;
  4393. enc_cfg.mode = mode;
  4394. channel_mapping = enc_cfg.channel_mapping;
  4395. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  4396. if (use_default_chmap) {
  4397. pr_debug("%s: setting default channel map for %d channels",
  4398. __func__, channels);
  4399. if (q6asm_map_channels(channel_mapping, channels,
  4400. use_back_flavor)) {
  4401. pr_err("%s: map channels failed %d\n",
  4402. __func__, channels);
  4403. rc = -EINVAL;
  4404. goto fail_cmd;
  4405. }
  4406. } else {
  4407. pr_debug("%s: Using pre-defined channel map", __func__);
  4408. memcpy(channel_mapping, channel_map,
  4409. PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  4410. }
  4411. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4412. if (rc < 0) {
  4413. pr_err("%s: Command open failed %d\n", __func__, rc);
  4414. goto fail_cmd;
  4415. }
  4416. rc = wait_event_timeout(ac->cmd_wait,
  4417. (atomic_read(&ac->cmd_state) >= 0), 5*HZ);
  4418. if (!rc) {
  4419. pr_err("%s: timeout opcode[0x%x]\n",
  4420. __func__, enc_cfg.hdr.opcode);
  4421. rc = -ETIMEDOUT;
  4422. goto fail_cmd;
  4423. }
  4424. if (atomic_read(&ac->cmd_state) > 0) {
  4425. pr_err("%s: DSP returned error[%s]\n",
  4426. __func__, adsp_err_get_err_str(
  4427. atomic_read(&ac->cmd_state)));
  4428. rc = adsp_err_get_lnx_err_code(
  4429. atomic_read(&ac->cmd_state));
  4430. goto fail_cmd;
  4431. }
  4432. return 0;
  4433. fail_cmd:
  4434. return rc;
  4435. }
  4436. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_v5);
  4437. /*
  4438. * q6asm_enc_cfg_blk_pcm_v4 - sends encoder configuration parameters
  4439. *
  4440. * @ac: Client session handle
  4441. * @rate: sample rate
  4442. * @channels: number of channels
  4443. * @bits_per_sample: bit width of encoder session
  4444. * @use_default_chmap: true if default channel map to be used
  4445. * @use_back_flavor: to configure back left and right channel
  4446. * @channel_map: input channel map
  4447. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4448. * @endianness: endianness of the pcm data
  4449. * @mode: Mode to provide additional info about the pcm input data
  4450. */
  4451. int q6asm_enc_cfg_blk_pcm_v4(struct audio_client *ac,
  4452. uint32_t rate, uint32_t channels,
  4453. uint16_t bits_per_sample, bool use_default_chmap,
  4454. bool use_back_flavor, u8 *channel_map,
  4455. uint16_t sample_word_size, uint16_t endianness,
  4456. uint16_t mode)
  4457. {
  4458. struct asm_multi_channel_pcm_enc_cfg_v4 enc_cfg;
  4459. struct asm_enc_cfg_blk_param_v2 enc_fg_blk;
  4460. u8 *channel_mapping;
  4461. u32 frames_per_buf = 0;
  4462. int rc;
  4463. if (!use_default_chmap && (channel_map == NULL)) {
  4464. pr_err("%s: No valid chan map and can't use default\n",
  4465. __func__);
  4466. rc = -EINVAL;
  4467. goto fail_cmd;
  4468. }
  4469. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  4470. ac->session, rate, channels,
  4471. bits_per_sample, sample_word_size);
  4472. memset(&enc_cfg, 0, sizeof(enc_cfg));
  4473. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4474. atomic_set(&ac->cmd_state, -1);
  4475. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4476. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4477. enc_cfg.encdec.param_size = sizeof(enc_cfg) - sizeof(enc_cfg.hdr) -
  4478. sizeof(enc_cfg.encdec);
  4479. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4480. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4481. sizeof(enc_fg_blk);
  4482. enc_cfg.num_channels = channels;
  4483. enc_cfg.bits_per_sample = bits_per_sample;
  4484. enc_cfg.sample_rate = rate;
  4485. enc_cfg.is_signed = 1;
  4486. enc_cfg.sample_word_size = sample_word_size;
  4487. enc_cfg.endianness = endianness;
  4488. enc_cfg.mode = mode;
  4489. channel_mapping = enc_cfg.channel_mapping;
  4490. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  4491. if (use_default_chmap) {
  4492. pr_debug("%s: setting default channel map for %d channels",
  4493. __func__, channels);
  4494. if (q6asm_map_channels(channel_mapping, channels,
  4495. use_back_flavor)) {
  4496. pr_err("%s: map channels failed %d\n",
  4497. __func__, channels);
  4498. rc = -EINVAL;
  4499. goto fail_cmd;
  4500. }
  4501. } else {
  4502. pr_debug("%s: Using pre-defined channel map", __func__);
  4503. memcpy(channel_mapping, channel_map,
  4504. PCM_FORMAT_MAX_NUM_CHANNEL);
  4505. }
  4506. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4507. if (rc < 0) {
  4508. pr_err("%s: Command open failed %d\n", __func__, rc);
  4509. goto fail_cmd;
  4510. }
  4511. rc = wait_event_timeout(ac->cmd_wait,
  4512. (atomic_read(&ac->cmd_state) >= 0),
  4513. msecs_to_jiffies(TIMEOUT_MS));
  4514. if (!rc) {
  4515. pr_err("%s: timeout opcode[0x%x]\n",
  4516. __func__, enc_cfg.hdr.opcode);
  4517. rc = -ETIMEDOUT;
  4518. goto fail_cmd;
  4519. }
  4520. if (atomic_read(&ac->cmd_state) > 0) {
  4521. pr_err("%s: DSP returned error[%s]\n",
  4522. __func__, adsp_err_get_err_str(
  4523. atomic_read(&ac->cmd_state)));
  4524. rc = adsp_err_get_lnx_err_code(
  4525. atomic_read(&ac->cmd_state));
  4526. goto fail_cmd;
  4527. }
  4528. return 0;
  4529. fail_cmd:
  4530. return rc;
  4531. }
  4532. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_v4);
  4533. /*
  4534. * q6asm_enc_cfg_blk_pcm_v3 - sends encoder configuration parameters
  4535. *
  4536. * @ac: Client session handle
  4537. * @rate: sample rate
  4538. * @channels: number of channels
  4539. * @bits_per_sample: bit width of encoder session
  4540. * @use_default_chmap: true if default channel map to be used
  4541. * @use_back_flavor: to configure back left and right channel
  4542. * @channel_map: input channel map
  4543. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4544. */
  4545. int q6asm_enc_cfg_blk_pcm_v3(struct audio_client *ac,
  4546. uint32_t rate, uint32_t channels,
  4547. uint16_t bits_per_sample, bool use_default_chmap,
  4548. bool use_back_flavor, u8 *channel_map,
  4549. uint16_t sample_word_size)
  4550. {
  4551. struct asm_multi_channel_pcm_enc_cfg_v3 enc_cfg;
  4552. struct asm_enc_cfg_blk_param_v2 enc_fg_blk;
  4553. u8 *channel_mapping;
  4554. u32 frames_per_buf = 0;
  4555. int rc;
  4556. if (!use_default_chmap && (channel_map == NULL)) {
  4557. pr_err("%s: No valid chan map and can't use default\n",
  4558. __func__);
  4559. rc = -EINVAL;
  4560. goto fail_cmd;
  4561. }
  4562. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  4563. ac->session, rate, channels,
  4564. bits_per_sample, sample_word_size);
  4565. memset(&enc_cfg, 0, sizeof(enc_cfg));
  4566. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4567. atomic_set(&ac->cmd_state, -1);
  4568. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4569. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4570. enc_cfg.encdec.param_size = sizeof(enc_cfg) - sizeof(enc_cfg.hdr) -
  4571. sizeof(enc_cfg.encdec);
  4572. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4573. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4574. sizeof(enc_fg_blk);
  4575. enc_cfg.num_channels = channels;
  4576. enc_cfg.bits_per_sample = bits_per_sample;
  4577. enc_cfg.sample_rate = rate;
  4578. enc_cfg.is_signed = 1;
  4579. enc_cfg.sample_word_size = sample_word_size;
  4580. channel_mapping = enc_cfg.channel_mapping;
  4581. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  4582. if (use_default_chmap) {
  4583. pr_debug("%s: setting default channel map for %d channels",
  4584. __func__, channels);
  4585. if (q6asm_map_channels(channel_mapping, channels,
  4586. use_back_flavor)) {
  4587. pr_err("%s: map channels failed %d\n",
  4588. __func__, channels);
  4589. rc = -EINVAL;
  4590. goto fail_cmd;
  4591. }
  4592. } else {
  4593. pr_debug("%s: Using pre-defined channel map", __func__);
  4594. memcpy(channel_mapping, channel_map,
  4595. PCM_FORMAT_MAX_NUM_CHANNEL);
  4596. }
  4597. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4598. if (rc < 0) {
  4599. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  4600. goto fail_cmd;
  4601. }
  4602. rc = wait_event_timeout(ac->cmd_wait,
  4603. (atomic_read(&ac->cmd_state) >= 0),
  4604. msecs_to_jiffies(TIMEOUT_MS));
  4605. if (!rc) {
  4606. pr_err("%s: timeout opcode[0x%x]\n",
  4607. __func__, enc_cfg.hdr.opcode);
  4608. rc = -ETIMEDOUT;
  4609. goto fail_cmd;
  4610. }
  4611. if (atomic_read(&ac->cmd_state) > 0) {
  4612. pr_err("%s: DSP returned error[%s]\n",
  4613. __func__, adsp_err_get_err_str(
  4614. atomic_read(&ac->cmd_state)));
  4615. rc = adsp_err_get_lnx_err_code(
  4616. atomic_read(&ac->cmd_state));
  4617. goto fail_cmd;
  4618. }
  4619. return 0;
  4620. fail_cmd:
  4621. return rc;
  4622. }
  4623. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_v3);
  4624. /**
  4625. * q6asm_enc_cfg_blk_pcm_v2 -
  4626. * command to set encode config block for pcm_v2
  4627. *
  4628. * @ac: Audio client handle
  4629. * @rate: sample rate
  4630. * @channels: number of channels
  4631. * @bits_per_sample: number of bits per sample
  4632. * @use_default_chmap: Flag indicating to use default ch_map or not
  4633. * @use_back_flavor: back flavor flag
  4634. * @channel_map: Custom channel map settings
  4635. *
  4636. * Returns 0 on success or error on failure
  4637. */
  4638. int q6asm_enc_cfg_blk_pcm_v2(struct audio_client *ac,
  4639. uint32_t rate, uint32_t channels, uint16_t bits_per_sample,
  4640. bool use_default_chmap, bool use_back_flavor, u8 *channel_map)
  4641. {
  4642. struct asm_multi_channel_pcm_enc_cfg_v2 enc_cfg;
  4643. u8 *channel_mapping;
  4644. u32 frames_per_buf = 0;
  4645. int rc = 0;
  4646. if (!use_default_chmap && (channel_map == NULL)) {
  4647. pr_err("%s: No valid chan map and can't use default\n",
  4648. __func__);
  4649. return -EINVAL;
  4650. }
  4651. pr_debug("%s: Session %d, rate = %d, channels = %d\n", __func__,
  4652. ac->session, rate, channels);
  4653. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4654. atomic_set(&ac->cmd_state, -1);
  4655. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4656. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4657. enc_cfg.encdec.param_size = sizeof(enc_cfg) - sizeof(enc_cfg.hdr) -
  4658. sizeof(enc_cfg.encdec);
  4659. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4660. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4661. sizeof(struct asm_enc_cfg_blk_param_v2);
  4662. enc_cfg.num_channels = channels;
  4663. enc_cfg.bits_per_sample = bits_per_sample;
  4664. enc_cfg.sample_rate = rate;
  4665. enc_cfg.is_signed = 1;
  4666. channel_mapping = enc_cfg.channel_mapping;
  4667. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  4668. if (use_default_chmap) {
  4669. pr_debug("%s: setting default channel map for %d channels",
  4670. __func__, channels);
  4671. if (q6asm_map_channels(channel_mapping, channels,
  4672. use_back_flavor)) {
  4673. pr_err("%s: map channels failed %d\n",
  4674. __func__, channels);
  4675. return -EINVAL;
  4676. }
  4677. } else {
  4678. pr_debug("%s: Using pre-defined channel map", __func__);
  4679. memcpy(channel_mapping, channel_map,
  4680. PCM_FORMAT_MAX_NUM_CHANNEL);
  4681. }
  4682. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4683. if (rc < 0) {
  4684. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  4685. rc = -EINVAL;
  4686. goto fail_cmd;
  4687. }
  4688. rc = wait_event_timeout(ac->cmd_wait,
  4689. (atomic_read(&ac->cmd_state) >= 0),
  4690. msecs_to_jiffies(TIMEOUT_MS));
  4691. if (!rc) {
  4692. pr_err("%s: timeout opcode[0x%x]\n",
  4693. __func__, enc_cfg.hdr.opcode);
  4694. rc = -ETIMEDOUT;
  4695. goto fail_cmd;
  4696. }
  4697. if (atomic_read(&ac->cmd_state) > 0) {
  4698. pr_err("%s: DSP returned error[%s]\n",
  4699. __func__, adsp_err_get_err_str(
  4700. atomic_read(&ac->cmd_state)));
  4701. rc = adsp_err_get_lnx_err_code(
  4702. atomic_read(&ac->cmd_state));
  4703. goto fail_cmd;
  4704. }
  4705. return 0;
  4706. fail_cmd:
  4707. return rc;
  4708. }
  4709. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_v2);
  4710. static int __q6asm_enc_cfg_blk_pcm_v5(struct audio_client *ac,
  4711. uint32_t rate, uint32_t channels,
  4712. uint16_t bits_per_sample,
  4713. uint16_t sample_word_size,
  4714. uint16_t endianness,
  4715. uint16_t mode)
  4716. {
  4717. return q6asm_enc_cfg_blk_pcm_v5(ac, rate, channels,
  4718. bits_per_sample, true, false, NULL,
  4719. sample_word_size, endianness, mode);
  4720. }
  4721. static int __q6asm_enc_cfg_blk_pcm_v4(struct audio_client *ac,
  4722. uint32_t rate, uint32_t channels,
  4723. uint16_t bits_per_sample,
  4724. uint16_t sample_word_size,
  4725. uint16_t endianness,
  4726. uint16_t mode)
  4727. {
  4728. return q6asm_enc_cfg_blk_pcm_v4(ac, rate, channels,
  4729. bits_per_sample, true, false, NULL,
  4730. sample_word_size, endianness, mode);
  4731. }
  4732. static int __q6asm_enc_cfg_blk_pcm_v3(struct audio_client *ac,
  4733. uint32_t rate, uint32_t channels,
  4734. uint16_t bits_per_sample,
  4735. uint16_t sample_word_size)
  4736. {
  4737. return q6asm_enc_cfg_blk_pcm_v3(ac, rate, channels,
  4738. bits_per_sample, true, false, NULL,
  4739. sample_word_size);
  4740. }
  4741. static int __q6asm_enc_cfg_blk_pcm(struct audio_client *ac,
  4742. uint32_t rate, uint32_t channels, uint16_t bits_per_sample)
  4743. {
  4744. return q6asm_enc_cfg_blk_pcm_v2(ac, rate, channels,
  4745. bits_per_sample, true, false, NULL);
  4746. }
  4747. /**
  4748. * q6asm_enc_cfg_blk_pcm -
  4749. * command to set encode config block for pcm
  4750. *
  4751. * @ac: Audio client handle
  4752. * @rate: sample rate
  4753. * @channels: number of channels
  4754. *
  4755. * Returns 0 on success or error on failure
  4756. */
  4757. int q6asm_enc_cfg_blk_pcm(struct audio_client *ac,
  4758. uint32_t rate, uint32_t channels)
  4759. {
  4760. return __q6asm_enc_cfg_blk_pcm(ac, rate, channels, 16);
  4761. }
  4762. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm);
  4763. int q6asm_enc_cfg_blk_pcm_format_support(struct audio_client *ac,
  4764. uint32_t rate, uint32_t channels, uint16_t bits_per_sample)
  4765. {
  4766. return __q6asm_enc_cfg_blk_pcm(ac, rate, channels, bits_per_sample);
  4767. }
  4768. /*
  4769. * q6asm_enc_cfg_blk_pcm_format_support_v3 - sends encoder configuration
  4770. * parameters
  4771. *
  4772. * @ac: Client session handle
  4773. * @rate: sample rate
  4774. * @channels: number of channels
  4775. * @bits_per_sample: bit width of encoder session
  4776. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4777. */
  4778. int q6asm_enc_cfg_blk_pcm_format_support_v3(struct audio_client *ac,
  4779. uint32_t rate, uint32_t channels,
  4780. uint16_t bits_per_sample,
  4781. uint16_t sample_word_size)
  4782. {
  4783. return __q6asm_enc_cfg_blk_pcm_v3(ac, rate, channels,
  4784. bits_per_sample, sample_word_size);
  4785. }
  4786. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_format_support_v3);
  4787. /*
  4788. * q6asm_enc_cfg_blk_pcm_format_support_v4 - sends encoder configuration
  4789. * parameters
  4790. *
  4791. * @ac: Client session handle
  4792. * @rate: sample rate
  4793. * @channels: number of channels
  4794. * @bits_per_sample: bit width of encoder session
  4795. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4796. * @endianness: endianness of the pcm data
  4797. * @mode: Mode to provide additional info about the pcm input data
  4798. */
  4799. int q6asm_enc_cfg_blk_pcm_format_support_v4(struct audio_client *ac,
  4800. uint32_t rate, uint32_t channels,
  4801. uint16_t bits_per_sample,
  4802. uint16_t sample_word_size,
  4803. uint16_t endianness,
  4804. uint16_t mode)
  4805. {
  4806. return __q6asm_enc_cfg_blk_pcm_v4(ac, rate, channels,
  4807. bits_per_sample, sample_word_size,
  4808. endianness, mode);
  4809. }
  4810. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_format_support_v4);
  4811. /*
  4812. * q6asm_enc_cfg_blk_pcm_format_support_v5 - sends encoder configuration
  4813. * parameters
  4814. *
  4815. * @ac: Client session handle
  4816. * @rate: sample rate
  4817. * @channels: number of channels
  4818. * @bits_per_sample: bit width of encoder session
  4819. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  4820. * @endianness: endianness of the pcm data
  4821. * @mode: Mode to provide additional info about the pcm input data
  4822. */
  4823. int q6asm_enc_cfg_blk_pcm_format_support_v5(struct audio_client *ac,
  4824. uint32_t rate, uint32_t channels,
  4825. uint16_t bits_per_sample,
  4826. uint16_t sample_word_size,
  4827. uint16_t endianness,
  4828. uint16_t mode)
  4829. {
  4830. return __q6asm_enc_cfg_blk_pcm_v5(ac, rate, channels,
  4831. bits_per_sample, sample_word_size,
  4832. endianness, mode);
  4833. }
  4834. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_format_support_v5);
  4835. /**
  4836. * q6asm_enc_cfg_blk_pcm_native -
  4837. * command to set encode config block for pcm_native
  4838. *
  4839. * @ac: Audio client handle
  4840. * @rate: sample rate
  4841. * @channels: number of channels
  4842. *
  4843. * Returns 0 on success or error on failure
  4844. */
  4845. int q6asm_enc_cfg_blk_pcm_native(struct audio_client *ac,
  4846. uint32_t rate, uint32_t channels)
  4847. {
  4848. struct asm_multi_channel_pcm_enc_cfg_v2 enc_cfg;
  4849. u8 *channel_mapping;
  4850. u32 frames_per_buf = 0;
  4851. int rc = 0;
  4852. pr_debug("%s: Session %d, rate = %d, channels = %d\n", __func__,
  4853. ac->session, rate, channels);
  4854. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  4855. atomic_set(&ac->cmd_state, -1);
  4856. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  4857. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  4858. enc_cfg.encdec.param_size = sizeof(enc_cfg) - sizeof(enc_cfg.hdr) -
  4859. sizeof(enc_cfg.encdec);
  4860. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  4861. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  4862. sizeof(struct asm_enc_cfg_blk_param_v2);
  4863. enc_cfg.num_channels = 0;/*channels;*/
  4864. enc_cfg.bits_per_sample = 16;
  4865. enc_cfg.sample_rate = 0;/*rate;*/
  4866. enc_cfg.is_signed = 1;
  4867. channel_mapping = enc_cfg.channel_mapping;
  4868. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  4869. if (q6asm_map_channels(channel_mapping, channels, false)) {
  4870. pr_err("%s: map channels failed %d\n", __func__, channels);
  4871. return -EINVAL;
  4872. }
  4873. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  4874. if (rc < 0) {
  4875. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  4876. rc = -EINVAL;
  4877. goto fail_cmd;
  4878. }
  4879. rc = wait_event_timeout(ac->cmd_wait,
  4880. (atomic_read(&ac->cmd_state) >= 0),
  4881. msecs_to_jiffies(TIMEOUT_MS));
  4882. if (!rc) {
  4883. pr_err("%s: timeout opcode[0x%x]\n",
  4884. __func__, enc_cfg.hdr.opcode);
  4885. rc = -ETIMEDOUT;
  4886. goto fail_cmd;
  4887. }
  4888. if (atomic_read(&ac->cmd_state) > 0) {
  4889. pr_err("%s: DSP returned error[%s]\n",
  4890. __func__, adsp_err_get_err_str(
  4891. atomic_read(&ac->cmd_state)));
  4892. rc = adsp_err_get_lnx_err_code(
  4893. atomic_read(&ac->cmd_state));
  4894. goto fail_cmd;
  4895. }
  4896. return 0;
  4897. fail_cmd:
  4898. return rc;
  4899. }
  4900. EXPORT_SYMBOL(q6asm_enc_cfg_blk_pcm_native);
  4901. static int q6asm_map_channels(u8 *channel_mapping, uint32_t channels,
  4902. bool use_back_flavor)
  4903. {
  4904. u8 *lchannel_mapping;
  4905. lchannel_mapping = channel_mapping;
  4906. pr_debug("%s: channels passed: %d\n", __func__, channels);
  4907. if (channels == 1) {
  4908. lchannel_mapping[0] = PCM_CHANNEL_FC;
  4909. } else if (channels == 2) {
  4910. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4911. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4912. } else if (channels == 3) {
  4913. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4914. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4915. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4916. } else if (channels == 4) {
  4917. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4918. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4919. lchannel_mapping[2] = use_back_flavor ?
  4920. PCM_CHANNEL_LB : PCM_CHANNEL_LS;
  4921. lchannel_mapping[3] = use_back_flavor ?
  4922. PCM_CHANNEL_RB : PCM_CHANNEL_RS;
  4923. } else if (channels == 5) {
  4924. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4925. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4926. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4927. lchannel_mapping[3] = use_back_flavor ?
  4928. PCM_CHANNEL_LB : PCM_CHANNEL_LS;
  4929. lchannel_mapping[4] = use_back_flavor ?
  4930. PCM_CHANNEL_RB : PCM_CHANNEL_RS;
  4931. } else if (channels == 6) {
  4932. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4933. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4934. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4935. lchannel_mapping[3] = PCM_CHANNEL_LFE;
  4936. lchannel_mapping[4] = use_back_flavor ?
  4937. PCM_CHANNEL_LB : PCM_CHANNEL_LS;
  4938. lchannel_mapping[5] = use_back_flavor ?
  4939. PCM_CHANNEL_RB : PCM_CHANNEL_RS;
  4940. } else if (channels == 7) {
  4941. /*
  4942. * Configured for 5.1 channel mapping + 1 channel for debug
  4943. * Can be customized based on DSP.
  4944. */
  4945. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4946. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4947. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4948. lchannel_mapping[3] = PCM_CHANNEL_LFE;
  4949. lchannel_mapping[4] = use_back_flavor ?
  4950. PCM_CHANNEL_LB : PCM_CHANNEL_LS;
  4951. lchannel_mapping[5] = use_back_flavor ?
  4952. PCM_CHANNEL_RB : PCM_CHANNEL_RS;
  4953. lchannel_mapping[6] = PCM_CHANNEL_CS;
  4954. } else if (channels == 8) {
  4955. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4956. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4957. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4958. lchannel_mapping[3] = PCM_CHANNEL_LFE;
  4959. lchannel_mapping[4] = PCM_CHANNEL_LB;
  4960. lchannel_mapping[5] = PCM_CHANNEL_RB;
  4961. lchannel_mapping[6] = PCM_CHANNEL_LS;
  4962. lchannel_mapping[7] = PCM_CHANNEL_RS;
  4963. } else if (channels == 12) {
  4964. /*
  4965. * Configured for 7.1.4 channel mapping
  4966. * Todo: Needs to be checked
  4967. */
  4968. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4969. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4970. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4971. lchannel_mapping[3] = PCM_CHANNEL_LFE;
  4972. lchannel_mapping[4] = PCM_CHANNEL_LB;
  4973. lchannel_mapping[5] = PCM_CHANNEL_RB;
  4974. lchannel_mapping[6] = PCM_CHANNEL_LS;
  4975. lchannel_mapping[7] = PCM_CHANNEL_RS;
  4976. lchannel_mapping[8] = PCM_CHANNEL_TFL;
  4977. lchannel_mapping[9] = PCM_CHANNEL_TFR;
  4978. lchannel_mapping[10] = PCM_CHANNEL_TSL;
  4979. lchannel_mapping[11] = PCM_CHANNEL_TSR;
  4980. } else if (channels == 16) {
  4981. /*
  4982. * Configured for 7.1.8 channel mapping
  4983. * Todo: Needs to be checked
  4984. */
  4985. lchannel_mapping[0] = PCM_CHANNEL_FL;
  4986. lchannel_mapping[1] = PCM_CHANNEL_FR;
  4987. lchannel_mapping[2] = PCM_CHANNEL_FC;
  4988. lchannel_mapping[3] = PCM_CHANNEL_LFE;
  4989. lchannel_mapping[4] = PCM_CHANNEL_LB;
  4990. lchannel_mapping[5] = PCM_CHANNEL_RB;
  4991. lchannel_mapping[6] = PCM_CHANNEL_LS;
  4992. lchannel_mapping[7] = PCM_CHANNEL_RS;
  4993. lchannel_mapping[8] = PCM_CHANNEL_TFL;
  4994. lchannel_mapping[9] = PCM_CHANNEL_TFR;
  4995. lchannel_mapping[10] = PCM_CHANNEL_TSL;
  4996. lchannel_mapping[11] = PCM_CHANNEL_TSR;
  4997. lchannel_mapping[12] = PCM_CHANNEL_FLC;
  4998. lchannel_mapping[13] = PCM_CHANNEL_FRC;
  4999. lchannel_mapping[14] = PCM_CHANNEL_RLC;
  5000. lchannel_mapping[15] = PCM_CHANNEL_RRC;
  5001. } else {
  5002. pr_err("%s: ERROR.unsupported num_ch = %u\n",
  5003. __func__, channels);
  5004. return -EINVAL;
  5005. }
  5006. return 0;
  5007. }
  5008. /**
  5009. * q6asm_enable_sbrps -
  5010. * command to enable sbrps for ASM
  5011. *
  5012. * @ac: Audio client handle
  5013. * @sbr_ps_enable: flag for sbr_ps enable or disable
  5014. *
  5015. * Returns 0 on success or error on failure
  5016. */
  5017. int q6asm_enable_sbrps(struct audio_client *ac,
  5018. uint32_t sbr_ps_enable)
  5019. {
  5020. struct asm_aac_sbr_ps_flag_param sbrps;
  5021. u32 frames_per_buf = 0;
  5022. int rc = 0;
  5023. pr_debug("%s: Session %d\n", __func__, ac->session);
  5024. q6asm_add_hdr(ac, &sbrps.hdr, sizeof(sbrps), TRUE);
  5025. atomic_set(&ac->cmd_state, -1);
  5026. sbrps.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5027. sbrps.encdec.param_id = ASM_PARAM_ID_AAC_SBR_PS_FLAG;
  5028. sbrps.encdec.param_size = sizeof(struct asm_aac_sbr_ps_flag_param) -
  5029. sizeof(struct asm_stream_cmd_set_encdec_param);
  5030. sbrps.encblk.frames_per_buf = frames_per_buf;
  5031. sbrps.encblk.enc_cfg_blk_size = sbrps.encdec.param_size -
  5032. sizeof(struct asm_enc_cfg_blk_param_v2);
  5033. sbrps.sbr_ps_flag = sbr_ps_enable;
  5034. rc = apr_send_pkt(ac->apr, (uint32_t *) &sbrps);
  5035. if (rc < 0) {
  5036. pr_err("%s: Command opcode[0x%x]paramid[0x%x] failed %d\n",
  5037. __func__,
  5038. ASM_STREAM_CMD_SET_ENCDEC_PARAM,
  5039. ASM_PARAM_ID_AAC_SBR_PS_FLAG, rc);
  5040. rc = -EINVAL;
  5041. goto fail_cmd;
  5042. }
  5043. rc = wait_event_timeout(ac->cmd_wait,
  5044. (atomic_read(&ac->cmd_state) >= 0),
  5045. msecs_to_jiffies(TIMEOUT_MS));
  5046. if (!rc) {
  5047. pr_err("%s: timeout opcode[0x%x] ", __func__, sbrps.hdr.opcode);
  5048. rc = -ETIMEDOUT;
  5049. goto fail_cmd;
  5050. }
  5051. if (atomic_read(&ac->cmd_state) > 0) {
  5052. pr_err("%s: DSP returned error[%s]\n",
  5053. __func__, adsp_err_get_err_str(
  5054. atomic_read(&ac->cmd_state)));
  5055. rc = adsp_err_get_lnx_err_code(
  5056. atomic_read(&ac->cmd_state));
  5057. goto fail_cmd;
  5058. }
  5059. return 0;
  5060. fail_cmd:
  5061. return rc;
  5062. }
  5063. EXPORT_SYMBOL(q6asm_enable_sbrps);
  5064. /**
  5065. * q6asm_cfg_dual_mono_aac -
  5066. * command to set config for dual mono aac
  5067. *
  5068. * @ac: Audio client handle
  5069. * @sce_left: left sce val
  5070. * @sce_right: right sce val
  5071. *
  5072. * Returns 0 on success or error on failure
  5073. */
  5074. int q6asm_cfg_dual_mono_aac(struct audio_client *ac,
  5075. uint16_t sce_left, uint16_t sce_right)
  5076. {
  5077. struct asm_aac_dual_mono_mapping_param dual_mono;
  5078. int rc = 0;
  5079. pr_debug("%s: Session %d, sce_left = %d, sce_right = %d\n",
  5080. __func__, ac->session, sce_left, sce_right);
  5081. q6asm_add_hdr(ac, &dual_mono.hdr, sizeof(dual_mono), TRUE);
  5082. atomic_set(&ac->cmd_state, -1);
  5083. dual_mono.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5084. dual_mono.encdec.param_id = ASM_PARAM_ID_AAC_DUAL_MONO_MAPPING;
  5085. dual_mono.encdec.param_size = sizeof(dual_mono.left_channel_sce) +
  5086. sizeof(dual_mono.right_channel_sce);
  5087. dual_mono.left_channel_sce = sce_left;
  5088. dual_mono.right_channel_sce = sce_right;
  5089. rc = apr_send_pkt(ac->apr, (uint32_t *) &dual_mono);
  5090. if (rc < 0) {
  5091. pr_err("%s: Command opcode[0x%x]paramid[0x%x] failed %d\n",
  5092. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM,
  5093. ASM_PARAM_ID_AAC_DUAL_MONO_MAPPING, rc);
  5094. rc = -EINVAL;
  5095. goto fail_cmd;
  5096. }
  5097. rc = wait_event_timeout(ac->cmd_wait,
  5098. (atomic_read(&ac->cmd_state) >= 0),
  5099. msecs_to_jiffies(TIMEOUT_MS));
  5100. if (!rc) {
  5101. pr_err("%s: timeout opcode[0x%x]\n", __func__,
  5102. dual_mono.hdr.opcode);
  5103. rc = -ETIMEDOUT;
  5104. goto fail_cmd;
  5105. }
  5106. if (atomic_read(&ac->cmd_state) > 0) {
  5107. pr_err("%s: DSP returned error[%s]\n",
  5108. __func__, adsp_err_get_err_str(
  5109. atomic_read(&ac->cmd_state)));
  5110. rc = adsp_err_get_lnx_err_code(
  5111. atomic_read(&ac->cmd_state));
  5112. goto fail_cmd;
  5113. }
  5114. return 0;
  5115. fail_cmd:
  5116. return rc;
  5117. }
  5118. EXPORT_SYMBOL(q6asm_cfg_dual_mono_aac);
  5119. /* Support for selecting stereo mixing coefficients for B family not done */
  5120. int q6asm_cfg_aac_sel_mix_coef(struct audio_client *ac, uint32_t mix_coeff)
  5121. {
  5122. struct asm_aac_stereo_mix_coeff_selection_param_v2 aac_mix_coeff;
  5123. int rc = 0;
  5124. q6asm_add_hdr(ac, &aac_mix_coeff.hdr, sizeof(aac_mix_coeff), TRUE);
  5125. atomic_set(&ac->cmd_state, -1);
  5126. aac_mix_coeff.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5127. aac_mix_coeff.param_id =
  5128. ASM_PARAM_ID_AAC_STEREO_MIX_COEFF_SELECTION_FLAG_V2;
  5129. aac_mix_coeff.param_size =
  5130. sizeof(struct asm_aac_stereo_mix_coeff_selection_param_v2);
  5131. aac_mix_coeff.aac_stereo_mix_coeff_flag = mix_coeff;
  5132. pr_debug("%s: mix_coeff = %u\n", __func__, mix_coeff);
  5133. rc = apr_send_pkt(ac->apr, (uint32_t *) &aac_mix_coeff);
  5134. if (rc < 0) {
  5135. pr_err("%s: Command opcode[0x%x]paramid[0x%x] failed %d\n",
  5136. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM,
  5137. ASM_PARAM_ID_AAC_STEREO_MIX_COEFF_SELECTION_FLAG_V2,
  5138. rc);
  5139. rc = -EINVAL;
  5140. goto fail_cmd;
  5141. }
  5142. rc = wait_event_timeout(ac->cmd_wait,
  5143. (atomic_read(&ac->cmd_state) >= 0),
  5144. msecs_to_jiffies(TIMEOUT_MS));
  5145. if (!rc) {
  5146. pr_err("%s: timeout opcode[0x%x]\n",
  5147. __func__, aac_mix_coeff.hdr.opcode);
  5148. rc = -ETIMEDOUT;
  5149. goto fail_cmd;
  5150. }
  5151. if (atomic_read(&ac->cmd_state) > 0) {
  5152. pr_err("%s: DSP returned error[%s]\n",
  5153. __func__, adsp_err_get_err_str(
  5154. atomic_read(&ac->cmd_state)));
  5155. rc = adsp_err_get_lnx_err_code(
  5156. atomic_read(&ac->cmd_state));
  5157. goto fail_cmd;
  5158. }
  5159. return 0;
  5160. fail_cmd:
  5161. return rc;
  5162. }
  5163. EXPORT_SYMBOL(q6asm_cfg_aac_sel_mix_coef);
  5164. /**
  5165. * q6asm_enc_cfg_blk_qcelp -
  5166. * command to set encode config block for QCELP
  5167. *
  5168. * @ac: Audio client handle
  5169. * @frames_per_buf: Number of frames per buffer
  5170. * @min_rate: Minimum Enc rate
  5171. * @max_rate: Maximum Enc rate
  5172. * reduced_rate_level: Reduced rate level
  5173. * @rate_modulation_cmd: rate modulation command
  5174. *
  5175. * Returns 0 on success or error on failure
  5176. */
  5177. int q6asm_enc_cfg_blk_qcelp(struct audio_client *ac, uint32_t frames_per_buf,
  5178. uint16_t min_rate, uint16_t max_rate,
  5179. uint16_t reduced_rate_level, uint16_t rate_modulation_cmd)
  5180. {
  5181. struct asm_v13k_enc_cfg enc_cfg;
  5182. int rc = 0;
  5183. pr_debug("%s: session[%d]frames[%d]min_rate[0x%4x]max_rate[0x%4x] reduced_rate_level[0x%4x]rate_modulation_cmd[0x%4x]\n",
  5184. __func__,
  5185. ac->session, frames_per_buf, min_rate, max_rate,
  5186. reduced_rate_level, rate_modulation_cmd);
  5187. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  5188. atomic_set(&ac->cmd_state, -1);
  5189. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5190. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  5191. enc_cfg.encdec.param_size = sizeof(struct asm_v13k_enc_cfg) -
  5192. sizeof(struct asm_stream_cmd_set_encdec_param);
  5193. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  5194. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  5195. sizeof(struct asm_enc_cfg_blk_param_v2);
  5196. enc_cfg.min_rate = min_rate;
  5197. enc_cfg.max_rate = max_rate;
  5198. enc_cfg.reduced_rate_cmd = reduced_rate_level;
  5199. enc_cfg.rate_mod_cmd = rate_modulation_cmd;
  5200. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  5201. if (rc < 0) {
  5202. pr_err("%s: Comamnd %d failed %d\n",
  5203. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  5204. rc = -EINVAL;
  5205. goto fail_cmd;
  5206. }
  5207. rc = wait_event_timeout(ac->cmd_wait,
  5208. (atomic_read(&ac->cmd_state) >= 0),
  5209. msecs_to_jiffies(TIMEOUT_MS));
  5210. if (!rc) {
  5211. pr_err("%s: timeout. waited for setencdec v13k resp\n",
  5212. __func__);
  5213. rc = -ETIMEDOUT;
  5214. goto fail_cmd;
  5215. }
  5216. if (atomic_read(&ac->cmd_state) > 0) {
  5217. pr_err("%s: DSP returned error[%s]\n",
  5218. __func__, adsp_err_get_err_str(
  5219. atomic_read(&ac->cmd_state)));
  5220. rc = adsp_err_get_lnx_err_code(
  5221. atomic_read(&ac->cmd_state));
  5222. goto fail_cmd;
  5223. }
  5224. return 0;
  5225. fail_cmd:
  5226. return rc;
  5227. }
  5228. EXPORT_SYMBOL(q6asm_enc_cfg_blk_qcelp);
  5229. /**
  5230. * q6asm_enc_cfg_blk_evrc -
  5231. * command to set encode config block for EVRC
  5232. *
  5233. * @ac: Audio client handle
  5234. * @frames_per_buf: Number of frames per buffer
  5235. * @min_rate: Minimum Enc rate
  5236. * @max_rate: Maximum Enc rate
  5237. * @rate_modulation_cmd: rate modulation command
  5238. *
  5239. * Returns 0 on success or error on failure
  5240. */
  5241. int q6asm_enc_cfg_blk_evrc(struct audio_client *ac, uint32_t frames_per_buf,
  5242. uint16_t min_rate, uint16_t max_rate,
  5243. uint16_t rate_modulation_cmd)
  5244. {
  5245. struct asm_evrc_enc_cfg enc_cfg;
  5246. int rc = 0;
  5247. pr_debug("%s: session[%d]frames[%d]min_rate[0x%4x]max_rate[0x%4x] rate_modulation_cmd[0x%4x]\n",
  5248. __func__, ac->session,
  5249. frames_per_buf, min_rate, max_rate, rate_modulation_cmd);
  5250. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  5251. atomic_set(&ac->cmd_state, -1);
  5252. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5253. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  5254. enc_cfg.encdec.param_size = sizeof(struct asm_evrc_enc_cfg) -
  5255. sizeof(struct asm_stream_cmd_set_encdec_param);
  5256. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  5257. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  5258. sizeof(struct asm_enc_cfg_blk_param_v2);
  5259. enc_cfg.min_rate = min_rate;
  5260. enc_cfg.max_rate = max_rate;
  5261. enc_cfg.rate_mod_cmd = rate_modulation_cmd;
  5262. enc_cfg.reserved = 0;
  5263. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  5264. if (rc < 0) {
  5265. pr_err("%s: Comamnd %d failed %d\n",
  5266. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  5267. rc = -EINVAL;
  5268. goto fail_cmd;
  5269. }
  5270. rc = wait_event_timeout(ac->cmd_wait,
  5271. (atomic_read(&ac->cmd_state) >= 0),
  5272. msecs_to_jiffies(TIMEOUT_MS));
  5273. if (!rc) {
  5274. pr_err("%s: timeout. waited for encdec evrc\n", __func__);
  5275. rc = -ETIMEDOUT;
  5276. goto fail_cmd;
  5277. }
  5278. if (atomic_read(&ac->cmd_state) > 0) {
  5279. pr_err("%s: DSP returned error[%s]\n",
  5280. __func__, adsp_err_get_err_str(
  5281. atomic_read(&ac->cmd_state)));
  5282. rc = adsp_err_get_lnx_err_code(
  5283. atomic_read(&ac->cmd_state));
  5284. goto fail_cmd;
  5285. }
  5286. return 0;
  5287. fail_cmd:
  5288. return rc;
  5289. }
  5290. EXPORT_SYMBOL(q6asm_enc_cfg_blk_evrc);
  5291. /**
  5292. * q6asm_enc_cfg_blk_amrnb -
  5293. * command to set encode config block for AMRNB
  5294. *
  5295. * @ac: Audio client handle
  5296. * @frames_per_buf: Number of frames per buffer
  5297. * @band_mode: Band mode used
  5298. * @dtx_enable: DTX en flag
  5299. *
  5300. * Returns 0 on success or error on failure
  5301. */
  5302. int q6asm_enc_cfg_blk_amrnb(struct audio_client *ac, uint32_t frames_per_buf,
  5303. uint16_t band_mode, uint16_t dtx_enable)
  5304. {
  5305. struct asm_amrnb_enc_cfg enc_cfg;
  5306. int rc = 0;
  5307. pr_debug("%s: session[%d]frames[%d]band_mode[0x%4x]dtx_enable[0x%4x]\n",
  5308. __func__, ac->session, frames_per_buf, band_mode, dtx_enable);
  5309. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  5310. atomic_set(&ac->cmd_state, -1);
  5311. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5312. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  5313. enc_cfg.encdec.param_size = sizeof(struct asm_amrnb_enc_cfg) -
  5314. sizeof(struct asm_stream_cmd_set_encdec_param);
  5315. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  5316. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  5317. sizeof(struct asm_enc_cfg_blk_param_v2);
  5318. enc_cfg.enc_mode = band_mode;
  5319. enc_cfg.dtx_mode = dtx_enable;
  5320. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  5321. if (rc < 0) {
  5322. pr_err("%s: Comamnd %d failed %d\n",
  5323. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  5324. rc = -EINVAL;
  5325. goto fail_cmd;
  5326. }
  5327. rc = wait_event_timeout(ac->cmd_wait,
  5328. (atomic_read(&ac->cmd_state) >= 0),
  5329. msecs_to_jiffies(TIMEOUT_MS));
  5330. if (!rc) {
  5331. pr_err("%s: timeout. waited for set encdec amrnb\n", __func__);
  5332. rc = -ETIMEDOUT;
  5333. goto fail_cmd;
  5334. }
  5335. if (atomic_read(&ac->cmd_state) > 0) {
  5336. pr_err("%s: DSP returned error[%s]\n",
  5337. __func__, adsp_err_get_err_str(
  5338. atomic_read(&ac->cmd_state)));
  5339. rc = adsp_err_get_lnx_err_code(
  5340. atomic_read(&ac->cmd_state));
  5341. goto fail_cmd;
  5342. }
  5343. return 0;
  5344. fail_cmd:
  5345. return rc;
  5346. }
  5347. EXPORT_SYMBOL(q6asm_enc_cfg_blk_amrnb);
  5348. /**
  5349. * q6asm_enc_cfg_blk_amrwb -
  5350. * command to set encode config block for AMRWB
  5351. *
  5352. * @ac: Audio client handle
  5353. * @frames_per_buf: Number of frames per buffer
  5354. * @band_mode: Band mode used
  5355. * @dtx_enable: DTX en flag
  5356. *
  5357. * Returns 0 on success or error on failure
  5358. */
  5359. int q6asm_enc_cfg_blk_amrwb(struct audio_client *ac, uint32_t frames_per_buf,
  5360. uint16_t band_mode, uint16_t dtx_enable)
  5361. {
  5362. struct asm_amrwb_enc_cfg enc_cfg;
  5363. int rc = 0;
  5364. pr_debug("%s: session[%d]frames[%d]band_mode[0x%4x]dtx_enable[0x%4x]\n",
  5365. __func__, ac->session, frames_per_buf, band_mode, dtx_enable);
  5366. q6asm_add_hdr(ac, &enc_cfg.hdr, sizeof(enc_cfg), TRUE);
  5367. atomic_set(&ac->cmd_state, -1);
  5368. enc_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  5369. enc_cfg.encdec.param_id = ASM_PARAM_ID_ENCDEC_ENC_CFG_BLK_V2;
  5370. enc_cfg.encdec.param_size = sizeof(struct asm_amrwb_enc_cfg) -
  5371. sizeof(struct asm_stream_cmd_set_encdec_param);
  5372. enc_cfg.encblk.frames_per_buf = frames_per_buf;
  5373. enc_cfg.encblk.enc_cfg_blk_size = enc_cfg.encdec.param_size -
  5374. sizeof(struct asm_enc_cfg_blk_param_v2);
  5375. enc_cfg.enc_mode = band_mode;
  5376. enc_cfg.dtx_mode = dtx_enable;
  5377. rc = apr_send_pkt(ac->apr, (uint32_t *) &enc_cfg);
  5378. if (rc < 0) {
  5379. pr_err("%s: Comamnd %d failed %d\n",
  5380. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  5381. rc = -EINVAL;
  5382. goto fail_cmd;
  5383. }
  5384. rc = wait_event_timeout(ac->cmd_wait,
  5385. (atomic_read(&ac->cmd_state) >= 0),
  5386. msecs_to_jiffies(TIMEOUT_MS));
  5387. if (!rc) {
  5388. pr_err("%s: timeout. waited for FORMAT_UPDATE\n", __func__);
  5389. rc = -ETIMEDOUT;
  5390. goto fail_cmd;
  5391. }
  5392. if (atomic_read(&ac->cmd_state) > 0) {
  5393. pr_err("%s: DSP returned error[%s]\n",
  5394. __func__, adsp_err_get_err_str(
  5395. atomic_read(&ac->cmd_state)));
  5396. rc = adsp_err_get_lnx_err_code(
  5397. atomic_read(&ac->cmd_state));
  5398. goto fail_cmd;
  5399. }
  5400. return 0;
  5401. fail_cmd:
  5402. return rc;
  5403. }
  5404. EXPORT_SYMBOL(q6asm_enc_cfg_blk_amrwb);
  5405. static int __q6asm_media_format_block_pcm(struct audio_client *ac,
  5406. uint32_t rate, uint32_t channels,
  5407. uint16_t bits_per_sample, int stream_id,
  5408. bool use_default_chmap, char *channel_map)
  5409. {
  5410. struct asm_multi_channel_pcm_fmt_blk_v2 fmt;
  5411. u8 *channel_mapping;
  5412. int rc = 0;
  5413. pr_debug("%s: session[%d]rate[%d]ch[%d]\n", __func__, ac->session, rate,
  5414. channels);
  5415. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  5416. atomic_set(&ac->cmd_state, -1);
  5417. /*
  5418. * Updated the token field with stream/session for compressed playback
  5419. * Platform driver must know the the stream with which the command is
  5420. * associated
  5421. */
  5422. if (ac->io_mode & COMPRESSED_STREAM_IO)
  5423. q6asm_update_token(&fmt.hdr.token,
  5424. ac->session,
  5425. stream_id,
  5426. 0, /* Buffer index is NA */
  5427. 0, /* Direction flag is NA */
  5428. WAIT_CMD);
  5429. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  5430. __func__, fmt.hdr.token, stream_id, ac->session);
  5431. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5432. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5433. sizeof(fmt.fmt_blk);
  5434. fmt.num_channels = channels;
  5435. fmt.bits_per_sample = bits_per_sample;
  5436. fmt.sample_rate = rate;
  5437. fmt.is_signed = 1;
  5438. channel_mapping = fmt.channel_mapping;
  5439. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  5440. if (use_default_chmap) {
  5441. if (q6asm_map_channels(channel_mapping, channels, false)) {
  5442. pr_err("%s: map channels failed %d\n",
  5443. __func__, channels);
  5444. return -EINVAL;
  5445. }
  5446. } else {
  5447. memcpy(channel_mapping, channel_map,
  5448. PCM_FORMAT_MAX_NUM_CHANNEL);
  5449. }
  5450. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5451. if (rc < 0) {
  5452. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5453. rc = -EINVAL;
  5454. goto fail_cmd;
  5455. }
  5456. rc = wait_event_timeout(ac->cmd_wait,
  5457. (atomic_read(&ac->cmd_state) >= 0),
  5458. msecs_to_jiffies(TIMEOUT_MS));
  5459. if (!rc) {
  5460. pr_err("%s: timeout. waited for format update\n", __func__);
  5461. rc = -ETIMEDOUT;
  5462. goto fail_cmd;
  5463. }
  5464. if (atomic_read(&ac->cmd_state) > 0) {
  5465. pr_err("%s: DSP returned error[%s]\n",
  5466. __func__, adsp_err_get_err_str(
  5467. atomic_read(&ac->cmd_state)));
  5468. rc = adsp_err_get_lnx_err_code(
  5469. atomic_read(&ac->cmd_state));
  5470. goto fail_cmd;
  5471. }
  5472. return 0;
  5473. fail_cmd:
  5474. return rc;
  5475. }
  5476. static int __q6asm_media_format_block_pcm_v3(struct audio_client *ac,
  5477. uint32_t rate, uint32_t channels,
  5478. uint16_t bits_per_sample,
  5479. int stream_id,
  5480. bool use_default_chmap,
  5481. char *channel_map,
  5482. uint16_t sample_word_size)
  5483. {
  5484. struct asm_multi_channel_pcm_fmt_blk_param_v3 fmt;
  5485. u8 *channel_mapping;
  5486. int rc;
  5487. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  5488. ac->session, rate, channels,
  5489. bits_per_sample, sample_word_size);
  5490. memset(&fmt, 0, sizeof(fmt));
  5491. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  5492. atomic_set(&ac->cmd_state, -1);
  5493. /*
  5494. * Updated the token field with stream/session for compressed playback
  5495. * Platform driver must know the the stream with which the command is
  5496. * associated
  5497. */
  5498. if (ac->io_mode & COMPRESSED_STREAM_IO)
  5499. fmt.hdr.token = ((ac->session << 8) & 0xFFFF00) |
  5500. (stream_id & 0xFF);
  5501. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  5502. __func__, fmt.hdr.token, stream_id, ac->session);
  5503. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5504. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5505. sizeof(fmt.fmt_blk);
  5506. fmt.param.num_channels = channels;
  5507. fmt.param.bits_per_sample = bits_per_sample;
  5508. fmt.param.sample_rate = rate;
  5509. fmt.param.is_signed = 1;
  5510. fmt.param.sample_word_size = sample_word_size;
  5511. channel_mapping = fmt.param.channel_mapping;
  5512. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  5513. if (use_default_chmap) {
  5514. if (q6asm_map_channels(channel_mapping, channels, false)) {
  5515. pr_err("%s: map channels failed %d\n",
  5516. __func__, channels);
  5517. rc = -EINVAL;
  5518. goto fail_cmd;
  5519. }
  5520. } else {
  5521. memcpy(channel_mapping, channel_map,
  5522. PCM_FORMAT_MAX_NUM_CHANNEL);
  5523. }
  5524. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5525. if (rc < 0) {
  5526. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5527. rc = -EINVAL;
  5528. goto fail_cmd;
  5529. }
  5530. rc = wait_event_timeout(ac->cmd_wait,
  5531. (atomic_read(&ac->cmd_state) >= 0),
  5532. msecs_to_jiffies(TIMEOUT_MS));
  5533. if (!rc) {
  5534. pr_err("%s: timeout. waited for format update\n", __func__);
  5535. rc = -ETIMEDOUT;
  5536. goto fail_cmd;
  5537. }
  5538. if (atomic_read(&ac->cmd_state) > 0) {
  5539. pr_err("%s: DSP returned error[%s]\n",
  5540. __func__, adsp_err_get_err_str(
  5541. atomic_read(&ac->cmd_state)));
  5542. rc = adsp_err_get_lnx_err_code(
  5543. atomic_read(&ac->cmd_state));
  5544. goto fail_cmd;
  5545. }
  5546. return 0;
  5547. fail_cmd:
  5548. return rc;
  5549. }
  5550. static int __q6asm_media_format_block_pcm_v4(struct audio_client *ac,
  5551. uint32_t rate, uint32_t channels,
  5552. uint16_t bits_per_sample,
  5553. int stream_id,
  5554. bool use_default_chmap,
  5555. char *channel_map,
  5556. uint16_t sample_word_size,
  5557. uint16_t endianness,
  5558. uint16_t mode)
  5559. {
  5560. struct asm_multi_channel_pcm_fmt_blk_param_v4 fmt;
  5561. u8 *channel_mapping;
  5562. int rc;
  5563. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  5564. ac->session, rate, channels,
  5565. bits_per_sample, sample_word_size);
  5566. memset(&fmt, 0, sizeof(fmt));
  5567. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  5568. atomic_set(&ac->cmd_state, -1);
  5569. /*
  5570. * Updated the token field with stream/session for compressed playback
  5571. * Platform driver must know the the stream with which the command is
  5572. * associated
  5573. */
  5574. if (ac->io_mode & COMPRESSED_STREAM_IO)
  5575. fmt.hdr.token = ((ac->session << 8) & 0xFFFF00) |
  5576. (stream_id & 0xFF);
  5577. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  5578. __func__, fmt.hdr.token, stream_id, ac->session);
  5579. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5580. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5581. sizeof(fmt.fmt_blk);
  5582. fmt.param.num_channels = channels;
  5583. fmt.param.bits_per_sample = bits_per_sample;
  5584. fmt.param.sample_rate = rate;
  5585. fmt.param.is_signed = 1;
  5586. fmt.param.sample_word_size = sample_word_size;
  5587. fmt.param.endianness = endianness;
  5588. fmt.param.mode = mode;
  5589. channel_mapping = fmt.param.channel_mapping;
  5590. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  5591. if (use_default_chmap) {
  5592. if (q6asm_map_channels(channel_mapping, channels, false)) {
  5593. pr_err("%s: map channels failed %d\n",
  5594. __func__, channels);
  5595. rc = -EINVAL;
  5596. goto fail_cmd;
  5597. }
  5598. } else {
  5599. memcpy(channel_mapping, channel_map,
  5600. PCM_FORMAT_MAX_NUM_CHANNEL);
  5601. }
  5602. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5603. if (rc < 0) {
  5604. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5605. rc = -EINVAL;
  5606. goto fail_cmd;
  5607. }
  5608. rc = wait_event_timeout(ac->cmd_wait,
  5609. (atomic_read(&ac->cmd_state) >= 0),
  5610. msecs_to_jiffies(TIMEOUT_MS));
  5611. if (!rc) {
  5612. pr_err("%s: timeout. waited for format update\n", __func__);
  5613. rc = -ETIMEDOUT;
  5614. goto fail_cmd;
  5615. }
  5616. if (atomic_read(&ac->cmd_state) > 0) {
  5617. pr_err("%s: DSP returned error[%s]\n",
  5618. __func__, adsp_err_get_err_str(
  5619. atomic_read(&ac->cmd_state)));
  5620. rc = adsp_err_get_lnx_err_code(
  5621. atomic_read(&ac->cmd_state));
  5622. goto fail_cmd;
  5623. }
  5624. return 0;
  5625. fail_cmd:
  5626. return rc;
  5627. }
  5628. static int __q6asm_media_format_block_pcm_v5(struct audio_client *ac,
  5629. uint32_t rate, uint32_t channels,
  5630. uint16_t bits_per_sample,
  5631. int stream_id,
  5632. bool use_default_chmap,
  5633. char *channel_map,
  5634. uint16_t sample_word_size,
  5635. uint16_t endianness,
  5636. uint16_t mode)
  5637. {
  5638. struct asm_multi_channel_pcm_fmt_blk_param_v5 fmt;
  5639. u8 *channel_mapping;
  5640. int rc;
  5641. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  5642. ac->session, rate, channels,
  5643. bits_per_sample, sample_word_size);
  5644. memset(&fmt, 0, sizeof(fmt));
  5645. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  5646. atomic_set(&ac->cmd_state, -1);
  5647. /*
  5648. * Updated the token field with stream/session for compressed playback
  5649. * Platform driver must know the the stream with which the command is
  5650. * associated
  5651. */
  5652. if (ac->io_mode & COMPRESSED_STREAM_IO)
  5653. fmt.hdr.token = ((ac->session << 8) & 0xFFFF00) |
  5654. (stream_id & 0xFF);
  5655. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  5656. __func__, fmt.hdr.token, stream_id, ac->session);
  5657. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5658. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5659. sizeof(fmt.fmt_blk);
  5660. fmt.param.num_channels = (uint16_t) channels & 0xFFFF;
  5661. fmt.param.bits_per_sample = bits_per_sample;
  5662. fmt.param.sample_rate = rate;
  5663. fmt.param.is_signed = 1;
  5664. fmt.param.sample_word_size = sample_word_size;
  5665. fmt.param.endianness = endianness;
  5666. fmt.param.mode = mode;
  5667. channel_mapping = fmt.param.channel_mapping;
  5668. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  5669. if (use_default_chmap) {
  5670. if (q6asm_map_channels(channel_mapping, fmt.param.num_channels, false)) {
  5671. pr_err("%s: map channels failed %d\n",
  5672. __func__, channels);
  5673. rc = -EINVAL;
  5674. goto fail_cmd;
  5675. }
  5676. } else {
  5677. memcpy(channel_mapping, channel_map,
  5678. PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  5679. }
  5680. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5681. if (rc < 0) {
  5682. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5683. rc = -EINVAL;
  5684. goto fail_cmd;
  5685. }
  5686. rc = wait_event_timeout(ac->cmd_wait,
  5687. (atomic_read(&ac->cmd_state) >= 0), 5*HZ);
  5688. if (!rc) {
  5689. pr_err("%s: timeout. waited for format update\n", __func__);
  5690. rc = -ETIMEDOUT;
  5691. goto fail_cmd;
  5692. }
  5693. if (atomic_read(&ac->cmd_state) > 0) {
  5694. pr_err("%s: DSP returned error[%s]\n",
  5695. __func__, adsp_err_get_err_str(
  5696. atomic_read(&ac->cmd_state)));
  5697. rc = adsp_err_get_lnx_err_code(
  5698. atomic_read(&ac->cmd_state));
  5699. goto fail_cmd;
  5700. }
  5701. return 0;
  5702. fail_cmd:
  5703. return rc;
  5704. }
  5705. /**
  5706. * q6asm_media_format_block_pcm -
  5707. * command to set mediafmt block for PCM on ASM stream
  5708. *
  5709. * @ac: Audio client handle
  5710. * @rate: sample rate
  5711. * @channels: number of ASM channels
  5712. *
  5713. * Returns 0 on success or error on failure
  5714. */
  5715. int q6asm_media_format_block_pcm(struct audio_client *ac,
  5716. uint32_t rate, uint32_t channels)
  5717. {
  5718. return __q6asm_media_format_block_pcm(ac, rate,
  5719. channels, 16, ac->stream_id,
  5720. true, NULL);
  5721. }
  5722. EXPORT_SYMBOL(q6asm_media_format_block_pcm);
  5723. /**
  5724. * q6asm_media_format_block_pcm_format_support -
  5725. * command to set mediafmt block for PCM format support
  5726. *
  5727. * @ac: Audio client handle
  5728. * @rate: sample rate
  5729. * @channels: number of ASM channels
  5730. * @bits_per_sample: number of bits per sample
  5731. *
  5732. * Returns 0 on success or error on failure
  5733. */
  5734. int q6asm_media_format_block_pcm_format_support(struct audio_client *ac,
  5735. uint32_t rate, uint32_t channels,
  5736. uint16_t bits_per_sample)
  5737. {
  5738. return __q6asm_media_format_block_pcm(ac, rate,
  5739. channels, bits_per_sample, ac->stream_id,
  5740. true, NULL);
  5741. }
  5742. EXPORT_SYMBOL(q6asm_media_format_block_pcm_format_support);
  5743. int q6asm_media_format_block_pcm_format_support_v2(struct audio_client *ac,
  5744. uint32_t rate, uint32_t channels,
  5745. uint16_t bits_per_sample, int stream_id,
  5746. bool use_default_chmap, char *channel_map)
  5747. {
  5748. if (!use_default_chmap && (channel_map == NULL)) {
  5749. pr_err("%s: No valid chan map and can't use default\n",
  5750. __func__);
  5751. return -EINVAL;
  5752. }
  5753. return __q6asm_media_format_block_pcm(ac, rate,
  5754. channels, bits_per_sample, stream_id,
  5755. use_default_chmap, channel_map);
  5756. }
  5757. /*
  5758. * q6asm_media_format_block_pcm_format_support_v3- sends pcm decoder
  5759. * configuration parameters
  5760. *
  5761. * @ac: Client session handle
  5762. * @rate: sample rate
  5763. * @channels: number of channels
  5764. * @bits_per_sample: bit width of encoder session
  5765. * @stream_id: stream id of stream to be associated with this session
  5766. * @use_default_chmap: true if default channel map to be used
  5767. * @channel_map: input channel map
  5768. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  5769. */
  5770. int q6asm_media_format_block_pcm_format_support_v3(struct audio_client *ac,
  5771. uint32_t rate,
  5772. uint32_t channels,
  5773. uint16_t bits_per_sample,
  5774. int stream_id,
  5775. bool use_default_chmap,
  5776. char *channel_map,
  5777. uint16_t sample_word_size)
  5778. {
  5779. if (!use_default_chmap && (channel_map == NULL)) {
  5780. pr_err("%s: No valid chan map and can't use default\n",
  5781. __func__);
  5782. return -EINVAL;
  5783. }
  5784. return __q6asm_media_format_block_pcm_v3(ac, rate,
  5785. channels, bits_per_sample, stream_id,
  5786. use_default_chmap, channel_map,
  5787. sample_word_size);
  5788. }
  5789. EXPORT_SYMBOL(q6asm_media_format_block_pcm_format_support_v3);
  5790. /*
  5791. * q6asm_media_format_block_pcm_format_support_v4- sends pcm decoder
  5792. * configuration parameters
  5793. *
  5794. * @ac: Client session handle
  5795. * @rate: sample rate
  5796. * @channels: number of channels
  5797. * @bits_per_sample: bit width of encoder session
  5798. * @stream_id: stream id of stream to be associated with this session
  5799. * @use_default_chmap: true if default channel map to be used
  5800. * @channel_map: input channel map
  5801. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  5802. * @endianness: endianness of the pcm data
  5803. * @mode: Mode to provide additional info about the pcm input data
  5804. */
  5805. int q6asm_media_format_block_pcm_format_support_v4(struct audio_client *ac,
  5806. uint32_t rate,
  5807. uint32_t channels,
  5808. uint16_t bits_per_sample,
  5809. int stream_id,
  5810. bool use_default_chmap,
  5811. char *channel_map,
  5812. uint16_t sample_word_size,
  5813. uint16_t endianness,
  5814. uint16_t mode)
  5815. {
  5816. if (!use_default_chmap && (channel_map == NULL)) {
  5817. pr_err("%s: No valid chan map and can't use default\n",
  5818. __func__);
  5819. return -EINVAL;
  5820. }
  5821. return __q6asm_media_format_block_pcm_v4(ac, rate,
  5822. channels, bits_per_sample, stream_id,
  5823. use_default_chmap, channel_map,
  5824. sample_word_size, endianness,
  5825. mode);
  5826. }
  5827. EXPORT_SYMBOL(q6asm_media_format_block_pcm_format_support_v4);
  5828. /*
  5829. * q6asm_media_format_block_pcm_format_support_v5- sends pcm decoder
  5830. * configuration parameters
  5831. *
  5832. * @ac: Client session handle
  5833. * @rate: sample rate
  5834. * @channels: number of channels
  5835. * @bits_per_sample: bit width of encoder session
  5836. * @stream_id: stream id of stream to be associated with this session
  5837. * @use_default_chmap: true if default channel map to be used
  5838. * @channel_map: input channel map
  5839. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  5840. * @endianness: endianness of the pcm data
  5841. * @mode: Mode to provide additional info about the pcm input data
  5842. */
  5843. int q6asm_media_format_block_pcm_format_support_v5(struct audio_client *ac,
  5844. uint32_t rate,
  5845. uint32_t channels,
  5846. uint16_t bits_per_sample,
  5847. int stream_id,
  5848. bool use_default_chmap,
  5849. char *channel_map,
  5850. uint16_t sample_word_size,
  5851. uint16_t endianness,
  5852. uint16_t mode)
  5853. {
  5854. if (!use_default_chmap && (channel_map == NULL)) {
  5855. pr_err("%s: No valid chan map and can't use default\n",
  5856. __func__);
  5857. return -EINVAL;
  5858. }
  5859. return __q6asm_media_format_block_pcm_v5(ac, rate,
  5860. channels, bits_per_sample, stream_id,
  5861. use_default_chmap, channel_map,
  5862. sample_word_size, endianness,
  5863. mode);
  5864. }
  5865. EXPORT_SYMBOL(q6asm_media_format_block_pcm_format_support_v5);
  5866. static int __q6asm_media_format_block_multi_ch_pcm(struct audio_client *ac,
  5867. uint32_t rate, uint32_t channels,
  5868. bool use_default_chmap, char *channel_map,
  5869. uint16_t bits_per_sample)
  5870. {
  5871. struct asm_multi_channel_pcm_fmt_blk_v2 fmt;
  5872. u8 *channel_mapping;
  5873. int rc = 0;
  5874. pr_debug("%s: session[%d]rate[%d]ch[%d]\n", __func__, ac->session, rate,
  5875. channels);
  5876. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  5877. atomic_set(&ac->cmd_state, -1);
  5878. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5879. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5880. sizeof(fmt.fmt_blk);
  5881. fmt.num_channels = channels;
  5882. fmt.bits_per_sample = bits_per_sample;
  5883. fmt.sample_rate = rate;
  5884. fmt.is_signed = 1;
  5885. channel_mapping = fmt.channel_mapping;
  5886. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  5887. if (use_default_chmap) {
  5888. if (q6asm_map_channels(channel_mapping, channels, false)) {
  5889. pr_err("%s: map channels failed %d\n",
  5890. __func__, channels);
  5891. return -EINVAL;
  5892. }
  5893. } else {
  5894. memcpy(channel_mapping, channel_map,
  5895. PCM_FORMAT_MAX_NUM_CHANNEL);
  5896. }
  5897. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5898. if (rc < 0) {
  5899. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5900. rc = -EINVAL;
  5901. goto fail_cmd;
  5902. }
  5903. rc = wait_event_timeout(ac->cmd_wait,
  5904. (atomic_read(&ac->cmd_state) >= 0),
  5905. msecs_to_jiffies(TIMEOUT_MS));
  5906. if (!rc) {
  5907. pr_err("%s: timeout. waited for format update\n", __func__);
  5908. rc = -ETIMEDOUT;
  5909. goto fail_cmd;
  5910. }
  5911. if (atomic_read(&ac->cmd_state) > 0) {
  5912. pr_err("%s: DSP returned error[%s]\n",
  5913. __func__, adsp_err_get_err_str(
  5914. atomic_read(&ac->cmd_state)));
  5915. rc = adsp_err_get_lnx_err_code(
  5916. atomic_read(&ac->cmd_state));
  5917. goto fail_cmd;
  5918. }
  5919. return 0;
  5920. fail_cmd:
  5921. return rc;
  5922. }
  5923. static int __q6asm_media_format_block_multi_ch_pcm_v3(struct audio_client *ac,
  5924. uint32_t rate,
  5925. uint32_t channels,
  5926. bool use_default_chmap,
  5927. char *channel_map,
  5928. uint16_t bits_per_sample,
  5929. uint16_t sample_word_size)
  5930. {
  5931. struct asm_multi_channel_pcm_fmt_blk_param_v3 fmt;
  5932. u8 *channel_mapping;
  5933. int rc;
  5934. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  5935. ac->session, rate, channels,
  5936. bits_per_sample, sample_word_size);
  5937. memset(&fmt, 0, sizeof(fmt));
  5938. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  5939. atomic_set(&ac->cmd_state, -1);
  5940. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  5941. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  5942. sizeof(fmt.fmt_blk);
  5943. fmt.param.num_channels = channels;
  5944. fmt.param.bits_per_sample = bits_per_sample;
  5945. fmt.param.sample_rate = rate;
  5946. fmt.param.is_signed = 1;
  5947. fmt.param.sample_word_size = sample_word_size;
  5948. channel_mapping = fmt.param.channel_mapping;
  5949. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  5950. if (use_default_chmap) {
  5951. if (q6asm_map_channels(channel_mapping, channels, false)) {
  5952. pr_err("%s: map channels failed %d\n",
  5953. __func__, channels);
  5954. rc = -EINVAL;
  5955. goto fail_cmd;
  5956. }
  5957. } else {
  5958. memcpy(channel_mapping, channel_map,
  5959. PCM_FORMAT_MAX_NUM_CHANNEL);
  5960. }
  5961. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  5962. if (rc < 0) {
  5963. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  5964. goto fail_cmd;
  5965. }
  5966. rc = wait_event_timeout(ac->cmd_wait,
  5967. (atomic_read(&ac->cmd_state) >= 0),
  5968. msecs_to_jiffies(TIMEOUT_MS));
  5969. if (!rc) {
  5970. pr_err("%s: timeout. waited for format update\n", __func__);
  5971. rc = -ETIMEDOUT;
  5972. goto fail_cmd;
  5973. }
  5974. if (atomic_read(&ac->cmd_state) > 0) {
  5975. pr_err("%s: DSP returned error[%s]\n",
  5976. __func__, adsp_err_get_err_str(
  5977. atomic_read(&ac->cmd_state)));
  5978. rc = adsp_err_get_lnx_err_code(
  5979. atomic_read(&ac->cmd_state));
  5980. goto fail_cmd;
  5981. }
  5982. return 0;
  5983. fail_cmd:
  5984. return rc;
  5985. }
  5986. static int __q6asm_media_format_block_multi_ch_pcm_v4(struct audio_client *ac,
  5987. uint32_t rate,
  5988. uint32_t channels,
  5989. bool use_default_chmap,
  5990. char *channel_map,
  5991. uint16_t bits_per_sample,
  5992. uint16_t sample_word_size,
  5993. uint16_t endianness,
  5994. uint16_t mode)
  5995. {
  5996. struct asm_multi_channel_pcm_fmt_blk_param_v4 fmt;
  5997. u8 *channel_mapping;
  5998. int rc;
  5999. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  6000. ac->session, rate, channels,
  6001. bits_per_sample, sample_word_size);
  6002. memset(&fmt, 0, sizeof(fmt));
  6003. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  6004. atomic_set(&ac->cmd_state, -1);
  6005. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6006. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6007. sizeof(fmt.fmt_blk);
  6008. fmt.param.num_channels = channels;
  6009. fmt.param.bits_per_sample = bits_per_sample;
  6010. fmt.param.sample_rate = rate;
  6011. fmt.param.is_signed = 1;
  6012. fmt.param.sample_word_size = sample_word_size;
  6013. fmt.param.endianness = endianness;
  6014. fmt.param.mode = mode;
  6015. channel_mapping = fmt.param.channel_mapping;
  6016. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  6017. if (use_default_chmap) {
  6018. if (q6asm_map_channels(channel_mapping, channels, false)) {
  6019. pr_err("%s: map channels failed %d\n",
  6020. __func__, channels);
  6021. rc = -EINVAL;
  6022. goto fail_cmd;
  6023. }
  6024. } else {
  6025. memcpy(channel_mapping, channel_map,
  6026. PCM_FORMAT_MAX_NUM_CHANNEL);
  6027. }
  6028. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6029. if (rc < 0) {
  6030. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6031. goto fail_cmd;
  6032. }
  6033. rc = wait_event_timeout(ac->cmd_wait,
  6034. (atomic_read(&ac->cmd_state) >= 0),
  6035. msecs_to_jiffies(TIMEOUT_MS));
  6036. if (!rc) {
  6037. pr_err("%s: timeout. waited for format update\n", __func__);
  6038. rc = -ETIMEDOUT;
  6039. goto fail_cmd;
  6040. }
  6041. if (atomic_read(&ac->cmd_state) > 0) {
  6042. pr_err("%s: DSP returned error[%s]\n",
  6043. __func__, adsp_err_get_err_str(
  6044. atomic_read(&ac->cmd_state)));
  6045. rc = adsp_err_get_lnx_err_code(
  6046. atomic_read(&ac->cmd_state));
  6047. goto fail_cmd;
  6048. }
  6049. return 0;
  6050. fail_cmd:
  6051. return rc;
  6052. }
  6053. static int __q6asm_media_format_block_multi_ch_pcm_v5(struct audio_client *ac,
  6054. uint32_t rate,
  6055. uint32_t channels,
  6056. bool use_default_chmap,
  6057. char *channel_map,
  6058. uint16_t bits_per_sample,
  6059. uint16_t sample_word_size,
  6060. uint16_t endianness,
  6061. uint16_t mode)
  6062. {
  6063. struct asm_multi_channel_pcm_fmt_blk_param_v5 fmt;
  6064. u8 *channel_mapping;
  6065. int rc;
  6066. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]wordsize[%d]\n", __func__,
  6067. ac->session, rate, channels,
  6068. bits_per_sample, sample_word_size);
  6069. memset(&fmt, 0, sizeof(fmt));
  6070. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  6071. atomic_set(&ac->cmd_state, -1);
  6072. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6073. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6074. sizeof(fmt.fmt_blk);
  6075. fmt.param.num_channels = channels;
  6076. fmt.param.bits_per_sample = bits_per_sample;
  6077. fmt.param.sample_rate = rate;
  6078. fmt.param.is_signed = 1;
  6079. fmt.param.sample_word_size = sample_word_size;
  6080. fmt.param.endianness = endianness;
  6081. fmt.param.mode = mode;
  6082. channel_mapping = fmt.param.channel_mapping;
  6083. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  6084. if (use_default_chmap) {
  6085. if (q6asm_map_channels(channel_mapping, channels, false)) {
  6086. pr_err("%s: map channels failed %d\n",
  6087. __func__, channels);
  6088. rc = -EINVAL;
  6089. goto fail_cmd;
  6090. }
  6091. } else {
  6092. memcpy(channel_mapping, channel_map,
  6093. PCM_FORMAT_MAX_NUM_CHANNEL_V8);
  6094. }
  6095. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6096. if (rc < 0) {
  6097. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6098. goto fail_cmd;
  6099. }
  6100. rc = wait_event_timeout(ac->cmd_wait,
  6101. (atomic_read(&ac->cmd_state) >= 0), 5*HZ);
  6102. if (!rc) {
  6103. pr_err("%s: timeout. waited for format update\n", __func__);
  6104. rc = -ETIMEDOUT;
  6105. goto fail_cmd;
  6106. }
  6107. if (atomic_read(&ac->cmd_state) > 0) {
  6108. pr_err("%s: DSP returned error[%s]\n",
  6109. __func__, adsp_err_get_err_str(
  6110. atomic_read(&ac->cmd_state)));
  6111. rc = adsp_err_get_lnx_err_code(
  6112. atomic_read(&ac->cmd_state));
  6113. goto fail_cmd;
  6114. }
  6115. return 0;
  6116. fail_cmd:
  6117. return rc;
  6118. }
  6119. int q6asm_media_format_block_multi_ch_pcm(struct audio_client *ac,
  6120. uint32_t rate, uint32_t channels,
  6121. bool use_default_chmap, char *channel_map)
  6122. {
  6123. return __q6asm_media_format_block_multi_ch_pcm(ac, rate,
  6124. channels, use_default_chmap, channel_map, 16);
  6125. }
  6126. int q6asm_media_format_block_multi_ch_pcm_v2(
  6127. struct audio_client *ac,
  6128. uint32_t rate, uint32_t channels,
  6129. bool use_default_chmap, char *channel_map,
  6130. uint16_t bits_per_sample)
  6131. {
  6132. return __q6asm_media_format_block_multi_ch_pcm(ac, rate,
  6133. channels, use_default_chmap, channel_map,
  6134. bits_per_sample);
  6135. }
  6136. /*
  6137. * q6asm_media_format_block_multi_ch_pcm_v3 - sends pcm decoder configuration
  6138. * parameters
  6139. *
  6140. * @ac: Client session handle
  6141. * @rate: sample rate
  6142. * @channels: number of channels
  6143. * @bits_per_sample: bit width of encoder session
  6144. * @use_default_chmap: true if default channel map to be used
  6145. * @channel_map: input channel map
  6146. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  6147. */
  6148. int q6asm_media_format_block_multi_ch_pcm_v3(struct audio_client *ac,
  6149. uint32_t rate, uint32_t channels,
  6150. bool use_default_chmap,
  6151. char *channel_map,
  6152. uint16_t bits_per_sample,
  6153. uint16_t sample_word_size)
  6154. {
  6155. return __q6asm_media_format_block_multi_ch_pcm_v3(ac, rate, channels,
  6156. use_default_chmap,
  6157. channel_map,
  6158. bits_per_sample,
  6159. sample_word_size);
  6160. }
  6161. EXPORT_SYMBOL(q6asm_media_format_block_multi_ch_pcm_v3);
  6162. /*
  6163. * q6asm_media_format_block_multi_ch_pcm_v4 - sends pcm decoder configuration
  6164. * parameters
  6165. *
  6166. * @ac: Client session handle
  6167. * @rate: sample rate
  6168. * @channels: number of channels
  6169. * @bits_per_sample: bit width of encoder session
  6170. * @use_default_chmap: true if default channel map to be used
  6171. * @channel_map: input channel map
  6172. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  6173. * @endianness: endianness of the pcm data
  6174. * @mode: Mode to provide additional info about the pcm input data
  6175. */
  6176. int q6asm_media_format_block_multi_ch_pcm_v4(struct audio_client *ac,
  6177. uint32_t rate, uint32_t channels,
  6178. bool use_default_chmap,
  6179. char *channel_map,
  6180. uint16_t bits_per_sample,
  6181. uint16_t sample_word_size,
  6182. uint16_t endianness,
  6183. uint16_t mode)
  6184. {
  6185. return __q6asm_media_format_block_multi_ch_pcm_v4(ac, rate, channels,
  6186. use_default_chmap,
  6187. channel_map,
  6188. bits_per_sample,
  6189. sample_word_size,
  6190. endianness,
  6191. mode);
  6192. }
  6193. EXPORT_SYMBOL(q6asm_media_format_block_multi_ch_pcm_v4);
  6194. /*
  6195. * q6asm_media_format_block_multi_ch_pcm_v5 - sends pcm decoder configuration
  6196. * parameters
  6197. *
  6198. * @ac: Client session handle
  6199. * @rate: sample rate
  6200. * @channels: number of channels
  6201. * @bits_per_sample: bit width of encoder session
  6202. * @use_default_chmap: true if default channel map to be used
  6203. * @channel_map: input channel map
  6204. * @sample_word_size: Size in bits of the word that holds a sample of a channel
  6205. * @endianness: endianness of the pcm data
  6206. * @mode: Mode to provide additional info about the pcm input data
  6207. */
  6208. int q6asm_media_format_block_multi_ch_pcm_v5(struct audio_client *ac,
  6209. uint32_t rate, uint32_t channels,
  6210. bool use_default_chmap,
  6211. char *channel_map,
  6212. uint16_t bits_per_sample,
  6213. uint16_t sample_word_size,
  6214. uint16_t endianness,
  6215. uint16_t mode)
  6216. {
  6217. return __q6asm_media_format_block_multi_ch_pcm_v5(ac, rate, channels,
  6218. use_default_chmap,
  6219. channel_map,
  6220. bits_per_sample,
  6221. sample_word_size,
  6222. endianness,
  6223. mode);
  6224. }
  6225. EXPORT_SYMBOL(q6asm_media_format_block_multi_ch_pcm_v5);
  6226. /*
  6227. * q6asm_media_format_block_gen_compr - set up generic compress format params
  6228. *
  6229. * @ac: Client session handle
  6230. * @rate: sample rate
  6231. * @channels: number of channels
  6232. * @use_default_chmap: true if default channel map to be used
  6233. * @channel_map: input channel map
  6234. * @bits_per_sample: bit width of gen compress stream
  6235. */
  6236. int q6asm_media_format_block_gen_compr(struct audio_client *ac,
  6237. uint32_t rate, uint32_t channels,
  6238. bool use_default_chmap, char *channel_map,
  6239. uint16_t bits_per_sample)
  6240. {
  6241. struct asm_generic_compressed_fmt_blk_t fmt;
  6242. u8 *channel_mapping;
  6243. int rc = 0;
  6244. pr_debug("%s: session[%d]rate[%d]ch[%d]bps[%d]\n",
  6245. __func__, ac->session, rate,
  6246. channels, bits_per_sample);
  6247. memset(&fmt, 0, sizeof(fmt));
  6248. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  6249. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6250. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6251. sizeof(fmt.fmt_blk);
  6252. fmt.num_channels = channels;
  6253. fmt.bits_per_sample = bits_per_sample;
  6254. fmt.sampling_rate = rate;
  6255. channel_mapping = fmt.channel_mapping;
  6256. memset(channel_mapping, 0, PCM_FORMAT_MAX_NUM_CHANNEL);
  6257. if (use_default_chmap) {
  6258. if (q6asm_map_channels(channel_mapping, channels, false)) {
  6259. pr_err("%s: map channels failed %d\n",
  6260. __func__, channels);
  6261. return -EINVAL;
  6262. }
  6263. } else {
  6264. memcpy(channel_mapping, channel_map,
  6265. PCM_FORMAT_MAX_NUM_CHANNEL);
  6266. }
  6267. atomic_set(&ac->cmd_state, -1);
  6268. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6269. if (rc < 0) {
  6270. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6271. rc = -EINVAL;
  6272. goto fail_cmd;
  6273. }
  6274. rc = wait_event_timeout(ac->cmd_wait,
  6275. (atomic_read(&ac->cmd_state) >= 0),
  6276. msecs_to_jiffies(TIMEOUT_MS));
  6277. if (!rc) {
  6278. pr_err("%s: timeout. waited for format update\n", __func__);
  6279. rc = -ETIMEDOUT;
  6280. goto fail_cmd;
  6281. }
  6282. if (atomic_read(&ac->cmd_state) > 0) {
  6283. pr_err("%s: DSP returned error[%s]\n",
  6284. __func__, adsp_err_get_err_str(
  6285. atomic_read(&ac->cmd_state)));
  6286. rc = adsp_err_get_lnx_err_code(
  6287. atomic_read(&ac->cmd_state));
  6288. }
  6289. return 0;
  6290. fail_cmd:
  6291. return rc;
  6292. }
  6293. EXPORT_SYMBOL(q6asm_media_format_block_gen_compr);
  6294. /*
  6295. * q6asm_media_format_block_iec - set up IEC61937 (compressed) or IEC60958
  6296. * (pcm) format params. Both audio standards
  6297. * use the same format and are used for
  6298. * HDMI or SPDIF.
  6299. *
  6300. * @ac: Client session handle
  6301. * @rate: sample rate
  6302. * @channels: number of channels
  6303. */
  6304. int q6asm_media_format_block_iec(struct audio_client *ac,
  6305. uint32_t rate, uint32_t channels)
  6306. {
  6307. struct asm_iec_compressed_fmt_blk_t fmt;
  6308. int rc = 0;
  6309. pr_debug("%s: session[%d]rate[%d]ch[%d]\n",
  6310. __func__, ac->session, rate,
  6311. channels);
  6312. memset(&fmt, 0, sizeof(fmt));
  6313. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  6314. fmt.hdr.opcode = ASM_DATA_CMD_IEC_60958_MEDIA_FMT;
  6315. fmt.num_channels = channels;
  6316. fmt.sampling_rate = rate;
  6317. atomic_set(&ac->cmd_state, -1);
  6318. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6319. if (rc < 0) {
  6320. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6321. rc = -EINVAL;
  6322. goto fail_cmd;
  6323. }
  6324. rc = wait_event_timeout(ac->cmd_wait,
  6325. (atomic_read(&ac->cmd_state) >= 0),
  6326. msecs_to_jiffies(TIMEOUT_MS));
  6327. if (!rc) {
  6328. pr_err("%s: timeout. waited for format update\n", __func__);
  6329. rc = -ETIMEDOUT;
  6330. goto fail_cmd;
  6331. }
  6332. if (atomic_read(&ac->cmd_state) > 0) {
  6333. pr_err("%s: DSP returned error[%s]\n",
  6334. __func__, adsp_err_get_err_str(
  6335. atomic_read(&ac->cmd_state)));
  6336. rc = adsp_err_get_lnx_err_code(
  6337. atomic_read(&ac->cmd_state));
  6338. }
  6339. return 0;
  6340. fail_cmd:
  6341. return rc;
  6342. }
  6343. EXPORT_SYMBOL(q6asm_media_format_block_iec);
  6344. static int __q6asm_media_format_block_multi_aac(struct audio_client *ac,
  6345. struct asm_aac_cfg *cfg, int stream_id)
  6346. {
  6347. struct asm_aac_fmt_blk_v2 fmt;
  6348. int rc = 0;
  6349. pr_debug("%s: session[%d]rate[%d]ch[%d]\n", __func__, ac->session,
  6350. cfg->sample_rate, cfg->ch_cfg);
  6351. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6352. atomic_set(&ac->cmd_state, -1);
  6353. /*
  6354. * Updated the token field with stream/session for compressed playback
  6355. * Platform driver must know the the stream with which the command is
  6356. * associated
  6357. */
  6358. if (ac->io_mode & COMPRESSED_STREAM_IO)
  6359. q6asm_update_token(&fmt.hdr.token,
  6360. ac->session,
  6361. stream_id,
  6362. 0, /* Buffer index is NA */
  6363. 0, /* Direction flag is NA */
  6364. WAIT_CMD);
  6365. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  6366. __func__, fmt.hdr.token, stream_id, ac->session);
  6367. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6368. fmt.fmt_blk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6369. sizeof(fmt.fmt_blk);
  6370. fmt.aac_fmt_flag = cfg->format;
  6371. fmt.audio_objype = cfg->aot;
  6372. /* If zero, PCE is assumed to be available in bitstream*/
  6373. fmt.total_size_of_PCE_bits = 0;
  6374. fmt.channel_config = cfg->ch_cfg;
  6375. fmt.sample_rate = cfg->sample_rate;
  6376. pr_debug("%s: format=0x%x cfg_size=%d aac-cfg=0x%x aot=%d ch=%d sr=%d\n",
  6377. __func__, fmt.aac_fmt_flag, fmt.fmt_blk.fmt_blk_size,
  6378. fmt.aac_fmt_flag,
  6379. fmt.audio_objype,
  6380. fmt.channel_config,
  6381. fmt.sample_rate);
  6382. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6383. if (rc < 0) {
  6384. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6385. rc = -EINVAL;
  6386. goto fail_cmd;
  6387. }
  6388. rc = wait_event_timeout(ac->cmd_wait,
  6389. (atomic_read(&ac->cmd_state) >= 0),
  6390. msecs_to_jiffies(TIMEOUT_MS));
  6391. if (!rc) {
  6392. pr_err("%s: timeout. waited for FORMAT_UPDATE\n", __func__);
  6393. rc = -ETIMEDOUT;
  6394. goto fail_cmd;
  6395. }
  6396. if (atomic_read(&ac->cmd_state) > 0) {
  6397. pr_err("%s: DSP returned error[%s]\n",
  6398. __func__, adsp_err_get_err_str(
  6399. atomic_read(&ac->cmd_state)));
  6400. rc = adsp_err_get_lnx_err_code(
  6401. atomic_read(&ac->cmd_state));
  6402. goto fail_cmd;
  6403. }
  6404. return 0;
  6405. fail_cmd:
  6406. return rc;
  6407. }
  6408. /**
  6409. * q6asm_media_format_block_multi_aac -
  6410. * command to set mediafmt block for multi_aac on ASM stream
  6411. *
  6412. * @ac: Audio client handle
  6413. * @cfg: multi_aac config
  6414. *
  6415. * Returns 0 on success or error on failure
  6416. */
  6417. int q6asm_media_format_block_multi_aac(struct audio_client *ac,
  6418. struct asm_aac_cfg *cfg)
  6419. {
  6420. return __q6asm_media_format_block_multi_aac(ac, cfg, ac->stream_id);
  6421. }
  6422. EXPORT_SYMBOL(q6asm_media_format_block_multi_aac);
  6423. /**
  6424. * q6asm_media_format_block_aac -
  6425. * command to set mediafmt block for aac on ASM
  6426. *
  6427. * @ac: Audio client handle
  6428. * @cfg: aac config
  6429. *
  6430. * Returns 0 on success or error on failure
  6431. */
  6432. int q6asm_media_format_block_aac(struct audio_client *ac,
  6433. struct asm_aac_cfg *cfg)
  6434. {
  6435. return __q6asm_media_format_block_multi_aac(ac, cfg, ac->stream_id);
  6436. }
  6437. EXPORT_SYMBOL(q6asm_media_format_block_aac);
  6438. /**
  6439. * q6asm_stream_media_format_block_aac -
  6440. * command to set mediafmt block for aac on ASM stream
  6441. *
  6442. * @ac: Audio client handle
  6443. * @cfg: aac config
  6444. * @stream_id: stream ID info
  6445. *
  6446. * Returns 0 on success or error on failure
  6447. */
  6448. int q6asm_stream_media_format_block_aac(struct audio_client *ac,
  6449. struct asm_aac_cfg *cfg, int stream_id)
  6450. {
  6451. return __q6asm_media_format_block_multi_aac(ac, cfg, stream_id);
  6452. }
  6453. EXPORT_SYMBOL(q6asm_stream_media_format_block_aac);
  6454. /**
  6455. * q6asm_media_format_block_wma -
  6456. * command to set mediafmt block for wma on ASM stream
  6457. *
  6458. * @ac: Audio client handle
  6459. * @cfg: wma config
  6460. * @stream_id: stream ID info
  6461. *
  6462. * Returns 0 on success or error on failure
  6463. */
  6464. int q6asm_media_format_block_wma(struct audio_client *ac,
  6465. void *cfg, int stream_id)
  6466. {
  6467. struct asm_wmastdv9_fmt_blk_v2 fmt;
  6468. struct asm_wma_cfg *wma_cfg = (struct asm_wma_cfg *)cfg;
  6469. int rc = 0;
  6470. pr_debug("session[%d]format_tag[0x%4x] rate[%d] ch[0x%4x] bps[%d], balign[0x%4x], bit_sample[0x%4x], ch_msk[%d], enc_opt[0x%4x]\n",
  6471. ac->session, wma_cfg->format_tag, wma_cfg->sample_rate,
  6472. wma_cfg->ch_cfg, wma_cfg->avg_bytes_per_sec,
  6473. wma_cfg->block_align, wma_cfg->valid_bits_per_sample,
  6474. wma_cfg->ch_mask, wma_cfg->encode_opt);
  6475. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6476. atomic_set(&ac->cmd_state, -1);
  6477. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6478. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6479. sizeof(fmt.fmtblk);
  6480. fmt.fmtag = wma_cfg->format_tag;
  6481. fmt.num_channels = wma_cfg->ch_cfg;
  6482. fmt.sample_rate = wma_cfg->sample_rate;
  6483. fmt.avg_bytes_per_sec = wma_cfg->avg_bytes_per_sec;
  6484. fmt.blk_align = wma_cfg->block_align;
  6485. fmt.bits_per_sample =
  6486. wma_cfg->valid_bits_per_sample;
  6487. fmt.channel_mask = wma_cfg->ch_mask;
  6488. fmt.enc_options = wma_cfg->encode_opt;
  6489. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6490. if (rc < 0) {
  6491. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6492. rc = -EINVAL;
  6493. goto fail_cmd;
  6494. }
  6495. rc = wait_event_timeout(ac->cmd_wait,
  6496. (atomic_read(&ac->cmd_state) >= 0),
  6497. msecs_to_jiffies(TIMEOUT_MS));
  6498. if (!rc) {
  6499. pr_err("%s: timeout. waited for FORMAT_UPDATE\n", __func__);
  6500. rc = -ETIMEDOUT;
  6501. goto fail_cmd;
  6502. }
  6503. if (atomic_read(&ac->cmd_state) > 0) {
  6504. pr_err("%s: DSP returned error[%s]\n",
  6505. __func__, adsp_err_get_err_str(
  6506. atomic_read(&ac->cmd_state)));
  6507. rc = adsp_err_get_lnx_err_code(
  6508. atomic_read(&ac->cmd_state));
  6509. goto fail_cmd;
  6510. }
  6511. return 0;
  6512. fail_cmd:
  6513. return rc;
  6514. }
  6515. EXPORT_SYMBOL(q6asm_media_format_block_wma);
  6516. /**
  6517. * q6asm_media_format_block_wmapro -
  6518. * command to set mediafmt block for wmapro on ASM stream
  6519. *
  6520. * @ac: Audio client handle
  6521. * @cfg: wmapro config
  6522. * @stream_id: stream ID info
  6523. *
  6524. * Returns 0 on success or error on failure
  6525. */
  6526. int q6asm_media_format_block_wmapro(struct audio_client *ac,
  6527. void *cfg, int stream_id)
  6528. {
  6529. struct asm_wmaprov10_fmt_blk_v2 fmt;
  6530. struct asm_wmapro_cfg *wmapro_cfg = (struct asm_wmapro_cfg *)cfg;
  6531. int rc = 0;
  6532. pr_debug("%s: session[%d]format_tag[0x%4x] rate[%d] ch[0x%4x] bps[%d], balign[0x%4x], bit_sample[0x%4x], ch_msk[%d], enc_opt[0x%4x], adv_enc_opt[0x%4x], adv_enc_opt2[0x%8x]\n",
  6533. __func__,
  6534. ac->session, wmapro_cfg->format_tag, wmapro_cfg->sample_rate,
  6535. wmapro_cfg->ch_cfg, wmapro_cfg->avg_bytes_per_sec,
  6536. wmapro_cfg->block_align, wmapro_cfg->valid_bits_per_sample,
  6537. wmapro_cfg->ch_mask, wmapro_cfg->encode_opt,
  6538. wmapro_cfg->adv_encode_opt, wmapro_cfg->adv_encode_opt2);
  6539. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6540. atomic_set(&ac->cmd_state, -1);
  6541. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6542. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6543. sizeof(fmt.fmtblk);
  6544. fmt.fmtag = wmapro_cfg->format_tag;
  6545. fmt.num_channels = wmapro_cfg->ch_cfg;
  6546. fmt.sample_rate = wmapro_cfg->sample_rate;
  6547. fmt.avg_bytes_per_sec =
  6548. wmapro_cfg->avg_bytes_per_sec;
  6549. fmt.blk_align = wmapro_cfg->block_align;
  6550. fmt.bits_per_sample = wmapro_cfg->valid_bits_per_sample;
  6551. fmt.channel_mask = wmapro_cfg->ch_mask;
  6552. fmt.enc_options = wmapro_cfg->encode_opt;
  6553. fmt.usAdvancedEncodeOpt = wmapro_cfg->adv_encode_opt;
  6554. fmt.advanced_enc_options2 = wmapro_cfg->adv_encode_opt2;
  6555. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6556. if (rc < 0) {
  6557. pr_err("%s: Comamnd open failed %d\n", __func__, rc);
  6558. rc = -EINVAL;
  6559. goto fail_cmd;
  6560. }
  6561. rc = wait_event_timeout(ac->cmd_wait,
  6562. (atomic_read(&ac->cmd_state) >= 0),
  6563. msecs_to_jiffies(TIMEOUT_MS));
  6564. if (!rc) {
  6565. pr_err("%s: timeout. waited for FORMAT_UPDATE\n", __func__);
  6566. rc = -ETIMEDOUT;
  6567. goto fail_cmd;
  6568. }
  6569. if (atomic_read(&ac->cmd_state) > 0) {
  6570. pr_err("%s: DSP returned error[%s]\n",
  6571. __func__, adsp_err_get_err_str(
  6572. atomic_read(&ac->cmd_state)));
  6573. rc = adsp_err_get_lnx_err_code(
  6574. atomic_read(&ac->cmd_state));
  6575. goto fail_cmd;
  6576. }
  6577. return 0;
  6578. fail_cmd:
  6579. return rc;
  6580. }
  6581. EXPORT_SYMBOL(q6asm_media_format_block_wmapro);
  6582. /**
  6583. * q6asm_media_format_block_amrwbplus -
  6584. * command to set mediafmt block for amrwbplus on ASM stream
  6585. *
  6586. * @ac: Audio client handle
  6587. * @cfg: amrwbplus config
  6588. * @stream_id: stream ID info
  6589. *
  6590. * Returns 0 on success or error on failure
  6591. */
  6592. int q6asm_media_format_block_amrwbplus(struct audio_client *ac,
  6593. struct asm_amrwbplus_cfg *cfg)
  6594. {
  6595. struct asm_amrwbplus_fmt_blk_v2 fmt;
  6596. int rc = 0;
  6597. pr_debug("%s: session[%d]band-mode[%d]frame-fmt[%d]ch[%d]\n",
  6598. __func__,
  6599. ac->session,
  6600. cfg->amr_band_mode,
  6601. cfg->amr_frame_fmt,
  6602. cfg->num_channels);
  6603. q6asm_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE);
  6604. atomic_set(&ac->cmd_state, -1);
  6605. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6606. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6607. sizeof(fmt.fmtblk);
  6608. fmt.amr_frame_fmt = cfg->amr_frame_fmt;
  6609. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6610. if (rc < 0) {
  6611. pr_err("%s: Comamnd media format update failed.. %d\n",
  6612. __func__, rc);
  6613. rc = -EINVAL;
  6614. goto fail_cmd;
  6615. }
  6616. rc = wait_event_timeout(ac->cmd_wait,
  6617. (atomic_read(&ac->cmd_state) >= 0),
  6618. msecs_to_jiffies(TIMEOUT_MS));
  6619. if (!rc) {
  6620. pr_err("%s: timeout. waited for FORMAT_UPDATE\n", __func__);
  6621. rc = -ETIMEDOUT;
  6622. goto fail_cmd;
  6623. }
  6624. if (atomic_read(&ac->cmd_state) > 0) {
  6625. pr_err("%s: DSP returned error[%s]\n",
  6626. __func__, adsp_err_get_err_str(
  6627. atomic_read(&ac->cmd_state)));
  6628. rc = adsp_err_get_lnx_err_code(
  6629. atomic_read(&ac->cmd_state));
  6630. goto fail_cmd;
  6631. }
  6632. return 0;
  6633. fail_cmd:
  6634. return rc;
  6635. }
  6636. EXPORT_SYMBOL(q6asm_media_format_block_amrwbplus);
  6637. /**
  6638. * q6asm_stream_media_format_block_flac -
  6639. * command to set mediafmt block for flac on ASM stream
  6640. *
  6641. * @ac: Audio client handle
  6642. * @cfg: FLAC config
  6643. * @stream_id: stream ID info
  6644. *
  6645. * Returns 0 on success or error on failure
  6646. */
  6647. int q6asm_stream_media_format_block_flac(struct audio_client *ac,
  6648. struct asm_flac_cfg *cfg, int stream_id)
  6649. {
  6650. struct asm_flac_fmt_blk_v2 fmt;
  6651. int rc = 0;
  6652. pr_debug("%s :session[%d] rate[%d] ch[%d] size[%d] stream_id[%d]\n",
  6653. __func__, ac->session, cfg->sample_rate, cfg->ch_cfg,
  6654. cfg->sample_size, stream_id);
  6655. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6656. atomic_set(&ac->cmd_state, -1);
  6657. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6658. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6659. sizeof(fmt.fmtblk);
  6660. fmt.is_stream_info_present = cfg->stream_info_present;
  6661. fmt.num_channels = cfg->ch_cfg;
  6662. fmt.min_blk_size = cfg->min_blk_size;
  6663. fmt.max_blk_size = cfg->max_blk_size;
  6664. fmt.sample_rate = cfg->sample_rate;
  6665. fmt.min_frame_size = cfg->min_frame_size;
  6666. fmt.max_frame_size = cfg->max_frame_size;
  6667. fmt.sample_size = cfg->sample_size;
  6668. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6669. if (rc < 0) {
  6670. pr_err("%s :Comamnd media format update failed %d\n",
  6671. __func__, rc);
  6672. goto fail_cmd;
  6673. }
  6674. rc = wait_event_timeout(ac->cmd_wait,
  6675. (atomic_read(&ac->cmd_state) >= 0),
  6676. msecs_to_jiffies(TIMEOUT_MS));
  6677. if (!rc) {
  6678. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  6679. rc = -ETIMEDOUT;
  6680. goto fail_cmd;
  6681. }
  6682. if (atomic_read(&ac->cmd_state) > 0) {
  6683. pr_err("%s: DSP returned error[%s]\n",
  6684. __func__, adsp_err_get_err_str(
  6685. atomic_read(&ac->cmd_state)));
  6686. rc = adsp_err_get_lnx_err_code(
  6687. atomic_read(&ac->cmd_state));
  6688. goto fail_cmd;
  6689. }
  6690. return 0;
  6691. fail_cmd:
  6692. return rc;
  6693. }
  6694. EXPORT_SYMBOL(q6asm_stream_media_format_block_flac);
  6695. /**
  6696. * q6asm_media_format_block_alac -
  6697. * command to set mediafmt block for alac on ASM stream
  6698. *
  6699. * @ac: Audio client handle
  6700. * @cfg: ALAC config
  6701. * @stream_id: stream ID info
  6702. *
  6703. * Returns 0 on success or error on failure
  6704. */
  6705. int q6asm_media_format_block_alac(struct audio_client *ac,
  6706. struct asm_alac_cfg *cfg, int stream_id)
  6707. {
  6708. struct asm_alac_fmt_blk_v2 fmt;
  6709. int rc = 0;
  6710. pr_debug("%s :session[%d]rate[%d]ch[%d]\n", __func__,
  6711. ac->session, cfg->sample_rate, cfg->num_channels);
  6712. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6713. atomic_set(&ac->cmd_state, -1);
  6714. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6715. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6716. sizeof(fmt.fmtblk);
  6717. fmt.frame_length = cfg->frame_length;
  6718. fmt.compatible_version = cfg->compatible_version;
  6719. fmt.bit_depth = cfg->bit_depth;
  6720. fmt.pb = cfg->pb;
  6721. fmt.mb = cfg->mb;
  6722. fmt.kb = cfg->kb;
  6723. fmt.num_channels = cfg->num_channels;
  6724. fmt.max_run = cfg->max_run;
  6725. fmt.max_frame_bytes = cfg->max_frame_bytes;
  6726. fmt.avg_bit_rate = cfg->avg_bit_rate;
  6727. fmt.sample_rate = cfg->sample_rate;
  6728. fmt.channel_layout_tag = cfg->channel_layout_tag;
  6729. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6730. if (rc < 0) {
  6731. pr_err("%s :Comamnd media format update failed %d\n",
  6732. __func__, rc);
  6733. goto fail_cmd;
  6734. }
  6735. rc = wait_event_timeout(ac->cmd_wait,
  6736. (atomic_read(&ac->cmd_state) >= 0),
  6737. msecs_to_jiffies(TIMEOUT_MS));
  6738. if (!rc) {
  6739. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  6740. rc = -ETIMEDOUT;
  6741. goto fail_cmd;
  6742. }
  6743. if (atomic_read(&ac->cmd_state) > 0) {
  6744. pr_err("%s: DSP returned error[%s]\n",
  6745. __func__, adsp_err_get_err_str(
  6746. atomic_read(&ac->cmd_state)));
  6747. rc = adsp_err_get_lnx_err_code(
  6748. atomic_read(&ac->cmd_state));
  6749. goto fail_cmd;
  6750. }
  6751. return 0;
  6752. fail_cmd:
  6753. return rc;
  6754. }
  6755. EXPORT_SYMBOL(q6asm_media_format_block_alac);
  6756. /*
  6757. * q6asm_media_format_block_g711 - sends g711 decoder configuration
  6758. * parameters
  6759. * @ac: Client session handle
  6760. * @cfg: Audio stream manager configuration parameters
  6761. * @stream_id: Stream id
  6762. */
  6763. int q6asm_media_format_block_g711(struct audio_client *ac,
  6764. struct asm_g711_dec_cfg *cfg, int stream_id)
  6765. {
  6766. struct asm_g711_dec_fmt_blk_v2 fmt;
  6767. int rc = 0;
  6768. if (!ac) {
  6769. pr_err("%s: audio client is null\n", __func__);
  6770. return -EINVAL;
  6771. }
  6772. if (!cfg) {
  6773. pr_err("%s: Invalid ASM config\n", __func__);
  6774. return -EINVAL;
  6775. }
  6776. if (stream_id <= 0) {
  6777. pr_err("%s: Invalid stream id\n", __func__);
  6778. return -EINVAL;
  6779. }
  6780. pr_debug("%s :session[%d]rate[%d]\n", __func__,
  6781. ac->session, cfg->sample_rate);
  6782. memset(&fmt, 0, sizeof(struct asm_g711_dec_fmt_blk_v2));
  6783. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6784. atomic_set(&ac->cmd_state, -1);
  6785. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6786. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6787. sizeof(fmt.fmtblk);
  6788. fmt.sample_rate = cfg->sample_rate;
  6789. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6790. if (rc < 0) {
  6791. pr_err("%s :Command media format update failed %d\n",
  6792. __func__, rc);
  6793. goto fail_cmd;
  6794. }
  6795. rc = wait_event_timeout(ac->cmd_wait,
  6796. (atomic_read(&ac->cmd_state) >= 0),
  6797. msecs_to_jiffies(TIMEOUT_MS));
  6798. if (!rc) {
  6799. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  6800. rc = -ETIMEDOUT;
  6801. goto fail_cmd;
  6802. }
  6803. if (atomic_read(&ac->cmd_state) > 0) {
  6804. pr_err("%s: DSP returned error[%s]\n",
  6805. __func__, adsp_err_get_err_str(
  6806. atomic_read(&ac->cmd_state)));
  6807. rc = adsp_err_get_lnx_err_code(
  6808. atomic_read(&ac->cmd_state));
  6809. goto fail_cmd;
  6810. }
  6811. return 0;
  6812. fail_cmd:
  6813. return rc;
  6814. }
  6815. EXPORT_SYMBOL(q6asm_media_format_block_g711);
  6816. /**
  6817. * q6asm_stream_media_format_block_vorbis -
  6818. * command to set mediafmt block for vorbis on ASM stream
  6819. *
  6820. * @ac: Audio client handle
  6821. * @cfg: vorbis config
  6822. * @stream_id: stream ID info
  6823. *
  6824. * Returns 0 on success or error on failure
  6825. */
  6826. int q6asm_stream_media_format_block_vorbis(struct audio_client *ac,
  6827. struct asm_vorbis_cfg *cfg, int stream_id)
  6828. {
  6829. struct asm_vorbis_fmt_blk_v2 fmt;
  6830. int rc = 0;
  6831. pr_debug("%s :session[%d] bit_stream_fmt[%d] stream_id[%d]\n",
  6832. __func__, ac->session, cfg->bit_stream_fmt, stream_id);
  6833. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6834. atomic_set(&ac->cmd_state, -1);
  6835. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6836. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6837. sizeof(fmt.fmtblk);
  6838. fmt.bit_stream_fmt = cfg->bit_stream_fmt;
  6839. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6840. if (rc < 0) {
  6841. pr_err("%s :Comamnd media format update failed %d\n",
  6842. __func__, rc);
  6843. goto fail_cmd;
  6844. }
  6845. rc = wait_event_timeout(ac->cmd_wait,
  6846. (atomic_read(&ac->cmd_state) >= 0),
  6847. msecs_to_jiffies(TIMEOUT_MS));
  6848. if (!rc) {
  6849. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  6850. rc = -ETIMEDOUT;
  6851. goto fail_cmd;
  6852. }
  6853. if (atomic_read(&ac->cmd_state) > 0) {
  6854. pr_err("%s: DSP returned error[%s]\n",
  6855. __func__, adsp_err_get_err_str(
  6856. atomic_read(&ac->cmd_state)));
  6857. rc = adsp_err_get_lnx_err_code(
  6858. atomic_read(&ac->cmd_state));
  6859. goto fail_cmd;
  6860. }
  6861. return 0;
  6862. fail_cmd:
  6863. return rc;
  6864. }
  6865. EXPORT_SYMBOL(q6asm_stream_media_format_block_vorbis);
  6866. /**
  6867. * q6asm_media_format_block_ape -
  6868. * command to set mediafmt block for APE on ASM stream
  6869. *
  6870. * @ac: Audio client handle
  6871. * @cfg: APE config
  6872. * @stream_id: stream ID info
  6873. *
  6874. * Returns 0 on success or error on failure
  6875. */
  6876. int q6asm_media_format_block_ape(struct audio_client *ac,
  6877. struct asm_ape_cfg *cfg, int stream_id)
  6878. {
  6879. struct asm_ape_fmt_blk_v2 fmt;
  6880. int rc = 0;
  6881. pr_debug("%s :session[%d]rate[%d]ch[%d]\n", __func__,
  6882. ac->session, cfg->sample_rate, cfg->num_channels);
  6883. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6884. atomic_set(&ac->cmd_state, -1);
  6885. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6886. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6887. sizeof(fmt.fmtblk);
  6888. fmt.compatible_version = cfg->compatible_version;
  6889. fmt.compression_level = cfg->compression_level;
  6890. fmt.format_flags = cfg->format_flags;
  6891. fmt.blocks_per_frame = cfg->blocks_per_frame;
  6892. fmt.final_frame_blocks = cfg->final_frame_blocks;
  6893. fmt.total_frames = cfg->total_frames;
  6894. fmt.bits_per_sample = cfg->bits_per_sample;
  6895. fmt.num_channels = cfg->num_channels;
  6896. fmt.sample_rate = cfg->sample_rate;
  6897. fmt.seek_table_present = cfg->seek_table_present;
  6898. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6899. if (rc < 0) {
  6900. pr_err("%s :Comamnd media format update failed %d\n",
  6901. __func__, rc);
  6902. goto fail_cmd;
  6903. }
  6904. rc = wait_event_timeout(ac->cmd_wait,
  6905. (atomic_read(&ac->cmd_state) >= 0),
  6906. msecs_to_jiffies(TIMEOUT_MS));
  6907. if (!rc) {
  6908. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  6909. rc = -ETIMEDOUT;
  6910. goto fail_cmd;
  6911. }
  6912. if (atomic_read(&ac->cmd_state) > 0) {
  6913. pr_err("%s: DSP returned error[%s]\n",
  6914. __func__, adsp_err_get_err_str(
  6915. atomic_read(&ac->cmd_state)));
  6916. rc = adsp_err_get_lnx_err_code(
  6917. atomic_read(&ac->cmd_state));
  6918. goto fail_cmd;
  6919. }
  6920. return 0;
  6921. fail_cmd:
  6922. return rc;
  6923. }
  6924. EXPORT_SYMBOL(q6asm_media_format_block_ape);
  6925. /*
  6926. * q6asm_media_format_block_dsd- Sends DSD Decoder
  6927. * configuration parameters
  6928. *
  6929. * @ac: Client session handle
  6930. * @cfg: DSD Media Format Configuration.
  6931. * @stream_id: stream id of stream to be associated with this session
  6932. *
  6933. * Return 0 on success or negative error code on failure
  6934. */
  6935. int q6asm_media_format_block_dsd(struct audio_client *ac,
  6936. struct asm_dsd_cfg *cfg, int stream_id)
  6937. {
  6938. struct asm_dsd_fmt_blk_v2 fmt;
  6939. int rc;
  6940. pr_debug("%s: session[%d] data_rate[%d] ch[%d]\n", __func__,
  6941. ac->session, cfg->dsd_data_rate, cfg->num_channels);
  6942. memset(&fmt, 0, sizeof(fmt));
  6943. q6asm_stream_add_hdr(ac, &fmt.hdr, sizeof(fmt), TRUE, stream_id);
  6944. fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  6945. fmt.fmtblk.fmt_blk_size = sizeof(fmt) - sizeof(fmt.hdr) -
  6946. sizeof(fmt.fmtblk);
  6947. fmt.num_version = cfg->num_version;
  6948. fmt.is_bitwise_big_endian = cfg->is_bitwise_big_endian;
  6949. fmt.dsd_channel_block_size = cfg->dsd_channel_block_size;
  6950. fmt.num_channels = cfg->num_channels;
  6951. fmt.dsd_data_rate = cfg->dsd_data_rate;
  6952. atomic_set(&ac->cmd_state, -1);
  6953. rc = apr_send_pkt(ac->apr, (uint32_t *) &fmt);
  6954. if (rc < 0) {
  6955. pr_err("%s: Command DSD media format update failed, err: %d\n",
  6956. __func__, rc);
  6957. goto done;
  6958. }
  6959. rc = wait_event_timeout(ac->cmd_wait,
  6960. (atomic_read(&ac->cmd_state) >= 0),
  6961. msecs_to_jiffies(TIMEOUT_MS));
  6962. if (!rc) {
  6963. pr_err("%s: timeout. waited for DSD FORMAT_UPDATE\n", __func__);
  6964. rc = -ETIMEDOUT;
  6965. goto done;
  6966. }
  6967. if (atomic_read(&ac->cmd_state) > 0) {
  6968. pr_err("%s: DSP returned error[%s]\n",
  6969. __func__, adsp_err_get_err_str(
  6970. atomic_read(&ac->cmd_state)));
  6971. rc = adsp_err_get_lnx_err_code(
  6972. atomic_read(&ac->cmd_state));
  6973. goto done;
  6974. }
  6975. return 0;
  6976. done:
  6977. return rc;
  6978. }
  6979. EXPORT_SYMBOL(q6asm_media_format_block_dsd);
  6980. /**
  6981. * q6asm_stream_media_format_block_aptx_dec -
  6982. * command to set mediafmt block for APTX dec on ASM stream
  6983. *
  6984. * @ac: Audio client handle
  6985. * @srate: sample rate
  6986. * @stream_id: stream ID info
  6987. *
  6988. * Returns 0 on success or error on failure
  6989. */
  6990. int q6asm_stream_media_format_block_aptx_dec(struct audio_client *ac,
  6991. uint32_t srate, int stream_id)
  6992. {
  6993. struct asm_aptx_dec_fmt_blk_v2 aptx_fmt;
  6994. int rc = 0;
  6995. if (!ac->session) {
  6996. pr_err("%s: ac session invalid\n", __func__);
  6997. rc = -EINVAL;
  6998. goto fail_cmd;
  6999. }
  7000. pr_debug("%s :session[%d] rate[%d] stream_id[%d]\n",
  7001. __func__, ac->session, srate, stream_id);
  7002. q6asm_stream_add_hdr(ac, &aptx_fmt.hdr, sizeof(aptx_fmt), TRUE,
  7003. stream_id);
  7004. atomic_set(&ac->cmd_state, -1);
  7005. aptx_fmt.hdr.opcode = ASM_DATA_CMD_MEDIA_FMT_UPDATE_V2;
  7006. aptx_fmt.fmtblk.fmt_blk_size = sizeof(aptx_fmt) - sizeof(aptx_fmt.hdr) -
  7007. sizeof(aptx_fmt.fmtblk);
  7008. aptx_fmt.sample_rate = srate;
  7009. rc = apr_send_pkt(ac->apr, (uint32_t *) &aptx_fmt);
  7010. if (rc < 0) {
  7011. pr_err("%s :Comamnd media format update failed %d\n",
  7012. __func__, rc);
  7013. goto fail_cmd;
  7014. }
  7015. rc = wait_event_timeout(ac->cmd_wait,
  7016. (atomic_read(&ac->cmd_state) >= 0),
  7017. msecs_to_jiffies(TIMEOUT_MS));
  7018. if (!rc) {
  7019. pr_err("%s :timeout. waited for FORMAT_UPDATE\n", __func__);
  7020. rc = -ETIMEDOUT;
  7021. goto fail_cmd;
  7022. }
  7023. if (atomic_read(&ac->cmd_state) > 0) {
  7024. pr_err("%s: DSP returned error[%s]\n",
  7025. __func__, adsp_err_get_err_str(
  7026. atomic_read(&ac->cmd_state)));
  7027. rc = adsp_err_get_lnx_err_code(
  7028. atomic_read(&ac->cmd_state));
  7029. goto fail_cmd;
  7030. }
  7031. rc = 0;
  7032. fail_cmd:
  7033. return rc;
  7034. }
  7035. EXPORT_SYMBOL(q6asm_stream_media_format_block_aptx_dec);
  7036. static int __q6asm_ds1_set_endp_params(struct audio_client *ac, int param_id,
  7037. int param_value, int stream_id)
  7038. {
  7039. struct asm_dec_ddp_endp_param_v2 ddp_cfg;
  7040. int rc = 0;
  7041. pr_debug("%s: session[%d] stream[%d],param_id[%d]param_value[%d]",
  7042. __func__, ac->session, stream_id, param_id, param_value);
  7043. q6asm_stream_add_hdr(ac, &ddp_cfg.hdr, sizeof(ddp_cfg), TRUE,
  7044. stream_id);
  7045. atomic_set(&ac->cmd_state, -1);
  7046. /*
  7047. * Updated the token field with stream/session for compressed playback
  7048. * Platform driver must know the stream with which the command is
  7049. * associated
  7050. */
  7051. if (ac->io_mode & COMPRESSED_STREAM_IO)
  7052. q6asm_update_token(&ddp_cfg.hdr.token,
  7053. ac->session,
  7054. stream_id,
  7055. 0, /* Buffer index is NA */
  7056. 0, /* Direction flag is NA */
  7057. WAIT_CMD);
  7058. ddp_cfg.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  7059. ddp_cfg.encdec.param_id = param_id;
  7060. ddp_cfg.encdec.param_size = sizeof(struct asm_dec_ddp_endp_param_v2) -
  7061. (sizeof(struct apr_hdr) +
  7062. sizeof(struct asm_stream_cmd_set_encdec_param));
  7063. ddp_cfg.endp_param_value = param_value;
  7064. rc = apr_send_pkt(ac->apr, (uint32_t *) &ddp_cfg);
  7065. if (rc < 0) {
  7066. pr_err("%s: Command opcode[0x%x] failed %d\n",
  7067. __func__, ASM_STREAM_CMD_SET_ENCDEC_PARAM, rc);
  7068. goto fail_cmd;
  7069. }
  7070. rc = wait_event_timeout(ac->cmd_wait,
  7071. (atomic_read(&ac->cmd_state) >= 0),
  7072. msecs_to_jiffies(TIMEOUT_MS));
  7073. if (!rc) {
  7074. pr_err("%s: timeout opcode[0x%x]\n", __func__,
  7075. ddp_cfg.hdr.opcode);
  7076. rc = -ETIMEDOUT;
  7077. goto fail_cmd;
  7078. }
  7079. if (atomic_read(&ac->cmd_state) > 0) {
  7080. pr_err("%s: DSP returned error[%s]\n",
  7081. __func__, adsp_err_get_err_str(
  7082. atomic_read(&ac->cmd_state)));
  7083. rc = adsp_err_get_lnx_err_code(
  7084. atomic_read(&ac->cmd_state));
  7085. goto fail_cmd;
  7086. }
  7087. return 0;
  7088. fail_cmd:
  7089. return rc;
  7090. }
  7091. /**
  7092. * q6asm_ds1_set_endp_params -
  7093. * command to set DS1 params for ASM
  7094. *
  7095. * @ac: Audio client handle
  7096. * @param_id: param id
  7097. * @param_value: value of param
  7098. *
  7099. * Returns 0 on success or error on failure
  7100. */
  7101. int q6asm_ds1_set_endp_params(struct audio_client *ac,
  7102. int param_id, int param_value)
  7103. {
  7104. return __q6asm_ds1_set_endp_params(ac, param_id, param_value,
  7105. ac->stream_id);
  7106. }
  7107. /**
  7108. * q6asm_ds1_set_stream_endp_params -
  7109. * command to set DS1 params for ASM stream
  7110. *
  7111. * @ac: Audio client handle
  7112. * @param_id: param id
  7113. * @param_value: value of param
  7114. * @stream_id: stream ID info
  7115. *
  7116. * Returns 0 on success or error on failure
  7117. */
  7118. int q6asm_ds1_set_stream_endp_params(struct audio_client *ac,
  7119. int param_id, int param_value,
  7120. int stream_id)
  7121. {
  7122. return __q6asm_ds1_set_endp_params(ac, param_id, param_value,
  7123. stream_id);
  7124. }
  7125. EXPORT_SYMBOL(q6asm_ds1_set_stream_endp_params);
  7126. /**
  7127. * q6asm_memory_map -
  7128. * command to send memory map for ASM
  7129. *
  7130. * @ac: Audio client handle
  7131. * @buf_add: buffer address to map
  7132. * @dir: RX or TX session
  7133. * @bufsz: size of each buffer
  7134. * @bufcnt: buffer count
  7135. *
  7136. * Returns 0 on success or error on failure
  7137. */
  7138. int q6asm_memory_map(struct audio_client *ac, phys_addr_t buf_add, int dir,
  7139. uint32_t bufsz, uint32_t bufcnt)
  7140. {
  7141. struct avs_cmd_shared_mem_map_regions *mmap_regions = NULL;
  7142. struct avs_shared_map_region_payload *mregions = NULL;
  7143. struct audio_port_data *port = NULL;
  7144. void *mmap_region_cmd = NULL;
  7145. void *payload = NULL;
  7146. struct asm_buffer_node *buffer_node = NULL;
  7147. int rc = 0;
  7148. int cmd_size = 0;
  7149. if (!ac) {
  7150. pr_err("%s: APR handle NULL\n", __func__);
  7151. return -EINVAL;
  7152. }
  7153. if (ac->mmap_apr == NULL) {
  7154. pr_err("%s: mmap APR handle NULL\n", __func__);
  7155. return -EINVAL;
  7156. }
  7157. pr_debug("%s: Session[%d]\n", __func__, ac->session);
  7158. buffer_node = kmalloc(sizeof(struct asm_buffer_node), GFP_KERNEL);
  7159. if (!buffer_node)
  7160. return -ENOMEM;
  7161. cmd_size = sizeof(struct avs_cmd_shared_mem_map_regions)
  7162. + sizeof(struct avs_shared_map_region_payload) * bufcnt;
  7163. mmap_region_cmd = kzalloc(cmd_size, GFP_KERNEL);
  7164. if (mmap_region_cmd == NULL) {
  7165. rc = -EINVAL;
  7166. kfree(buffer_node);
  7167. return rc;
  7168. }
  7169. mmap_regions = (struct avs_cmd_shared_mem_map_regions *)
  7170. mmap_region_cmd;
  7171. q6asm_add_mmaphdr(ac, &mmap_regions->hdr, cmd_size, dir);
  7172. atomic_set(&ac->mem_state, -1);
  7173. mmap_regions->hdr.opcode = ASM_CMD_SHARED_MEM_MAP_REGIONS;
  7174. mmap_regions->mem_pool_id = ADSP_MEMORY_MAP_SHMEM8_4K_POOL;
  7175. mmap_regions->num_regions = bufcnt & 0x00ff;
  7176. mmap_regions->property_flag = 0x00;
  7177. payload = ((u8 *) mmap_region_cmd +
  7178. sizeof(struct avs_cmd_shared_mem_map_regions));
  7179. mregions = (struct avs_shared_map_region_payload *)payload;
  7180. ac->port[dir].tmp_hdl = 0;
  7181. port = &ac->port[dir];
  7182. pr_debug("%s: buf_add 0x%pK, bufsz: %d\n", __func__,
  7183. &buf_add, bufsz);
  7184. mregions->shm_addr_lsw = lower_32_bits(buf_add);
  7185. mregions->shm_addr_msw = msm_audio_populate_upper_32_bits(buf_add);
  7186. mregions->mem_size_bytes = bufsz;
  7187. ++mregions;
  7188. rc = apr_send_pkt(ac->mmap_apr, (uint32_t *) mmap_region_cmd);
  7189. if (rc < 0) {
  7190. pr_err("%s: mmap op[0x%x]rc[%d]\n", __func__,
  7191. mmap_regions->hdr.opcode, rc);
  7192. rc = -EINVAL;
  7193. kfree(buffer_node);
  7194. goto fail_cmd;
  7195. }
  7196. rc = wait_event_timeout(ac->mem_wait,
  7197. (atomic_read(&ac->mem_state) >= 0 &&
  7198. ac->port[dir].tmp_hdl),
  7199. msecs_to_jiffies(TIMEOUT_MS));
  7200. if (!rc) {
  7201. pr_err("%s: timeout. waited for memory_map\n", __func__);
  7202. rc = -ETIMEDOUT;
  7203. kfree(buffer_node);
  7204. goto fail_cmd;
  7205. }
  7206. if (atomic_read(&ac->mem_state) > 0) {
  7207. pr_err("%s: DSP returned error[%s] for memory_map\n",
  7208. __func__, adsp_err_get_err_str(
  7209. atomic_read(&ac->mem_state)));
  7210. rc = adsp_err_get_lnx_err_code(
  7211. atomic_read(&ac->mem_state));
  7212. kfree(buffer_node);
  7213. goto fail_cmd;
  7214. }
  7215. buffer_node->buf_phys_addr = buf_add;
  7216. buffer_node->mmap_hdl = ac->port[dir].tmp_hdl;
  7217. list_add_tail(&buffer_node->list, &ac->port[dir].mem_map_handle);
  7218. ac->port[dir].tmp_hdl = 0;
  7219. rc = 0;
  7220. fail_cmd:
  7221. kfree(mmap_region_cmd);
  7222. return rc;
  7223. }
  7224. EXPORT_SYMBOL(q6asm_memory_map);
  7225. /**
  7226. * q6asm_memory_unmap -
  7227. * command to send memory unmap for ASM
  7228. *
  7229. * @ac: Audio client handle
  7230. * @buf_add: buffer address to unmap
  7231. * @dir: RX or TX session
  7232. *
  7233. * Returns 0 on success or error on failure
  7234. */
  7235. int q6asm_memory_unmap(struct audio_client *ac, phys_addr_t buf_add, int dir)
  7236. {
  7237. struct avs_cmd_shared_mem_unmap_regions mem_unmap;
  7238. struct asm_buffer_node *buf_node = NULL;
  7239. struct list_head *ptr, *next;
  7240. int rc = 0;
  7241. if (!ac) {
  7242. pr_err("%s: APR handle NULL\n", __func__);
  7243. return -EINVAL;
  7244. }
  7245. if (this_mmap.apr == NULL) {
  7246. pr_err("%s: APR handle NULL\n", __func__);
  7247. return -EINVAL;
  7248. }
  7249. pr_debug("%s: Session[%d]\n", __func__, ac->session);
  7250. q6asm_add_mmaphdr(ac, &mem_unmap.hdr,
  7251. sizeof(struct avs_cmd_shared_mem_unmap_regions),
  7252. dir);
  7253. atomic_set(&ac->mem_state, -1);
  7254. mem_unmap.hdr.opcode = ASM_CMD_SHARED_MEM_UNMAP_REGIONS;
  7255. mem_unmap.mem_map_handle = 0;
  7256. list_for_each_safe(ptr, next, &ac->port[dir].mem_map_handle) {
  7257. buf_node = list_entry(ptr, struct asm_buffer_node,
  7258. list);
  7259. if (buf_node->buf_phys_addr == buf_add) {
  7260. pr_debug("%s: Found the element\n", __func__);
  7261. mem_unmap.mem_map_handle = buf_node->mmap_hdl;
  7262. break;
  7263. }
  7264. }
  7265. pr_debug("%s: mem_unmap-mem_map_handle: 0x%x\n",
  7266. __func__, mem_unmap.mem_map_handle);
  7267. if (mem_unmap.mem_map_handle == 0) {
  7268. pr_err("%s: Do not send null mem handle to DSP\n", __func__);
  7269. rc = 0;
  7270. goto fail_cmd;
  7271. }
  7272. rc = apr_send_pkt(ac->mmap_apr, (uint32_t *) &mem_unmap);
  7273. if (rc < 0) {
  7274. pr_err("%s: mem_unmap op[0x%x]rc[%d]\n", __func__,
  7275. mem_unmap.hdr.opcode, rc);
  7276. rc = -EINVAL;
  7277. goto fail_cmd;
  7278. }
  7279. rc = wait_event_timeout(ac->mem_wait,
  7280. (atomic_read(&ac->mem_state) >= 0),
  7281. msecs_to_jiffies(TIMEOUT_MS));
  7282. if (!rc) {
  7283. pr_err("%s: timeout. waited for memory_unmap of handle 0x%x\n",
  7284. __func__, mem_unmap.mem_map_handle);
  7285. rc = -ETIMEDOUT;
  7286. goto fail_cmd;
  7287. } else if (atomic_read(&ac->mem_state) > 0) {
  7288. pr_err("%s DSP returned error [%s] map handle 0x%x\n",
  7289. __func__, adsp_err_get_err_str(
  7290. atomic_read(&ac->mem_state)),
  7291. mem_unmap.mem_map_handle);
  7292. rc = adsp_err_get_lnx_err_code(
  7293. atomic_read(&ac->mem_state));
  7294. goto fail_cmd;
  7295. } else if (atomic_read(&ac->unmap_cb_success) == 0) {
  7296. pr_err("%s: Error in mem unmap callback of handle 0x%x\n",
  7297. __func__, mem_unmap.mem_map_handle);
  7298. rc = -EINVAL;
  7299. goto fail_cmd;
  7300. }
  7301. rc = 0;
  7302. fail_cmd:
  7303. list_for_each_safe(ptr, next, &ac->port[dir].mem_map_handle) {
  7304. buf_node = list_entry(ptr, struct asm_buffer_node,
  7305. list);
  7306. if (buf_node->buf_phys_addr == buf_add) {
  7307. list_del(&buf_node->list);
  7308. kfree(buf_node);
  7309. break;
  7310. }
  7311. }
  7312. return rc;
  7313. }
  7314. EXPORT_SYMBOL(q6asm_memory_unmap);
  7315. /**
  7316. * q6asm_memory_map_regions -
  7317. * command to send memory map regions for ASM
  7318. *
  7319. * @ac: Audio client handle
  7320. * @dir: RX or TX session
  7321. * @bufsz: size of each buffer
  7322. * @bufcnt: buffer count
  7323. * @is_contiguous: alloc contiguous mem or not
  7324. *
  7325. * Returns 0 on success or error on failure
  7326. */
  7327. static int q6asm_memory_map_regions(struct audio_client *ac, int dir,
  7328. uint32_t bufsz, uint32_t bufcnt,
  7329. bool is_contiguous)
  7330. {
  7331. struct avs_cmd_shared_mem_map_regions *mmap_regions = NULL;
  7332. struct avs_shared_map_region_payload *mregions = NULL;
  7333. struct audio_port_data *port = NULL;
  7334. struct audio_buffer *ab = NULL;
  7335. void *mmap_region_cmd = NULL;
  7336. void *payload = NULL;
  7337. struct asm_buffer_node *buffer_node = NULL;
  7338. int rc = 0;
  7339. int i = 0;
  7340. uint32_t cmd_size = 0;
  7341. uint32_t bufcnt_t;
  7342. uint32_t bufsz_t;
  7343. if (!ac) {
  7344. pr_err("%s: APR handle NULL\n", __func__);
  7345. return -EINVAL;
  7346. }
  7347. if (ac->mmap_apr == NULL) {
  7348. pr_err("%s: mmap APR handle NULL\n", __func__);
  7349. return -EINVAL;
  7350. }
  7351. pr_debug("%s: Session[%d]\n", __func__, ac->session);
  7352. bufcnt_t = (is_contiguous) ? 1 : bufcnt;
  7353. bufsz_t = (is_contiguous) ? (bufsz * bufcnt) : bufsz;
  7354. if (is_contiguous) {
  7355. /* The size to memory map should be multiple of 4K bytes */
  7356. bufsz_t = PAGE_ALIGN(bufsz_t);
  7357. }
  7358. if (bufcnt_t > (UINT_MAX
  7359. - sizeof(struct avs_cmd_shared_mem_map_regions))
  7360. / sizeof(struct avs_shared_map_region_payload)) {
  7361. pr_err("%s: Unsigned Integer Overflow. bufcnt_t = %u\n",
  7362. __func__, bufcnt_t);
  7363. return -EINVAL;
  7364. }
  7365. cmd_size = sizeof(struct avs_cmd_shared_mem_map_regions)
  7366. + (sizeof(struct avs_shared_map_region_payload)
  7367. * bufcnt_t);
  7368. if (bufcnt > (UINT_MAX / sizeof(struct asm_buffer_node))) {
  7369. pr_err("%s: Unsigned Integer Overflow. bufcnt = %u\n",
  7370. __func__, bufcnt);
  7371. return -EINVAL;
  7372. }
  7373. buffer_node = kzalloc(sizeof(struct asm_buffer_node) * bufcnt,
  7374. GFP_KERNEL);
  7375. if (!buffer_node)
  7376. return -ENOMEM;
  7377. mmap_region_cmd = kzalloc(cmd_size, GFP_KERNEL);
  7378. if (mmap_region_cmd == NULL) {
  7379. rc = -EINVAL;
  7380. kfree(buffer_node);
  7381. return rc;
  7382. }
  7383. mmap_regions = (struct avs_cmd_shared_mem_map_regions *)
  7384. mmap_region_cmd;
  7385. q6asm_add_mmaphdr(ac, &mmap_regions->hdr, cmd_size, dir);
  7386. atomic_set(&ac->mem_state, -1);
  7387. pr_debug("%s: mmap_region=0x%pK token=0x%x\n", __func__,
  7388. mmap_regions, ((ac->session << 8) | dir));
  7389. mmap_regions->hdr.opcode = ASM_CMD_SHARED_MEM_MAP_REGIONS;
  7390. mmap_regions->mem_pool_id = ADSP_MEMORY_MAP_SHMEM8_4K_POOL;
  7391. mmap_regions->num_regions = bufcnt_t; /*bufcnt & 0x00ff; */
  7392. mmap_regions->property_flag = 0x00;
  7393. pr_debug("%s: map_regions->nregions = %d\n", __func__,
  7394. mmap_regions->num_regions);
  7395. payload = ((u8 *) mmap_region_cmd +
  7396. sizeof(struct avs_cmd_shared_mem_map_regions));
  7397. mregions = (struct avs_shared_map_region_payload *)payload;
  7398. ac->port[dir].tmp_hdl = 0;
  7399. port = &ac->port[dir];
  7400. for (i = 0; i < bufcnt_t; i++) {
  7401. ab = &port->buf[i];
  7402. mregions->shm_addr_lsw = lower_32_bits(ab->phys);
  7403. mregions->shm_addr_msw =
  7404. msm_audio_populate_upper_32_bits(ab->phys);
  7405. mregions->mem_size_bytes = bufsz_t;
  7406. ++mregions;
  7407. }
  7408. rc = apr_send_pkt(ac->mmap_apr, (uint32_t *) mmap_region_cmd);
  7409. if (rc < 0) {
  7410. pr_err("%s: mmap_regions op[0x%x]rc[%d]\n", __func__,
  7411. mmap_regions->hdr.opcode, rc);
  7412. rc = -EINVAL;
  7413. kfree(buffer_node);
  7414. goto fail_cmd;
  7415. }
  7416. rc = wait_event_timeout(ac->mem_wait,
  7417. (atomic_read(&ac->mem_state) >= 0 &&
  7418. ac->port[dir].tmp_hdl),
  7419. msecs_to_jiffies(TIMEOUT_MS));
  7420. if (!rc) {
  7421. pr_err("%s: timeout. waited for memory_map\n", __func__);
  7422. rc = -ETIMEDOUT;
  7423. kfree(buffer_node);
  7424. goto fail_cmd;
  7425. }
  7426. if (atomic_read(&ac->mem_state) > 0) {
  7427. pr_err("%s DSP returned error for memory_map [%s]\n",
  7428. __func__, adsp_err_get_err_str(
  7429. atomic_read(&ac->mem_state)));
  7430. rc = adsp_err_get_lnx_err_code(
  7431. atomic_read(&ac->mem_state));
  7432. kfree(buffer_node);
  7433. goto fail_cmd;
  7434. }
  7435. mutex_lock(&ac->cmd_lock);
  7436. for (i = 0; i < bufcnt; i++) {
  7437. ab = &port->buf[i];
  7438. buffer_node[i].buf_phys_addr = ab->phys;
  7439. buffer_node[i].mmap_hdl = ac->port[dir].tmp_hdl;
  7440. list_add_tail(&buffer_node[i].list,
  7441. &ac->port[dir].mem_map_handle);
  7442. pr_debug("%s: i=%d, bufadd[i] = 0x%pK, maphdl[i] = 0x%x\n",
  7443. __func__, i, &buffer_node[i].buf_phys_addr,
  7444. buffer_node[i].mmap_hdl);
  7445. }
  7446. ac->port[dir].tmp_hdl = 0;
  7447. mutex_unlock(&ac->cmd_lock);
  7448. rc = 0;
  7449. fail_cmd:
  7450. kfree(mmap_region_cmd);
  7451. return rc;
  7452. }
  7453. EXPORT_SYMBOL(q6asm_memory_map_regions);
  7454. /**
  7455. * q6asm_memory_unmap_regions -
  7456. * command to send memory unmap regions for ASM
  7457. *
  7458. * @ac: Audio client handle
  7459. * @dir: RX or TX session
  7460. *
  7461. * Returns 0 on success or error on failure
  7462. */
  7463. static int q6asm_memory_unmap_regions(struct audio_client *ac, int dir)
  7464. {
  7465. struct avs_cmd_shared_mem_unmap_regions mem_unmap;
  7466. struct audio_port_data *port = NULL;
  7467. struct asm_buffer_node *buf_node = NULL;
  7468. struct list_head *ptr, *next;
  7469. phys_addr_t buf_add;
  7470. int rc = 0;
  7471. int cmd_size = 0;
  7472. if (!ac) {
  7473. pr_err("%s: APR handle NULL\n", __func__);
  7474. return -EINVAL;
  7475. }
  7476. if (ac->mmap_apr == NULL) {
  7477. pr_err("%s: mmap APR handle NULL\n", __func__);
  7478. return -EINVAL;
  7479. }
  7480. pr_debug("%s: Session[%d]\n", __func__, ac->session);
  7481. cmd_size = sizeof(struct avs_cmd_shared_mem_unmap_regions);
  7482. q6asm_add_mmaphdr(ac, &mem_unmap.hdr, cmd_size, dir);
  7483. atomic_set(&ac->mem_state, -1);
  7484. port = &ac->port[dir];
  7485. buf_add = port->buf->phys;
  7486. mem_unmap.hdr.opcode = ASM_CMD_SHARED_MEM_UNMAP_REGIONS;
  7487. mem_unmap.mem_map_handle = 0;
  7488. list_for_each_safe(ptr, next, &ac->port[dir].mem_map_handle) {
  7489. buf_node = list_entry(ptr, struct asm_buffer_node,
  7490. list);
  7491. if (buf_node->buf_phys_addr == buf_add) {
  7492. pr_debug("%s: Found the element\n", __func__);
  7493. mem_unmap.mem_map_handle = buf_node->mmap_hdl;
  7494. break;
  7495. }
  7496. }
  7497. pr_debug("%s: mem_unmap-mem_map_handle: 0x%x\n",
  7498. __func__, mem_unmap.mem_map_handle);
  7499. if (mem_unmap.mem_map_handle == 0) {
  7500. pr_err("%s: Do not send null mem handle to DSP\n", __func__);
  7501. rc = 0;
  7502. goto fail_cmd;
  7503. }
  7504. rc = apr_send_pkt(ac->mmap_apr, (uint32_t *) &mem_unmap);
  7505. if (rc < 0) {
  7506. pr_err("mmap_regions op[0x%x]rc[%d]\n",
  7507. mem_unmap.hdr.opcode, rc);
  7508. goto fail_cmd;
  7509. }
  7510. rc = wait_event_timeout(ac->mem_wait,
  7511. (atomic_read(&ac->mem_state) >= 0),
  7512. msecs_to_jiffies(TIMEOUT_MS));
  7513. if (!rc) {
  7514. pr_err("%s: timeout. waited for memory_unmap of handle 0x%x\n",
  7515. __func__, mem_unmap.mem_map_handle);
  7516. rc = -ETIMEDOUT;
  7517. goto fail_cmd;
  7518. } else if (atomic_read(&ac->mem_state) > 0) {
  7519. pr_err("%s: DSP returned error[%s]\n",
  7520. __func__, adsp_err_get_err_str(
  7521. atomic_read(&ac->mem_state)));
  7522. rc = adsp_err_get_lnx_err_code(
  7523. atomic_read(&ac->mem_state));
  7524. goto fail_cmd;
  7525. } else if (atomic_read(&ac->unmap_cb_success) == 0) {
  7526. pr_err("%s: Error in mem unmap callback of handle 0x%x\n",
  7527. __func__, mem_unmap.mem_map_handle);
  7528. rc = -EINVAL;
  7529. goto fail_cmd;
  7530. }
  7531. rc = 0;
  7532. fail_cmd:
  7533. list_for_each_safe(ptr, next, &ac->port[dir].mem_map_handle) {
  7534. buf_node = list_entry(ptr, struct asm_buffer_node,
  7535. list);
  7536. if (buf_node->buf_phys_addr == buf_add) {
  7537. list_del(&buf_node->list);
  7538. kfree(buf_node);
  7539. break;
  7540. }
  7541. }
  7542. return rc;
  7543. }
  7544. EXPORT_SYMBOL(q6asm_memory_unmap_regions);
  7545. int q6asm_set_lrgain(struct audio_client *ac, int left_gain, int right_gain)
  7546. {
  7547. struct asm_volume_ctrl_multichannel_gain multi_ch_gain;
  7548. struct param_hdr_v3 param_info;
  7549. int rc = 0;
  7550. memset(&param_info, 0, sizeof(param_info));
  7551. memset(&multi_ch_gain, 0, sizeof(multi_ch_gain));
  7552. param_info.module_id = ASM_MODULE_ID_VOL_CTRL;
  7553. param_info.instance_id = INSTANCE_ID_0;
  7554. param_info.param_id = ASM_PARAM_ID_MULTICHANNEL_GAIN;
  7555. param_info.param_size = sizeof(multi_ch_gain);
  7556. multi_ch_gain.gain_data[0].channeltype = PCM_CHANNEL_FL;
  7557. multi_ch_gain.gain_data[0].gain = left_gain << 15;
  7558. multi_ch_gain.gain_data[1].channeltype = PCM_CHANNEL_FR;
  7559. multi_ch_gain.gain_data[1].gain = right_gain << 15;
  7560. multi_ch_gain.num_channels = 2;
  7561. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info,
  7562. (u8 *) &multi_ch_gain);
  7563. if (rc < 0)
  7564. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7565. __func__, param_info.param_id, rc);
  7566. return rc;
  7567. }
  7568. /*
  7569. * q6asm_set_multich_gain: set multiple channel gains on an ASM session
  7570. * @ac: audio client handle
  7571. * @channels: number of channels caller intends to set gains
  7572. * @gains: list of gains of audio channels
  7573. * @ch_map: list of channel mapping. Only valid if use_default is false
  7574. * @use_default: flag to indicate whether to use default mapping
  7575. */
  7576. int q6asm_set_multich_gain(struct audio_client *ac, uint32_t channels,
  7577. uint32_t *gains, uint8_t *ch_map, bool use_default)
  7578. {
  7579. struct asm_volume_ctrl_multichannel_gain multich_gain;
  7580. struct param_hdr_v3 param_info;
  7581. int rc = 0;
  7582. int i;
  7583. u8 default_chmap[VOLUME_CONTROL_MAX_CHANNELS];
  7584. if (ac == NULL) {
  7585. pr_err("%s: Audio client is NULL\n", __func__);
  7586. return -EINVAL;
  7587. }
  7588. if (gains == NULL) {
  7589. dev_err(ac->dev, "%s: gain_list is NULL\n", __func__);
  7590. rc = -EINVAL;
  7591. goto done;
  7592. }
  7593. if (channels > VOLUME_CONTROL_MAX_CHANNELS) {
  7594. dev_err(ac->dev, "%s: Invalid channel count %d\n",
  7595. __func__, channels);
  7596. rc = -EINVAL;
  7597. goto done;
  7598. }
  7599. if (!use_default && ch_map == NULL) {
  7600. dev_err(ac->dev, "%s: NULL channel map\n", __func__);
  7601. rc = -EINVAL;
  7602. goto done;
  7603. }
  7604. memset(&param_info, 0, sizeof(param_info));
  7605. memset(&multich_gain, 0, sizeof(multich_gain));
  7606. param_info.module_id = ASM_MODULE_ID_VOL_CTRL;
  7607. param_info.instance_id = INSTANCE_ID_0;
  7608. param_info.param_id = ASM_PARAM_ID_MULTICHANNEL_GAIN;
  7609. param_info.param_size = sizeof(multich_gain);
  7610. if (use_default) {
  7611. rc = q6asm_map_channels(default_chmap, channels, false);
  7612. if (rc < 0)
  7613. goto done;
  7614. for (i = 0; i < channels; i++) {
  7615. multich_gain.gain_data[i].channeltype =
  7616. default_chmap[i];
  7617. multich_gain.gain_data[i].gain = gains[i] << 15;
  7618. }
  7619. } else {
  7620. for (i = 0; i < channels; i++) {
  7621. multich_gain.gain_data[i].channeltype = ch_map[i];
  7622. multich_gain.gain_data[i].gain = gains[i] << 15;
  7623. }
  7624. }
  7625. multich_gain.num_channels = channels;
  7626. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info,
  7627. (u8 *) &multich_gain);
  7628. if (rc)
  7629. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7630. __func__, param_info.param_id, rc);
  7631. done:
  7632. return rc;
  7633. }
  7634. EXPORT_SYMBOL(q6asm_set_multich_gain);
  7635. /**
  7636. * q6asm_set_mute -
  7637. * command to set mute for ASM
  7638. *
  7639. * @ac: Audio client handle
  7640. * @muteflag: mute value
  7641. *
  7642. * Returns 0 on success or error on failure
  7643. */
  7644. int q6asm_set_mute(struct audio_client *ac, int muteflag)
  7645. {
  7646. struct asm_volume_ctrl_mute_config mute;
  7647. struct param_hdr_v3 param_info;
  7648. int rc = 0;
  7649. memset(&mute, 0, sizeof(mute));
  7650. memset(&param_info, 0, sizeof(param_info));
  7651. param_info.module_id = ASM_MODULE_ID_VOL_CTRL;
  7652. param_info.instance_id = INSTANCE_ID_0;
  7653. param_info.param_id = ASM_PARAM_ID_VOL_CTRL_MUTE_CONFIG;
  7654. param_info.param_size = sizeof(mute);
  7655. mute.mute_flag = muteflag;
  7656. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info, (u8 *) &mute);
  7657. if (rc)
  7658. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7659. __func__, param_info.param_id, rc);
  7660. return rc;
  7661. }
  7662. EXPORT_SYMBOL(q6asm_set_mute);
  7663. static int __q6asm_set_volume(struct audio_client *ac, int volume, int instance)
  7664. {
  7665. struct asm_volume_ctrl_master_gain vol;
  7666. struct param_hdr_v3 param_info;
  7667. int rc = 0;
  7668. memset(&vol, 0, sizeof(vol));
  7669. memset(&param_info, 0, sizeof(param_info));
  7670. rc = q6asm_set_soft_volume_module_instance_ids(instance, &param_info);
  7671. if (rc) {
  7672. pr_err("%s: Failed to pack soft volume module and instance IDs, error %d\n",
  7673. __func__, rc);
  7674. return rc;
  7675. }
  7676. param_info.param_id = ASM_PARAM_ID_VOL_CTRL_MASTER_GAIN;
  7677. param_info.param_size = sizeof(vol);
  7678. vol.master_gain = volume;
  7679. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info, (u8 *) &vol);
  7680. if (rc)
  7681. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7682. __func__, param_info.param_id, rc);
  7683. return rc;
  7684. }
  7685. /**
  7686. * q6asm_set_volume -
  7687. * command to set volume for ASM
  7688. *
  7689. * @ac: Audio client handle
  7690. * @volume: volume level
  7691. *
  7692. * Returns 0 on success or error on failure
  7693. */
  7694. int q6asm_set_volume(struct audio_client *ac, int volume)
  7695. {
  7696. return __q6asm_set_volume(ac, volume, SOFT_VOLUME_INSTANCE_1);
  7697. }
  7698. EXPORT_SYMBOL(q6asm_set_volume);
  7699. int q6asm_set_volume_v2(struct audio_client *ac, int volume, int instance)
  7700. {
  7701. return __q6asm_set_volume(ac, volume, instance);
  7702. }
  7703. /**
  7704. * q6asm_set_aptx_dec_bt_addr -
  7705. * command to aptx decoder BT addr for ASM
  7706. *
  7707. * @ac: Audio client handle
  7708. * @cfg: APTX decoder bt addr config
  7709. *
  7710. * Returns 0 on success or error on failure
  7711. */
  7712. int q6asm_set_aptx_dec_bt_addr(struct audio_client *ac,
  7713. struct aptx_dec_bt_addr_cfg *cfg)
  7714. {
  7715. struct aptx_dec_bt_dev_addr paylod;
  7716. int sz = 0;
  7717. int rc = 0;
  7718. pr_debug("%s: BT addr nap %d, uap %d, lap %d\n", __func__, cfg->nap,
  7719. cfg->uap, cfg->lap);
  7720. if (ac == NULL) {
  7721. pr_err("%s: AC handle NULL\n", __func__);
  7722. rc = -EINVAL;
  7723. goto fail_cmd;
  7724. }
  7725. if (ac->apr == NULL) {
  7726. pr_err("%s: AC APR handle NULL\n", __func__);
  7727. rc = -EINVAL;
  7728. goto fail_cmd;
  7729. }
  7730. sz = sizeof(struct aptx_dec_bt_dev_addr);
  7731. q6asm_add_hdr_async(ac, &paylod.hdr, sz, TRUE);
  7732. atomic_set(&ac->cmd_state, -1);
  7733. paylod.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM;
  7734. paylod.encdec.param_id = APTX_DECODER_BT_ADDRESS;
  7735. paylod.encdec.param_size = sz - sizeof(paylod.hdr)
  7736. - sizeof(paylod.encdec);
  7737. paylod.bt_addr_cfg.lap = cfg->lap;
  7738. paylod.bt_addr_cfg.uap = cfg->uap;
  7739. paylod.bt_addr_cfg.nap = cfg->nap;
  7740. rc = apr_send_pkt(ac->apr, (uint32_t *) &paylod);
  7741. if (rc < 0) {
  7742. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7743. __func__, paylod.encdec.param_id, rc);
  7744. rc = -EINVAL;
  7745. goto fail_cmd;
  7746. }
  7747. rc = wait_event_timeout(ac->cmd_wait,
  7748. (atomic_read(&ac->cmd_state) >= 0),
  7749. msecs_to_jiffies(TIMEOUT_MS));
  7750. if (!rc) {
  7751. pr_err("%s: timeout, set-params paramid[0x%x]\n", __func__,
  7752. paylod.encdec.param_id);
  7753. rc = -ETIMEDOUT;
  7754. goto fail_cmd;
  7755. }
  7756. if (atomic_read(&ac->cmd_state) > 0) {
  7757. pr_err("%s: DSP returned error[%s] set-params paramid[0x%x]\n",
  7758. __func__, adsp_err_get_err_str(
  7759. atomic_read(&ac->cmd_state)),
  7760. paylod.encdec.param_id);
  7761. rc = adsp_err_get_lnx_err_code(
  7762. atomic_read(&ac->cmd_state));
  7763. goto fail_cmd;
  7764. }
  7765. pr_debug("%s: set BT addr is success\n", __func__);
  7766. rc = 0;
  7767. fail_cmd:
  7768. return rc;
  7769. }
  7770. EXPORT_SYMBOL(q6asm_set_aptx_dec_bt_addr);
  7771. /**
  7772. * q6asm_send_ion_fd -
  7773. * command to send ION memory map for ASM
  7774. *
  7775. * @ac: Audio client handle
  7776. * @fd: ION file desc
  7777. *
  7778. * Returns 0 on success or error on failure
  7779. */
  7780. int q6asm_send_ion_fd(struct audio_client *ac, int fd)
  7781. {
  7782. struct dma_buf *dma_buf;
  7783. dma_addr_t paddr;
  7784. size_t pa_len = 0;
  7785. void *vaddr;
  7786. int ret;
  7787. int sz = 0;
  7788. struct avs_rtic_shared_mem_addr shm;
  7789. if (ac == NULL) {
  7790. pr_err("%s: APR handle NULL\n", __func__);
  7791. ret = -EINVAL;
  7792. goto fail_cmd;
  7793. }
  7794. if (ac->apr == NULL) {
  7795. pr_err("%s: AC APR handle NULL\n", __func__);
  7796. ret = -EINVAL;
  7797. goto fail_cmd;
  7798. }
  7799. ret = msm_audio_ion_import(&dma_buf,
  7800. fd,
  7801. NULL,
  7802. 0,
  7803. &paddr,
  7804. &pa_len,
  7805. &vaddr);
  7806. if (ret) {
  7807. pr_err("%s: audio ION import failed, rc = %d\n",
  7808. __func__, ret);
  7809. ret = -ENOMEM;
  7810. goto fail_cmd;
  7811. }
  7812. /* get payload length */
  7813. sz = sizeof(struct avs_rtic_shared_mem_addr);
  7814. q6asm_add_hdr_async(ac, &shm.hdr, sz, TRUE);
  7815. atomic_set(&ac->cmd_state, -1);
  7816. shm.shm_buf_addr_lsw = lower_32_bits(paddr);
  7817. shm.shm_buf_addr_msw = msm_audio_populate_upper_32_bits(paddr);
  7818. shm.buf_size = pa_len;
  7819. shm.shm_buf_num_regions = 1;
  7820. shm.shm_buf_mem_pool_id = ADSP_MEMORY_MAP_SHMEM8_4K_POOL;
  7821. shm.shm_buf_flag = 0x00;
  7822. shm.encdec.param_id = AVS_PARAM_ID_RTIC_SHARED_MEMORY_ADDR;
  7823. shm.encdec.param_size = sizeof(struct avs_rtic_shared_mem_addr) -
  7824. sizeof(struct apr_hdr) -
  7825. sizeof(struct asm_stream_cmd_set_encdec_param_v2);
  7826. shm.encdec.service_id = OUT;
  7827. shm.encdec.reserved = 0;
  7828. shm.map_region.shm_addr_lsw = shm.shm_buf_addr_lsw;
  7829. shm.map_region.shm_addr_msw = shm.shm_buf_addr_msw;
  7830. shm.map_region.mem_size_bytes = pa_len;
  7831. shm.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM_V2;
  7832. ret = apr_send_pkt(ac->apr, (uint32_t *) &shm);
  7833. if (ret < 0) {
  7834. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7835. __func__, shm.encdec.param_id, ret);
  7836. ret = -EINVAL;
  7837. goto fail_cmd;
  7838. }
  7839. ret = wait_event_timeout(ac->cmd_wait,
  7840. (atomic_read(&ac->cmd_state) >= 0),
  7841. msecs_to_jiffies(TIMEOUT_MS));
  7842. if (!ret) {
  7843. pr_err("%s: timeout, shm.encdec paramid[0x%x]\n", __func__,
  7844. shm.encdec.param_id);
  7845. ret = -ETIMEDOUT;
  7846. goto fail_cmd;
  7847. }
  7848. if (atomic_read(&ac->cmd_state) > 0) {
  7849. pr_err("%s: DSP returned error[%s] shm.encdec paramid[0x%x]\n",
  7850. __func__,
  7851. adsp_err_get_err_str(atomic_read(&ac->cmd_state)),
  7852. shm.encdec.param_id);
  7853. ret = adsp_err_get_lnx_err_code(atomic_read(&ac->cmd_state));
  7854. goto fail_cmd;
  7855. }
  7856. ret = 0;
  7857. fail_cmd:
  7858. return ret;
  7859. }
  7860. EXPORT_SYMBOL(q6asm_send_ion_fd);
  7861. /**
  7862. * q6asm_send_rtic_event_ack -
  7863. * command to send RTIC event ack
  7864. *
  7865. * @ac: Audio client handle
  7866. * @param: params for event ack
  7867. * @params_length: length of params
  7868. *
  7869. * Returns 0 on success or error on failure
  7870. */
  7871. int q6asm_send_rtic_event_ack(struct audio_client *ac,
  7872. void *param, uint32_t params_length)
  7873. {
  7874. char *asm_params = NULL;
  7875. int sz, rc;
  7876. struct avs_param_rtic_event_ack ack;
  7877. if (!param || !ac) {
  7878. pr_err("%s: %s is NULL\n", __func__,
  7879. (!param) ? "param" : "ac");
  7880. rc = -EINVAL;
  7881. goto done;
  7882. }
  7883. sz = sizeof(struct avs_param_rtic_event_ack) + params_length;
  7884. asm_params = kzalloc(sz, GFP_KERNEL);
  7885. if (!asm_params) {
  7886. rc = -ENOMEM;
  7887. goto done;
  7888. }
  7889. q6asm_add_hdr_async(ac, &ack.hdr,
  7890. sizeof(struct avs_param_rtic_event_ack) +
  7891. params_length, TRUE);
  7892. atomic_set(&ac->cmd_state, -1);
  7893. ack.hdr.opcode = ASM_STREAM_CMD_SET_ENCDEC_PARAM_V2;
  7894. ack.encdec.param_id = AVS_PARAM_ID_RTIC_EVENT_ACK;
  7895. ack.encdec.param_size = params_length;
  7896. ack.encdec.reserved = 0;
  7897. ack.encdec.service_id = OUT;
  7898. memcpy(asm_params, &ack, sizeof(struct avs_param_rtic_event_ack));
  7899. memcpy(asm_params + sizeof(struct avs_param_rtic_event_ack),
  7900. param, params_length);
  7901. rc = apr_send_pkt(ac->apr, (uint32_t *) asm_params);
  7902. if (rc < 0) {
  7903. pr_err("%s: apr pkt failed for rtic event ack\n", __func__);
  7904. rc = -EINVAL;
  7905. goto fail_send_param;
  7906. }
  7907. rc = wait_event_timeout(ac->cmd_wait,
  7908. (atomic_read(&ac->cmd_state) >= 0),
  7909. msecs_to_jiffies(TIMEOUT_MS));
  7910. if (!rc) {
  7911. pr_err("%s: timeout for rtic event ack cmd\n", __func__);
  7912. rc = -ETIMEDOUT;
  7913. goto fail_send_param;
  7914. }
  7915. if (atomic_read(&ac->cmd_state) > 0) {
  7916. pr_err("%s: DSP returned error[%s] for rtic event ack cmd\n",
  7917. __func__, adsp_err_get_err_str(
  7918. atomic_read(&ac->cmd_state)));
  7919. rc = adsp_err_get_lnx_err_code(
  7920. atomic_read(&ac->cmd_state));
  7921. goto fail_send_param;
  7922. }
  7923. rc = 0;
  7924. fail_send_param:
  7925. kfree(asm_params);
  7926. done:
  7927. return rc;
  7928. }
  7929. EXPORT_SYMBOL(q6asm_send_rtic_event_ack);
  7930. /**
  7931. * q6asm_set_softpause -
  7932. * command to set pause for ASM
  7933. *
  7934. * @ac: Audio client handle
  7935. * @pause_param: params for pause
  7936. *
  7937. * Returns 0 on success or error on failure
  7938. */
  7939. int q6asm_set_softpause(struct audio_client *ac,
  7940. struct asm_softpause_params *pause_param)
  7941. {
  7942. struct asm_soft_pause_params softpause;
  7943. struct param_hdr_v3 param_info;
  7944. int rc = 0;
  7945. memset(&softpause, 0, sizeof(softpause));
  7946. memset(&param_info, 0, sizeof(param_info));
  7947. param_info.module_id = ASM_MODULE_ID_VOL_CTRL;
  7948. param_info.instance_id = INSTANCE_ID_0;
  7949. param_info.param_id = ASM_PARAM_ID_SOFT_PAUSE_PARAMETERS;
  7950. param_info.param_size = sizeof(softpause);
  7951. softpause.enable_flag = pause_param->enable;
  7952. softpause.period = pause_param->period;
  7953. softpause.step = pause_param->step;
  7954. softpause.ramping_curve = pause_param->rampingcurve;
  7955. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info,
  7956. (u8 *) &softpause);
  7957. if (rc)
  7958. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7959. __func__, param_info.param_id, rc);
  7960. return rc;
  7961. }
  7962. EXPORT_SYMBOL(q6asm_set_softpause);
  7963. static int __q6asm_set_softvolume(struct audio_client *ac,
  7964. struct asm_softvolume_params *softvol_param,
  7965. int instance)
  7966. {
  7967. struct asm_soft_step_volume_params softvol;
  7968. struct param_hdr_v3 param_info;
  7969. int rc = 0;
  7970. memset(&softvol, 0, sizeof(softvol));
  7971. memset(&param_info, 0, sizeof(param_info));
  7972. rc = q6asm_set_soft_volume_module_instance_ids(instance, &param_info);
  7973. if (rc) {
  7974. pr_err("%s: Failed to pack soft volume module and instance IDs, error %d\n",
  7975. __func__, rc);
  7976. return rc;
  7977. }
  7978. param_info.param_id = ASM_PARAM_ID_SOFT_VOL_STEPPING_PARAMETERS;
  7979. param_info.param_size = sizeof(softvol);
  7980. softvol.period = softvol_param->period;
  7981. softvol.step = softvol_param->step;
  7982. softvol.ramping_curve = softvol_param->rampingcurve;
  7983. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info,
  7984. (u8 *) &softvol);
  7985. if (rc)
  7986. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  7987. __func__, param_info.param_id, rc);
  7988. return rc;
  7989. }
  7990. /**
  7991. * q6asm_set_softvolume -
  7992. * command to set softvolume for ASM
  7993. *
  7994. * @ac: Audio client handle
  7995. * @softvol_param: params for softvol
  7996. *
  7997. * Returns 0 on success or error on failure
  7998. */
  7999. int q6asm_set_softvolume(struct audio_client *ac,
  8000. struct asm_softvolume_params *softvol_param)
  8001. {
  8002. return __q6asm_set_softvolume(ac, softvol_param,
  8003. SOFT_VOLUME_INSTANCE_1);
  8004. }
  8005. EXPORT_SYMBOL(q6asm_set_softvolume);
  8006. /**
  8007. * q6asm_set_softvolume_v2 -
  8008. * command to set softvolume V2 for ASM
  8009. *
  8010. * @ac: Audio client handle
  8011. * @softvol_param: params for softvol
  8012. * @instance: instance to apply softvol
  8013. *
  8014. * Returns 0 on success or error on failure
  8015. */
  8016. int q6asm_set_softvolume_v2(struct audio_client *ac,
  8017. struct asm_softvolume_params *softvol_param,
  8018. int instance)
  8019. {
  8020. return __q6asm_set_softvolume(ac, softvol_param, instance);
  8021. }
  8022. EXPORT_SYMBOL(q6asm_set_softvolume_v2);
  8023. /**
  8024. * q6asm_equalizer -
  8025. * command to set equalizer for ASM
  8026. *
  8027. * @ac: Audio client handle
  8028. * @eq_p: Equalizer params
  8029. *
  8030. * Returns 0 on success or error on failure
  8031. */
  8032. int q6asm_equalizer(struct audio_client *ac, void *eq_p)
  8033. {
  8034. struct asm_eq_params eq;
  8035. struct msm_audio_eq_stream_config *eq_params = NULL;
  8036. struct param_hdr_v3 param_info;
  8037. int i = 0;
  8038. int rc = 0;
  8039. if (ac == NULL) {
  8040. pr_err("%s: Audio client is NULL\n", __func__);
  8041. return -EINVAL;
  8042. }
  8043. if (eq_p == NULL) {
  8044. pr_err("%s: [%d]: Invalid Eq param\n", __func__, ac->session);
  8045. rc = -EINVAL;
  8046. goto fail_cmd;
  8047. }
  8048. memset(&eq, 0, sizeof(eq));
  8049. memset(&param_info, 0, sizeof(param_info));
  8050. eq_params = (struct msm_audio_eq_stream_config *) eq_p;
  8051. param_info.module_id = ASM_MODULE_ID_EQUALIZER;
  8052. param_info.instance_id = INSTANCE_ID_0;
  8053. param_info.param_id = ASM_PARAM_ID_EQUALIZER_PARAMETERS;
  8054. param_info.param_size = sizeof(eq);
  8055. eq.enable_flag = eq_params->enable;
  8056. eq.num_bands = eq_params->num_bands;
  8057. pr_debug("%s: enable:%d numbands:%d\n", __func__, eq_params->enable,
  8058. eq_params->num_bands);
  8059. for (i = 0; i < eq_params->num_bands; i++) {
  8060. eq.eq_bands[i].band_idx =
  8061. eq_params->eq_bands[i].band_idx;
  8062. eq.eq_bands[i].filterype =
  8063. eq_params->eq_bands[i].filter_type;
  8064. eq.eq_bands[i].center_freq_hz =
  8065. eq_params->eq_bands[i].center_freq_hz;
  8066. eq.eq_bands[i].filter_gain =
  8067. eq_params->eq_bands[i].filter_gain;
  8068. eq.eq_bands[i].q_factor =
  8069. eq_params->eq_bands[i].q_factor;
  8070. pr_debug("%s: filter_type:%u bandnum:%d\n", __func__,
  8071. eq_params->eq_bands[i].filter_type, i);
  8072. pr_debug("%s: center_freq_hz:%u bandnum:%d\n", __func__,
  8073. eq_params->eq_bands[i].center_freq_hz, i);
  8074. pr_debug("%s: filter_gain:%d bandnum:%d\n", __func__,
  8075. eq_params->eq_bands[i].filter_gain, i);
  8076. pr_debug("%s: q_factor:%d bandnum:%d\n", __func__,
  8077. eq_params->eq_bands[i].q_factor, i);
  8078. }
  8079. rc = q6asm_pack_and_set_pp_param_in_band(ac, param_info, (u8 *) &eq);
  8080. if (rc)
  8081. pr_err("%s: set-params send failed paramid[0x%x] rc %d\n",
  8082. __func__, param_info.param_id, rc);
  8083. fail_cmd:
  8084. return rc;
  8085. }
  8086. EXPORT_SYMBOL(q6asm_equalizer);
  8087. static int __q6asm_read(struct audio_client *ac, bool is_custom_len_reqd,
  8088. int len)
  8089. {
  8090. struct asm_data_cmd_read_v2 read;
  8091. struct asm_buffer_node *buf_node = NULL;
  8092. struct list_head *ptr, *next;
  8093. struct audio_buffer *ab;
  8094. int dsp_buf;
  8095. struct audio_port_data *port;
  8096. int rc;
  8097. if (ac == NULL) {
  8098. pr_err("%s: APR handle NULL\n", __func__);
  8099. return -EINVAL;
  8100. }
  8101. if (ac->apr == NULL) {
  8102. pr_err("%s: AC APR handle NULL\n", __func__);
  8103. return -EINVAL;
  8104. }
  8105. if (ac->io_mode & SYNC_IO_MODE) {
  8106. port = &ac->port[OUT];
  8107. q6asm_add_hdr(ac, &read.hdr, sizeof(read), FALSE);
  8108. mutex_lock(&port->lock);
  8109. dsp_buf = port->dsp_buf;
  8110. if (port->buf == NULL) {
  8111. pr_err("%s: buf is NULL\n", __func__);
  8112. mutex_unlock(&port->lock);
  8113. return -EINVAL;
  8114. }
  8115. ab = &port->buf[dsp_buf];
  8116. dev_vdbg(ac->dev, "%s: session[%d]dsp-buf[%d][%pK]cpu_buf[%d][%pK]\n",
  8117. __func__,
  8118. ac->session,
  8119. dsp_buf,
  8120. port->buf[dsp_buf].data,
  8121. port->cpu_buf,
  8122. &port->buf[port->cpu_buf].phys);
  8123. read.hdr.opcode = ASM_DATA_CMD_READ_V2;
  8124. read.buf_addr_lsw = lower_32_bits(ab->phys);
  8125. read.buf_addr_msw = msm_audio_populate_upper_32_bits(ab->phys);
  8126. list_for_each_safe(ptr, next, &ac->port[OUT].mem_map_handle) {
  8127. buf_node = list_entry(ptr, struct asm_buffer_node,
  8128. list);
  8129. if (buf_node->buf_phys_addr == ab->phys) {
  8130. read.mem_map_handle = buf_node->mmap_hdl;
  8131. break;
  8132. }
  8133. }
  8134. dev_vdbg(ac->dev, "memory_map handle in q6asm_read: [%0x]:",
  8135. read.mem_map_handle);
  8136. read.buf_size = is_custom_len_reqd ? len : ab->size;
  8137. read.seq_id = port->dsp_buf;
  8138. q6asm_update_token(&read.hdr.token,
  8139. 0, /* Session ID is NA */
  8140. 0, /* Stream ID is NA */
  8141. port->dsp_buf,
  8142. 0, /* Direction flag is NA */
  8143. WAIT_CMD);
  8144. port->dsp_buf = q6asm_get_next_buf(ac, port->dsp_buf,
  8145. port->max_buf_cnt);
  8146. mutex_unlock(&port->lock);
  8147. dev_vdbg(ac->dev, "%s: buf add[%pK] token[0x%x] uid[%d]\n",
  8148. __func__, &ab->phys, read.hdr.token,
  8149. read.seq_id);
  8150. rc = apr_send_pkt(ac->apr, (uint32_t *) &read);
  8151. if (rc < 0) {
  8152. pr_err("%s: read op[0x%x]rc[%d]\n",
  8153. __func__, read.hdr.opcode, rc);
  8154. goto fail_cmd;
  8155. }
  8156. return 0;
  8157. }
  8158. fail_cmd:
  8159. return -EINVAL;
  8160. }
  8161. /**
  8162. * q6asm_read -
  8163. * command to read buffer data from DSP
  8164. *
  8165. * @ac: Audio client handle
  8166. *
  8167. * Returns 0 on success or error on failure
  8168. */
  8169. int q6asm_read(struct audio_client *ac)
  8170. {
  8171. return __q6asm_read(ac, false/*is_custom_len_reqd*/, 0);
  8172. }
  8173. EXPORT_SYMBOL(q6asm_read);
  8174. /**
  8175. * q6asm_read_v2 -
  8176. * command to read buffer data from DSP
  8177. *
  8178. * @ac: Audio client handle
  8179. * @len: buffer size to read
  8180. *
  8181. * Returns 0 on success or error on failure
  8182. */
  8183. int q6asm_read_v2(struct audio_client *ac, uint32_t len)
  8184. {
  8185. return __q6asm_read(ac, true /*is_custom_len_reqd*/, len);
  8186. }
  8187. EXPORT_SYMBOL(q6asm_read_v2);
  8188. /**
  8189. * q6asm_read_nolock -
  8190. * command to read buffer data from DSP
  8191. * with no wait for ack.
  8192. *
  8193. * @ac: Audio client handle
  8194. *
  8195. * Returns 0 on success or error on failure
  8196. */
  8197. int q6asm_read_nolock(struct audio_client *ac)
  8198. {
  8199. struct asm_data_cmd_read_v2 read;
  8200. struct asm_buffer_node *buf_node = NULL;
  8201. struct list_head *ptr, *next;
  8202. struct audio_buffer *ab;
  8203. int dsp_buf;
  8204. struct audio_port_data *port;
  8205. int rc;
  8206. if (ac == NULL) {
  8207. pr_err("%s: APR handle NULL\n", __func__);
  8208. return -EINVAL;
  8209. }
  8210. if (ac->apr == NULL) {
  8211. pr_err("%s: AC APR handle NULL\n", __func__);
  8212. return -EINVAL;
  8213. }
  8214. if (ac->io_mode & SYNC_IO_MODE) {
  8215. port = &ac->port[OUT];
  8216. q6asm_add_hdr_async(ac, &read.hdr, sizeof(read), FALSE);
  8217. dsp_buf = port->dsp_buf;
  8218. ab = &port->buf[dsp_buf];
  8219. dev_vdbg(ac->dev, "%s: session[%d]dsp-buf[%d][%pK]cpu_buf[%d][%pK]\n",
  8220. __func__,
  8221. ac->session,
  8222. dsp_buf,
  8223. port->buf[dsp_buf].data,
  8224. port->cpu_buf,
  8225. &port->buf[port->cpu_buf].phys);
  8226. read.hdr.opcode = ASM_DATA_CMD_READ_V2;
  8227. read.buf_addr_lsw = lower_32_bits(ab->phys);
  8228. read.buf_addr_msw = msm_audio_populate_upper_32_bits(ab->phys);
  8229. read.buf_size = ab->size;
  8230. read.seq_id = port->dsp_buf;
  8231. q6asm_update_token(&read.hdr.token,
  8232. 0, /* Session ID is NA */
  8233. 0, /* Stream ID is NA */
  8234. port->dsp_buf,
  8235. 0, /* Direction flag is NA */
  8236. WAIT_CMD);
  8237. list_for_each_safe(ptr, next, &ac->port[OUT].mem_map_handle) {
  8238. buf_node = list_entry(ptr, struct asm_buffer_node,
  8239. list);
  8240. if (buf_node->buf_phys_addr == ab->phys) {
  8241. read.mem_map_handle = buf_node->mmap_hdl;
  8242. break;
  8243. }
  8244. }
  8245. port->dsp_buf = q6asm_get_next_buf(ac, port->dsp_buf,
  8246. port->max_buf_cnt);
  8247. dev_vdbg(ac->dev, "%s: buf add[%pK] token[0x%x] uid[%d]\n",
  8248. __func__, &ab->phys, read.hdr.token,
  8249. read.seq_id);
  8250. rc = apr_send_pkt(ac->apr, (uint32_t *) &read);
  8251. if (rc < 0) {
  8252. pr_err("%s: read op[0x%x]rc[%d]\n",
  8253. __func__, read.hdr.opcode, rc);
  8254. goto fail_cmd;
  8255. }
  8256. return 0;
  8257. }
  8258. fail_cmd:
  8259. return -EINVAL;
  8260. }
  8261. EXPORT_SYMBOL(q6asm_read_nolock);
  8262. /**
  8263. * q6asm_async_write -
  8264. * command to write DSP buffer
  8265. *
  8266. * @ac: Audio client handle
  8267. * @param: params for async write
  8268. *
  8269. * Returns 0 on success or error on failure
  8270. */
  8271. int q6asm_async_write(struct audio_client *ac,
  8272. struct audio_aio_write_param *param)
  8273. {
  8274. int rc = 0;
  8275. struct asm_data_cmd_write_v2 write;
  8276. struct asm_buffer_node *buf_node = NULL;
  8277. struct list_head *ptr, *next;
  8278. struct audio_buffer *ab;
  8279. struct audio_port_data *port;
  8280. phys_addr_t lbuf_phys_addr;
  8281. u32 liomode;
  8282. u32 io_compressed;
  8283. u32 io_compressed_stream;
  8284. if (ac == NULL) {
  8285. pr_err("%s: APR handle NULL\n", __func__);
  8286. return -EINVAL;
  8287. }
  8288. if (ac->apr == NULL) {
  8289. pr_err_ratelimited("%s: AC APR handle NULL\n", __func__);
  8290. return -EINVAL;
  8291. }
  8292. q6asm_stream_add_hdr_async(
  8293. ac, &write.hdr, sizeof(write), TRUE, ac->stream_id);
  8294. port = &ac->port[IN];
  8295. ab = &port->buf[port->dsp_buf];
  8296. /* Pass session id as token for AIO scheme */
  8297. write.hdr.token = param->uid;
  8298. write.hdr.opcode = ASM_DATA_CMD_WRITE_V2;
  8299. write.buf_addr_lsw = lower_32_bits(param->paddr);
  8300. write.buf_addr_msw = msm_audio_populate_upper_32_bits(param->paddr);
  8301. write.buf_size = param->len;
  8302. write.timestamp_msw = param->msw_ts;
  8303. write.timestamp_lsw = param->lsw_ts;
  8304. liomode = (ASYNC_IO_MODE | NT_MODE);
  8305. io_compressed = (ASYNC_IO_MODE | COMPRESSED_IO);
  8306. io_compressed_stream = (ASYNC_IO_MODE | COMPRESSED_STREAM_IO);
  8307. if (ac->io_mode == liomode)
  8308. lbuf_phys_addr = (param->paddr - 32);
  8309. else if (ac->io_mode == io_compressed ||
  8310. ac->io_mode == io_compressed_stream)
  8311. lbuf_phys_addr = (param->paddr - param->metadata_len);
  8312. else {
  8313. if (param->flags & SET_TIMESTAMP)
  8314. lbuf_phys_addr = param->paddr -
  8315. sizeof(struct snd_codec_metadata);
  8316. else
  8317. lbuf_phys_addr = param->paddr;
  8318. }
  8319. dev_vdbg(ac->dev, "%s: token[0x%x], buf_addr[%pK], buf_size[0x%x], ts_msw[0x%x], ts_lsw[0x%x], lbuf_phys_addr: 0x[%pK]\n",
  8320. __func__,
  8321. write.hdr.token, &param->paddr,
  8322. write.buf_size, write.timestamp_msw,
  8323. write.timestamp_lsw, &lbuf_phys_addr);
  8324. /* Use 0xFF00 for disabling timestamps */
  8325. if (param->flags == 0xFF00)
  8326. write.flags = (0x00000000 | (param->flags & 0x800000FF));
  8327. else
  8328. write.flags = (0x80000000 | param->flags);
  8329. write.flags |= param->last_buffer << ASM_SHIFT_LAST_BUFFER_FLAG;
  8330. write.seq_id = param->uid;
  8331. list_for_each_safe(ptr, next, &ac->port[IN].mem_map_handle) {
  8332. buf_node = list_entry(ptr, struct asm_buffer_node,
  8333. list);
  8334. if (buf_node->buf_phys_addr == lbuf_phys_addr) {
  8335. write.mem_map_handle = buf_node->mmap_hdl;
  8336. break;
  8337. }
  8338. }
  8339. rc = apr_send_pkt(ac->apr, (uint32_t *) &write);
  8340. if (rc < 0) {
  8341. pr_err("%s: write op[0x%x]rc[%d]\n", __func__,
  8342. write.hdr.opcode, rc);
  8343. goto fail_cmd;
  8344. }
  8345. return 0;
  8346. fail_cmd:
  8347. return -EINVAL;
  8348. }
  8349. EXPORT_SYMBOL(q6asm_async_write);
  8350. /**
  8351. * q6asm_async_read -
  8352. * command to read DSP buffer
  8353. *
  8354. * @ac: Audio client handle
  8355. * @param: params for async read
  8356. *
  8357. * Returns 0 on success or error on failure
  8358. */
  8359. int q6asm_async_read(struct audio_client *ac,
  8360. struct audio_aio_read_param *param)
  8361. {
  8362. int rc = 0;
  8363. struct asm_data_cmd_read_v2 read;
  8364. struct asm_buffer_node *buf_node = NULL;
  8365. struct list_head *ptr, *next;
  8366. phys_addr_t lbuf_phys_addr;
  8367. u32 liomode;
  8368. u32 io_compressed;
  8369. int dir = 0;
  8370. if (ac == NULL) {
  8371. pr_err("%s: APR handle NULL\n", __func__);
  8372. return -EINVAL;
  8373. }
  8374. if (ac->apr == NULL) {
  8375. pr_err_ratelimited("%s: AC APR handle NULL\n", __func__);
  8376. return -EINVAL;
  8377. }
  8378. q6asm_add_hdr_async(ac, &read.hdr, sizeof(read), FALSE);
  8379. /* Pass session id as token for AIO scheme */
  8380. read.hdr.token = param->uid;
  8381. read.hdr.opcode = ASM_DATA_CMD_READ_V2;
  8382. read.buf_addr_lsw = lower_32_bits(param->paddr);
  8383. read.buf_addr_msw = msm_audio_populate_upper_32_bits(param->paddr);
  8384. read.buf_size = param->len;
  8385. read.seq_id = param->uid;
  8386. liomode = (NT_MODE | ASYNC_IO_MODE);
  8387. io_compressed = (ASYNC_IO_MODE | COMPRESSED_IO);
  8388. if (ac->io_mode == liomode) {
  8389. lbuf_phys_addr = (param->paddr - 32);
  8390. /*legacy wma driver case*/
  8391. dir = IN;
  8392. } else if (ac->io_mode == io_compressed) {
  8393. lbuf_phys_addr = (param->paddr - 64);
  8394. dir = OUT;
  8395. } else {
  8396. if (param->flags & COMPRESSED_TIMESTAMP_FLAG)
  8397. lbuf_phys_addr = param->paddr -
  8398. sizeof(struct snd_codec_metadata);
  8399. else
  8400. lbuf_phys_addr = param->paddr;
  8401. dir = OUT;
  8402. }
  8403. list_for_each_safe(ptr, next, &ac->port[dir].mem_map_handle) {
  8404. buf_node = list_entry(ptr, struct asm_buffer_node,
  8405. list);
  8406. if (buf_node->buf_phys_addr == lbuf_phys_addr) {
  8407. read.mem_map_handle = buf_node->mmap_hdl;
  8408. break;
  8409. }
  8410. }
  8411. rc = apr_send_pkt(ac->apr, (uint32_t *) &read);
  8412. if (rc < 0) {
  8413. pr_err("%s: read op[0x%x]rc[%d]\n", __func__,
  8414. read.hdr.opcode, rc);
  8415. goto fail_cmd;
  8416. }
  8417. return 0;
  8418. fail_cmd:
  8419. return -EINVAL;
  8420. }
  8421. EXPORT_SYMBOL(q6asm_async_read);
  8422. /**
  8423. * q6asm_write -
  8424. * command to write buffer data to DSP
  8425. *
  8426. * @ac: Audio client handle
  8427. * @len: buffer size
  8428. * @msw_ts: upper 32bits of timestamp
  8429. * @lsw_ts: lower 32bits of timestamp
  8430. * @flags: Flags for timestamp mode
  8431. *
  8432. * Returns 0 on success or error on failure
  8433. */
  8434. int q6asm_write(struct audio_client *ac, uint32_t len, uint32_t msw_ts,
  8435. uint32_t lsw_ts, uint32_t flags)
  8436. {
  8437. int rc = 0;
  8438. struct asm_data_cmd_write_v2 write;
  8439. struct asm_buffer_node *buf_node = NULL;
  8440. struct audio_port_data *port;
  8441. struct audio_buffer *ab;
  8442. int dsp_buf = 0;
  8443. if (ac == NULL) {
  8444. pr_err("%s: APR handle NULL\n", __func__);
  8445. return -EINVAL;
  8446. }
  8447. if (ac->apr == NULL) {
  8448. pr_err("%s: AC APR handle NULL\n", __func__);
  8449. return -EINVAL;
  8450. }
  8451. dev_vdbg(ac->dev, "%s: session[%d] len=%d\n",
  8452. __func__, ac->session, len);
  8453. if (ac->io_mode & SYNC_IO_MODE) {
  8454. port = &ac->port[IN];
  8455. q6asm_add_hdr(ac, &write.hdr, sizeof(write),
  8456. FALSE);
  8457. mutex_lock(&port->lock);
  8458. dsp_buf = port->dsp_buf;
  8459. ab = &port->buf[dsp_buf];
  8460. q6asm_update_token(&write.hdr.token,
  8461. 0, /* Session ID is NA */
  8462. 0, /* Stream ID is NA */
  8463. port->dsp_buf,
  8464. 0, /* Direction flag is NA */
  8465. NO_WAIT_CMD);
  8466. write.hdr.opcode = ASM_DATA_CMD_WRITE_V2;
  8467. write.buf_addr_lsw = lower_32_bits(ab->phys);
  8468. write.buf_addr_msw = msm_audio_populate_upper_32_bits(ab->phys);
  8469. write.buf_size = len;
  8470. write.seq_id = port->dsp_buf;
  8471. write.timestamp_lsw = lsw_ts;
  8472. write.timestamp_msw = msw_ts;
  8473. /* Use 0xFF00 for disabling timestamps */
  8474. if (flags == 0xFF00)
  8475. write.flags = (0x00000000 | (flags & 0x800000FF));
  8476. else
  8477. write.flags = (0x80000000 | flags);
  8478. port->dsp_buf = q6asm_get_next_buf(ac, port->dsp_buf,
  8479. port->max_buf_cnt);
  8480. buf_node = list_first_entry(&ac->port[IN].mem_map_handle,
  8481. struct asm_buffer_node,
  8482. list);
  8483. write.mem_map_handle = buf_node->mmap_hdl;
  8484. dev_vdbg(ac->dev, "%s: ab->phys[%pK]bufadd[0x%x] token[0x%x]buf_id[0x%x]buf_size[0x%x]mmaphdl[0x%x]"
  8485. , __func__,
  8486. &ab->phys,
  8487. write.buf_addr_lsw,
  8488. write.hdr.token,
  8489. write.seq_id,
  8490. write.buf_size,
  8491. write.mem_map_handle);
  8492. mutex_unlock(&port->lock);
  8493. config_debug_fs_write(ab);
  8494. rc = apr_send_pkt(ac->apr, (uint32_t *) &write);
  8495. if (rc < 0) {
  8496. pr_err("%s: write op[0x%x]rc[%d]\n",
  8497. __func__, write.hdr.opcode, rc);
  8498. goto fail_cmd;
  8499. }
  8500. return 0;
  8501. }
  8502. fail_cmd:
  8503. return -EINVAL;
  8504. }
  8505. EXPORT_SYMBOL(q6asm_write);
  8506. /**
  8507. * q6asm_write_nolock -
  8508. * command to write buffer data to DSP
  8509. * with no wait for ack.
  8510. *
  8511. * @ac: Audio client handle
  8512. * @len: buffer size
  8513. * @msw_ts: upper 32bits of timestamp
  8514. * @lsw_ts: lower 32bits of timestamp
  8515. * @flags: Flags for timestamp mode
  8516. *
  8517. * Returns 0 on success or error on failure
  8518. */
  8519. int q6asm_write_nolock(struct audio_client *ac, uint32_t len, uint32_t msw_ts,
  8520. uint32_t lsw_ts, uint32_t flags)
  8521. {
  8522. int rc = 0;
  8523. struct asm_data_cmd_write_v2 write;
  8524. struct asm_buffer_node *buf_node = NULL;
  8525. struct audio_port_data *port;
  8526. struct audio_buffer *ab;
  8527. int dsp_buf = 0;
  8528. if (ac == NULL) {
  8529. pr_err("%s: APR handle NULL\n", __func__);
  8530. return -EINVAL;
  8531. }
  8532. if (ac->apr == NULL) {
  8533. pr_err("%s: AC APR handle NULL\n", __func__);
  8534. return -EINVAL;
  8535. }
  8536. dev_vdbg(ac->dev, "%s: session[%d] len=%d\n",
  8537. __func__, ac->session, len);
  8538. if (ac->io_mode & SYNC_IO_MODE) {
  8539. port = &ac->port[IN];
  8540. q6asm_add_hdr_async(ac, &write.hdr, sizeof(write),
  8541. FALSE);
  8542. dsp_buf = port->dsp_buf;
  8543. ab = &port->buf[dsp_buf];
  8544. q6asm_update_token(&write.hdr.token,
  8545. 0, /* Session ID is NA */
  8546. 0, /* Stream ID is NA */
  8547. port->dsp_buf,
  8548. 0, /* Direction flag is NA */
  8549. NO_WAIT_CMD);
  8550. write.hdr.opcode = ASM_DATA_CMD_WRITE_V2;
  8551. write.buf_addr_lsw = lower_32_bits(ab->phys);
  8552. write.buf_addr_msw = msm_audio_populate_upper_32_bits(ab->phys);
  8553. write.buf_size = len;
  8554. write.seq_id = port->dsp_buf;
  8555. write.timestamp_lsw = lsw_ts;
  8556. write.timestamp_msw = msw_ts;
  8557. buf_node = list_first_entry(&ac->port[IN].mem_map_handle,
  8558. struct asm_buffer_node,
  8559. list);
  8560. write.mem_map_handle = buf_node->mmap_hdl;
  8561. /* Use 0xFF00 for disabling timestamps */
  8562. if (flags == 0xFF00)
  8563. write.flags = (0x00000000 | (flags & 0x800000FF));
  8564. else
  8565. write.flags = (0x80000000 | flags);
  8566. port->dsp_buf = q6asm_get_next_buf(ac, port->dsp_buf,
  8567. port->max_buf_cnt);
  8568. dev_vdbg(ac->dev, "%s: ab->phys[%pK]bufadd[0x%x]token[0x%x] buf_id[0x%x]buf_size[0x%x]mmaphdl[0x%x]"
  8569. , __func__,
  8570. &ab->phys,
  8571. write.buf_addr_lsw,
  8572. write.hdr.token,
  8573. write.seq_id,
  8574. write.buf_size,
  8575. write.mem_map_handle);
  8576. rc = apr_send_pkt(ac->apr, (uint32_t *) &write);
  8577. if (rc < 0) {
  8578. pr_err("%s: write op[0x%x]rc[%d]\n",
  8579. __func__, write.hdr.opcode, rc);
  8580. goto fail_cmd;
  8581. }
  8582. return 0;
  8583. }
  8584. fail_cmd:
  8585. return -EINVAL;
  8586. }
  8587. EXPORT_SYMBOL(q6asm_write_nolock);
  8588. /**
  8589. * q6asm_get_session_time -
  8590. * command to retrieve timestamp info
  8591. *
  8592. * @ac: Audio client handle
  8593. * @tstamp: pointer to fill with timestamp info
  8594. *
  8595. * Returns 0 on success or error on failure
  8596. */
  8597. int q6asm_get_session_time(struct audio_client *ac, uint64_t *tstamp)
  8598. {
  8599. struct asm_mtmx_strtr_get_params mtmx_params;
  8600. int rc;
  8601. if (ac == NULL) {
  8602. pr_err("%s: APR handle NULL\n", __func__);
  8603. return -EINVAL;
  8604. }
  8605. if (ac->apr == NULL) {
  8606. pr_err("%s: AC APR handle NULL\n", __func__);
  8607. return -EINVAL;
  8608. }
  8609. if (tstamp == NULL) {
  8610. pr_err("%s: tstamp NULL\n", __func__);
  8611. return -EINVAL;
  8612. }
  8613. q6asm_add_hdr(ac, &mtmx_params.hdr, sizeof(mtmx_params), TRUE);
  8614. mtmx_params.hdr.opcode = ASM_SESSION_CMD_GET_MTMX_STRTR_PARAMS_V2;
  8615. mtmx_params.param_info.data_payload_addr_lsw = 0;
  8616. mtmx_params.param_info.data_payload_addr_msw = 0;
  8617. mtmx_params.param_info.mem_map_handle = 0;
  8618. mtmx_params.param_info.direction = (ac->io_mode & TUN_READ_IO_MODE
  8619. ? 1 : 0);
  8620. mtmx_params.param_info.module_id =
  8621. ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC;
  8622. mtmx_params.param_info.param_id =
  8623. ASM_SESSION_MTMX_STRTR_PARAM_SESSION_TIME_V3;
  8624. mtmx_params.param_info.param_max_size =
  8625. sizeof(struct param_hdr_v1) +
  8626. sizeof(struct asm_session_mtmx_strtr_param_session_time_v3_t);
  8627. atomic_set(&ac->time_flag, 1);
  8628. dev_vdbg(ac->dev, "%s: session[%d]opcode[0x%x]\n", __func__,
  8629. ac->session, mtmx_params.hdr.opcode);
  8630. rc = apr_send_pkt(ac->apr, (uint32_t *) &mtmx_params);
  8631. if (rc < 0) {
  8632. pr_err("%s: Commmand 0x%x failed %d\n", __func__,
  8633. mtmx_params.hdr.opcode, rc);
  8634. goto fail_cmd;
  8635. }
  8636. rc = wait_event_timeout(ac->time_wait,
  8637. (atomic_read(&ac->time_flag) == 0),
  8638. msecs_to_jiffies(TIMEOUT_MS));
  8639. if (!rc) {
  8640. pr_err("%s: timeout in getting session time from DSP\n",
  8641. __func__);
  8642. goto fail_cmd;
  8643. }
  8644. *tstamp = ac->time_stamp;
  8645. return 0;
  8646. fail_cmd:
  8647. return -EINVAL;
  8648. }
  8649. EXPORT_SYMBOL(q6asm_get_session_time);
  8650. /**
  8651. * q6asm_get_session_time_legacy -
  8652. * command to retrieve timestamp info
  8653. *
  8654. * @ac: Audio client handle
  8655. * @tstamp: pointer to fill with timestamp info
  8656. *
  8657. * Returns 0 on success or error on failure
  8658. */
  8659. int q6asm_get_session_time_legacy(struct audio_client *ac, uint64_t *tstamp)
  8660. {
  8661. struct apr_hdr hdr;
  8662. int rc;
  8663. if (ac == NULL) {
  8664. pr_err("%s: APR handle NULL\n", __func__);
  8665. return -EINVAL;
  8666. }
  8667. if (ac->apr == NULL) {
  8668. pr_err("%s: AC APR handle NULL\n", __func__);
  8669. return -EINVAL;
  8670. }
  8671. if (tstamp == NULL) {
  8672. pr_err("%s: tstamp NULL\n", __func__);
  8673. return -EINVAL;
  8674. }
  8675. q6asm_add_hdr(ac, &hdr, sizeof(hdr), TRUE);
  8676. hdr.opcode = ASM_SESSION_CMD_GET_SESSIONTIME_V3;
  8677. atomic_set(&ac->time_flag, 1);
  8678. dev_vdbg(ac->dev, "%s: session[%d]opcode[0x%x]\n", __func__,
  8679. ac->session,
  8680. hdr.opcode);
  8681. rc = apr_send_pkt(ac->apr, (uint32_t *) &hdr);
  8682. if (rc < 0) {
  8683. pr_err("%s: Commmand 0x%x failed %d\n",
  8684. __func__, hdr.opcode, rc);
  8685. goto fail_cmd;
  8686. }
  8687. rc = wait_event_timeout(ac->time_wait,
  8688. (atomic_read(&ac->time_flag) == 0),
  8689. msecs_to_jiffies(TIMEOUT_MS));
  8690. if (!rc) {
  8691. pr_err("%s: timeout in getting session time from DSP\n",
  8692. __func__);
  8693. goto fail_cmd;
  8694. }
  8695. *tstamp = ac->time_stamp;
  8696. return 0;
  8697. fail_cmd:
  8698. return -EINVAL;
  8699. }
  8700. EXPORT_SYMBOL(q6asm_get_session_time_legacy);
  8701. /**
  8702. * q6asm_send_mtmx_strtr_window -
  8703. * command to send matrix for window params
  8704. *
  8705. * @ac: Audio client handle
  8706. * @window_param: window params
  8707. * @param_id: param id for window
  8708. *
  8709. * Returns 0 on success or error on failure
  8710. */
  8711. int q6asm_send_mtmx_strtr_window(struct audio_client *ac,
  8712. struct asm_session_mtmx_strtr_param_window_v2_t *window_param,
  8713. uint32_t param_id)
  8714. {
  8715. struct asm_mtmx_strtr_params matrix;
  8716. int sz = 0;
  8717. int rc = 0;
  8718. pr_debug("%s: Window lsw is %d, window msw is %d\n", __func__,
  8719. window_param->window_lsw, window_param->window_msw);
  8720. if (!ac) {
  8721. pr_err("%s: audio client handle is NULL\n", __func__);
  8722. rc = -EINVAL;
  8723. goto fail_cmd;
  8724. }
  8725. if (ac->apr == NULL) {
  8726. pr_err("%s: ac->apr is NULL", __func__);
  8727. rc = -EINVAL;
  8728. goto fail_cmd;
  8729. }
  8730. sz = sizeof(struct asm_mtmx_strtr_params);
  8731. q6asm_add_hdr(ac, &matrix.hdr, sz, TRUE);
  8732. atomic_set(&ac->cmd_state, -1);
  8733. matrix.hdr.opcode = ASM_SESSION_CMD_SET_MTMX_STRTR_PARAMS_V2;
  8734. matrix.param.data_payload_addr_lsw = 0;
  8735. matrix.param.data_payload_addr_msw = 0;
  8736. matrix.param.mem_map_handle = 0;
  8737. matrix.param.data_payload_size =
  8738. sizeof(struct param_hdr_v1) +
  8739. sizeof(struct asm_session_mtmx_strtr_param_window_v2_t);
  8740. matrix.param.direction = 0; /* RX */
  8741. matrix.data.module_id = ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC;
  8742. matrix.data.param_id = param_id;
  8743. matrix.data.param_size =
  8744. sizeof(struct asm_session_mtmx_strtr_param_window_v2_t);
  8745. matrix.data.reserved = 0;
  8746. memcpy(&(matrix.config.window_param),
  8747. window_param,
  8748. sizeof(struct asm_session_mtmx_strtr_param_window_v2_t));
  8749. rc = apr_send_pkt(ac->apr, (uint32_t *) &matrix);
  8750. if (rc < 0) {
  8751. pr_err("%s: Render window start send failed paramid [0x%x]\n",
  8752. __func__, matrix.data.param_id);
  8753. rc = -EINVAL;
  8754. goto fail_cmd;
  8755. }
  8756. rc = wait_event_timeout(ac->cmd_wait,
  8757. (atomic_read(&ac->cmd_state) >= 0),
  8758. msecs_to_jiffies(TIMEOUT_MS));
  8759. if (!rc) {
  8760. pr_err("%s: timeout, Render window start paramid[0x%x]\n",
  8761. __func__, matrix.data.param_id);
  8762. rc = -ETIMEDOUT;
  8763. goto fail_cmd;
  8764. }
  8765. if (atomic_read(&ac->cmd_state) > 0) {
  8766. pr_err("%s: DSP returned error[%s]\n",
  8767. __func__, adsp_err_get_err_str(
  8768. atomic_read(&ac->cmd_state)));
  8769. rc = adsp_err_get_lnx_err_code(
  8770. atomic_read(&ac->cmd_state));
  8771. goto fail_cmd;
  8772. }
  8773. rc = 0;
  8774. fail_cmd:
  8775. return rc;
  8776. }
  8777. EXPORT_SYMBOL(q6asm_send_mtmx_strtr_window);
  8778. /**
  8779. * q6asm_send_mtmx_strtr_render_mode -
  8780. * command to send matrix for render mode
  8781. *
  8782. * @ac: Audio client handle
  8783. * @render_mode: rendering mode
  8784. *
  8785. * Returns 0 on success or error on failure
  8786. */
  8787. int q6asm_send_mtmx_strtr_render_mode(struct audio_client *ac,
  8788. uint32_t render_mode)
  8789. {
  8790. struct asm_mtmx_strtr_params matrix;
  8791. struct asm_session_mtmx_strtr_param_render_mode_t render_param;
  8792. int sz = 0;
  8793. int rc = 0;
  8794. pr_debug("%s: render mode is %d\n", __func__, render_mode);
  8795. if (!ac) {
  8796. pr_err("%s: audio client handle is NULL\n", __func__);
  8797. rc = -EINVAL;
  8798. goto exit;
  8799. }
  8800. if (ac->apr == NULL) {
  8801. pr_err("%s: ac->apr is NULL\n", __func__);
  8802. rc = -EINVAL;
  8803. goto exit;
  8804. }
  8805. if ((render_mode != ASM_SESSION_MTMX_STRTR_PARAM_RENDER_DEFAULT) &&
  8806. (render_mode != ASM_SESSION_MTMX_STRTR_PARAM_RENDER_LOCAL_STC)) {
  8807. pr_err("%s: Invalid render mode %d\n", __func__, render_mode);
  8808. rc = -EINVAL;
  8809. goto exit;
  8810. }
  8811. memset(&render_param, 0,
  8812. sizeof(struct asm_session_mtmx_strtr_param_render_mode_t));
  8813. render_param.flags = render_mode;
  8814. memset(&matrix, 0, sizeof(struct asm_mtmx_strtr_params));
  8815. sz = sizeof(struct asm_mtmx_strtr_params);
  8816. q6asm_add_hdr(ac, &matrix.hdr, sz, TRUE);
  8817. atomic_set(&ac->cmd_state, -1);
  8818. matrix.hdr.opcode = ASM_SESSION_CMD_SET_MTMX_STRTR_PARAMS_V2;
  8819. matrix.param.data_payload_addr_lsw = 0;
  8820. matrix.param.data_payload_addr_msw = 0;
  8821. matrix.param.mem_map_handle = 0;
  8822. matrix.param.data_payload_size =
  8823. sizeof(struct param_hdr_v1) +
  8824. sizeof(struct asm_session_mtmx_strtr_param_render_mode_t);
  8825. matrix.param.direction = 0; /* RX */
  8826. matrix.data.module_id = ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC;
  8827. matrix.data.param_id = ASM_SESSION_MTMX_STRTR_PARAM_RENDER_MODE_CMD;
  8828. matrix.data.param_size =
  8829. sizeof(struct asm_session_mtmx_strtr_param_render_mode_t);
  8830. matrix.data.reserved = 0;
  8831. memcpy(&(matrix.config.render_param),
  8832. &render_param,
  8833. sizeof(struct asm_session_mtmx_strtr_param_render_mode_t));
  8834. rc = apr_send_pkt(ac->apr, (uint32_t *) &matrix);
  8835. if (rc < 0) {
  8836. pr_err("%s: Render mode send failed paramid [0x%x]\n",
  8837. __func__, matrix.data.param_id);
  8838. rc = -EINVAL;
  8839. goto exit;
  8840. }
  8841. rc = wait_event_timeout(ac->cmd_wait,
  8842. (atomic_read(&ac->cmd_state) >= 0),
  8843. msecs_to_jiffies(TIMEOUT_MS));
  8844. if (!rc) {
  8845. pr_err("%s: timeout, Render mode send paramid [0x%x]\n",
  8846. __func__, matrix.data.param_id);
  8847. rc = -ETIMEDOUT;
  8848. goto exit;
  8849. }
  8850. if (atomic_read(&ac->cmd_state) > 0) {
  8851. pr_err("%s: DSP returned error[%s]\n",
  8852. __func__, adsp_err_get_err_str(
  8853. atomic_read(&ac->cmd_state)));
  8854. rc = adsp_err_get_lnx_err_code(
  8855. atomic_read(&ac->cmd_state));
  8856. goto exit;
  8857. }
  8858. rc = 0;
  8859. exit:
  8860. return rc;
  8861. }
  8862. EXPORT_SYMBOL(q6asm_send_mtmx_strtr_render_mode);
  8863. /**
  8864. * q6asm_send_mtmx_strtr_clk_rec_mode -
  8865. * command to send matrix for clock rec
  8866. *
  8867. * @ac: Audio client handle
  8868. * @clk_rec_mode: mode for clock rec
  8869. *
  8870. * Returns 0 on success or error on failure
  8871. */
  8872. int q6asm_send_mtmx_strtr_clk_rec_mode(struct audio_client *ac,
  8873. uint32_t clk_rec_mode)
  8874. {
  8875. struct asm_mtmx_strtr_params matrix;
  8876. struct asm_session_mtmx_strtr_param_clk_rec_t clk_rec_param;
  8877. int sz = 0;
  8878. int rc = 0;
  8879. pr_debug("%s: clk rec mode is %d\n", __func__, clk_rec_mode);
  8880. if (!ac) {
  8881. pr_err("%s: audio client handle is NULL\n", __func__);
  8882. rc = -EINVAL;
  8883. goto exit;
  8884. }
  8885. if (ac->apr == NULL) {
  8886. pr_err("%s: ac->apr is NULL\n", __func__);
  8887. rc = -EINVAL;
  8888. goto exit;
  8889. }
  8890. if ((clk_rec_mode != ASM_SESSION_MTMX_STRTR_PARAM_CLK_REC_NONE) &&
  8891. (clk_rec_mode != ASM_SESSION_MTMX_STRTR_PARAM_CLK_REC_AUTO)) {
  8892. pr_err("%s: Invalid clk rec mode %d\n", __func__, clk_rec_mode);
  8893. rc = -EINVAL;
  8894. goto exit;
  8895. }
  8896. memset(&clk_rec_param, 0,
  8897. sizeof(struct asm_session_mtmx_strtr_param_clk_rec_t));
  8898. clk_rec_param.flags = clk_rec_mode;
  8899. memset(&matrix, 0, sizeof(struct asm_mtmx_strtr_params));
  8900. sz = sizeof(struct asm_mtmx_strtr_params);
  8901. q6asm_add_hdr(ac, &matrix.hdr, sz, TRUE);
  8902. atomic_set(&ac->cmd_state, -1);
  8903. matrix.hdr.opcode = ASM_SESSION_CMD_SET_MTMX_STRTR_PARAMS_V2;
  8904. matrix.param.data_payload_addr_lsw = 0;
  8905. matrix.param.data_payload_addr_msw = 0;
  8906. matrix.param.mem_map_handle = 0;
  8907. matrix.param.data_payload_size =
  8908. sizeof(struct param_hdr_v1) +
  8909. sizeof(struct asm_session_mtmx_strtr_param_clk_rec_t);
  8910. matrix.param.direction = 0; /* RX */
  8911. matrix.data.module_id = ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC;
  8912. matrix.data.param_id = ASM_SESSION_MTMX_STRTR_PARAM_CLK_REC_CMD;
  8913. matrix.data.param_size =
  8914. sizeof(struct asm_session_mtmx_strtr_param_clk_rec_t);
  8915. matrix.data.reserved = 0;
  8916. memcpy(&(matrix.config.clk_rec_param),
  8917. &clk_rec_param,
  8918. sizeof(struct asm_session_mtmx_strtr_param_clk_rec_t));
  8919. rc = apr_send_pkt(ac->apr, (uint32_t *) &matrix);
  8920. if (rc < 0) {
  8921. pr_err("%s: clk rec mode send failed paramid [0x%x]\n",
  8922. __func__, matrix.data.param_id);
  8923. rc = -EINVAL;
  8924. goto exit;
  8925. }
  8926. rc = wait_event_timeout(ac->cmd_wait,
  8927. (atomic_read(&ac->cmd_state) >= 0),
  8928. msecs_to_jiffies(TIMEOUT_MS));
  8929. if (!rc) {
  8930. pr_err("%s: timeout, clk rec mode send paramid [0x%x]\n",
  8931. __func__, matrix.data.param_id);
  8932. rc = -ETIMEDOUT;
  8933. goto exit;
  8934. }
  8935. if (atomic_read(&ac->cmd_state) > 0) {
  8936. pr_err("%s: DSP returned error[%s]\n",
  8937. __func__, adsp_err_get_err_str(
  8938. atomic_read(&ac->cmd_state)));
  8939. rc = adsp_err_get_lnx_err_code(
  8940. atomic_read(&ac->cmd_state));
  8941. goto exit;
  8942. }
  8943. rc = 0;
  8944. exit:
  8945. return rc;
  8946. }
  8947. EXPORT_SYMBOL(q6asm_send_mtmx_strtr_clk_rec_mode);
  8948. /**
  8949. * q6asm_send_mtmx_strtr_enable_adjust_session_clock -
  8950. * command to send matrix for adjust time
  8951. *
  8952. * @ac: Audio client handle
  8953. * @enable: flag to adjust time or not
  8954. *
  8955. * Returns 0 on success or error on failure
  8956. */
  8957. int q6asm_send_mtmx_strtr_enable_adjust_session_clock(struct audio_client *ac,
  8958. bool enable)
  8959. {
  8960. struct asm_mtmx_strtr_params matrix;
  8961. struct asm_session_mtmx_param_adjust_session_time_ctl_t adjust_time;
  8962. int sz = 0;
  8963. int rc = 0;
  8964. pr_debug("%s: adjust session enable %d\n", __func__, enable);
  8965. if (!ac) {
  8966. pr_err("%s: audio client handle is NULL\n", __func__);
  8967. rc = -EINVAL;
  8968. goto exit;
  8969. }
  8970. if (ac->apr == NULL) {
  8971. pr_err("%s: ac->apr is NULL\n", __func__);
  8972. rc = -EINVAL;
  8973. goto exit;
  8974. }
  8975. adjust_time.enable = enable;
  8976. memset(&matrix, 0, sizeof(struct asm_mtmx_strtr_params));
  8977. sz = sizeof(struct asm_mtmx_strtr_params);
  8978. q6asm_add_hdr(ac, &matrix.hdr, sz, TRUE);
  8979. atomic_set(&ac->cmd_state, -1);
  8980. matrix.hdr.opcode = ASM_SESSION_CMD_SET_MTMX_STRTR_PARAMS_V2;
  8981. matrix.param.data_payload_addr_lsw = 0;
  8982. matrix.param.data_payload_addr_msw = 0;
  8983. matrix.param.mem_map_handle = 0;
  8984. matrix.param.data_payload_size =
  8985. sizeof(struct param_hdr_v1) +
  8986. sizeof(struct asm_session_mtmx_param_adjust_session_time_ctl_t);
  8987. matrix.param.direction = 0; /* RX */
  8988. matrix.data.module_id = ASM_SESSION_MTMX_STRTR_MODULE_ID_AVSYNC;
  8989. matrix.data.param_id = ASM_SESSION_MTMX_PARAM_ADJUST_SESSION_TIME_CTL;
  8990. matrix.data.param_size =
  8991. sizeof(struct asm_session_mtmx_param_adjust_session_time_ctl_t);
  8992. matrix.data.reserved = 0;
  8993. matrix.config.adj_time_param.enable = adjust_time.enable;
  8994. rc = apr_send_pkt(ac->apr, (uint32_t *) &matrix);
  8995. if (rc < 0) {
  8996. pr_err("%s: enable adjust session failed failed paramid [0x%x]\n",
  8997. __func__, matrix.data.param_id);
  8998. rc = -EINVAL;
  8999. goto exit;
  9000. }
  9001. rc = wait_event_timeout(ac->cmd_wait,
  9002. (atomic_read(&ac->cmd_state) >= 0),
  9003. msecs_to_jiffies(TIMEOUT_MS));
  9004. if (!rc) {
  9005. pr_err("%s: enable adjust session failed failed paramid [0x%x]\n",
  9006. __func__, matrix.data.param_id);
  9007. rc = -ETIMEDOUT;
  9008. goto exit;
  9009. }
  9010. if (atomic_read(&ac->cmd_state) > 0) {
  9011. pr_err("%s: DSP returned error[%s]\n",
  9012. __func__, adsp_err_get_err_str(
  9013. atomic_read(&ac->cmd_state)));
  9014. rc = adsp_err_get_lnx_err_code(
  9015. atomic_read(&ac->cmd_state));
  9016. goto exit;
  9017. }
  9018. rc = 0;
  9019. exit:
  9020. return rc;
  9021. }
  9022. EXPORT_SYMBOL(q6asm_send_mtmx_strtr_enable_adjust_session_clock);
  9023. static int __q6asm_cmd(struct audio_client *ac, int cmd, uint32_t stream_id)
  9024. {
  9025. struct apr_hdr hdr;
  9026. int rc;
  9027. atomic_t *state;
  9028. int cnt = 0;
  9029. if (!ac) {
  9030. pr_err_ratelimited("%s: APR handle NULL\n", __func__);
  9031. return -EINVAL;
  9032. }
  9033. if (ac->apr == NULL) {
  9034. pr_err_ratelimited("%s: AC APR handle NULL\n", __func__);
  9035. return -EINVAL;
  9036. }
  9037. q6asm_stream_add_hdr(ac, &hdr, sizeof(hdr), TRUE, stream_id);
  9038. atomic_set(&ac->cmd_state, -1);
  9039. /*
  9040. * Updated the token field with stream/session for compressed playback
  9041. * Platform driver must know the the stream with which the command is
  9042. * associated
  9043. */
  9044. if (ac->io_mode & COMPRESSED_STREAM_IO)
  9045. q6asm_update_token(&hdr.token,
  9046. ac->session,
  9047. stream_id,
  9048. 0, /* Buffer index is NA */
  9049. 0, /* Direction flag is NA */
  9050. WAIT_CMD);
  9051. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  9052. __func__, hdr.token, stream_id, ac->session);
  9053. switch (cmd) {
  9054. case CMD_PAUSE:
  9055. pr_debug("%s: CMD_PAUSE\n", __func__);
  9056. hdr.opcode = ASM_SESSION_CMD_PAUSE;
  9057. state = &ac->cmd_state;
  9058. break;
  9059. case CMD_SUSPEND:
  9060. pr_debug("%s: CMD_SUSPEND\n", __func__);
  9061. hdr.opcode = ASM_SESSION_CMD_SUSPEND;
  9062. state = &ac->cmd_state;
  9063. break;
  9064. case CMD_FLUSH:
  9065. pr_debug("%s: CMD_FLUSH\n", __func__);
  9066. hdr.opcode = ASM_STREAM_CMD_FLUSH;
  9067. state = &ac->cmd_state;
  9068. break;
  9069. case CMD_OUT_FLUSH:
  9070. pr_debug("%s: CMD_OUT_FLUSH\n", __func__);
  9071. hdr.opcode = ASM_STREAM_CMD_FLUSH_READBUFS;
  9072. state = &ac->cmd_state;
  9073. break;
  9074. case CMD_EOS:
  9075. pr_debug("%s: CMD_EOS\n", __func__);
  9076. hdr.opcode = ASM_DATA_CMD_EOS;
  9077. atomic_set(&ac->cmd_state, 0);
  9078. state = &ac->cmd_state;
  9079. break;
  9080. case CMD_CLOSE:
  9081. pr_debug("%s: CMD_CLOSE\n", __func__);
  9082. hdr.opcode = ASM_STREAM_CMD_CLOSE;
  9083. state = &ac->cmd_state;
  9084. break;
  9085. default:
  9086. pr_err("%s: Invalid format[%d]\n", __func__, cmd);
  9087. rc = -EINVAL;
  9088. goto fail_cmd;
  9089. }
  9090. pr_debug("%s: session[%d]opcode[0x%x]\n", __func__,
  9091. ac->session,
  9092. hdr.opcode);
  9093. rc = apr_send_pkt(ac->apr, (uint32_t *) &hdr);
  9094. if (rc < 0) {
  9095. pr_err("%s: Commmand 0x%x failed %d\n",
  9096. __func__, hdr.opcode, rc);
  9097. rc = -EINVAL;
  9098. goto fail_cmd;
  9099. }
  9100. rc = wait_event_timeout(ac->cmd_wait, (atomic_read(state) >= 0),
  9101. msecs_to_jiffies(TIMEOUT_MS));
  9102. if (!rc) {
  9103. pr_err("%s: timeout. waited for response opcode[0x%x]\n",
  9104. __func__, hdr.opcode);
  9105. rc = -ETIMEDOUT;
  9106. goto fail_cmd;
  9107. }
  9108. if (atomic_read(state) > 0) {
  9109. pr_err("%s: DSP returned error[%s] opcode %d\n",
  9110. __func__, adsp_err_get_err_str(
  9111. atomic_read(state)),
  9112. hdr.opcode);
  9113. rc = adsp_err_get_lnx_err_code(atomic_read(state));
  9114. goto fail_cmd;
  9115. }
  9116. if (cmd == CMD_FLUSH)
  9117. q6asm_reset_buf_state(ac);
  9118. if (cmd == CMD_CLOSE) {
  9119. /* check if DSP return all buffers */
  9120. if (ac->port[IN].buf) {
  9121. for (cnt = 0; cnt < ac->port[IN].max_buf_cnt;
  9122. cnt++) {
  9123. if (ac->port[IN].buf[cnt].used == IN) {
  9124. dev_vdbg(ac->dev, "Write Buf[%d] not returned\n",
  9125. cnt);
  9126. }
  9127. }
  9128. }
  9129. if (ac->port[OUT].buf) {
  9130. for (cnt = 0; cnt < ac->port[OUT].max_buf_cnt; cnt++) {
  9131. if (ac->port[OUT].buf[cnt].used == OUT) {
  9132. dev_vdbg(ac->dev, "Read Buf[%d] not returned\n",
  9133. cnt);
  9134. }
  9135. }
  9136. }
  9137. }
  9138. return 0;
  9139. fail_cmd:
  9140. return rc;
  9141. }
  9142. /**
  9143. * q6asm_cmd -
  9144. * Function used to send commands for
  9145. * ASM with wait for ack.
  9146. *
  9147. * @ac: Audio client handle
  9148. * @cmd: command to send
  9149. *
  9150. * Returns 0 on success or error on failure
  9151. */
  9152. int q6asm_cmd(struct audio_client *ac, int cmd)
  9153. {
  9154. return __q6asm_cmd(ac, cmd, ac->stream_id);
  9155. }
  9156. EXPORT_SYMBOL(q6asm_cmd);
  9157. /**
  9158. * q6asm_stream_cmd -
  9159. * Function used to send commands for
  9160. * ASM stream with wait for ack.
  9161. *
  9162. * @ac: Audio client handle
  9163. * @cmd: command to send
  9164. * @stream_id: Stream ID
  9165. *
  9166. * Returns 0 on success or error on failure
  9167. */
  9168. int q6asm_stream_cmd(struct audio_client *ac, int cmd, uint32_t stream_id)
  9169. {
  9170. return __q6asm_cmd(ac, cmd, stream_id);
  9171. }
  9172. EXPORT_SYMBOL(q6asm_stream_cmd);
  9173. /**
  9174. * q6asm_cmd_nowait -
  9175. * Function used to send commands for
  9176. * ASM stream without wait for ack.
  9177. *
  9178. * @ac: Audio client handle
  9179. * @cmd: command to send
  9180. * @stream_id: Stream ID
  9181. *
  9182. * Returns 0 on success or error on failure
  9183. */
  9184. static int __q6asm_cmd_nowait(struct audio_client *ac, int cmd,
  9185. uint32_t stream_id)
  9186. {
  9187. struct apr_hdr hdr;
  9188. int rc;
  9189. if (!ac) {
  9190. pr_err_ratelimited("%s: APR handle NULL\n", __func__);
  9191. return -EINVAL;
  9192. }
  9193. if (ac->apr == NULL) {
  9194. pr_err_ratelimited("%s: AC APR handle NULL\n", __func__);
  9195. return -EINVAL;
  9196. }
  9197. q6asm_stream_add_hdr_async(ac, &hdr, sizeof(hdr), TRUE, stream_id);
  9198. atomic_set(&ac->cmd_state, 1);
  9199. /*
  9200. * Updated the token field with stream/session for compressed playback
  9201. * Platform driver must know the the stream with which the command is
  9202. * associated
  9203. */
  9204. if (ac->io_mode & COMPRESSED_STREAM_IO)
  9205. q6asm_update_token(&hdr.token,
  9206. ac->session,
  9207. stream_id,
  9208. 0, /* Buffer index is NA */
  9209. 0, /* Direction flag is NA */
  9210. NO_WAIT_CMD);
  9211. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  9212. __func__, hdr.token, stream_id, ac->session);
  9213. switch (cmd) {
  9214. case CMD_PAUSE:
  9215. pr_debug("%s: CMD_PAUSE\n", __func__);
  9216. hdr.opcode = ASM_SESSION_CMD_PAUSE;
  9217. break;
  9218. case CMD_EOS:
  9219. pr_debug("%s: CMD_EOS\n", __func__);
  9220. hdr.opcode = ASM_DATA_CMD_EOS;
  9221. break;
  9222. case CMD_CLOSE:
  9223. pr_debug("%s: CMD_CLOSE\n", __func__);
  9224. hdr.opcode = ASM_STREAM_CMD_CLOSE;
  9225. break;
  9226. default:
  9227. pr_err("%s: Invalid format[%d]\n", __func__, cmd);
  9228. goto fail_cmd;
  9229. }
  9230. pr_debug("%s: session[%d]opcode[0x%x]\n", __func__,
  9231. ac->session,
  9232. hdr.opcode);
  9233. rc = apr_send_pkt(ac->apr, (uint32_t *) &hdr);
  9234. if (rc < 0) {
  9235. pr_err("%s: Commmand 0x%x failed %d\n",
  9236. __func__, hdr.opcode, rc);
  9237. goto fail_cmd;
  9238. }
  9239. return 0;
  9240. fail_cmd:
  9241. return -EINVAL;
  9242. }
  9243. int q6asm_cmd_nowait(struct audio_client *ac, int cmd)
  9244. {
  9245. pr_debug("%s: stream_id: %d\n", __func__, ac->stream_id);
  9246. return __q6asm_cmd_nowait(ac, cmd, ac->stream_id);
  9247. }
  9248. EXPORT_SYMBOL(q6asm_cmd_nowait);
  9249. /**
  9250. * q6asm_stream_cmd_nowait -
  9251. * Function used to send commands for
  9252. * ASM stream without wait for ack.
  9253. *
  9254. * @ac: Audio client handle
  9255. * @cmd: command to send
  9256. * @stream_id: Stream ID
  9257. *
  9258. * Returns 0 on success or error on failure
  9259. */
  9260. int q6asm_stream_cmd_nowait(struct audio_client *ac, int cmd,
  9261. uint32_t stream_id)
  9262. {
  9263. pr_debug("%s: stream_id: %d\n", __func__, stream_id);
  9264. return __q6asm_cmd_nowait(ac, cmd, stream_id);
  9265. }
  9266. EXPORT_SYMBOL(q6asm_stream_cmd_nowait);
  9267. int __q6asm_send_meta_data(struct audio_client *ac, uint32_t stream_id,
  9268. uint32_t initial_samples, uint32_t trailing_samples)
  9269. {
  9270. struct asm_data_cmd_remove_silence silence;
  9271. int rc = 0;
  9272. if (!ac) {
  9273. pr_err_ratelimited("%s: APR handle NULL\n", __func__);
  9274. return -EINVAL;
  9275. }
  9276. if (ac->apr == NULL) {
  9277. pr_err_ratelimited("%s: AC APR handle NULL\n", __func__);
  9278. return -EINVAL;
  9279. }
  9280. pr_debug("%s: session[%d]\n", __func__, ac->session);
  9281. q6asm_stream_add_hdr_async(ac, &silence.hdr, sizeof(silence), TRUE,
  9282. stream_id);
  9283. /*
  9284. * Updated the token field with stream/session for compressed playback
  9285. * Platform driver must know the the stream with which the command is
  9286. * associated
  9287. */
  9288. if (ac->io_mode & COMPRESSED_STREAM_IO)
  9289. q6asm_update_token(&silence.hdr.token,
  9290. ac->session,
  9291. stream_id,
  9292. 0, /* Buffer index is NA */
  9293. 0, /* Direction flag is NA */
  9294. NO_WAIT_CMD);
  9295. pr_debug("%s: token = 0x%x, stream_id %d, session 0x%x\n",
  9296. __func__, silence.hdr.token, stream_id, ac->session);
  9297. silence.hdr.opcode = ASM_DATA_CMD_REMOVE_INITIAL_SILENCE;
  9298. silence.num_samples_to_remove = initial_samples;
  9299. rc = apr_send_pkt(ac->apr, (uint32_t *) &silence);
  9300. if (rc < 0) {
  9301. pr_err("%s: Commmand silence failed[%d]", __func__, rc);
  9302. goto fail_cmd;
  9303. }
  9304. silence.hdr.opcode = ASM_DATA_CMD_REMOVE_TRAILING_SILENCE;
  9305. silence.num_samples_to_remove = trailing_samples;
  9306. rc = apr_send_pkt(ac->apr, (uint32_t *) &silence);
  9307. if (rc < 0) {
  9308. pr_err("%s: Commmand silence failed[%d]", __func__, rc);
  9309. goto fail_cmd;
  9310. }
  9311. return 0;
  9312. fail_cmd:
  9313. return -EINVAL;
  9314. }
  9315. /**
  9316. * q6asm_stream_send_meta_data -
  9317. * command to send meta data for stream
  9318. *
  9319. * @ac: Audio client handle
  9320. * @stream_id: Stream ID
  9321. * @initial_samples: Initial samples of stream
  9322. * @trailing_samples: Trailing samples of stream
  9323. *
  9324. * Returns 0 on success or error on failure
  9325. */
  9326. int q6asm_stream_send_meta_data(struct audio_client *ac, uint32_t stream_id,
  9327. uint32_t initial_samples, uint32_t trailing_samples)
  9328. {
  9329. return __q6asm_send_meta_data(ac, stream_id, initial_samples,
  9330. trailing_samples);
  9331. }
  9332. EXPORT_SYMBOL(q6asm_stream_send_meta_data);
  9333. int q6asm_send_meta_data(struct audio_client *ac, uint32_t initial_samples,
  9334. uint32_t trailing_samples)
  9335. {
  9336. return __q6asm_send_meta_data(ac, ac->stream_id, initial_samples,
  9337. trailing_samples);
  9338. }
  9339. static void q6asm_reset_buf_state(struct audio_client *ac)
  9340. {
  9341. int cnt = 0;
  9342. int loopcnt = 0;
  9343. int used;
  9344. struct audio_port_data *port = NULL;
  9345. if (ac->io_mode & SYNC_IO_MODE) {
  9346. used = (ac->io_mode & TUN_WRITE_IO_MODE ? 1 : 0);
  9347. mutex_lock(&ac->cmd_lock);
  9348. for (loopcnt = 0; loopcnt <= OUT; loopcnt++) {
  9349. port = &ac->port[loopcnt];
  9350. cnt = port->max_buf_cnt - 1;
  9351. port->dsp_buf = 0;
  9352. port->cpu_buf = 0;
  9353. while (cnt >= 0) {
  9354. if (!port->buf)
  9355. continue;
  9356. port->buf[cnt].used = used;
  9357. cnt--;
  9358. }
  9359. }
  9360. mutex_unlock(&ac->cmd_lock);
  9361. }
  9362. }
  9363. /**
  9364. * q6asm_reg_tx_overflow -
  9365. * command to register for TX overflow events
  9366. *
  9367. * @ac: Audio client handle
  9368. * @enable: flag to enable or disable events
  9369. *
  9370. * Returns 0 on success or error on failure
  9371. */
  9372. int q6asm_reg_tx_overflow(struct audio_client *ac, uint16_t enable)
  9373. {
  9374. struct asm_session_cmd_regx_overflow tx_overflow;
  9375. int rc;
  9376. if (!ac) {
  9377. pr_err("%s: APR handle NULL\n", __func__);
  9378. return -EINVAL;
  9379. }
  9380. if (ac->apr == NULL) {
  9381. pr_err("%s: AC APR handle NULL\n", __func__);
  9382. return -EINVAL;
  9383. }
  9384. pr_debug("%s: session[%d]enable[%d]\n", __func__,
  9385. ac->session, enable);
  9386. q6asm_add_hdr(ac, &tx_overflow.hdr, sizeof(tx_overflow), TRUE);
  9387. atomic_set(&ac->cmd_state, -1);
  9388. tx_overflow.hdr.opcode =
  9389. ASM_SESSION_CMD_REGISTER_FORX_OVERFLOW_EVENTS;
  9390. /* tx overflow event: enable */
  9391. tx_overflow.enable_flag = enable;
  9392. rc = apr_send_pkt(ac->apr, (uint32_t *) &tx_overflow);
  9393. if (rc < 0) {
  9394. pr_err("%s: tx overflow op[0x%x]rc[%d]\n",
  9395. __func__, tx_overflow.hdr.opcode, rc);
  9396. rc = -EINVAL;
  9397. goto fail_cmd;
  9398. }
  9399. rc = wait_event_timeout(ac->cmd_wait,
  9400. (atomic_read(&ac->cmd_state) >= 0),
  9401. msecs_to_jiffies(TIMEOUT_MS));
  9402. if (!rc) {
  9403. pr_err("%s: timeout. waited for tx overflow\n", __func__);
  9404. rc = -ETIMEDOUT;
  9405. goto fail_cmd;
  9406. }
  9407. if (atomic_read(&ac->cmd_state) > 0) {
  9408. pr_err("%s: DSP returned error[%s]\n",
  9409. __func__, adsp_err_get_err_str(
  9410. atomic_read(&ac->cmd_state)));
  9411. rc = adsp_err_get_lnx_err_code(
  9412. atomic_read(&ac->cmd_state));
  9413. goto fail_cmd;
  9414. }
  9415. return 0;
  9416. fail_cmd:
  9417. return rc;
  9418. }
  9419. EXPORT_SYMBOL(q6asm_reg_tx_overflow);
  9420. int q6asm_reg_rx_underflow(struct audio_client *ac, uint16_t enable)
  9421. {
  9422. struct asm_session_cmd_rgstr_rx_underflow rx_underflow;
  9423. int rc;
  9424. if (!ac) {
  9425. pr_err("%s: AC APR handle NULL\n", __func__);
  9426. return -EINVAL;
  9427. }
  9428. if (ac->apr == NULL) {
  9429. pr_err("%s: APR handle NULL\n", __func__);
  9430. return -EINVAL;
  9431. }
  9432. pr_debug("%s: session[%d]enable[%d]\n", __func__,
  9433. ac->session, enable);
  9434. q6asm_add_hdr_async(ac, &rx_underflow.hdr, sizeof(rx_underflow), FALSE);
  9435. rx_underflow.hdr.opcode =
  9436. ASM_SESSION_CMD_REGISTER_FOR_RX_UNDERFLOW_EVENTS;
  9437. /* tx overflow event: enable */
  9438. rx_underflow.enable_flag = enable;
  9439. rc = apr_send_pkt(ac->apr, (uint32_t *) &rx_underflow);
  9440. if (rc < 0) {
  9441. pr_err("%s: tx overflow op[0x%x]rc[%d]\n",
  9442. __func__, rx_underflow.hdr.opcode, rc);
  9443. goto fail_cmd;
  9444. }
  9445. return 0;
  9446. fail_cmd:
  9447. return -EINVAL;
  9448. }
  9449. /**
  9450. * q6asm_adjust_session_clock -
  9451. * command to adjust session clock
  9452. *
  9453. * @ac: Audio client handle
  9454. * @adjust_time_lsw: lower 32bits
  9455. * @adjust_time_msw: upper 32bits
  9456. *
  9457. * Returns 0 on success or error on failure
  9458. */
  9459. int q6asm_adjust_session_clock(struct audio_client *ac,
  9460. uint32_t adjust_time_lsw,
  9461. uint32_t adjust_time_msw)
  9462. {
  9463. int rc = 0;
  9464. int sz = 0;
  9465. struct asm_session_cmd_adjust_session_clock_v2 adjust_clock;
  9466. pr_debug("%s: adjust_time_lsw is %x, adjust_time_msw is %x\n", __func__,
  9467. adjust_time_lsw, adjust_time_msw);
  9468. if (!ac) {
  9469. pr_err("%s: audio client handle is NULL\n", __func__);
  9470. rc = -EINVAL;
  9471. goto fail_cmd;
  9472. }
  9473. if (ac->apr == NULL) {
  9474. pr_err("%s: ac->apr is NULL", __func__);
  9475. rc = -EINVAL;
  9476. goto fail_cmd;
  9477. }
  9478. sz = sizeof(struct asm_session_cmd_adjust_session_clock_v2);
  9479. q6asm_add_hdr(ac, &adjust_clock.hdr, sz, TRUE);
  9480. atomic_set(&ac->cmd_state, -1);
  9481. adjust_clock.hdr.opcode = ASM_SESSION_CMD_ADJUST_SESSION_CLOCK_V2;
  9482. adjust_clock.adjustime_lsw = adjust_time_lsw;
  9483. adjust_clock.adjustime_msw = adjust_time_msw;
  9484. rc = apr_send_pkt(ac->apr, (uint32_t *) &adjust_clock);
  9485. if (rc < 0) {
  9486. pr_err("%s: adjust_clock send failed paramid [0x%x]\n",
  9487. __func__, adjust_clock.hdr.opcode);
  9488. rc = -EINVAL;
  9489. goto fail_cmd;
  9490. }
  9491. rc = wait_event_timeout(ac->cmd_wait,
  9492. (atomic_read(&ac->cmd_state) >= 0),
  9493. msecs_to_jiffies(TIMEOUT_MS));
  9494. if (!rc) {
  9495. pr_err("%s: timeout, adjust_clock paramid[0x%x]\n",
  9496. __func__, adjust_clock.hdr.opcode);
  9497. rc = -ETIMEDOUT;
  9498. goto fail_cmd;
  9499. }
  9500. if (atomic_read(&ac->cmd_state) > 0) {
  9501. pr_err("%s: DSP returned error[%s]\n",
  9502. __func__, adsp_err_get_err_str(
  9503. atomic_read(&ac->cmd_state)));
  9504. rc = adsp_err_get_lnx_err_code(
  9505. atomic_read(&ac->cmd_state));
  9506. goto fail_cmd;
  9507. }
  9508. rc = 0;
  9509. fail_cmd:
  9510. return rc;
  9511. }
  9512. EXPORT_SYMBOL(q6asm_adjust_session_clock);
  9513. /*
  9514. * q6asm_get_path_delay() - get the path delay for an audio session
  9515. * @ac: audio client handle
  9516. *
  9517. * Retrieves the current audio DSP path delay for the given audio session.
  9518. *
  9519. * Return: 0 on success, error code otherwise
  9520. */
  9521. int q6asm_get_path_delay(struct audio_client *ac)
  9522. {
  9523. int rc = 0;
  9524. struct apr_hdr hdr;
  9525. if (!ac || ac->apr == NULL) {
  9526. pr_err("%s: invalid audio client\n", __func__);
  9527. return -EINVAL;
  9528. }
  9529. hdr.opcode = ASM_SESSION_CMD_GET_PATH_DELAY_V2;
  9530. q6asm_add_hdr(ac, &hdr, sizeof(hdr), TRUE);
  9531. atomic_set(&ac->cmd_state, -1);
  9532. rc = apr_send_pkt(ac->apr, (uint32_t *) &hdr);
  9533. if (rc < 0) {
  9534. pr_err("%s: Commmand 0x%x failed %d\n", __func__,
  9535. hdr.opcode, rc);
  9536. return rc;
  9537. }
  9538. rc = wait_event_timeout(ac->cmd_wait,
  9539. (atomic_read(&ac->cmd_state) >= 0),
  9540. msecs_to_jiffies(TIMEOUT_MS));
  9541. if (!rc) {
  9542. pr_err("%s: timeout. waited for response opcode[0x%x]\n",
  9543. __func__, hdr.opcode);
  9544. return -ETIMEDOUT;
  9545. }
  9546. if (atomic_read(&ac->cmd_state) > 0) {
  9547. pr_err("%s: DSP returned error[%s]\n",
  9548. __func__, adsp_err_get_err_str(
  9549. atomic_read(&ac->cmd_state)));
  9550. rc = adsp_err_get_lnx_err_code(
  9551. atomic_read(&ac->cmd_state));
  9552. return rc;
  9553. }
  9554. return 0;
  9555. }
  9556. EXPORT_SYMBOL(q6asm_get_path_delay);
  9557. int q6asm_get_apr_service_id(int session_id)
  9558. {
  9559. pr_debug("%s:\n", __func__);
  9560. if (session_id <= 0 || session_id > ASM_ACTIVE_STREAMS_ALLOWED) {
  9561. pr_err("%s: invalid session_id = %d\n", __func__, session_id);
  9562. return -EINVAL;
  9563. }
  9564. return ((struct apr_svc *)(session[session_id].ac)->apr)->id;
  9565. }
  9566. int q6asm_get_asm_topology(int session_id)
  9567. {
  9568. int topology = -EINVAL;
  9569. if (session_id <= 0 || session_id > ASM_ACTIVE_STREAMS_ALLOWED) {
  9570. pr_err("%s: invalid session_id = %d\n", __func__, session_id);
  9571. goto done;
  9572. }
  9573. if (session[session_id].ac == NULL) {
  9574. pr_err("%s: session not created for session id = %d\n",
  9575. __func__, session_id);
  9576. goto done;
  9577. }
  9578. topology = (session[session_id].ac)->topology;
  9579. done:
  9580. return topology;
  9581. }
  9582. int q6asm_get_asm_app_type(int session_id)
  9583. {
  9584. int app_type = -EINVAL;
  9585. if (session_id <= 0 || session_id > ASM_ACTIVE_STREAMS_ALLOWED) {
  9586. pr_err("%s: invalid session_id = %d\n", __func__, session_id);
  9587. goto done;
  9588. }
  9589. if (session[session_id].ac == NULL) {
  9590. pr_err("%s: session not created for session id = %d\n",
  9591. __func__, session_id);
  9592. goto done;
  9593. }
  9594. app_type = (session[session_id].ac)->app_type;
  9595. done:
  9596. return app_type;
  9597. }
  9598. /*
  9599. * Retrieving cal_block will mark cal_block as stale.
  9600. * Hence it cannot be reused or resent unless the flag
  9601. * is reset.
  9602. */
  9603. static int q6asm_get_asm_topology_apptype(struct q6asm_cal_info *cal_info)
  9604. {
  9605. struct cal_block_data *cal_block = NULL;
  9606. cal_info->topology_id = DEFAULT_POPP_TOPOLOGY;
  9607. cal_info->app_type = DEFAULT_APP_TYPE;
  9608. if (cal_data[ASM_TOPOLOGY_CAL] == NULL)
  9609. goto done;
  9610. mutex_lock(&cal_data[ASM_TOPOLOGY_CAL]->lock);
  9611. cal_block = cal_utils_get_only_cal_block(cal_data[ASM_TOPOLOGY_CAL]);
  9612. if (cal_block == NULL || cal_utils_is_cal_stale(cal_block))
  9613. goto unlock;
  9614. cal_info->topology_id = ((struct audio_cal_info_asm_top *)
  9615. cal_block->cal_info)->topology;
  9616. cal_info->app_type = ((struct audio_cal_info_asm_top *)
  9617. cal_block->cal_info)->app_type;
  9618. cal_utils_mark_cal_used(cal_block);
  9619. unlock:
  9620. mutex_unlock(&cal_data[ASM_TOPOLOGY_CAL]->lock);
  9621. done:
  9622. pr_debug("%s: Using topology %d app_type %d\n", __func__,
  9623. cal_info->topology_id, cal_info->app_type);
  9624. return 0;
  9625. }
  9626. /**
  9627. * q6asm_send_cal -
  9628. * command to send ASM calibration
  9629. *
  9630. * @ac: Audio client handle
  9631. *
  9632. * Returns 0 on success or error on failure
  9633. */
  9634. int q6asm_send_cal(struct audio_client *ac)
  9635. {
  9636. struct cal_block_data *cal_block = NULL;
  9637. struct mem_mapping_hdr mem_hdr;
  9638. u32 payload_size = 0;
  9639. int rc = -EINVAL;
  9640. pr_debug("%s:\n", __func__);
  9641. if (!ac) {
  9642. pr_err("%s: Audio client is NULL\n", __func__);
  9643. return -EINVAL;
  9644. }
  9645. if (ac->io_mode & NT_MODE) {
  9646. pr_debug("%s: called for NT MODE, exiting\n", __func__);
  9647. goto done;
  9648. }
  9649. if (cal_data[ASM_AUDSTRM_CAL] == NULL)
  9650. goto done;
  9651. if (ac->perf_mode == ULTRA_LOW_LATENCY_PCM_MODE) {
  9652. rc = 0; /* no cal is required, not error case */
  9653. goto done;
  9654. }
  9655. memset(&mem_hdr, 0, sizeof(mem_hdr));
  9656. mutex_lock(&cal_data[ASM_AUDSTRM_CAL]->lock);
  9657. cal_block = cal_utils_get_only_cal_block(cal_data[ASM_AUDSTRM_CAL]);
  9658. if (cal_block == NULL) {
  9659. pr_err("%s: cal_block is NULL\n",
  9660. __func__);
  9661. goto unlock;
  9662. }
  9663. if (cal_utils_is_cal_stale(cal_block)) {
  9664. rc = 0; /* not error case */
  9665. pr_debug("%s: cal_block is stale\n",
  9666. __func__);
  9667. goto unlock;
  9668. }
  9669. if (cal_block->cal_data.size == 0) {
  9670. rc = 0; /* not error case */
  9671. pr_debug("%s: cal_data.size is 0, don't send cal data\n",
  9672. __func__);
  9673. goto unlock;
  9674. }
  9675. rc = remap_cal_data(ASM_AUDSTRM_CAL_TYPE, cal_block);
  9676. if (rc) {
  9677. pr_err("%s: Remap_cal_data failed for cal %d!\n",
  9678. __func__, ASM_AUDSTRM_CAL);
  9679. goto unlock;
  9680. }
  9681. mem_hdr.data_payload_addr_lsw =
  9682. lower_32_bits(cal_block->cal_data.paddr);
  9683. mem_hdr.data_payload_addr_msw =
  9684. msm_audio_populate_upper_32_bits(cal_block->cal_data.paddr);
  9685. mem_hdr.mem_map_handle = cal_block->map_data.q6map_handle;
  9686. payload_size = cal_block->cal_data.size;
  9687. pr_debug("%s: phyaddr lsw = %x msw = %x, maphdl = %x calsize = %d\n",
  9688. __func__, mem_hdr.data_payload_addr_lsw,
  9689. mem_hdr.data_payload_addr_msw, mem_hdr.mem_map_handle,
  9690. payload_size);
  9691. rc = q6asm_set_pp_params(ac, &mem_hdr, NULL, payload_size);
  9692. if (rc) {
  9693. pr_err("%s: audio audstrm cal send failed\n", __func__);
  9694. goto unlock;
  9695. }
  9696. if (cal_block)
  9697. cal_utils_mark_cal_used(cal_block);
  9698. rc = 0;
  9699. unlock:
  9700. mutex_unlock(&cal_data[ASM_AUDSTRM_CAL]->lock);
  9701. done:
  9702. return rc;
  9703. }
  9704. EXPORT_SYMBOL(q6asm_send_cal);
  9705. static int get_cal_type_index(int32_t cal_type)
  9706. {
  9707. int ret = -EINVAL;
  9708. switch (cal_type) {
  9709. case ASM_TOPOLOGY_CAL_TYPE:
  9710. ret = ASM_TOPOLOGY_CAL;
  9711. break;
  9712. case ASM_CUST_TOPOLOGY_CAL_TYPE:
  9713. ret = ASM_CUSTOM_TOP_CAL;
  9714. break;
  9715. case ASM_AUDSTRM_CAL_TYPE:
  9716. ret = ASM_AUDSTRM_CAL;
  9717. break;
  9718. case ASM_RTAC_APR_CAL_TYPE:
  9719. ret = ASM_RTAC_APR_CAL;
  9720. break;
  9721. default:
  9722. pr_err("%s: invalid cal type %d!\n", __func__, cal_type);
  9723. }
  9724. return ret;
  9725. }
  9726. static int q6asm_alloc_cal(int32_t cal_type,
  9727. size_t data_size, void *data)
  9728. {
  9729. int ret = 0;
  9730. int cal_index;
  9731. pr_debug("%s:\n", __func__);
  9732. cal_index = get_cal_type_index(cal_type);
  9733. if (cal_index < 0) {
  9734. pr_err("%s: could not get cal index %d!\n",
  9735. __func__, cal_index);
  9736. ret = -EINVAL;
  9737. goto done;
  9738. }
  9739. ret = cal_utils_alloc_cal(data_size, data,
  9740. cal_data[cal_index], 0, NULL);
  9741. if (ret < 0) {
  9742. pr_err("%s: cal_utils_alloc_block failed, ret = %d, cal type = %d!\n",
  9743. __func__, ret, cal_type);
  9744. ret = -EINVAL;
  9745. goto done;
  9746. }
  9747. done:
  9748. return ret;
  9749. }
  9750. static int q6asm_dealloc_cal(int32_t cal_type,
  9751. size_t data_size, void *data)
  9752. {
  9753. int ret = 0;
  9754. int cal_index;
  9755. pr_debug("%s:\n", __func__);
  9756. cal_index = get_cal_type_index(cal_type);
  9757. if (cal_index < 0) {
  9758. pr_err("%s: could not get cal index %d!\n",
  9759. __func__, cal_index);
  9760. ret = -EINVAL;
  9761. goto done;
  9762. }
  9763. ret = cal_utils_dealloc_cal(data_size, data,
  9764. cal_data[cal_index]);
  9765. if (ret < 0) {
  9766. pr_err("%s: cal_utils_dealloc_block failed, ret = %d, cal type = %d!\n",
  9767. __func__, ret, cal_type);
  9768. ret = -EINVAL;
  9769. goto done;
  9770. }
  9771. done:
  9772. return ret;
  9773. }
  9774. static int q6asm_set_cal(int32_t cal_type,
  9775. size_t data_size, void *data)
  9776. {
  9777. int ret = 0;
  9778. int cal_index;
  9779. pr_debug("%s:\n", __func__);
  9780. cal_index = get_cal_type_index(cal_type);
  9781. if (cal_index < 0) {
  9782. pr_err("%s: could not get cal index %d!\n",
  9783. __func__, cal_index);
  9784. ret = -EINVAL;
  9785. goto done;
  9786. }
  9787. ret = cal_utils_set_cal(data_size, data,
  9788. cal_data[cal_index], 0, NULL);
  9789. if (ret < 0) {
  9790. pr_err("%s: cal_utils_set_cal failed, ret = %d, cal type = %d!\n",
  9791. __func__, ret, cal_type);
  9792. ret = -EINVAL;
  9793. goto done;
  9794. }
  9795. if (cal_index == ASM_CUSTOM_TOP_CAL) {
  9796. mutex_lock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  9797. set_custom_topology = 1;
  9798. mutex_unlock(&cal_data[ASM_CUSTOM_TOP_CAL]->lock);
  9799. }
  9800. done:
  9801. return ret;
  9802. }
  9803. static void q6asm_delete_cal_data(void)
  9804. {
  9805. pr_debug("%s:\n", __func__);
  9806. cal_utils_destroy_cal_types(ASM_MAX_CAL_TYPES, cal_data);
  9807. }
  9808. static int q6asm_init_cal_data(void)
  9809. {
  9810. int ret = 0;
  9811. struct cal_type_info cal_type_info[] = {
  9812. {{ASM_TOPOLOGY_CAL_TYPE,
  9813. {NULL, NULL, NULL,
  9814. q6asm_set_cal, NULL, NULL} },
  9815. {NULL, NULL, cal_utils_match_buf_num} },
  9816. {{ASM_CUST_TOPOLOGY_CAL_TYPE,
  9817. {q6asm_alloc_cal, q6asm_dealloc_cal, NULL,
  9818. q6asm_set_cal, NULL, NULL} },
  9819. {NULL, q6asm_unmap_cal_memory, cal_utils_match_buf_num} },
  9820. {{ASM_AUDSTRM_CAL_TYPE,
  9821. {q6asm_alloc_cal, q6asm_dealloc_cal, NULL,
  9822. q6asm_set_cal, NULL, NULL} },
  9823. {NULL, q6asm_unmap_cal_memory, cal_utils_match_buf_num} },
  9824. {{ASM_RTAC_APR_CAL_TYPE,
  9825. {NULL, NULL, NULL, NULL, NULL, NULL} },
  9826. {NULL, NULL, cal_utils_match_buf_num} }
  9827. };
  9828. pr_debug("%s\n", __func__);
  9829. ret = cal_utils_create_cal_types(ASM_MAX_CAL_TYPES, cal_data,
  9830. cal_type_info);
  9831. if (ret < 0) {
  9832. pr_err("%s: could not create cal type! %d\n",
  9833. __func__, ret);
  9834. ret = -EINVAL;
  9835. goto err;
  9836. }
  9837. return ret;
  9838. err:
  9839. q6asm_delete_cal_data();
  9840. return ret;
  9841. }
  9842. static int q6asm_is_valid_session(struct apr_client_data *data, void *priv)
  9843. {
  9844. struct audio_client *ac = (struct audio_client *)priv;
  9845. union asm_token_struct asm_token;
  9846. asm_token.token = data->token;
  9847. if (asm_token._token.session_id != ac->session) {
  9848. pr_err("%s: Invalid session[%d] rxed expected[%d]",
  9849. __func__, asm_token._token.session_id, ac->session);
  9850. return -EINVAL;
  9851. }
  9852. return 0;
  9853. }
  9854. int __init q6asm_init(void)
  9855. {
  9856. int lcnt, ret;
  9857. pr_debug("%s:\n", __func__);
  9858. memset(session, 0, sizeof(struct audio_session) *
  9859. (ASM_ACTIVE_STREAMS_ALLOWED + 1));
  9860. for (lcnt = 0; lcnt <= ASM_ACTIVE_STREAMS_ALLOWED; lcnt++) {
  9861. spin_lock_init(&(session[lcnt].session_lock));
  9862. mutex_init(&(session[lcnt].mutex_lock_per_session));
  9863. }
  9864. set_custom_topology = 1;
  9865. /*setup common client used for cal mem map */
  9866. common_client.session = ASM_CONTROL_SESSION;
  9867. common_client.port[0].buf = &common_buf[0];
  9868. common_client.port[1].buf = &common_buf[1];
  9869. init_waitqueue_head(&common_client.cmd_wait);
  9870. init_waitqueue_head(&common_client.time_wait);
  9871. init_waitqueue_head(&common_client.mem_wait);
  9872. atomic_set(&common_client.time_flag, 1);
  9873. INIT_LIST_HEAD(&common_client.port[0].mem_map_handle);
  9874. INIT_LIST_HEAD(&common_client.port[1].mem_map_handle);
  9875. mutex_init(&common_client.cmd_lock);
  9876. for (lcnt = 0; lcnt <= OUT; lcnt++) {
  9877. mutex_init(&common_client.port[lcnt].lock);
  9878. spin_lock_init(&common_client.port[lcnt].dsp_lock);
  9879. }
  9880. atomic_set(&common_client.cmd_state, 0);
  9881. atomic_set(&common_client.mem_state, 0);
  9882. ret = q6asm_init_cal_data();
  9883. if (ret)
  9884. pr_err("%s: could not init cal data! ret %d\n",
  9885. __func__, ret);
  9886. config_debug_fs_init();
  9887. return 0;
  9888. }
  9889. void q6asm_exit(void)
  9890. {
  9891. q6asm_delete_cal_data();
  9892. }