primary_display.c 284 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868786978707871787278737874787578767877787878797880788178827883788478857886788778887889789078917892789378947895789678977898789979007901790279037904790579067907790879097910791179127913791479157916791779187919792079217922792379247925792679277928792979307931793279337934793579367937793879397940794179427943794479457946794779487949795079517952795379547955795679577958795979607961796279637964796579667967796879697970797179727973797479757976797779787979798079817982798379847985798679877988798979907991799279937994799579967997799879998000800180028003800480058006800780088009801080118012801380148015801680178018801980208021802280238024802580268027802880298030803180328033803480358036803780388039804080418042804380448045804680478048804980508051805280538054805580568057805880598060806180628063806480658066806780688069807080718072807380748075807680778078807980808081808280838084808580868087808880898090809180928093809480958096809780988099810081018102810381048105810681078108810981108111811281138114811581168117811881198120812181228123812481258126812781288129813081318132813381348135813681378138813981408141814281438144814581468147814881498150815181528153815481558156815781588159816081618162816381648165816681678168816981708171817281738174817581768177817881798180818181828183818481858186818781888189819081918192819381948195819681978198819982008201820282038204820582068207820882098210821182128213821482158216821782188219822082218222822382248225822682278228822982308231823282338234823582368237823882398240824182428243824482458246824782488249825082518252825382548255825682578258825982608261826282638264826582668267826882698270827182728273827482758276827782788279828082818282828382848285828682878288828982908291829282938294829582968297829882998300830183028303830483058306830783088309831083118312831383148315831683178318831983208321832283238324832583268327832883298330833183328333833483358336833783388339834083418342834383448345834683478348834983508351835283538354835583568357835883598360836183628363836483658366836783688369837083718372837383748375837683778378837983808381838283838384838583868387838883898390839183928393839483958396839783988399840084018402840384048405840684078408840984108411841284138414841584168417841884198420842184228423842484258426842784288429843084318432843384348435843684378438843984408441844284438444844584468447844884498450845184528453845484558456845784588459846084618462846384648465846684678468846984708471847284738474847584768477847884798480848184828483848484858486848784888489849084918492849384948495849684978498849985008501850285038504850585068507850885098510851185128513851485158516851785188519852085218522852385248525852685278528852985308531853285338534853585368537853885398540854185428543854485458546854785488549855085518552855385548555855685578558855985608561856285638564856585668567856885698570857185728573857485758576857785788579858085818582858385848585858685878588858985908591859285938594859585968597859885998600860186028603860486058606860786088609861086118612861386148615861686178618861986208621862286238624862586268627862886298630863186328633863486358636863786388639864086418642864386448645864686478648864986508651865286538654865586568657865886598660866186628663866486658666866786688669867086718672867386748675867686778678867986808681868286838684868586868687868886898690869186928693869486958696869786988699870087018702870387048705870687078708870987108711871287138714871587168717871887198720872187228723872487258726872787288729873087318732873387348735873687378738873987408741874287438744874587468747874887498750875187528753875487558756875787588759876087618762876387648765876687678768876987708771877287738774877587768777877887798780878187828783878487858786878787888789879087918792879387948795879687978798879988008801880288038804880588068807880888098810881188128813881488158816881788188819882088218822882388248825882688278828882988308831883288338834883588368837883888398840884188428843884488458846884788488849885088518852885388548855885688578858885988608861886288638864886588668867886888698870887188728873887488758876887788788879888088818882888388848885888688878888888988908891889288938894889588968897889888998900890189028903890489058906890789088909891089118912891389148915891689178918891989208921892289238924892589268927892889298930893189328933893489358936893789388939894089418942894389448945894689478948894989508951895289538954895589568957895889598960896189628963896489658966896789688969897089718972897389748975897689778978897989808981898289838984898589868987898889898990899189928993899489958996899789988999900090019002900390049005900690079008900990109011901290139014901590169017901890199020902190229023902490259026902790289029903090319032903390349035903690379038903990409041904290439044904590469047904890499050905190529053905490559056905790589059906090619062906390649065906690679068906990709071907290739074907590769077907890799080908190829083908490859086908790889089909090919092909390949095909690979098909991009101910291039104910591069107910891099110911191129113911491159116911791189119912091219122912391249125912691279128912991309131913291339134913591369137913891399140914191429143914491459146914791489149915091519152915391549155915691579158915991609161916291639164916591669167916891699170917191729173917491759176917791789179918091819182918391849185918691879188918991909191919291939194919591969197919891999200920192029203920492059206920792089209921092119212921392149215921692179218921992209221922292239224922592269227922892299230923192329233923492359236923792389239924092419242924392449245924692479248924992509251925292539254925592569257925892599260926192629263926492659266926792689269927092719272927392749275927692779278927992809281928292839284928592869287928892899290929192929293929492959296929792989299930093019302930393049305930693079308930993109311931293139314931593169317931893199320932193229323932493259326932793289329933093319332933393349335933693379338933993409341934293439344934593469347934893499350935193529353935493559356935793589359936093619362936393649365936693679368936993709371937293739374937593769377937893799380938193829383938493859386938793889389939093919392939393949395939693979398939994009401940294039404940594069407940894099410941194129413941494159416941794189419942094219422942394249425942694279428942994309431943294339434943594369437943894399440944194429443944494459446944794489449945094519452945394549455945694579458945994609461946294639464946594669467946894699470947194729473947494759476947794789479948094819482948394849485948694879488948994909491949294939494949594969497949894999500950195029503950495059506950795089509951095119512
  1. #include <linux/delay.h>
  2. #include <linux/sched.h>
  3. #include <linux/semaphore.h>
  4. #include <linux/module.h>
  5. #include <linux/wait.h>
  6. #include <linux/kthread.h>
  7. #include <linux/mutex.h>
  8. #include <linux/types.h>
  9. #include <linux/ktime.h>
  10. #include <linux/of.h>
  11. #include <linux/of_irq.h>
  12. #include <linux/vmalloc.h>
  13. #include <linux/slab.h>
  14. #include <linux/switch.h>
  15. #include "disp_drv_platform.h"
  16. #include "ion_drv.h"
  17. #include "mtk_ion.h"
  18. #include "m4u.h"
  19. #include "mt_idle.h"
  20. #include "mt_spm_idle.h"
  21. #include "mt_spm.h" /* for sodi reg addr define */
  22. #define mt_eint_set_hw_debounce(eint_num, ms) (void)0
  23. /* #include <linux/rtpm_prio.h> */
  24. /* #include "mach/eint.h" */
  25. #include <mt-plat/mt_gpio.h>
  26. #ifdef CONFIG_MTK_CLKMGR
  27. #include <mach/mt_clkmgr.h>
  28. #endif
  29. #if defined(CONFIG_MTK_LEGACY)
  30. /* #include <cust_eint.h> */
  31. /* #include <cust_gpio_usage.h> */
  32. #endif
  33. #include "mtk_sync.h"
  34. #include "mtkfb.h"
  35. #include "mtkfb_fence.h"
  36. #include "disp_session.h"
  37. #include "debug.h"
  38. #include "disp_drv_log.h"
  39. #include "disp_lcm.h"
  40. #include "disp_utils.h"
  41. #include "display_recorder.h"
  42. #include "fbconfig_kdebug.h"
  43. #include "primary_display.h"
  44. #include "disp_helper.h"
  45. #include "ddp_hal.h"
  46. #include "ddp_dump.h"
  47. #include "ddp_path.h"
  48. #include "ddp_drv.h"
  49. #include "ddp_ovl.h"
  50. #include "ddp_rdma.h"
  51. #include "ddp_manager.h"
  52. #include "ddp_mmp.h"
  53. #include "ddp_reg.h"
  54. #include "ddp_debug.h"
  55. #include "ddp_irq.h"
  56. #ifndef MTK_FB_CMDQ_DISABLE
  57. #include "cmdq_def.h"
  58. #include "cmdq_record.h"
  59. #include "cmdq_reg.h"
  60. #include "cmdq_core.h"
  61. #endif
  62. #include "disp_assert_layer.h"
  63. #include "ddp_dsi.h"
  64. #include "mtk_disp_mgr.h"
  65. #include "ddp_wdma.h"
  66. #include "ddp_wdma_ex.h"
  67. #include "mt_clk_id.h"
  68. #include "mmdvfs_mgr.h"
  69. #include "mt_smi.h"
  70. #include <mach/mt_freqhopping.h>
  71. typedef void (*fence_release_callback) (unsigned int data);
  72. unsigned int is_hwc_enabled = 0;
  73. static int is_hwc_update_frame;
  74. unsigned int gEnableLowPowerFeature = 0;
  75. /* int _trigger_display_interface(int blocking, void *callback, unsigned int userdata); */
  76. int primary_display_use_cmdq = CMDQ_DISABLE;
  77. int primary_display_use_m4u = 1;
  78. #if defined(MTK_OVL_DECOUPLE_SUPPORT)
  79. DISP_PRIMARY_PATH_MODE primary_display_mode = DECOUPLE_MODE;
  80. #else
  81. DISP_PRIMARY_PATH_MODE primary_display_mode = DIRECT_LINK_MODE;
  82. #endif
  83. static unsigned long dim_layer_mva;
  84. /* wdma dump thread */
  85. static unsigned int primary_dump_wdma;
  86. static struct task_struct *primary_display_wdma_out;
  87. static unsigned long dc_vAddr[DISP_INTERNAL_BUFFER_COUNT];
  88. static disp_internal_buffer_info *decouple_buffer_info[DISP_INTERNAL_BUFFER_COUNT];
  89. static RDMA_CONFIG_STRUCT decouple_rdma_config;
  90. static WDMA_CONFIG_STRUCT decouple_wdma_config;
  91. static disp_mem_output_config mem_config;
  92. static unsigned int primary_session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  93. unsigned int ext_session_id = MAKE_DISP_SESSION(DISP_SESSION_MEMORY, 2);
  94. /* primary display uses itself's abs macro */
  95. #ifdef abs
  96. #undef abs
  97. #define abs(a) (((a) < 0) ? -(a) : (a))
  98. #endif
  99. #define FRM_UPDATE_SEQ_CACHE_NUM (DISP_INTERNAL_BUFFER_COUNT+1)
  100. static disp_frm_seq_info frm_update_sequence[FRM_UPDATE_SEQ_CACHE_NUM];
  101. static unsigned int frm_update_cnt;
  102. /* DDP_SCENARIO_ENUM ddp_scenario = DDP_SCENARIO_SUB_RDMA1_DISP; */
  103. #ifdef DISP_SWITCH_DST_MODE
  104. int primary_display_def_dst_mode = 0;
  105. int primary_display_cur_dst_mode = 0;
  106. #endif
  107. #ifdef CONFIG_OF
  108. /* extern unsigned int islcmconnected; */
  109. #endif
  110. #define ALIGN_TO(x, n) \
  111. (((x) + ((n) - 1)) & ~((n) - 1))
  112. int primary_trigger_cnt = 0;
  113. #define PRIMARY_DISPLAY_TRIGGER_CNT (1)
  114. unsigned int gEnterSodiAfterEOF = 0;
  115. unsigned int WDMA0_FRAME_START_FLAG = 0;
  116. unsigned int NEW_BUF_IDX = 0;
  117. unsigned int ALL_LAYER_DISABLE_STEP = 0;
  118. unsigned long long last_primary_trigger_time = 0xffffffffffffffff;
  119. unsigned int isIdlePowerOff = 0;
  120. unsigned int isDSIOff = 0;
  121. unsigned int gPresentFenceIndex = 0;
  122. unsigned int gTriggerDispMode = 0; /* 0: normal, 1: lcd only, 2: none of lcd and lcm */
  123. disp_ddp_path_config last_primary_config;
  124. static atomic_t DvfsIsHPM = ATOMIC_INIT(1);
  125. #ifdef CONFIG_TRUSTONIC_TRUSTED_UI
  126. static struct switch_dev disp_switch_data;
  127. #endif
  128. void enqueue_buffer(display_primary_path_context *ctx, struct list_head *head,
  129. disp_internal_buffer_info *buf)
  130. {
  131. if (ctx && head && buf)
  132. list_add_tail(&buf->list, head);
  133. /* DISPMSG("enqueue_buffer, head=0x%08x, buf=0x%08x, mva=0x%08x\n", head, buf, buf->mva); */
  134. }
  135. void reset_buffer(display_primary_path_context *ctx, disp_internal_buffer_info *buf)
  136. {
  137. if (ctx && buf)
  138. list_del_init(&buf->list);
  139. /* DISPMSG("reset_buffer, buf=0x%08x, mva=0x%08x\n", buf, buf->mva); */
  140. }
  141. disp_internal_buffer_info *dequeue_buffer(display_primary_path_context *ctx,
  142. struct list_head *head)
  143. {
  144. disp_internal_buffer_info *temp = NULL;
  145. if (ctx && head) {
  146. if (!list_empty(head)) {
  147. temp = list_entry(head->prev, disp_internal_buffer_info, list);
  148. /* DISPMSG("dequeue_buffer, head=0x%08x, buf=0x%08x, mva=0x%08x\n", head, temp, temp->mva); */
  149. list_del_init(&temp->list);
  150. if (list_empty(head))
  151. ; /* DISPMSG("after dequeue_buffer, head:0x%08x is empty\n", head); */
  152. } else {
  153. DISPMSG("list is empty, alloc new buffer\n");
  154. return NULL;
  155. }
  156. }
  157. return temp;
  158. }
  159. disp_internal_buffer_info *find_buffer_by_mva(display_primary_path_context *ctx,
  160. struct list_head *head, uint32_t mva)
  161. {
  162. disp_internal_buffer_info *buf = NULL;
  163. disp_internal_buffer_info *temp = NULL;
  164. if (ctx && head && mva) {
  165. if (!list_empty(head)) {
  166. list_for_each_entry(temp, head, list) {
  167. /* DISPMSG("find buffer: temp=0x%08x, mva=0x%08x\n", temp, temp->mva); */
  168. if (mva == temp->mva) {
  169. buf = temp;
  170. break;
  171. }
  172. }
  173. }
  174. /* DISPMSG("find buffer by mva, head=0x%08x, buf=0x%08x\n", head, buf); */
  175. }
  176. return buf;
  177. }
  178. #define pgc _get_context()
  179. static display_primary_path_context *_get_context(void)
  180. {
  181. static int is_context_inited;
  182. static display_primary_path_context g_context;
  183. if (!is_context_inited) {
  184. memset((void *)&g_context, 0, sizeof(display_primary_path_context));
  185. is_context_inited = 1;
  186. }
  187. return &g_context;
  188. }
  189. static inline int _is_mirror_mode(DISP_MODE mode)
  190. {
  191. if (mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE || mode == DISP_SESSION_DECOUPLE_MIRROR_MODE)
  192. return 1;
  193. else
  194. return 0;
  195. }
  196. static inline int _is_decouple_mode(DISP_MODE mode)
  197. {
  198. if (mode == DISP_SESSION_DECOUPLE_MODE || mode == DISP_SESSION_DECOUPLE_MIRROR_MODE)
  199. return 1;
  200. else
  201. return 0;
  202. }
  203. struct mutex esd_mode_switch_lock;
  204. static void _primary_path_esd_check_lock(void)
  205. {
  206. mutex_lock(&esd_mode_switch_lock);
  207. }
  208. static void _primary_path_esd_check_unlock(void)
  209. {
  210. mutex_unlock(&esd_mode_switch_lock);
  211. }
  212. display_primary_path_context *primary_display_path_lock(const char *caller)
  213. {
  214. dprec_logger_start(DPREC_LOGGER_PRIMARY_MUTEX, 0, 0);
  215. disp_sw_mutex_lock(&(pgc->lock));
  216. pgc->mutex_locker = (char *)caller;
  217. return pgc;
  218. }
  219. void primary_display_path_unlock(const char *caller)
  220. {
  221. pgc->mutex_locker = NULL;
  222. disp_sw_mutex_unlock(&(pgc->lock));
  223. dprec_logger_done(DPREC_LOGGER_PRIMARY_MUTEX, 0, 0);
  224. }
  225. static void _primary_path_cmd_lock(void)
  226. {
  227. mutex_lock(&(pgc->cmd_lock));
  228. }
  229. static void _primary_path_cmd_unlock(void)
  230. {
  231. mutex_unlock(&(pgc->cmd_lock));
  232. }
  233. static void _primary_path_lock(const char *caller)
  234. {
  235. dprec_logger_start(DPREC_LOGGER_PRIMARY_MUTEX, 0, 0);
  236. disp_sw_mutex_lock(&(pgc->lock));
  237. pgc->mutex_locker = (char *)caller;
  238. }
  239. static void _primary_path_unlock(const char *caller)
  240. {
  241. pgc->mutex_locker = NULL;
  242. disp_sw_mutex_unlock(&(pgc->lock));
  243. dprec_logger_done(DPREC_LOGGER_PRIMARY_MUTEX, 0, 0);
  244. }
  245. static DECLARE_WAIT_QUEUE_HEAD(display_state_wait_queue);
  246. static DISP_POWER_STATE primary_get_state(void)
  247. {
  248. return pgc->state;
  249. }
  250. static DISP_POWER_STATE primary_set_state(DISP_POWER_STATE new_state)
  251. {
  252. DISP_POWER_STATE old_state = pgc->state;
  253. pgc->state = new_state;
  254. DISPMSG("%s %d to %d\n", __func__, old_state, new_state);
  255. wake_up(&display_state_wait_queue);
  256. return old_state;
  257. }
  258. /* use MAX_SCHEDULE_TIMEOUT to wait for ever
  259. * NOTES: primary_path_lock should NOT be held when call this func !!!!!!!!
  260. */
  261. #define __primary_display_wait_state(condition, timeout) \
  262. ({ \
  263. wait_event_timeout(display_state_wait_queue, condition, timeout);\
  264. })
  265. long primary_display_wait_state(DISP_POWER_STATE state, long timeout)
  266. {
  267. long ret;
  268. ret = __primary_display_wait_state(primary_get_state() == state, timeout);
  269. return ret;
  270. }
  271. long primary_display_wait_not_state(DISP_POWER_STATE state, long timeout)
  272. {
  273. long ret;
  274. ret = __primary_display_wait_state(primary_get_state() != state, timeout);
  275. return ret;
  276. }
  277. static void _primary_path_vsync_lock(void)
  278. {
  279. mutex_lock(&(pgc->vsync_lock));
  280. }
  281. static void _primary_path_vsync_unlock(void)
  282. {
  283. mutex_unlock(&(pgc->vsync_lock));
  284. }
  285. static void _primary_path_set_dvfsHPM(bool bForcedinHPM, unsigned int needLock)
  286. {
  287. if (needLock)
  288. _primary_path_lock(__func__);
  289. if (bForcedinHPM)
  290. atomic_set(&DvfsIsHPM, 1);
  291. else
  292. atomic_set(&DvfsIsHPM, 0);
  293. if (needLock)
  294. _primary_path_unlock(__func__);
  295. }
  296. #ifdef MTK_DISP_IDLE_LP
  297. static bool _primary_path_IsForcedHPM(unsigned int needLock)
  298. {
  299. bool bResult;
  300. if (needLock)
  301. _primary_path_lock(__func__);
  302. if (atomic_read(&DvfsIsHPM) == 1)
  303. bResult = true;
  304. else
  305. bResult = false;
  306. if (needLock)
  307. _primary_path_unlock(__func__);
  308. return bResult;
  309. }
  310. #endif
  311. static void _cmdq_flush_config_handle_mira(void *handle, int blocking);
  312. #ifdef MTK_DISP_IDLE_LP
  313. static atomic_t isDdp_Idle = ATOMIC_INIT(0);
  314. static atomic_t idle_detect_flag = ATOMIC_INIT(0);
  315. static struct mutex idle_lock;
  316. static struct task_struct *primary_display_idle_detect_task;
  317. #define DISP_DSI_REG_VFP 0x28
  318. static DECLARE_WAIT_QUEUE_HEAD(idle_detect_wq);
  319. static int _disp_primary_path_idle_clock_on(unsigned int level)
  320. {
  321. dpmgr_path_idle_on(pgc->dpmgr_handle, NULL, level);
  322. return 0;
  323. }
  324. static int _disp_primary_path_idle_clock_off(unsigned int level)
  325. {
  326. dpmgr_path_idle_off(pgc->dpmgr_handle, NULL, level);
  327. return 0;
  328. }
  329. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  330. static void primary_suspend_release_present_fence(void)
  331. {
  332. int fence_increment = 0;
  333. disp_sync_info *layer_info = NULL;
  334. /* if session not created, do not release present fence*/
  335. if (pgc->session_id == 0) {
  336. DISPDBG("_get_sync_info fail in present_fence_release thread\n");
  337. return;
  338. }
  339. layer_info = _get_sync_info(pgc->session_id, disp_sync_get_present_timeline_id());
  340. if (layer_info == NULL) {
  341. DISPERR("_get_sync_info fail in present_fence_release thread\n");
  342. return;
  343. }
  344. _primary_path_lock(__func__);
  345. fence_increment = gPresentFenceIndex-layer_info->timeline->value;
  346. if (fence_increment > 0) {
  347. timeline_inc(layer_info->timeline, fence_increment);
  348. MMProfileLogEx(ddp_mmp_get_events()->present_fence_release,
  349. MMProfileFlagPulse, gPresentFenceIndex, fence_increment);
  350. }
  351. _primary_path_unlock(__func__);
  352. DISPPR_FENCE("RPF/%d/%d\n", gPresentFenceIndex, gPresentFenceIndex-layer_info->timeline->value);
  353. }
  354. #endif
  355. static DEFINE_SPINLOCK(gLockTopClockOff);
  356. static int _disp_primary_path_dsi_clock_on(unsigned int level)
  357. {
  358. if (!primary_display_is_video_mode()) {
  359. unsigned long flags;
  360. #ifndef CONFIG_MTK_CLKMGR
  361. ddp_clk_prepare(DISP_MTCMOS_CLK);
  362. #endif
  363. spin_lock_irqsave(&gLockTopClockOff, flags);
  364. isDSIOff = 0;
  365. dpmgr_path_dsi_on(pgc->dpmgr_handle, NULL, level);
  366. spin_unlock_irqrestore(&gLockTopClockOff, flags);
  367. }
  368. return 0;
  369. }
  370. static int _disp_primary_path_dsi_clock_off(unsigned int level)
  371. {
  372. if (!primary_display_is_video_mode()) {
  373. unsigned long flags;
  374. spin_lock_irqsave(&gLockTopClockOff, flags);
  375. dpmgr_path_dsi_off(pgc->dpmgr_handle, NULL, level);
  376. isDSIOff = 1;
  377. spin_unlock_irqrestore(&gLockTopClockOff, flags);
  378. #ifndef CONFIG_MTK_CLKMGR
  379. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  380. #endif
  381. }
  382. return 0;
  383. }
  384. int _disp_primary_path_set_vfp(int enter)
  385. {
  386. int ret = 0;
  387. if (primary_display_is_video_mode()) {
  388. LCM_PARAMS *lcm_param = disp_lcm_get_params(pgc->plcm);
  389. cmdqRecHandle cmdq_handle_vfp = NULL;
  390. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle_vfp);
  391. if (ret != 0) {
  392. DISPCHECK("fail to create primary cmdq handle for set vfp\n");
  393. return -1;
  394. }
  395. DISPCHECK("primary set vfp, handle=%p\n", cmdq_handle_vfp);
  396. cmdqRecReset(cmdq_handle_vfp);
  397. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle_vfp);
  398. if (enter)
  399. dpmgr_path_ioctl(pgc->dpmgr_handle, cmdq_handle_vfp, DDP_DSI_VFP_LP,
  400. (unsigned long *)&(lcm_param->dsi.vertical_vfp_lp));
  401. else
  402. dpmgr_path_ioctl(pgc->dpmgr_handle, cmdq_handle_vfp, DDP_DSI_VFP_LP,
  403. (unsigned long *)&(lcm_param->dsi.vertical_frontporch));
  404. MMProfileLogEx(ddp_mmp_get_events()->dal_clean, MMProfileFlagPulse, 0, enter);
  405. _cmdq_flush_config_handle_mira(cmdq_handle_vfp, 1);
  406. DISPCHECK("[VFP]cmdq_handle_vfp ret=%d\n", ret);
  407. cmdqRecDestroy(cmdq_handle_vfp);
  408. cmdq_handle_vfp = NULL;
  409. } else {
  410. DISPCHECK("CMD mode don't set vfp for lows\n");
  411. }
  412. return ret;
  413. }
  414. int primary_display_save_power_for_idle(int enter, unsigned int need_primary_lock)
  415. {
  416. static unsigned int isLowPowerMode;
  417. if (is_hwc_enabled == 0)
  418. return 0;
  419. if (need_primary_lock) /* if outer api has add primary lock, do not have to lock again */
  420. _primary_path_lock(__func__);
  421. if (primary_get_state() == DISP_SLEPT) {
  422. DISPMSG("suspend mode can not enable low power.\n");
  423. goto end;
  424. }
  425. if (enter == 1 && isLowPowerMode == 1) {
  426. DISPMSG("already in low power mode.\n");
  427. goto end;
  428. }
  429. if (enter == 0 && isLowPowerMode == 0) {
  430. DISPMSG("already not in low power mode.\n");
  431. goto end;
  432. }
  433. isLowPowerMode = enter;
  434. if (pgc->plcm == NULL) {
  435. DISPERR("lcm handle is null\n");
  436. goto end;
  437. }
  438. DISPMSG("low power in, enter=%d.\n", enter);
  439. if (disp_low_power_enlarge_blanking == 1) {
  440. #if 0
  441. if (pgc->plcm->params) {
  442. vfp_for_low_power = pgc->plcm->params->dsi.vertical_frontporch_for_low_power;
  443. vfp_original = pgc->plcm->params->dsi.vertical_frontporch;
  444. }
  445. if (enter)
  446. DISP_REG_SET(pgc->cmdq_handle_config, dispsys_reg[DISP_REG_DSI0] + DISP_DSI_REG_VFP,
  447. vfp_for_low_power);
  448. else
  449. DISP_REG_SET(pgc->cmdq_handle_config, dispsys_reg[DISP_REG_DSI0] + DISP_DSI_REG_VFP,
  450. vfp_original);
  451. #else
  452. if (primary_display_is_video_mode() == 1)
  453. _disp_primary_path_set_vfp(enter);
  454. #endif
  455. }
  456. if (1 == disp_low_power_disable_ddp_clock) {/* open */
  457. if (primary_display_is_video_mode() == 0 /* && gEnableLowPowerFeature==1 */) {
  458. static unsigned int disp_low_power_disable_ddp_clock_cnt;
  459. DISPDBG("MM clock, disp_low_power_disable_ddp_clock enter %d.\n",
  460. disp_low_power_disable_ddp_clock_cnt++);
  461. if (1 == enter) { /* only for command mode */
  462. if (isIdlePowerOff == 0) {
  463. /* extern void clk_stat_bug(void); */
  464. unsigned long flags;
  465. DISPMSG("off MM clock start.\n");
  466. spin_lock_irqsave(&gLockTopClockOff, flags);
  467. _disp_primary_path_idle_clock_off(0); /* parameter represent level */
  468. isIdlePowerOff = 1;
  469. spin_unlock_irqrestore(&gLockTopClockOff, flags);
  470. #ifndef CONFIG_MTK_CLKMGR
  471. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  472. #endif
  473. /* DISPMSG("off MM clock end.\n"); */
  474. DISPMSG("***start dump regs! clk_stat_check\n");
  475. #ifdef CONFIG_MTK_CLKMGR
  476. clk_stat_check(SYS_DIS);
  477. #endif
  478. /*DISPMSG("---- start dump regs! clk_stat_bug.\n"); */
  479. /* clk_stat_bug(); */
  480. }
  481. } else {
  482. if (isIdlePowerOff == 1) {
  483. unsigned long flags;
  484. DISPMSG("on MM clock start.\n");
  485. #ifndef CONFIG_MTK_CLKMGR
  486. ddp_clk_prepare(DISP_MTCMOS_CLK);
  487. #endif
  488. spin_lock_irqsave(&gLockTopClockOff, flags);
  489. isIdlePowerOff = 0;
  490. _disp_primary_path_idle_clock_on(0); /* parameter represent level */
  491. spin_unlock_irqrestore(&gLockTopClockOff, flags);
  492. DISPMSG("on MM clock end.\n");
  493. }
  494. }
  495. }
  496. }
  497. if (disp_low_power_disable_fence_thread == 1)
  498. ; /* already implemented. */
  499. /* no need idle lock, cause primary lock will be used inside switch_mode */
  500. if (disp_low_power_remove_ovl == 1) {
  501. if (primary_display_is_video_mode() == 1) {/* only for video mode */
  502. /* DISPMSG("[LP] OVL pgc->session_mode=%d, enter=%d\n", pgc->session_mode, enter); */
  503. if (enter) {
  504. if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  505. DISPDBG("[LP]remove ovl.\n");
  506. primary_display_switch_mode_nolock(DISP_SESSION_DECOUPLE_MODE,
  507. pgc->session_id, 1);
  508. }
  509. /* DISPMSG("disp_low_power_remove_ovl 2\n"); */
  510. } else {
  511. if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE) {
  512. DISPDBG("[LP]add ovl.\n");
  513. primary_display_switch_mode_nolock(DISP_SESSION_DIRECT_LINK_MODE,
  514. pgc->session_id, 1);
  515. }
  516. /* DISPMSG("disp_low_power_remove_ovl 4\n"); */
  517. }
  518. }
  519. }
  520. if (is_mmdvfs_supported() && mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D1_PLUS) {
  521. if (_primary_path_IsForcedHPM(0) == false) {
  522. DISPMSG("MMDVFS enter:%d\n", enter);
  523. if (enter)
  524. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_LOW); /* Vote to LPM mode */
  525. else if (primary_display_get_width() > 800)
  526. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_HIGH); /* Enter HPM mode */
  527. } else {
  528. DISPMSG("MMDVFS is forced into HPM\n");
  529. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_HIGH); /* Enter HPM mode */
  530. }
  531. }
  532. end:
  533. if (enter)
  534. atomic_set(&isDdp_Idle, 1);
  535. if (need_primary_lock)
  536. _primary_path_unlock(__func__);
  537. return 0;
  538. }
  539. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  540. static int release_idle_lp_dc_buffer(unsigned int need_primary_lock);
  541. static int allocate_idle_lp_dc_buffer(void);
  542. #endif
  543. void _disp_primary_path_exit_idle(const char *caller, unsigned int need_primary_lock)
  544. {
  545. /* _disp_primary_idle_lock(); */
  546. if (atomic_read(&isDdp_Idle) == 1) {
  547. DISPMSG("[ddp_idle_on]_disp_primary_path_exit_idle (%s) &&&\n", caller);
  548. primary_display_save_power_for_idle(0, need_primary_lock);
  549. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  550. if (primary_display_is_video_mode())
  551. release_idle_lp_dc_buffer(need_primary_lock);
  552. #endif
  553. atomic_set(&isDdp_Idle, 0);
  554. atomic_set(&idle_detect_flag, 1);
  555. wake_up(&idle_detect_wq);
  556. }
  557. /* _disp_primary_idle_unlock(); */
  558. }
  559. /* used by aal/pwm to exit idle before config register(for "screen idle top clock off" feature) */
  560. void disp_exit_idle_ex(const char *caller)
  561. {
  562. if (primary_display_is_video_mode() == 0) {
  563. disp_update_trigger_time();
  564. _disp_primary_path_exit_idle(caller, 0);
  565. }
  566. }
  567. static int _disp_primary_path_idle_detect_thread(void *data)
  568. {
  569. int ret = 0, idle_time;
  570. static int enter_cnt;
  571. static int end_cnt;
  572. static int stop_cnt;
  573. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  574. idle_time = 2000;
  575. #else
  576. idle_time = 500;
  577. #endif
  578. while (1) {
  579. msleep(idle_time); /* 0.5s trigger once */
  580. /* DISPMSG("[ddp_idle]_disp_primary_path_idle_detect start 1\n"); */
  581. if (gSkipIdleDetect || atomic_read(&isDdp_Idle) == 1 ||
  582. is_hwc_update_frame == 0) /* skip is already in idle mode */
  583. continue;
  584. _primary_path_lock(__func__);
  585. if (primary_get_state() != DISP_ALIVE) {
  586. MMProfileLogEx(ddp_mmp_get_events()->esd_check_t, MMProfileFlagPulse, 1, 0);
  587. /* DISPCHECK("[ddp_idle]primary display path is slept?? -- skip ddp_idle\n"); */
  588. _primary_path_unlock(__func__);
  589. continue;
  590. }
  591. if (pgc->session_mode != DISP_SESSION_DIRECT_LINK_MODE) {
  592. DISPCHECK("[ddp_idle]primary display path is decouple mode. -- skip ddp_idle\n");
  593. _primary_path_unlock(__func__);
  594. continue;
  595. }
  596. _primary_path_unlock(__func__);
  597. /* _disp_primary_idle_lock(); */
  598. _primary_path_esd_check_lock();
  599. if (((sched_clock() - last_primary_trigger_time) / 1000) > idle_time * 1000) {
  600. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  601. /* Dynamically allocate decouple buffer. */
  602. if (primary_display_is_video_mode()) {
  603. ret = 0;
  604. if (decouple_buffer_info[0] == NULL)
  605. ret = allocate_idle_lp_dc_buffer();
  606. if (ret < 0) {
  607. DISPCHECK("[ddp_idle]allocate dc buffer fail\n");
  608. _primary_path_esd_check_unlock();
  609. continue;
  610. }
  611. }
  612. #endif
  613. enter_cnt++;
  614. pr_debug("[LP] - enter: %d, flag:%d,%d\n", enter_cnt,
  615. atomic_read(&isDdp_Idle), atomic_read(&idle_detect_flag));
  616. primary_display_save_power_for_idle(1, 1);
  617. _primary_path_set_dvfsHPM(false, 0);
  618. _primary_path_esd_check_unlock();
  619. } else {
  620. /* _disp_primary_idle_unlock(); */
  621. _primary_path_set_dvfsHPM(false, 0);
  622. _primary_path_esd_check_unlock();
  623. continue;
  624. }
  625. /* _disp_primary_idle_unlock(); */
  626. ret = wait_event_interruptible(idle_detect_wq, (atomic_read(&idle_detect_flag) != 0));
  627. atomic_set(&idle_detect_flag, 0);
  628. /* printk("[ddp_idle]ret=%d\n", ret); */
  629. if (kthread_should_stop()) {
  630. stop_cnt++;
  631. pr_debug("[LP] stop: %d, flag:%d,%d\n", stop_cnt,
  632. atomic_read(&isDdp_Idle), atomic_read(&idle_detect_flag));
  633. break;
  634. }
  635. end_cnt++;
  636. pr_debug("[LP] end: %d, flag:%d,%d\n", end_cnt, atomic_read(&isDdp_Idle),
  637. atomic_read(&idle_detect_flag));
  638. if (enter_cnt != end_cnt) {
  639. pr_err("[LP][ASSERT]%d, %d, %d\n", enter_cnt, end_cnt, stop_cnt);
  640. ASSERT(0);
  641. }
  642. }
  643. return ret;
  644. }
  645. int primary_display_cmdq_set_reg(unsigned int addr, unsigned int val)
  646. {
  647. int ret = 0;
  648. cmdqRecHandle handle = NULL;
  649. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &handle);
  650. cmdqRecReset(handle);
  651. _cmdq_insert_wait_frame_done_token_mira(handle);
  652. cmdqRecWrite(handle, addr & 0x1fffffff, val, ~0);
  653. cmdqRecFlushAsync(handle);
  654. cmdqRecDestroy(handle);
  655. return 0;
  656. }
  657. #endif
  658. unsigned int dvfs_test = 0xFF;
  659. static atomic_t gMmsysClk = ATOMIC_INIT(0xFF);
  660. static void set_mmsys_clk(unsigned int clk)
  661. {
  662. if (atomic_read(&gMmsysClk) != clk)
  663. atomic_set(&gMmsysClk, clk);
  664. }
  665. static unsigned int get_mmsys_clk(void)
  666. {
  667. return atomic_read(&gMmsysClk);
  668. }
  669. static int _switch_mmsys_clk(int mmsys_clk_old, int mmsys_clk_new)
  670. {
  671. int ret = -1;
  672. cmdqRecHandle handle;
  673. DISPMSG("[LP]: %s\n", __func__);
  674. if (mmsys_clk_new == get_mmsys_clk())
  675. return ret;
  676. if (pgc->state == DISP_SLEPT) {
  677. DISP_REG_MASK(NULL, DISP_REG_CONFIG_C13, 0x07000000, 0x07000000); /* clear */
  678. DISP_REG_MASK(NULL, DISP_REG_CONFIG_C12, 0x06000000, 0x07000000); /* set syspll2_d2 */
  679. DISPMSG("[DISP] DVFS mode=%d, profile=%d, state=0x%x\n", mmsys_clk_new,
  680. mmdvfs_get_mmdvfs_profile(), pgc->state);
  681. switch (mmsys_clk_new) {
  682. case MMSYS_CLK_LOW:
  683. DISP_REG_MASK(NULL, DISP_REG_VENCPLL_CON1, 0x830E0000, 0xFFFFFFFF); /* update */
  684. break;
  685. case MMSYS_CLK_MEDIUM:
  686. /* by frequency hopping */
  687. goto cpu_d;
  688. case MMSYS_CLK_HIGH:
  689. if (mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_P_PLUS) {
  690. DISP_REG_MASK(NULL, DISP_REG_VENCPLL_CON1, 0x82110000, 0xFFFFFFFF); /* update */
  691. } else if (mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_M_PLUS) {
  692. DISP_REG_MASK(NULL, DISP_REG_VENCPLL_CON1, 0x820F0000, 0xFFFFFFFF); /* update */
  693. }
  694. break;
  695. default:
  696. DISPERR("[DISP] DVFS mode=%d\n", mmsys_clk_new);
  697. goto cpu_d;
  698. }
  699. udelay(40);
  700. DISP_REG_MASK(NULL, DISP_REG_CONFIG_C13, 0x07000000, 0x07000000); /* clear */
  701. DISP_REG_MASK(NULL, DISP_REG_CONFIG_C12, 0x01000000, 0x07000000); /* set vencpll_ck */
  702. cpu_d:
  703. /* set rdma golden setting parameters */
  704. set_mmsys_clk(mmsys_clk_new);
  705. return get_mmsys_clk();
  706. }
  707. /* 1.create and reset cmdq */
  708. cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &handle);
  709. cmdqRecReset(handle);
  710. /* wait eof */
  711. _cmdq_insert_wait_frame_done_token_mira(handle);
  712. cmdqRecWrite(handle, 0x10210048, 0x07000000, 0x07000000); /* clear */
  713. cmdqRecWrite(handle, 0x10210044, 0x06000000, 0x07000000); /* set syspll2_d2 */
  714. DISPMSG("[DISP] DVFS mode=%d, profile=%d, state=0x%x\n", mmsys_clk_new,
  715. mmdvfs_get_mmdvfs_profile(), pgc->state);
  716. switch (mmsys_clk_new) {
  717. case MMSYS_CLK_LOW:
  718. cmdqRecWrite(handle, 0x10209254, 0x830E0000, 0xFFFFFFFF); /* update */
  719. break;
  720. case MMSYS_CLK_MEDIUM:
  721. /* by frequency hopping */
  722. goto cmdq_d;
  723. case MMSYS_CLK_HIGH:
  724. if (mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_P_PLUS) {
  725. cmdqRecWrite(handle, 0x10209254, 0x82110000, 0xFFFFFFFF); /* update */
  726. } else if (mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_M_PLUS) {
  727. cmdqRecWrite(handle, 0x10209254, 0x820F0000, 0xFFFFFFFF); /* update */
  728. }
  729. break;
  730. default:
  731. DISPERR("[DISP] DVFS mode=%d\n", mmsys_clk_new);
  732. goto cmdq_d;
  733. }
  734. /* wait rdma0_sof: only used for video mode & trigger loop need wait and clear rdma0 sof */
  735. cmdqRecWaitNoClear(handle, CMDQ_EVENT_DISP_RDMA0_SOF);
  736. cmdqRecWrite(handle, 0x10210048, 0x07000000, 0x07000000); /* clear */
  737. cmdqRecWrite(handle, 0x10210044, 0x01000000, 0x07000000); /* set vencpll_ck */
  738. cmdqRecFlush(handle);
  739. cmdq_d:
  740. cmdqRecDestroy(handle);
  741. /* set rdma golden setting parameters */
  742. set_mmsys_clk(mmsys_clk_new);
  743. return get_mmsys_clk();
  744. }
  745. int primary_display_switch_mmsys_clk(int mmsys_clk_old, int mmsys_clk_new)
  746. {
  747. int ret = -1;
  748. /* need lock */
  749. DISPMSG("[LP] %s\n", __func__);
  750. primary_display_manual_lock();
  751. ret = _switch_mmsys_clk(mmsys_clk_old, mmsys_clk_new);
  752. primary_display_manual_unlock();
  753. return ret;
  754. }
  755. int primary_display_set_secondary_display(int add, DISP_SESSION_TYPE type)
  756. {
  757. if (add) {
  758. #ifdef MTK_DISP_IDLE_LP
  759. gSkipIdleDetect = 1;
  760. _disp_primary_path_exit_idle(__func__, 0);
  761. #endif
  762. } else {
  763. #ifdef MTK_DISP_IDLE_LP
  764. gSkipIdleDetect = 0;
  765. #endif
  766. }
  767. return 0;
  768. }
  769. static void update_frm_seq_info(unsigned int addr, unsigned int addr_offset, unsigned int seq,
  770. DISP_FRM_SEQ_STATE state)
  771. {
  772. int i = 0;
  773. if (FRM_CONFIG == state) {
  774. frm_update_sequence[frm_update_cnt].state = state;
  775. frm_update_sequence[frm_update_cnt].mva = addr;
  776. frm_update_sequence[frm_update_cnt].max_offset = addr_offset;
  777. if (seq > 0)
  778. frm_update_sequence[frm_update_cnt].seq = seq;
  779. MMProfileLogEx(ddp_mmp_get_events()->primary_seq_config, MMProfileFlagPulse, addr, seq);
  780. } else if (FRM_TRIGGER == state) {
  781. frm_update_sequence[frm_update_cnt].state = FRM_TRIGGER;
  782. MMProfileLogEx(ddp_mmp_get_events()->primary_seq_trigger, MMProfileFlagPulse,
  783. frm_update_cnt, frm_update_sequence[frm_update_cnt].seq);
  784. dprec_logger_frame_seq_begin(pgc->session_id, frm_update_sequence[frm_update_cnt].seq);
  785. frm_update_cnt++;
  786. frm_update_cnt %= FRM_UPDATE_SEQ_CACHE_NUM;
  787. } else if (FRM_START == state) {
  788. for (i = 0; i < FRM_UPDATE_SEQ_CACHE_NUM; i++) {
  789. if ((abs(addr - frm_update_sequence[i].mva) <= frm_update_sequence[i].max_offset) &&
  790. (frm_update_sequence[i].state == FRM_TRIGGER)) {
  791. MMProfileLogEx(ddp_mmp_get_events()->primary_seq_rdma_irq, MMProfileFlagPulse,
  792. frm_update_sequence[i].mva | seq, frm_update_sequence[i].state);
  793. frm_update_sequence[i].state = FRM_START;
  794. /* break; */
  795. }
  796. }
  797. } else if (FRM_END == state) {
  798. for (i = 0; i < FRM_UPDATE_SEQ_CACHE_NUM; i++) {
  799. if (FRM_START == frm_update_sequence[i].state) {
  800. frm_update_sequence[i].state = FRM_END;
  801. dprec_logger_frame_seq_end(pgc->session_id, frm_update_sequence[i].seq);
  802. MMProfileLogEx(ddp_mmp_get_events()->primary_seq_release, MMProfileFlagPulse,
  803. frm_update_sequence[i].mva, frm_update_sequence[i].seq);
  804. }
  805. }
  806. }
  807. }
  808. static int _config_wdma_output(WDMA_CONFIG_STRUCT *wdma_config, disp_path_handle disp_handle,
  809. cmdqRecHandle cmdq_handle)
  810. {
  811. disp_ddp_path_config *pconfig = dpmgr_path_get_last_config(disp_handle);
  812. pconfig->wdma_config = *wdma_config;
  813. pconfig->wdma_dirty = 1;
  814. dpmgr_path_config(disp_handle, pconfig, cmdq_handle);
  815. return 0;
  816. }
  817. static int _config_rdma_input_data(RDMA_CONFIG_STRUCT *rdma_config, disp_path_handle disp_handle,
  818. cmdqRecHandle cmdq_handle)
  819. {
  820. disp_ddp_path_config *pconfig = dpmgr_path_get_last_config(disp_handle);
  821. pconfig->rdma_config = *rdma_config;
  822. pconfig->rdma_dirty = 1;
  823. dpmgr_path_config(disp_handle, pconfig, cmdq_handle);
  824. return 0;
  825. }
  826. #if 0
  827. /**
  828. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  829. */
  830. static unsigned int get_frm_seq_by_addr(unsigned int addr, DISP_FRM_SEQ_STATE state)
  831. {
  832. int i = 0;
  833. for (i = 0; i < FRM_UPDATE_SEQ_CACHE_NUM; i++) {
  834. if (addr == frm_update_sequence[i].mva)
  835. return frm_update_sequence[i].seq;
  836. }
  837. return 0;
  838. }
  839. #endif
  840. #if 0
  841. /**
  842. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  843. */
  844. static unsigned int release_started_frm_seq(unsigned int addr, DISP_FRM_SEQ_STATE FRM_END)
  845. {
  846. int i = 0;
  847. for (i = 0; i < FRM_UPDATE_SEQ_CACHE_NUM; i++) {
  848. if (FRM_END == frm_update_sequence[i].state)
  849. return frm_update_sequence[i].seq;
  850. }
  851. return 0;
  852. }
  853. #endif
  854. #ifdef DISP_SWITCH_DST_MODE
  855. unsigned long long last_primary_trigger_time;
  856. bool is_switched_dst_mode = false;
  857. static struct task_struct *primary_display_switch_dst_mode_task;
  858. static void _primary_path_switch_dst_lock(void)
  859. {
  860. mutex_lock(&(pgc->switch_dst_lock));
  861. }
  862. static void _primary_path_switch_dst_unlock(void)
  863. {
  864. mutex_unlock(&(pgc->switch_dst_lock));
  865. }
  866. static int _disp_primary_path_switch_dst_mode_thread(void *data)
  867. {
  868. int ret = 0;
  869. while (1) {
  870. msleep(1000);
  871. if (((sched_clock() - last_primary_trigger_time) / 1000) > 500000) { /* 500ms not trigger disp */
  872. primary_display_switch_dst_mode(0); /* switch to cmd mode */
  873. is_switched_dst_mode = true;
  874. }
  875. if (kthread_should_stop())
  876. break;
  877. }
  878. return 0;
  879. }
  880. #endif
  881. unsigned long long idlemgr_last_kick_time = 0xffffffffffffffff;
  882. bool is_switched_dst_mode = false;
  883. /*----------------------------------------------------------------
  884. defined but not used because some functions are not used either.
  885. ----------------------------------------------------------------
  886. static wait_queue_head_t idlemgr_wq; // For display idle manager
  887. static atomic_t idlemgr_wakeup = ATOMIC_INIT(0); // For display idle manager
  888. ----------------------------------------------------------------*/
  889. struct task_struct *primary_display_idlemgr_task = NULL;
  890. #if 0
  891. /**
  892. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  893. */
  894. static void _idlemgr_wait_for_wakeup(void)
  895. {
  896. int ret = 0;
  897. atomic_set(&idlemgr_wakeup, 0);
  898. ret = wait_event_interruptible(idlemgr_wq, atomic_read(&idlemgr_wakeup));
  899. if (ret < 0)
  900. DISPERR("display idle manager thread waked up by signal\n");
  901. }
  902. #endif
  903. #if 0
  904. /**
  905. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  906. */
  907. static void _idlemgr_wakeup(void)
  908. {
  909. atomic_set(&idlemgr_wakeup, 1);
  910. wake_up_interruptible(&idlemgr_wq);
  911. }
  912. #endif
  913. #if 0
  914. /**
  915. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  916. */
  917. static void _idlemgr_update_last_kick_time(void)
  918. {
  919. idlemgr_last_kick_time = sched_clock();
  920. }
  921. #endif
  922. #if 0
  923. void primary_display_idlemgr_enter_idle(void)
  924. {
  925. MMProfileLogEx(ddp_mmp_get_events()->idlemgr_enter_idle, MMProfileFlagPulse, 0, 0);
  926. if (!primary_display_is_video_mode()) {
  927. spm_enable_sodi(1);
  928. spm_sodi_mempll_pwr_mode(0);
  929. MMProfileLogEx(ddp_mmp_get_events()->sodi_enable, MMProfileFlagPulse, 0, 0);
  930. }
  931. }
  932. void primary_display_idlemgr_leave_idle(void)
  933. {
  934. MMProfileLogEx(ddp_mmp_get_events()->idlemgr_leave_idle, MMProfileFlagPulse, 0, 0);
  935. if (!primary_display_is_video_mode()) {
  936. spm_enable_sodi(0);
  937. MMProfileLogEx(ddp_mmp_get_events()->sodi_disable, MMProfileFlagPulse, 0, 0);
  938. }
  939. }
  940. static int _primary_path_idlemgr_monitor_thread(void *data)
  941. {
  942. int ret = 0;
  943. init_waitqueue_head(&idlemgr_wq);
  944. while (1) {
  945. msleep(100);
  946. if (((sched_clock() - idlemgr_last_kick_time) / 1000) > 500000) {/* 500ms not trigger disp */
  947. primary_display_idlemgr_enter_idle(); /* switch to cmd mode */
  948. _idlemgr_wait_for_wakeup();
  949. }
  950. if (kthread_should_stop())
  951. break;
  952. }
  953. return 0;
  954. }
  955. #else
  956. void primary_display_idlemgr_kick(char *source)
  957. {
  958. }
  959. void primary_display_idlemgr_enter_idle(void)
  960. {
  961. }
  962. void primary_display_idlemgr_leave_idle(void)
  963. {
  964. }
  965. #if 0
  966. /**
  967. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  968. */
  969. static int _primary_path_idlemgr_monitor_thread(void *data)
  970. {
  971. return 0;
  972. }
  973. #endif
  974. #endif
  975. /* extern int disp_od_is_enabled(void); */
  976. int primary_display_get_debug_state(char *stringbuf, int buf_len)
  977. {
  978. int len = 0;
  979. LCM_PARAMS *lcm_param = disp_lcm_get_params(pgc->plcm);
  980. LCM_DRIVER *lcm_drv = pgc->plcm->drv;
  981. len += scnprintf(stringbuf + len, buf_len - len,
  982. "|--------------------------------------------------------------------------------------|\n");
  983. len += scnprintf(stringbuf + len, buf_len - len,
  984. "|********Primary Display Path General Information********\n");
  985. len += scnprintf(stringbuf + len, buf_len - len, "|Primary Display is %s\n",
  986. dpmgr_path_is_idle(pgc->dpmgr_handle) ? "idle" : "busy");
  987. if (mutex_trylock(&(pgc->lock))) {
  988. mutex_unlock(&(pgc->lock));
  989. len += scnprintf(stringbuf + len, buf_len - len,
  990. "|primary path global mutex is free\n");
  991. } else {
  992. len += scnprintf(stringbuf + len, buf_len - len,
  993. "|primary path global mutex is hold by [%s]\n", pgc->mutex_locker);
  994. }
  995. if (lcm_param && lcm_drv)
  996. len += scnprintf(stringbuf + len, buf_len - len,
  997. "|LCM Driver=[%s]\tResolution=%dx%d,Interface:%s\n", lcm_drv->name,
  998. lcm_param->width, lcm_param->height,
  999. (lcm_param->type == LCM_TYPE_DSI) ? "DSI" : "Other");
  1000. /* no OD */
  1001. /* len += scnprintf(stringbuf+len, buf_len - len, "|OD is %s\n", disp_od_is_enabled()?"enabled":"disabled"); */
  1002. len += scnprintf(stringbuf + len, buf_len - len, "|session_mode is 0x%08x\n", pgc->session_mode);
  1003. len += scnprintf(stringbuf + len, buf_len - len,
  1004. "|State=%s\tlcm_fps=%d\tmax_layer=%d\tmode:%d\tvsync_drop=%d\n",
  1005. pgc->state == DISP_ALIVE ? "Alive" : "Sleep", pgc->lcm_fps, pgc->max_layer,
  1006. pgc->mode, pgc->vsync_drop);
  1007. len += scnprintf(stringbuf + len, buf_len - len,
  1008. "|cmdq_handle_config=%p\tcmdq_handle_trigger=%p\tdpmgr_handle=%p\tovl2mem_path_handle=%p\n",
  1009. pgc->cmdq_handle_config, pgc->cmdq_handle_trigger, pgc->dpmgr_handle,
  1010. pgc->ovl2mem_path_handle);
  1011. len += scnprintf(stringbuf + len, buf_len - len, "|Current display driver status=%s + %s\n",
  1012. primary_display_is_video_mode() ? "video mode" : "cmd mode",
  1013. primary_display_cmdq_enabled() ? "CMDQ Enabled" : "CMDQ Disabled");
  1014. return len;
  1015. }
  1016. static char _dump_buffer_string[512];
  1017. static void _dump_internal_buffer_into(display_primary_path_context *ctx, char *dump_reason)
  1018. {
  1019. int n = 0;
  1020. int len = 0;
  1021. disp_internal_buffer_info *buf = NULL;
  1022. struct list_head *head = NULL;
  1023. int i = 0;
  1024. len = sizeof(_dump_buffer_string);
  1025. n += scnprintf(_dump_buffer_string + n, len - n, "dump for %s\n",
  1026. dump_reason ? dump_reason : "unknown");
  1027. if (ctx) {
  1028. mutex_lock(&(pgc->dc_lock));
  1029. head = &(pgc->dc_free_list);
  1030. i = 0;
  1031. n += scnprintf(_dump_buffer_string + n, len - n, "free list: ");
  1032. if (!list_empty(head)) {
  1033. list_for_each_entry(buf, head, list)
  1034. n += scnprintf(_dump_buffer_string + n, len - n, "\t0x%08x", buf->mva);
  1035. }
  1036. n += scnprintf(_dump_buffer_string + n, len - n, "\n");
  1037. #if 0
  1038. head = &(pgc->dc_writing_list);
  1039. i = 0;
  1040. n += scnprintf(_dump_buffer_string + n, len - n, "writing list: ");
  1041. if (!list_empty(head)) {
  1042. list_for_each_entry(buf, head, list)
  1043. n += scnprintf(_dump_buffer_string + n, len - n, "\t0x%08x", buf->mva);
  1044. }
  1045. n += scnprintf(_dump_buffer_string + n, len - n, "\n");
  1046. #endif
  1047. head = &(pgc->dc_reading_list);
  1048. i = 0;
  1049. n += scnprintf(_dump_buffer_string + n, len - n, "reading list: ");
  1050. if (!list_empty(head)) {
  1051. list_for_each_entry(buf, head, list)
  1052. n += scnprintf(_dump_buffer_string + n, len - n, "\t0x%08x", buf->mva);
  1053. }
  1054. n += scnprintf(_dump_buffer_string + n, len - n, "\n");
  1055. DISPMSG("%s", _dump_buffer_string);
  1056. mutex_unlock(&(pgc->dc_lock));
  1057. }
  1058. }
  1059. int32_t decouple_rdma_worker_callback(unsigned long data)
  1060. {
  1061. disp_internal_buffer_info *reading_buf = NULL;
  1062. disp_internal_buffer_info *temp = NULL;
  1063. disp_internal_buffer_info *n = NULL;
  1064. unsigned int current_reading_addr = 0;
  1065. mutex_lock(&(pgc->dc_lock));
  1066. current_reading_addr = ddp_ovl_get_cur_addr(1, 0);
  1067. DISPMSG("rdma applied, 0x%lx, 0x%08x\n", data, current_reading_addr);
  1068. reading_buf = find_buffer_by_mva(pgc, &pgc->dc_reading_list, data);
  1069. if (!reading_buf) {
  1070. /* DISPERR("fatal error, we can't find related buffer info with callback data:0x%08x\n", data); */
  1071. mutex_unlock(&(pgc->dc_lock));
  1072. return 0;
  1073. }
  1074. list_for_each_entry_safe(temp, n, &(pgc->dc_reading_list), list) {
  1075. if (temp && temp->mva != current_reading_addr) {
  1076. DISPMSG("temp=0x%p, temp->mva=0x%08x, temp->list=0x%lx\n", temp, temp->mva,
  1077. (unsigned long)(&temp->list));
  1078. reset_buffer(pgc, temp);
  1079. enqueue_buffer(pgc, &(pgc->dc_free_list), temp);
  1080. }
  1081. }
  1082. mutex_unlock(&(pgc->dc_lock));
  1083. _dump_internal_buffer_into(pgc, "interface applied");
  1084. return 0;
  1085. }
  1086. #if 0
  1087. /**
  1088. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  1089. */
  1090. static int _disp_primary_path_decouple_worker_thread(void *data)
  1091. {
  1092. int ret = 0;
  1093. primary_disp_input_config input;
  1094. disp_internal_buffer_info *writing_buf = NULL;
  1095. disp_internal_buffer_info *reading_buf = NULL;
  1096. uint32_t current_writing_mva = 0;
  1097. disp_ddp_path_config *pconfig = NULL;
  1098. struct sched_param param = { .sched_priority = 90 };
  1099. sched_setscheduler(current, SCHED_RR, &param);
  1100. while (1) {
  1101. /* shuold wait FRAME_START here, but MUTEX1's sof always issued with MUTEX0's sof,
  1102. * root cause still unkonwn. so use FRAME_COMPLETE instead
  1103. */
  1104. dpmgr_wait_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE);
  1105. pconfig = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  1106. current_writing_mva = pconfig->wdma_config.dstAddress;
  1107. _dump_internal_buffer_into(pgc, "ovl2mem done");
  1108. disp_sw_mutex_lock(&(pgc->dc_lock));
  1109. writing_buf = find_buffer_by_mva(pgc, &pgc->dc_writing_list, current_writing_mva);
  1110. if (!writing_buf) {
  1111. DISPERR("fatal error, we can't find related buffer info with current_writing_mva:0x%08x\n",
  1112. current_writing_mva);
  1113. disp_sw_mutex_unlock(&(pgc->dc_lock));
  1114. continue;
  1115. } else {
  1116. reset_buffer(pgc, writing_buf);
  1117. enqueue_buffer(pgc, &(pgc->dc_reading_list), writing_buf);
  1118. reading_buf = dequeue_buffer(pgc, &(pgc->dc_free_list));
  1119. if (!reading_buf) {
  1120. DISPERR("dc_free_list is empty!!\n");
  1121. disp_sw_mutex_unlock(&(pgc->dc_lock));
  1122. continue;
  1123. } else {
  1124. enqueue_buffer(pgc, &(pgc->dc_writing_list), reading_buf);
  1125. }
  1126. disp_sw_mutex_unlock(&(pgc->dc_lock));
  1127. }
  1128. _dump_internal_buffer_into(pgc, "trigger interface");
  1129. pconfig->wdma_config.dstAddress = reading_buf->mva;
  1130. pconfig->wdma_config.srcHeight = primary_display_get_height();
  1131. pconfig->wdma_config.srcWidth = primary_display_get_width();
  1132. pconfig->wdma_config.clipX = 0;
  1133. pconfig->wdma_config.clipY = 0;
  1134. pconfig->wdma_config.clipHeight = primary_display_get_height();
  1135. pconfig->wdma_config.clipWidth = primary_display_get_width();
  1136. pconfig->wdma_config.outputFormat = eRGB888;
  1137. pconfig->wdma_config.useSpecifiedAlpha = 1;
  1138. pconfig->wdma_config.alpha = 0xFF;
  1139. pconfig->wdma_config.dstPitch = primary_display_get_width() *
  1140. DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  1141. pconfig->wdma_dirty = 1;
  1142. ret = dpmgr_path_config(pgc->ovl2mem_path_handle, pconfig, CMDQ_DISABLE);
  1143. memset((void *)&input, 0, sizeof(primary_disp_input_config));
  1144. input.addr = (unsigned int) writing_buf->mva;
  1145. input.src_x = 0;
  1146. input.src_y = 0;
  1147. input.src_w = primary_display_get_width();
  1148. input.src_h = primary_display_get_height();
  1149. input.dst_x = 0;
  1150. input.dst_y = 0;
  1151. input.dst_w = primary_display_get_width();
  1152. input.dst_h = primary_display_get_height();
  1153. input.fmt = eRGB888;
  1154. input.alpha = 0xFF;
  1155. input.src_pitch = primary_display_get_width() * 3;
  1156. input.isDirty = 1;
  1157. MMProfileLogEx(ddp_mmp_get_events()->interface_trigger, MMProfileFlagPulse,
  1158. input.addr, pconfig->wdma_config.dstAddress);
  1159. ret = primary_display_config_interface_input(&input);
  1160. ret = _trigger_display_interface(FALSE, decouple_rdma_worker_callback, ddp_ovl_get_cur_addr(1, 0));
  1161. if (kthread_should_stop())
  1162. break;
  1163. }
  1164. return 0;
  1165. }
  1166. #endif
  1167. static DISP_MODULE_ENUM _get_dst_module_by_lcm(disp_lcm_handle *plcm)
  1168. {
  1169. if (plcm == NULL) {
  1170. DISPERR("plcm is null\n");
  1171. return DISP_MODULE_UNKNOWN;
  1172. }
  1173. if (plcm->params->type == LCM_TYPE_DSI) {
  1174. if (plcm->lcm_if_id == LCM_INTERFACE_DSI0)
  1175. return DISP_MODULE_DSI0;
  1176. else if (plcm->lcm_if_id == LCM_INTERFACE_DSI1)
  1177. return DISP_MODULE_DSI1;
  1178. else if (plcm->lcm_if_id == LCM_INTERFACE_DSI_DUAL)
  1179. return DISP_MODULE_DSIDUAL;
  1180. else
  1181. return DISP_MODULE_DSI0;
  1182. } else if (plcm->params->type == LCM_TYPE_DPI) {
  1183. return DISP_MODULE_DPI;
  1184. }
  1185. DISPERR("can't find primary path dst module\n");
  1186. return DISP_MODULE_UNKNOWN;
  1187. }
  1188. #define AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
  1189. /**
  1190. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1191. * 1.wait idle: N N Y Y
  1192. * 2.lcm update: N Y N Y
  1193. * 3.path start: idle->Y Y idle->Y Y
  1194. * 4.path trigger: idle->Y Y idle->Y Y
  1195. * 5.mutex enable: N N idle->Y Y
  1196. * 6.set cmdq dirty: N Y N N
  1197. * 7.flush cmdq: Y Y N N
  1198. */
  1199. /**
  1200. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1201. * 1.wait idle: N N Y Y
  1202. */
  1203. int _should_wait_path_idle(void)
  1204. {
  1205. if (primary_display_cmdq_enabled()) {
  1206. if (primary_display_is_video_mode())
  1207. return 0;
  1208. else
  1209. return 0;
  1210. } else {
  1211. if (primary_display_is_video_mode())
  1212. return dpmgr_path_is_busy(pgc->dpmgr_handle);
  1213. else
  1214. return dpmgr_path_is_busy(pgc->dpmgr_handle);
  1215. }
  1216. }
  1217. /**
  1218. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1219. * 2.lcm update: N Y N Y
  1220. */
  1221. int _should_update_lcm(void)
  1222. {
  1223. if (primary_display_cmdq_enabled()) {
  1224. if (primary_display_is_video_mode())
  1225. return 0;
  1226. /* TODO: lcm_update can't use cmdq now */
  1227. return 0;
  1228. }
  1229. if (primary_display_is_video_mode())
  1230. return 0;
  1231. else
  1232. return 1;
  1233. }
  1234. /**
  1235. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1236. * 3.path start: idle->Y Y idle->Y Y
  1237. */
  1238. int _should_start_path(void)
  1239. {
  1240. if (primary_display_cmdq_enabled()) {
  1241. if (primary_display_is_video_mode())
  1242. return 0;
  1243. /* return dpmgr_path_is_idle(pgc->dpmgr_handle); */
  1244. else
  1245. return 0;
  1246. } else {
  1247. if (primary_display_is_video_mode())
  1248. return dpmgr_path_is_idle(pgc->dpmgr_handle);
  1249. else
  1250. return 1;
  1251. }
  1252. }
  1253. /**
  1254. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1255. * 4.path trigger: idle->Y Y idle->Y Y
  1256. * 5.mutex enable: N N idle->Y Y
  1257. */
  1258. int _should_trigger_path(void)
  1259. {
  1260. /* this is not a perfect design, we can't decide path trigger(ovl/rdma/dsi..) separately with mutex enable */
  1261. /* but it's lucky because path trigger and mutex enable is the same w/o cmdq, and it's correct w/ CMDQ(Y+N). */
  1262. if (primary_display_cmdq_enabled()) {
  1263. if (primary_display_is_video_mode())
  1264. return 0;
  1265. /* return dpmgr_path_is_idle(pgc->dpmgr_handle); */
  1266. else
  1267. return 0;
  1268. } else {
  1269. if (primary_display_is_video_mode())
  1270. return dpmgr_path_is_idle(pgc->dpmgr_handle);
  1271. else
  1272. return 1;
  1273. }
  1274. }
  1275. /**
  1276. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1277. * 6.set cmdq dirty: N Y N N
  1278. */
  1279. int _should_set_cmdq_dirty(void)
  1280. {
  1281. if (primary_display_cmdq_enabled()) {
  1282. if (primary_display_is_video_mode())
  1283. return 0;
  1284. else
  1285. return 1;
  1286. } else {
  1287. if (primary_display_is_video_mode())
  1288. return 0;
  1289. else
  1290. return 0;
  1291. }
  1292. }
  1293. /**
  1294. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1295. * 7.flush cmdq: Y Y N N
  1296. */
  1297. int _should_flush_cmdq_config_handle(void)
  1298. {
  1299. if (primary_display_cmdq_enabled()) {
  1300. if (primary_display_is_video_mode())
  1301. return 1;
  1302. else
  1303. return 1;
  1304. } else {
  1305. if (primary_display_is_video_mode())
  1306. return 0;
  1307. else
  1308. return 0;
  1309. }
  1310. }
  1311. int _should_reset_cmdq_config_handle(void)
  1312. {
  1313. if (primary_display_cmdq_enabled()) {
  1314. if (primary_display_is_video_mode())
  1315. return 1;
  1316. else
  1317. return 1;
  1318. } else {
  1319. if (primary_display_is_video_mode())
  1320. return 0;
  1321. else
  1322. return 0;
  1323. }
  1324. }
  1325. /**
  1326. * trigger operation: VDO+CMDQ CMD+CMDQ VDO+CPU CMD+CPU
  1327. * 7.flush cmdq: Y Y N N
  1328. */
  1329. int _should_insert_wait_frame_done_token(void)
  1330. {
  1331. if (primary_display_cmdq_enabled()) {
  1332. if (primary_display_is_video_mode())
  1333. return 1;
  1334. else
  1335. return 1;
  1336. } else {
  1337. if (primary_display_is_video_mode())
  1338. return 0;
  1339. else
  1340. return 0;
  1341. }
  1342. }
  1343. int _should_trigger_interface(void)
  1344. {
  1345. if (pgc->mode == DECOUPLE_MODE)
  1346. return 0;
  1347. else
  1348. return 1;
  1349. }
  1350. int _should_config_ovl_input(void)
  1351. {
  1352. /* should extend this when display path dynamic switch is ready */
  1353. if (pgc->mode == SINGLE_LAYER_MODE || pgc->mode == DEBUG_RDMA1_DSI0_MODE)
  1354. return 0;
  1355. else
  1356. return 1;
  1357. }
  1358. int _should_config_ovl_to_memory(display_primary_path_context *ctx)
  1359. {
  1360. if (ctx == NULL) {
  1361. DISP_FATAL_ERR("DISP", "Context is NULL!\n");
  1362. return 0;
  1363. }
  1364. if (ctx->mode == DECOUPLE_MODE)
  1365. return 1;
  1366. else
  1367. return 0;
  1368. }
  1369. #define OOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOO
  1370. static unsigned long get_current_time_us(void)
  1371. {
  1372. struct timeval t;
  1373. do_gettimeofday(&t);
  1374. return (t.tv_sec & 0xFFF) * 1000000 + t.tv_usec;
  1375. }
  1376. static struct hrtimer cmd_mode_update_timer;
  1377. static int is_fake_timer_inited;
  1378. static enum hrtimer_restart _DISP_CmdModeTimer_handler(struct hrtimer *timer)
  1379. {
  1380. dpmgr_signal_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  1381. #if 0
  1382. if ((get_current_time_us() - pgc->last_vsync_tick) > 16666) {
  1383. dpmgr_signal_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  1384. pgc->last_vsync_tick = get_current_time_us();
  1385. }
  1386. #endif
  1387. hrtimer_forward_now(timer, ns_to_ktime(16666666));
  1388. return HRTIMER_RESTART;
  1389. }
  1390. int _init_vsync_fake_monitor(int fps)
  1391. {
  1392. if (is_fake_timer_inited)
  1393. return 0;
  1394. is_fake_timer_inited = 1;
  1395. if (fps == 0)
  1396. fps = 6000;
  1397. hrtimer_init(&cmd_mode_update_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
  1398. cmd_mode_update_timer.function = _DISP_CmdModeTimer_handler;
  1399. hrtimer_start(&cmd_mode_update_timer, ns_to_ktime(16666666), HRTIMER_MODE_REL);
  1400. DISPMSG("fake timer, init\n");
  1401. return 0;
  1402. }
  1403. #define DISP_REG_SODI_PA 0x10006b0c
  1404. /* extern unsigned int gEnableSODIControl; */
  1405. /* extern unsigned int gPrefetchControl; */
  1406. void disp_set_sodi(unsigned int enable, void *cmdq_handle)
  1407. {
  1408. #ifndef CONFIG_FPGA_EARLY_PORTING
  1409. #ifdef FORCE_SODI_BY_SW
  1410. if (gEnableSODIControl == 1) {
  1411. if (cmdq_handle != NULL) {
  1412. if (enable == 1)
  1413. cmdqRecWrite(cmdq_handle, DISP_REG_SODI_PA, 0, 1);
  1414. else
  1415. cmdqRecWrite(cmdq_handle, DISP_REG_SODI_PA, 1, 1);
  1416. } else {
  1417. if (enable == 1)
  1418. DISP_REG_SET(0, SPM_PCM_SRC_REQ,
  1419. DISP_REG_GET(SPM_PCM_SRC_REQ) & (~0x1));
  1420. else
  1421. DISP_REG_SET(0, SPM_PCM_SRC_REQ,
  1422. DISP_REG_GET(SPM_PCM_SRC_REQ) | 0x1);
  1423. }
  1424. }
  1425. #endif
  1426. #endif
  1427. }
  1428. /* extern unsigned int gEnableSWTrigger; */
  1429. /* extern unsigned int gEnableMutexRisingEdge; */
  1430. /* extern unsigned int gDisableSODIForTriggerLoop; */
  1431. static void _cmdq_build_trigger_loop(void)
  1432. {
  1433. #ifndef MTK_FB_CMDQ_DISABLE
  1434. int ret = 0;
  1435. int i = 0;
  1436. if (pgc->cmdq_handle_trigger == NULL) {
  1437. cmdqRecCreate(CMDQ_SCENARIO_TRIGGER_LOOP, &(pgc->cmdq_handle_trigger));
  1438. DISPMSG("primary path trigger thread cmd handle=%p\n", pgc->cmdq_handle_trigger);
  1439. }
  1440. cmdqRecReset(pgc->cmdq_handle_trigger);
  1441. if (primary_display_is_video_mode()) {
  1442. /* wait and clear stream_done, HW will assert mutex enable automatically in frame done reset. */
  1443. /* todo: should let dpmanager to decide wait which mutex's eof. */
  1444. ret = cmdqRecWait(pgc->cmdq_handle_trigger, CMDQ_EVENT_DISP_RDMA0_EOF);
  1445. ret = cmdqRecWait(pgc->cmdq_handle_trigger,
  1446. dpmgr_path_get_mutex(pgc->dpmgr_handle) +
  1447. CMDQ_EVENT_MUTEX0_STREAM_EOF);
  1448. /* wait and clear rdma0_sof for vfp change */
  1449. cmdqRecClearEventToken(pgc->cmdq_handle_trigger, CMDQ_EVENT_DISP_RDMA0_SOF);
  1450. if (gEnableSWTrigger == 1)
  1451. DISP_REG_SET(pgc->cmdq_handle_trigger, DISP_REG_CONFIG_MUTEX_EN(DISP_OVL_SEPARATE_MUTEX_ID), 1);
  1452. /* for some module(like COLOR) to read hw register to GPR after frame done */
  1453. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_trigger,
  1454. CMDQ_AFTER_STREAM_EOF);
  1455. } else {
  1456. ret = cmdqRecWaitNoClear(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_CABC_EOF);
  1457. /* DSI command mode doesn't have mutex_stream_eof, need use CMDQ token instead */
  1458. ret = cmdqRecWait(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
  1459. if (_need_wait_esd_eof())
  1460. /* Wait esd config thread done. */
  1461. ret = cmdqRecWaitNoClear(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_ESD_EOF);
  1462. /* ret = cmdqRecWait(pgc->cmdq_handle_trigger, CMDQ_EVENT_MDP_DSI0_TE_SOF); */
  1463. /* for operations before frame transfer, such as waiting for DSI TE */
  1464. if (islcmconnected)
  1465. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_trigger, CMDQ_BEFORE_STREAM_SOF);
  1466. /* pull DSI clk lane back to HS */
  1467. DSI_manual_enter_HS(pgc->cmdq_handle_trigger);
  1468. /* cleat frame done token, now the config thread will not allowed to config registers. */
  1469. /* remember that config thread's priority is higher than trigger thread,
  1470. * so all the config queued before will be applied then STREAM_EOF token be cleared
  1471. */
  1472. /* this is what CMDQ did as "Merge" */
  1473. ret = cmdqRecClearEventToken(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_STREAM_EOF);
  1474. ret = cmdqRecClearEventToken(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
  1475. /* enable mutex, only cmd mode need this */
  1476. /* this is what CMDQ did as "Trigger" */
  1477. /* clear rdma EOF token before wait */
  1478. ret = cmdqRecClearEventToken(pgc->cmdq_handle_trigger, CMDQ_EVENT_DISP_RDMA0_EOF);
  1479. #ifdef FORCE_SODI_BY_SW
  1480. /* TODO: use separate api for sodi control flow later */
  1481. /* Enable EMI Force On(Equal to SODI Disable) */
  1482. cmdqRecWrite(pgc->cmdq_handle_trigger, 0x10006b0c, 1, 1);
  1483. #endif
  1484. #ifndef FORCE_SODI_CG_MODE
  1485. /* Enable SPM CG Mode(Force 30+ times to ensure write success, need find root cause and fix later) */
  1486. cmdqRecWrite(pgc->cmdq_handle_trigger, 0x10006b04, 0x80, 0x80);
  1487. /* Polling EMI Status to ensure EMI is enabled */
  1488. cmdqRecPoll(pgc->cmdq_handle_trigger, 0x100063b4, 0, 0x00200000);
  1489. #endif
  1490. #ifdef FORCE_SODI_BY_SW
  1491. /* Clear EMI Force on, Let SPM control EMI state now */
  1492. cmdqRecWrite(pgc->cmdq_handle_trigger, 0x10006b0c, 0, 1);
  1493. #endif
  1494. dpmgr_path_trigger(pgc->dpmgr_handle, pgc->cmdq_handle_trigger, CMDQ_ENABLE);
  1495. /* ret = cmdqRecWrite(pgc->cmdq_handle_trigger,
  1496. (unsigned int)(DISP_REG_CONFIG_MUTEX_EN(0))&0x1fffffff, 1, ~0);
  1497. */
  1498. /* backup DSI state register to slot */
  1499. if (gEnableDSIStateCheck == 1) {
  1500. for (i = 0; i < 10; i++) {
  1501. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->dsi_state_info, i,
  1502. DISP_REG_DSI_STATE);
  1503. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->rdma_state_info, i * 5 + 0,
  1504. DISP_REG_RDMA_INT_STATUS);
  1505. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->rdma_state_info, i * 5 + 1,
  1506. DISP_REG_RDMA_IN_P_CNT);
  1507. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->rdma_state_info, i * 5 + 2,
  1508. DISP_REG_RDMA_IN_LINE_CNT);
  1509. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->rdma_state_info, i * 5 + 3,
  1510. DISP_REG_RDMA_OUT_P_CNT);
  1511. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_trigger, pgc->rdma_state_info, i * 5 + 4,
  1512. DISP_REG_RDMA_OUT_LINE_CNT);
  1513. }
  1514. }
  1515. /* SODI is disabled in config thread, so mutex enable/dsi start/CPU wait
  1516. * TE will not be blocked by SODI
  1517. */
  1518. /* should enable SODI here, */
  1519. if (gDisableSODIForTriggerLoop == 1)
  1520. disp_set_sodi(1, pgc->cmdq_handle_trigger);
  1521. /* waiting for frame done, because we can't use mutex stream eof here,
  1522. * so need to let dpmanager help to decide which event to wait
  1523. */
  1524. /* most time we wait rdmax frame done event. */
  1525. ret = cmdqRecWait(pgc->cmdq_handle_trigger, CMDQ_EVENT_DISP_RDMA0_EOF);
  1526. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_trigger, CMDQ_WAIT_STREAM_EOF_EVENT);
  1527. /* dsi is not idle rightly after rdma frame done,
  1528. * so we need to polling about 1us for dsi returns to idle
  1529. */
  1530. /* do not polling dsi idle directly which will decrease CMDQ performance */
  1531. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_trigger, CMDQ_CHECK_IDLE_AFTER_STREAM_EOF);
  1532. /* for some module(like COLOR) to read hw register to GPR after frame done */
  1533. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_trigger, CMDQ_AFTER_STREAM_EOF);
  1534. #ifndef FORCE_SODI_CG_MODE
  1535. /* Enable EMI Power Down Mode */
  1536. cmdqRecWrite(pgc->cmdq_handle_trigger, 0x10006b04, 0, 0x80);
  1537. #endif
  1538. /* pull DSI clk lane to LP */
  1539. DSI_sw_clk_trail_cmdq(0, pgc->cmdq_handle_trigger);
  1540. /* polling DSI idle */
  1541. /* ret = cmdqRecPoll(pgc->cmdq_handle_trigger, 0x1401b00c, 0, 0x80000000); */
  1542. /* polling wdma frame done */
  1543. /* ret = cmdqRecPoll(pgc->cmdq_handle_trigger, 0x140060A0, 1, 0x1); */
  1544. /* now frame done, config thread is allowed to config register now */
  1545. ret = cmdqRecSetEventToken(pgc->cmdq_handle_trigger, CMDQ_SYNC_TOKEN_STREAM_EOF);
  1546. /* RUN forever!!!! */
  1547. BUG_ON(ret < 0);
  1548. }
  1549. /* dump trigger loop instructions to check whether dpmgr_path_build_cmdq works correctly */
  1550. DISPPRINT("primary display BUILD cmdq trigger loop finished\n");
  1551. return;
  1552. #endif
  1553. }
  1554. void disp_spm_enter_cg_mode(void)
  1555. {
  1556. MMProfileLogEx(ddp_mmp_get_events()->cg_mode, MMProfileFlagPulse, 0, 0);
  1557. }
  1558. void disp_spm_enter_power_down_mode(void)
  1559. {
  1560. MMProfileLogEx(ddp_mmp_get_events()->power_down_mode, MMProfileFlagPulse, 0, 0);
  1561. }
  1562. #if 0
  1563. /**
  1564. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  1565. */
  1566. static void _cmdq_build_monitor_loop(void)
  1567. {
  1568. int ret = 0;
  1569. cmdqRecHandle g_cmdq_handle_monitor;
  1570. cmdqRecCreate(CMDQ_SCENARIO_DISP_SCREEN_CAPTURE, &(g_cmdq_handle_monitor));
  1571. DISPMSG("primary path monitor thread cmd handle=%p\n", g_cmdq_handle_monitor);
  1572. cmdqRecReset(g_cmdq_handle_monitor);
  1573. /* wait and clear stream_done, HW will assert mutex enable automatically in frame done reset. */
  1574. /* todo: should let dpmanager to decide wait which mutex's eof. */
  1575. ret = cmdqRecWait(g_cmdq_handle_monitor, CMDQ_EVENT_DISP_RDMA0_UNDERRUN);
  1576. cmdqRecReadToDataRegister(g_cmdq_handle_monitor, 0x10006b0c, CMDQ_DATA_REG_2D_SHARPNESS_1_DST);
  1577. cmdqRecWriteFromDataRegister(g_cmdq_handle_monitor, CMDQ_DATA_REG_2D_SHARPNESS_1_DST, 0x1401b280);
  1578. cmdqRecReadToDataRegister(g_cmdq_handle_monitor, 0x10006b08, CMDQ_DATA_REG_2D_SHARPNESS_1_DST);
  1579. cmdqRecWriteFromDataRegister(g_cmdq_handle_monitor, CMDQ_DATA_REG_2D_SHARPNESS_1_DST, 0x1401b284);
  1580. cmdqRecReadToDataRegister(g_cmdq_handle_monitor, 0x10006b04, CMDQ_DATA_REG_2D_SHARPNESS_1_DST);
  1581. cmdqRecWriteFromDataRegister(g_cmdq_handle_monitor, CMDQ_DATA_REG_2D_SHARPNESS_1_DST, 0x1401b288);
  1582. cmdqRecReadToDataRegister(g_cmdq_handle_monitor, 0x1401b16c, CMDQ_DATA_REG_2D_SHARPNESS_1_DST);
  1583. cmdqRecWriteFromDataRegister(g_cmdq_handle_monitor, CMDQ_DATA_REG_2D_SHARPNESS_1_DST, 0x1401b28C);
  1584. ret = cmdqRecStartLoop(g_cmdq_handle_monitor);
  1585. }
  1586. #endif
  1587. static void _cmdq_start_trigger_loop(void)
  1588. {
  1589. #ifndef MTK_FB_CMDQ_DISABLE
  1590. int ret = 0;
  1591. cmdqRecDumpCommand(pgc->cmdq_handle_trigger);
  1592. /* this should be called only once because trigger loop will nevet stop */
  1593. ret = cmdqRecStartLoop(pgc->cmdq_handle_trigger);
  1594. if (!primary_display_is_video_mode()) {
  1595. /* Need set esd check eof synctoken to let trigger loop go. */
  1596. if (_need_wait_esd_eof())
  1597. cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_ESD_EOF);
  1598. /* need to set STREAM_EOF for the first time, otherwise we will stuck in dead loop */
  1599. cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_STREAM_EOF);
  1600. cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_CABC_EOF);
  1601. dprec_event_op(DPREC_EVENT_CMDQ_SET_EVENT_ALLOW);
  1602. } else {
  1603. #if 0
  1604. if (dpmgr_path_is_idle(pgc->dpmgr_handle))
  1605. cmdqCoreSetEvent(CMDQ_EVENT_MUTEX0_STREAM_EOF);
  1606. #endif
  1607. }
  1608. if (_is_decouple_mode(pgc->session_mode))
  1609. cmdqCoreSetEvent(CMDQ_EVENT_DISP_WDMA0_EOF);
  1610. DISPPRINT("primary display START cmdq trigger loop finished\n");
  1611. #endif
  1612. }
  1613. static void _cmdq_stop_trigger_loop(void)
  1614. {
  1615. #ifndef MTK_FB_CMDQ_DISABLE
  1616. int ret = 0;
  1617. /* this should be called only once because trigger loop will nevet stop */
  1618. ret = cmdqRecStopLoop(pgc->cmdq_handle_trigger);
  1619. DISPCHECK("primary display STOP cmdq trigger loop finished\n");
  1620. #endif
  1621. }
  1622. static void _cmdq_set_config_handle_dirty(void)
  1623. {
  1624. #ifndef MTK_FB_CMDQ_DISABLE
  1625. if (!primary_display_is_video_mode()) {
  1626. dprec_logger_start(DPREC_LOGGER_PRIMARY_CMDQ_SET_DIRTY, 0, 0);
  1627. /* only command mode need to set dirty */
  1628. cmdqRecSetEventToken(pgc->cmdq_handle_config, CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
  1629. dprec_event_op(DPREC_EVENT_CMDQ_SET_DIRTY);
  1630. dprec_logger_done(DPREC_LOGGER_PRIMARY_CMDQ_SET_DIRTY, 0, 0);
  1631. }
  1632. #endif
  1633. }
  1634. static void _cmdq_set_config_handle_dirty_mira(void *handle)
  1635. {
  1636. #ifndef MTK_FB_CMDQ_DISABLE
  1637. if (!primary_display_is_video_mode()) {
  1638. dprec_logger_start(DPREC_LOGGER_PRIMARY_CMDQ_SET_DIRTY, 0, 0);
  1639. /* only command mode need to set dirty */
  1640. cmdqRecSetEventToken(handle, CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
  1641. dprec_event_op(DPREC_EVENT_CMDQ_SET_DIRTY);
  1642. dprec_logger_done(DPREC_LOGGER_PRIMARY_CMDQ_SET_DIRTY, 0, 0);
  1643. }
  1644. #endif
  1645. }
  1646. static void _cmdq_reset_config_handle(void)
  1647. {
  1648. #ifndef MTK_FB_CMDQ_DISABLE
  1649. cmdqRecReset(pgc->cmdq_handle_config);
  1650. dprec_event_op(DPREC_EVENT_CMDQ_RESET);
  1651. #endif
  1652. }
  1653. static void _cmdq_flush_config_handle(int blocking, void *callback, unsigned int userdata)
  1654. {
  1655. #ifndef MTK_FB_CMDQ_DISABLE
  1656. dprec_logger_start(DPREC_LOGGER_PRIMARY_CMDQ_FLUSH, blocking,
  1657. (unsigned int)(unsigned long)callback);
  1658. if (blocking) {
  1659. /* DISPERR("Should not use blocking cmdq flush,
  1660. * may block primary display path for 1 frame period\n");
  1661. */
  1662. cmdqRecFlush(pgc->cmdq_handle_config);
  1663. } else {
  1664. if (callback)
  1665. cmdqRecFlushAsyncCallback(pgc->cmdq_handle_config, callback, userdata);
  1666. else
  1667. cmdqRecFlushAsync(pgc->cmdq_handle_config);
  1668. }
  1669. dprec_event_op(DPREC_EVENT_CMDQ_FLUSH);
  1670. dprec_logger_done(DPREC_LOGGER_PRIMARY_CMDQ_FLUSH, userdata, 0);
  1671. #endif
  1672. }
  1673. static void _cmdq_flush_config_handle_mira(void *handle, int blocking)
  1674. {
  1675. #ifndef MTK_FB_CMDQ_DISABLE
  1676. dprec_logger_start(DPREC_LOGGER_PRIMARY_CMDQ_FLUSH, 0, 0);
  1677. if (blocking)
  1678. cmdqRecFlush(handle);
  1679. else
  1680. cmdqRecFlushAsync(handle);
  1681. dprec_event_op(DPREC_EVENT_CMDQ_FLUSH);
  1682. dprec_logger_done(DPREC_LOGGER_PRIMARY_CMDQ_FLUSH, 0, 0);
  1683. #endif
  1684. }
  1685. static void _cmdq_insert_wait_frame_done_token(void)
  1686. {
  1687. #ifndef MTK_FB_CMDQ_DISABLE
  1688. if (primary_display_is_video_mode())
  1689. cmdqRecWaitNoClear(pgc->cmdq_handle_config, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  1690. else
  1691. cmdqRecWaitNoClear(pgc->cmdq_handle_config, CMDQ_SYNC_TOKEN_STREAM_EOF);
  1692. dprec_event_op(DPREC_EVENT_CMDQ_WAIT_STREAM_EOF);
  1693. #endif
  1694. }
  1695. void _cmdq_insert_wait_frame_done_token_mira(void *handle)
  1696. {
  1697. #ifndef MTK_FB_CMDQ_DISABLE
  1698. if (primary_display_is_video_mode())
  1699. cmdqRecWaitNoClear(handle, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  1700. else
  1701. cmdqRecWaitNoClear(handle, CMDQ_SYNC_TOKEN_STREAM_EOF);
  1702. dprec_event_op(DPREC_EVENT_CMDQ_WAIT_STREAM_EOF);
  1703. #endif
  1704. }
  1705. static int primary_display_is_secure_path(DISP_SESSION_TYPE session_type)
  1706. {
  1707. int i;
  1708. disp_session_input_config *session_input;
  1709. session_input = &cached_session_input[session_type - 1];
  1710. for (i = 0; i < session_input->config_layer_num; i++) {
  1711. if (session_input->config[i].layer_enable &&
  1712. (session_input->config[i].security == DISP_SECURE_BUFFER))
  1713. return 1;
  1714. }
  1715. return 0;
  1716. }
  1717. static void directlink_path_add_memory(WDMA_CONFIG_STRUCT *p_wdma)
  1718. {
  1719. int ret = 0;
  1720. int secure_path_on = primary_display_is_secure_path(DISP_SESSION_PRIMARY);
  1721. cmdqRecHandle cmdq_handle = NULL;
  1722. cmdqRecHandle cmdq_wait_handle = NULL;
  1723. disp_ddp_path_config *pconfig = NULL;
  1724. /*create config thread*/
  1725. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  1726. if (ret != 0) {
  1727. DISPCHECK("dl_to_dc capture:Fail to create cmdq handle\n");
  1728. ret = -1;
  1729. goto out;
  1730. }
  1731. cmdqRecReset(cmdq_handle);
  1732. /*create wait thread*/
  1733. ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_SCREEN_CAPTURE, &cmdq_wait_handle);
  1734. if (ret != 0) {
  1735. DISPCHECK("dl_to_dc capture:Fail to create cmdq wait handle\n");
  1736. ret = -1;
  1737. goto out;
  1738. }
  1739. cmdqRecReset(cmdq_wait_handle);
  1740. /*configure config thread*/
  1741. if (!secure_path_on)
  1742. cmdqRecEnablePrefetch(cmdq_handle);
  1743. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  1744. dpmgr_path_add_memout(pgc->dpmgr_handle, ENGINE_OVL0, cmdq_handle);
  1745. pconfig = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  1746. pconfig->wdma_config = *p_wdma;
  1747. pconfig->wdma_dirty = 1;
  1748. ret = dpmgr_path_config(pgc->dpmgr_handle, pconfig, cmdq_handle);
  1749. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  1750. dpmgr_wdma_path_force_power_on();
  1751. if (!secure_path_on)
  1752. cmdqRecDisablePrefetch(cmdq_handle);
  1753. _cmdq_flush_config_handle_mira(cmdq_handle, 0);
  1754. DISPMSG("dl_to_dc capture:Flush add memout mva(0x%lx)\n", p_wdma->dstAddress);
  1755. /*wait wdma0 sof*/
  1756. cmdqRecWait(cmdq_wait_handle, CMDQ_EVENT_DISP_WDMA0_SOF);
  1757. cmdqRecFlush(cmdq_wait_handle);
  1758. DISPMSG("dl_to_dc capture:Flush wait wdma sof\n");
  1759. #if 0
  1760. cmdqRecReset(cmdq_handle);
  1761. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  1762. dpmgr_path_remove_memout(pgc->dpmgr_handle, cmdq_handle);
  1763. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  1764. /* flush remove memory to cmdq */
  1765. _cmdq_flush_config_handle_mira(cmdq_handle, 0);
  1766. DISPMSG("dl_to_dc capture: Flush remove memout\n");
  1767. dpmgr_path_memout_clock(pgc->dpmgr_handle, 0);
  1768. #endif
  1769. out:
  1770. cmdqRecDestroy(cmdq_handle);
  1771. cmdqRecDestroy(cmdq_wait_handle);
  1772. }
  1773. static unsigned int _get_switch_dc_buffer(void)
  1774. {
  1775. #ifdef CONFIG_MTK_GMO_RAM_OPTIMIZE
  1776. return pgc->dc_buf[0];
  1777. #else
  1778. if (primary_display_is_secure_path(DISP_SESSION_PRIMARY))
  1779. return pgc->session_buf[pgc->session_buf_id];
  1780. else
  1781. return pgc->dc_buf[pgc->dc_buf_id];
  1782. #endif
  1783. }
  1784. static int _DL_switch_to_DC_fast(void)
  1785. {
  1786. int ret = 0;
  1787. int secure_path_on = primary_display_is_secure_path(DISP_SESSION_PRIMARY);
  1788. RDMA_CONFIG_STRUCT rdma_config = decouple_rdma_config;
  1789. WDMA_CONFIG_STRUCT wdma_config = decouple_wdma_config;
  1790. disp_ddp_path_config *data_config_dl = NULL;
  1791. disp_ddp_path_config *data_config_dc = NULL;
  1792. unsigned int mva;
  1793. mva = _get_switch_dc_buffer();
  1794. wdma_config.dstAddress = mva;
  1795. if (secure_path_on)
  1796. wdma_config.security = DISP_SECURE_BUFFER;
  1797. else
  1798. wdma_config.security = DISP_NORMAL_BUFFER;
  1799. /* disable SODI by CPU to prevent underflow */
  1800. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  1801. spm_enable_sodi(0);
  1802. #endif
  1803. /* 1.save a temp frame to intermediate buffer */
  1804. directlink_path_add_memory(&wdma_config);
  1805. /* 2.reset primary handle */
  1806. _cmdq_reset_config_handle();
  1807. if (!secure_path_on)
  1808. cmdqRecEnablePrefetch(pgc->cmdq_handle_config);
  1809. _cmdq_insert_wait_frame_done_token();
  1810. /* 3.modify interface path handle to new scenario(rdma->dsi) */
  1811. #ifdef CONFIG_FOR_SOURCE_PQ
  1812. dpmgr_modify_path(pgc->dpmgr_handle, DDP_SCENARIO_PRIMARY_RDMA0_DISP, pgc->cmdq_handle_config,
  1813. primary_display_is_video_mode() ? DDP_VIDEO_MODE : DDP_CMD_MODE);
  1814. #else
  1815. dpmgr_modify_path(pgc->dpmgr_handle, DDP_SCENARIO_PRIMARY_RDMA0_COLOR0_DISP, pgc->cmdq_handle_config,
  1816. primary_display_is_video_mode() ? DDP_VIDEO_MODE : DDP_CMD_MODE);
  1817. #endif
  1818. /* 4.config rdma from directlink mode to memory mode */
  1819. rdma_config.address = mva;
  1820. rdma_config.security = DISP_NORMAL_BUFFER;
  1821. rdma_config.pitch = primary_display_get_width() * DP_COLOR_BITS_PER_PIXEL(rdma_config.inputFormat) >> 3;
  1822. if (secure_path_on) {
  1823. rdma_config.address = pgc->session_buf[pgc->session_buf_id];
  1824. rdma_config.security = DISP_SECURE_BUFFER;
  1825. }
  1826. data_config_dl = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  1827. data_config_dl->rdma_config = rdma_config;
  1828. data_config_dl->rdma_dirty = 1;
  1829. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config_dl, pgc->cmdq_handle_config);
  1830. /* 5. backup rdma address to slots */
  1831. cmdqRecBackupUpdateSlot(pgc->cmdq_handle_config, pgc->rdma_buff_info, 0, mva);
  1832. /* 6 .flush to cmdq */
  1833. _cmdq_set_config_handle_dirty();
  1834. if (!secure_path_on)
  1835. cmdqRecDisablePrefetch(pgc->cmdq_handle_config);
  1836. _cmdq_flush_config_handle(1, NULL, 0);
  1837. /* ddp_mmp_rdma_layer(&rdma_config, 0, 20, 20); */
  1838. /* 7.reset cmdq */
  1839. _cmdq_reset_config_handle();
  1840. _cmdq_insert_wait_frame_done_token();
  1841. /* 9. create ovl2mem path handle */
  1842. cmdqRecReset(pgc->cmdq_handle_ovl1to2_config);
  1843. #ifdef CONFIG_FOR_SOURCE_PQ
  1844. pgc->ovl2mem_path_handle = dpmgr_create_path(DDP_SCENARIO_PRIMARY_DITHER_MEMOUT,
  1845. pgc->cmdq_handle_ovl1to2_config);
  1846. #else
  1847. pgc->ovl2mem_path_handle = dpmgr_create_path(DDP_SCENARIO_PRIMARY_OVL_MEMOUT,
  1848. pgc->cmdq_handle_ovl1to2_config);
  1849. #endif
  1850. if (pgc->ovl2mem_path_handle) {
  1851. DISPCHECK("dpmgr create ovl memout path SUCCESS(%p)\n", pgc->ovl2mem_path_handle);
  1852. } else {
  1853. DISPCHECK("dpmgr create path FAIL\n");
  1854. return -1;
  1855. }
  1856. dpmgr_path_set_video_mode(pgc->ovl2mem_path_handle, 0);
  1857. #ifndef CONFIG_MTK_CLKMGR
  1858. ddp_clk_prepare(DISP_MTCMOS_CLK);
  1859. #endif
  1860. dpmgr_path_init(pgc->ovl2mem_path_handle, CMDQ_ENABLE);
  1861. data_config_dc = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  1862. data_config_dc->dst_w = rdma_config.width;
  1863. data_config_dc->dst_h = rdma_config.height;
  1864. data_config_dc->dst_dirty = 1;
  1865. /* move ovl config info from dl to dc */
  1866. memcpy(data_config_dc->ovl_config, data_config_dl->ovl_config, sizeof(data_config_dl->ovl_config));
  1867. ret = dpmgr_path_config(pgc->ovl2mem_path_handle, data_config_dc, pgc->cmdq_handle_ovl1to2_config);
  1868. ret = dpmgr_path_start(pgc->ovl2mem_path_handle, CMDQ_ENABLE);
  1869. /* use blocking flush to make sure all config is done. */
  1870. /* cmdqRecDumpCommand(pgc->cmdq_handle_ovl1to2_config); */
  1871. cmdqRecClearEventToken(pgc->cmdq_handle_ovl1to2_config, CMDQ_EVENT_DISP_WDMA0_EOF);
  1872. _cmdq_flush_config_handle_mira(pgc->cmdq_handle_ovl1to2_config, 1);
  1873. cmdqRecReset(pgc->cmdq_handle_ovl1to2_config);
  1874. /* 11..enable event for new path */
  1875. /* dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE); */
  1876. /* dpmgr_map_event_to_irq(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START, DDP_IRQ_WDMA0_FRAME_COMPLETE); */
  1877. /* dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START); */
  1878. if (primary_display_is_video_mode())
  1879. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC, DDP_IRQ_RDMA0_DONE);
  1880. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  1881. /* dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE); */
  1882. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  1883. /* enable SODI after switch */
  1884. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  1885. spm_enable_sodi(1);
  1886. #endif
  1887. return ret;
  1888. }
  1889. static int _DC_switch_to_DL_fast(void)
  1890. {
  1891. int ret = 0;
  1892. int layer = 0;
  1893. int secure_path_on = primary_display_is_secure_path(DISP_SESSION_PRIMARY);
  1894. disp_ddp_path_config *data_config_dl = NULL;
  1895. disp_ddp_path_config *data_config_dc = NULL;
  1896. OVL_CONFIG_STRUCT ovl_config[OVL_LAYER_NUM];
  1897. if(!pgc->ovl2mem_path_handle)
  1898. return -1;
  1899. /* 1. disable SODI */
  1900. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  1901. spm_enable_sodi(0);
  1902. #endif
  1903. /* 2.enable ovl/wdma clock */
  1904. /* 3.destroy ovl->mem path. */
  1905. #if defined(OVL_TIME_SHARING)
  1906. data_config_dc = &last_primary_config;
  1907. #else
  1908. data_config_dc = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  1909. #endif
  1910. /*save ovl info */;
  1911. memcpy(ovl_config, data_config_dc->ovl_config, sizeof(ovl_config));
  1912. dpmgr_path_deinit(pgc->ovl2mem_path_handle,
  1913. (int)(unsigned long)pgc->cmdq_handle_ovl1to2_config);
  1914. #ifndef CONFIG_MTK_CLKMGR
  1915. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  1916. #endif
  1917. dpmgr_destroy_path(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config);
  1918. pgc->ovl2mem_path_handle = NULL;
  1919. /*clear sof token for next dl to dc */
  1920. cmdqRecClearEventToken(pgc->cmdq_handle_ovl1to2_config, CMDQ_EVENT_DISP_WDMA0_SOF);
  1921. _cmdq_flush_config_handle_mira(pgc->cmdq_handle_ovl1to2_config, 1);
  1922. cmdqRecReset(pgc->cmdq_handle_ovl1to2_config);
  1923. /* release output buffer */
  1924. layer = disp_sync_get_output_timeline_id();
  1925. mtkfb_release_layer_fence(primary_session_id, layer);
  1926. /* 4.modify interface path handle to new scenario(rdma->dsi) */
  1927. _cmdq_reset_config_handle();
  1928. if (!secure_path_on)
  1929. cmdqRecEnablePrefetch(pgc->cmdq_handle_config);
  1930. _cmdq_insert_wait_frame_done_token();
  1931. dpmgr_modify_path(pgc->dpmgr_handle, DDP_SCENARIO_PRIMARY_DISP, pgc->cmdq_handle_config,
  1932. primary_display_is_video_mode() ? DDP_VIDEO_MODE : DDP_CMD_MODE);
  1933. /* 5.config rdma from memory mode to directlink mode */
  1934. data_config_dl = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  1935. data_config_dl->rdma_config = decouple_rdma_config;
  1936. data_config_dl->rdma_config.address = 0;
  1937. data_config_dl->rdma_config.pitch = 0;
  1938. data_config_dl->rdma_config.security = DISP_NORMAL_BUFFER;
  1939. data_config_dl->rdma_dirty = 1;
  1940. memcpy(data_config_dl->ovl_config, ovl_config, sizeof(ovl_config));
  1941. data_config_dl->ovl_dirty = 1;
  1942. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config_dl, pgc->cmdq_handle_config);
  1943. /* use blocking flush to make sure all config is done, then stop/start trigger loop */
  1944. cmdqRecBackupUpdateSlot(pgc->cmdq_handle_config, pgc->rdma_buff_info, 0, 0);
  1945. /* cmdqRecDumpCommand(pgc->cmdq_handle_config); */
  1946. _cmdq_set_config_handle_dirty();
  1947. if (!secure_path_on)
  1948. cmdqRecDisablePrefetch(pgc->cmdq_handle_config);
  1949. _cmdq_flush_config_handle(1, NULL, 0);
  1950. /* release output buffer */
  1951. layer = disp_sync_get_output_interface_timeline_id();
  1952. mtkfb_release_layer_fence(primary_session_id, layer);
  1953. _cmdq_reset_config_handle();
  1954. _cmdq_insert_wait_frame_done_token();
  1955. /* 9.enable event for new path */
  1956. if (primary_display_is_video_mode())
  1957. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC, DDP_IRQ_RDMA0_DONE);
  1958. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  1959. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE);
  1960. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  1961. /* 1. enable SODI */
  1962. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  1963. spm_enable_sodi(1);
  1964. #endif
  1965. return ret;
  1966. }
  1967. const char *session_mode_spy(unsigned int mode)
  1968. {
  1969. switch (mode) {
  1970. case DISP_SESSION_DIRECT_LINK_MODE:
  1971. return "DIRECT_LINK";
  1972. case DISP_SESSION_DIRECT_LINK_MIRROR_MODE:
  1973. return "DIRECT_LINK_MIRROR";
  1974. case DISP_SESSION_DECOUPLE_MODE:
  1975. return "DECOUPLE";
  1976. case DISP_SESSION_DECOUPLE_MIRROR_MODE:
  1977. return "DECOUPLE_MIRROR";
  1978. default:
  1979. return "UNKNOWN";
  1980. }
  1981. }
  1982. static int config_display_m4u_port(void)
  1983. {
  1984. int ret = 0;
  1985. M4U_PORT_STRUCT sPort;
  1986. sPort.ePortID = M4U_PORT_DISP_OVL0;
  1987. sPort.Virtuality = primary_display_use_m4u;
  1988. sPort.Security = 0;
  1989. sPort.Distance = 1;
  1990. sPort.Direction = 0;
  1991. ret = m4u_config_port(&sPort);
  1992. if (ret == 0) {
  1993. DISPCHECK("config M4U Port %s to %s SUCCESS\n",
  1994. ddp_get_module_name(DISP_MODULE_OVL0),
  1995. primary_display_use_m4u ? "virtual" : "physical");
  1996. } else {
  1997. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  1998. ddp_get_module_name(DISP_MODULE_OVL0),
  1999. primary_display_use_m4u ? "virtual" : "physical", ret);
  2000. return -1;
  2001. }
  2002. #ifdef OVL_CASCADE_SUPPORT
  2003. sPort.ePortID = M4U_PORT_DISP_OVL1;
  2004. ret = m4u_config_port(&sPort);
  2005. if (ret) {
  2006. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  2007. ddp_get_module_name(DISP_MODULE_OVL1),
  2008. primary_display_use_m4u ? "virtual" : "physical", ret);
  2009. return -1;
  2010. }
  2011. #endif
  2012. sPort.ePortID = M4U_PORT_DISP_RDMA0;
  2013. ret = m4u_config_port(&sPort);
  2014. if (ret) {
  2015. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  2016. ddp_get_module_name(DISP_MODULE_RDMA0),
  2017. primary_display_use_m4u ? "virtual" : "physical", ret);
  2018. return -1;
  2019. }
  2020. sPort.ePortID = M4U_PORT_DISP_WDMA0;
  2021. ret = m4u_config_port(&sPort);
  2022. if (ret) {
  2023. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  2024. ddp_get_module_name(DISP_MODULE_WDMA0),
  2025. primary_display_use_m4u ? "virtual" : "physical", ret);
  2026. return -1;
  2027. }
  2028. return ret;
  2029. }
  2030. static disp_internal_buffer_info *allocat_decouple_buffer(int size)
  2031. {
  2032. void *buffer_va = NULL;
  2033. unsigned int buffer_mva = 0;
  2034. unsigned int mva_size = 0;
  2035. struct ion_client *client = NULL;
  2036. struct ion_handle *handle = NULL;
  2037. disp_internal_buffer_info *buf_info = NULL;
  2038. struct ion_mm_data mm_data;
  2039. memset((void *)&mm_data, 0, sizeof(struct ion_mm_data));
  2040. client = ion_client_create(g_ion_device, "disp_decouple");
  2041. buf_info = kzalloc(sizeof(disp_internal_buffer_info), GFP_KERNEL);
  2042. if (buf_info) {
  2043. handle = ion_alloc(client, size, 0, ION_HEAP_MULTIMEDIA_MASK, 0);
  2044. if (IS_ERR(handle)) {
  2045. DISPERR("Fatal Error, ion_alloc for size %d failed\n", size);
  2046. ion_free(client, handle);
  2047. ion_client_destroy(client);
  2048. kfree(buf_info);
  2049. return NULL;
  2050. }
  2051. buffer_va = ion_map_kernel(client, handle);
  2052. if (buffer_va == NULL) {
  2053. DISPERR("ion_map_kernrl failed\n");
  2054. ion_free(client, handle);
  2055. ion_client_destroy(client);
  2056. kfree(buf_info);
  2057. return NULL;
  2058. }
  2059. mm_data.config_buffer_param.kernel_handle = handle;
  2060. mm_data.mm_cmd = ION_MM_CONFIG_BUFFER;
  2061. if (ion_kernel_ioctl(client, ION_CMD_MULTIMEDIA, (unsigned long)&mm_data) < 0) {
  2062. DISPERR("ion_test_drv: Config buffer failed.\n");
  2063. ion_free(client, handle);
  2064. ion_client_destroy(client);
  2065. kfree(buf_info);
  2066. return NULL;
  2067. }
  2068. ion_phys(client, handle, (ion_phys_addr_t *) &buffer_mva, (size_t *) &mva_size);
  2069. if (buffer_mva == 0) {
  2070. DISPERR("Fatal Error, get mva failed\n");
  2071. ion_free(client, handle);
  2072. ion_client_destroy(client);
  2073. kfree(buf_info);
  2074. return NULL;
  2075. }
  2076. buf_info->handle = handle;
  2077. buf_info->mva = buffer_mva;
  2078. buf_info->size = mva_size;
  2079. buf_info->va = buffer_va;
  2080. buf_info->client = client;
  2081. } else {
  2082. DISPERR("Fatal error, kzalloc internal buffer info failed!!\n");
  2083. kfree(buf_info);
  2084. return NULL;
  2085. }
  2086. return buf_info;
  2087. }
  2088. static int init_decouple_buffers(void)
  2089. {
  2090. int i = 0;
  2091. int height = primary_display_get_height();
  2092. int width = primary_display_get_width();
  2093. int bpp = primary_display_get_dc_bpp();
  2094. int buffer_size = width * height * bpp / 8;
  2095. for (i = 0; i < DISP_INTERNAL_BUFFER_COUNT; i++) {
  2096. decouple_buffer_info[i] = allocat_decouple_buffer(buffer_size);
  2097. if (decouple_buffer_info[i] != NULL) {
  2098. pgc->dc_buf[i] = decouple_buffer_info[i]->mva;
  2099. dc_vAddr[i] = (unsigned long)decouple_buffer_info[i]->va;
  2100. }
  2101. }
  2102. /*initialize rdma config */
  2103. decouple_rdma_config.height = height;
  2104. decouple_rdma_config.width = width;
  2105. decouple_rdma_config.idx = 0;
  2106. decouple_rdma_config.inputFormat = eRGB888;
  2107. decouple_rdma_config.pitch = width * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2108. decouple_rdma_config.security = DISP_NORMAL_BUFFER;
  2109. /*initialize wdma config */
  2110. decouple_wdma_config.srcHeight = height;
  2111. decouple_wdma_config.srcWidth = width;
  2112. decouple_wdma_config.clipX = 0;
  2113. decouple_wdma_config.clipY = 0;
  2114. decouple_wdma_config.clipHeight = height;
  2115. decouple_wdma_config.clipWidth = width;
  2116. decouple_wdma_config.outputFormat = eRGB888;
  2117. decouple_wdma_config.useSpecifiedAlpha = 1;
  2118. decouple_wdma_config.alpha = 0xFF;
  2119. decouple_wdma_config.dstPitch = width * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2120. decouple_wdma_config.security = DISP_NORMAL_BUFFER;
  2121. return 0;
  2122. }
  2123. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  2124. static int allocate_idle_lp_dc_buffer(void)
  2125. {
  2126. int height = primary_display_get_height();
  2127. int width = primary_display_get_width();
  2128. int bpp = primary_display_get_dc_bpp();
  2129. int buffer_size = width * height * bpp / 8;
  2130. decouple_buffer_info[0] = allocat_decouple_buffer(buffer_size);
  2131. if (decouple_buffer_info[0] != NULL) {
  2132. pgc->dc_buf[0] = decouple_buffer_info[0]->mva;
  2133. dc_vAddr[0] = (unsigned long)decouple_buffer_info[0]->va;
  2134. } else {
  2135. return -1;
  2136. }
  2137. /*initialize rdma config*/
  2138. decouple_rdma_config.height = height;
  2139. decouple_rdma_config.width = width;
  2140. decouple_rdma_config.idx = 0;
  2141. decouple_rdma_config.inputFormat = eRGB888;
  2142. decouple_rdma_config.pitch = width * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2143. decouple_rdma_config.security = DISP_NORMAL_BUFFER;
  2144. /*initialize wdma config*/
  2145. decouple_wdma_config.srcHeight = height;
  2146. decouple_wdma_config.srcWidth = width;
  2147. decouple_wdma_config.clipX = 0;
  2148. decouple_wdma_config.clipY = 0;
  2149. decouple_wdma_config.clipHeight = height;
  2150. decouple_wdma_config.clipWidth = width;
  2151. decouple_wdma_config.outputFormat = eRGB888;
  2152. decouple_wdma_config.useSpecifiedAlpha = 1;
  2153. decouple_wdma_config.alpha = 0xFF;
  2154. decouple_wdma_config.dstPitch = width * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2155. decouple_wdma_config.security = DISP_NORMAL_BUFFER;
  2156. return 0;
  2157. }
  2158. static int release_idle_lp_dc_buffer(unsigned int need_primary_lock)
  2159. {
  2160. if (need_primary_lock)
  2161. _primary_path_lock(__func__);
  2162. if (decouple_buffer_info[0]) {
  2163. ion_free(decouple_buffer_info[0]->client, decouple_buffer_info[0]->handle);
  2164. ion_client_destroy(decouple_buffer_info[0]->client);
  2165. kfree(decouple_buffer_info[0]);
  2166. decouple_buffer_info[0] = NULL;
  2167. }
  2168. if (need_primary_lock)
  2169. _primary_path_unlock(__func__);
  2170. return 0;
  2171. }
  2172. #endif
  2173. static int __build_path_direct_link(void)
  2174. {
  2175. int ret = 0;
  2176. DISP_MODULE_ENUM dst_module = 0;
  2177. /* DISPFUNC();*/
  2178. DISPPRINT("%s\n", __func__);
  2179. pgc->mode = DIRECT_LINK_MODE;
  2180. pgc->dpmgr_handle = dpmgr_create_path(DDP_SCENARIO_PRIMARY_DISP, pgc->cmdq_handle_config);
  2181. if (pgc->dpmgr_handle) {
  2182. DISPPRINT("dpmgr create path SUCCESS(0x%p)\n", pgc->dpmgr_handle);
  2183. } else {
  2184. DISPERR("dpmgr create path FAIL\n");
  2185. return -1;
  2186. }
  2187. dst_module = _get_dst_module_by_lcm(pgc->plcm);
  2188. dpmgr_path_set_dst_module(pgc->dpmgr_handle, dst_module);
  2189. DISPCHECK("dpmgr set dst module FINISHED(%s)\n", ddp_get_module_name(dst_module));
  2190. #ifndef MTKFB_NO_M4U
  2191. config_display_m4u_port();
  2192. #if !defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  2193. init_decouple_buffers();
  2194. #endif
  2195. #endif
  2196. dpmgr_set_lcm_utils(pgc->dpmgr_handle, pgc->plcm->drv);
  2197. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  2198. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE);
  2199. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  2200. /* set video mode must before path_init */
  2201. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  2202. #ifndef CONFIG_MTK_CLKMGR
  2203. ddp_clk_prepare(DISP_MTCMOS_CLK);
  2204. #endif
  2205. #ifndef MTK_FB_CMDQ_DISABLE
  2206. dpmgr_path_init(pgc->dpmgr_handle, CMDQ_ENABLE);
  2207. /* cmdqRecFlush(pgc->cmdq_handle_config); */
  2208. #else
  2209. dpmgr_path_init(pgc->dpmgr_handle, CMDQ_DISABLE);
  2210. #endif
  2211. return ret;
  2212. }
  2213. static void copy_lklogo_to_dc_buf(void)
  2214. {
  2215. unsigned int line, k, src_argb8888;
  2216. unsigned int *s;
  2217. unsigned char *d;
  2218. for (line = 0; line < primary_display_get_height(); line++) {
  2219. d = (char *)(dc_vAddr[0] + line * primary_display_get_width() * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8);
  2220. s = (int *)(pgc->framebuffer_va + line * ALIGN_TO(primary_display_get_width(),
  2221. MTK_FB_ALIGNMENT) * primary_display_get_dc_bpp() / 8);
  2222. for (k = 0; k < primary_display_get_width(); k++) {
  2223. src_argb8888 = *s++;
  2224. *d++ = ((src_argb8888 & 0xFF));
  2225. *d++ = ((src_argb8888 & 0xFF00) >> 8);
  2226. *d++ = ((src_argb8888 & 0xFF0000) >> 16);
  2227. }
  2228. }
  2229. }
  2230. static int _build_path_rdma_to_dsi(void)
  2231. {
  2232. DISP_MODULE_ENUM dst_module = 0;
  2233. uint32_t writing_mva = 0;
  2234. #ifdef CONFIG_FOR_SOURCE_PQ
  2235. pgc->dpmgr_handle =
  2236. dpmgr_create_path(DDP_SCENARIO_PRIMARY_RDMA0_DISP, pgc->cmdq_handle_config);
  2237. #else
  2238. pgc->dpmgr_handle =
  2239. dpmgr_create_path(DDP_SCENARIO_PRIMARY_RDMA0_COLOR0_DISP, pgc->cmdq_handle_config);
  2240. #endif
  2241. if (pgc->dpmgr_handle) {
  2242. DISPCHECK("dpmgr create interface path SUCCESS\n");
  2243. } else {
  2244. DISPCHECK("dpmgr create path FAIL\n");
  2245. return -1;
  2246. }
  2247. /* VSYNC event will be provided to hwc for system vsync hw source
  2248. * FRAME_DONE will be used in esd/suspend/resume for path status check
  2249. */
  2250. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  2251. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE);
  2252. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  2253. dpmgr_set_lcm_utils(pgc->dpmgr_handle, pgc->plcm->drv);
  2254. dst_module = _get_dst_module_by_lcm(pgc->plcm);
  2255. dpmgr_path_set_dst_module(pgc->dpmgr_handle, dst_module);
  2256. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  2257. if (primary_display_is_video_mode())
  2258. _cmdq_insert_wait_frame_done_token();
  2259. #ifndef CONFIG_MTK_CLKMGR
  2260. ddp_clk_prepare(DISP_MTCMOS_CLK);
  2261. #endif
  2262. dpmgr_path_init(pgc->dpmgr_handle, CMDQ_ENABLE);
  2263. writing_mva = pgc->dc_buf[pgc->dc_buf_id];
  2264. DISPMSG("writing_mva = 0x%08x\n", writing_mva);
  2265. pgc->dc_buf_id++;
  2266. pgc->dc_buf_id %= DISP_INTERNAL_BUFFER_COUNT;
  2267. /*initialize rdma config */
  2268. decouple_rdma_config.address = (unsigned int)writing_mva;
  2269. decouple_rdma_config.width = primary_display_get_width();
  2270. decouple_rdma_config.height = primary_display_get_height();
  2271. decouple_rdma_config.idx = 0;
  2272. decouple_rdma_config.inputFormat = eRGB888;
  2273. decouple_rdma_config.pitch = primary_display_get_width() * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2274. _config_rdma_input_data(&decouple_rdma_config, pgc->dpmgr_handle, pgc->cmdq_handle_config);
  2275. cmdqRecFlush(pgc->cmdq_handle_config);
  2276. DISPCHECK("dpmgr set dst module FINISHED(%s)\n", ddp_get_module_name(dst_module));
  2277. return 0;
  2278. }
  2279. static int _build_path_ovl_to_wdma(void)
  2280. {
  2281. int ret = 0;
  2282. uint32_t writing_mva = 0;
  2283. #ifdef CONFIG_FOR_SOURCE_PQ
  2284. pgc->ovl2mem_path_handle = dpmgr_create_path(DDP_SCENARIO_PRIMARY_DITHER_MEMOUT,
  2285. pgc->cmdq_handle_ovl1to2_config);
  2286. #else
  2287. pgc->ovl2mem_path_handle = dpmgr_create_path(DDP_SCENARIO_PRIMARY_OVL_MEMOUT,
  2288. pgc->cmdq_handle_ovl1to2_config);
  2289. #endif
  2290. /*
  2291. * FRAME_START will be used in decouple-mirror mode, for post-path fence release(rdma->dsi)
  2292. */
  2293. dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE);
  2294. dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START);
  2295. if (pgc->ovl2mem_path_handle) {
  2296. DISPCHECK("dpmgr create ovl memout path SUCCESS\n");
  2297. } else {
  2298. DISPCHECK("dpmgr create path FAIL\n");
  2299. return -1;
  2300. }
  2301. dpmgr_path_set_video_mode(pgc->ovl2mem_path_handle, 0);
  2302. dpmgr_path_init(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  2303. writing_mva = pgc->dc_buf[pgc->dc_buf_id];
  2304. /*initialize wdma config */
  2305. decouple_wdma_config.dstAddress = writing_mva;
  2306. decouple_wdma_config.srcHeight = primary_display_get_height();
  2307. decouple_wdma_config.srcWidth = primary_display_get_width();
  2308. decouple_wdma_config.clipX = 0;
  2309. decouple_wdma_config.clipY = 0;
  2310. decouple_wdma_config.clipHeight = primary_display_get_height();
  2311. decouple_wdma_config.clipWidth = primary_display_get_width();
  2312. decouple_wdma_config.outputFormat = eRGB888;
  2313. decouple_wdma_config.useSpecifiedAlpha = 1;
  2314. decouple_wdma_config.alpha = 0xFF;
  2315. decouple_wdma_config.dstPitch = primary_display_get_width() * DP_COLOR_BITS_PER_PIXEL(eRGB888) / 8;
  2316. dpmgr_path_reset(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  2317. _config_wdma_output(&decouple_wdma_config, pgc->ovl2mem_path_handle, NULL);
  2318. ret = dpmgr_path_start(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  2319. DISPCHECK("build rdma to dsi path finished\n");
  2320. return ret;
  2321. }
  2322. /* #define CONFIG_USE_CMDQ */
  2323. static int __build_path_decouple(void)
  2324. {
  2325. int ret = 0;
  2326. /* DISPFUNC(); */
  2327. DISPPRINT("%s\n", __func__);
  2328. pgc->mode = DECOUPLE_MODE;
  2329. /* 1. Allocate decouple buffers and copy the LK logo to dc buffer. */
  2330. init_decouple_buffers();
  2331. copy_lklogo_to_dc_buf();
  2332. mutex_init(&(pgc->dc_lock));
  2333. /*
  2334. * 2. Enable m4u of RDMA0
  2335. * The OVL engine would be used in VDO mode.
  2336. * Before turn on all the m4u of display modules,
  2337. * the data path need to change to rdma0->dsi.
  2338. */
  2339. {
  2340. M4U_PORT_STRUCT sPort;
  2341. sPort.ePortID = M4U_PORT_DISP_RDMA0;
  2342. sPort.Virtuality = primary_display_use_m4u;
  2343. sPort.Security = 0;
  2344. sPort.Distance = 1;
  2345. sPort.Direction = 0;
  2346. ret = m4u_config_port(&sPort);
  2347. if (ret == 0) {
  2348. DISPCHECK("config M4U Port %s to %s SUCCESS\n",
  2349. ddp_get_module_name(DISP_MODULE_OVL0),
  2350. primary_display_use_m4u ? "virtual" : "physical");
  2351. } else {
  2352. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  2353. ddp_get_module_name(DISP_MODULE_OVL0),
  2354. primary_display_use_m4u ? "virtual" : "physical", ret);
  2355. return -1;
  2356. }
  2357. }
  2358. /* 3. Build and switch display data path from ovl->dsi to rdma->dsi. */
  2359. ret = _build_path_rdma_to_dsi();
  2360. if (ret) {
  2361. pr_debug("build path rdma to dsi fail\n");
  2362. return ret;
  2363. }
  2364. /*
  2365. * 4. Enable all display engine m4u.
  2366. * Since the display data path has been changed to rdma->dsi,
  2367. * all the display engine m4u can be enabled safely.
  2368. */
  2369. config_display_m4u_port();
  2370. /* 5. Build/Connect ovl->wdma display data path. */
  2371. ret = _build_path_ovl_to_wdma();
  2372. if (ret) {
  2373. pr_debug("build path ovl to wdma fail\n");
  2374. return ret;
  2375. }
  2376. DISPCHECK("build decouple path finished\n");
  2377. return ret;
  2378. }
  2379. static int __build_path_single_layer(void)
  2380. {
  2381. return 0; /* avoid build warning. */
  2382. }
  2383. static int __build_path_debug_rdma1_dsi0(void)
  2384. {
  2385. int ret = 0;
  2386. #if defined(MTK_FB_RDMA1_SUPPORT)
  2387. DISP_MODULE_ENUM dst_module = 0;
  2388. pgc->mode = DEBUG_RDMA1_DSI0_MODE;
  2389. pgc->dpmgr_handle = dpmgr_create_path(DDP_SCENARIO_SUB_RDMA1_DISP, pgc->cmdq_handle_config);
  2390. if (pgc->dpmgr_handle) {
  2391. DISPCHECK("dpmgr create path SUCCESS(0x%p)\n", pgc->dpmgr_handle);
  2392. } else {
  2393. DISPCHECK("dpmgr create path FAIL\n");
  2394. return -1;
  2395. }
  2396. dst_module = _get_dst_module_by_lcm(pgc->plcm);
  2397. dpmgr_path_set_dst_module(pgc->dpmgr_handle, dst_module);
  2398. DISPCHECK("dpmgr set dst module FINISHED(%s)\n", ddp_get_module_name(dst_module));
  2399. #ifndef MTKFB_NO_M4U
  2400. {
  2401. #ifdef MTK_FB_RDMA1_SUPPORT
  2402. M4U_PORT_STRUCT sPort;
  2403. sPort.ePortID = M4U_PORT_DISP_RDMA1;
  2404. sPort.Virtuality = primary_display_use_m4u;
  2405. sPort.Security = 0;
  2406. sPort.Distance = 1;
  2407. sPort.Direction = 0;
  2408. ret = m4u_config_port(&sPort);
  2409. if (ret == 0) {
  2410. DISPCHECK("config M4U Port %s to %s SUCCESS\n",
  2411. ddp_get_module_name(DISP_MODULE_RDMA1),
  2412. primary_display_use_m4u ? "virtual" : "physical");
  2413. } else {
  2414. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  2415. ddp_get_module_name(DISP_MODULE_RDMA1),
  2416. primary_display_use_m4u ? "virtual" : "physical", ret);
  2417. return -1;
  2418. }
  2419. #endif
  2420. }
  2421. #endif
  2422. dpmgr_set_lcm_utils(pgc->dpmgr_handle, pgc->plcm->drv);
  2423. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  2424. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE);
  2425. #endif
  2426. return ret;
  2427. }
  2428. int disp_fmt_to_hw_fmt(DISP_FORMAT src_fmt, unsigned int *hw_fmt,
  2429. unsigned int *Bpp, unsigned int *bpp)
  2430. {
  2431. switch (src_fmt) {
  2432. case DISP_FORMAT_YUV422:
  2433. *hw_fmt = eYUY2;
  2434. *Bpp = 2;
  2435. *bpp = 16;
  2436. break;
  2437. case DISP_FORMAT_RGB565:
  2438. *hw_fmt = eRGB565;
  2439. *Bpp = 2;
  2440. *bpp = 16;
  2441. break;
  2442. case DISP_FORMAT_RGB888:
  2443. *hw_fmt = eRGB888;
  2444. *Bpp = 3;
  2445. *bpp = 24;
  2446. break;
  2447. case DISP_FORMAT_BGR888:
  2448. *hw_fmt = eBGR888;
  2449. *Bpp = 3;
  2450. *bpp = 24;
  2451. break;
  2452. case DISP_FORMAT_ARGB8888:
  2453. *hw_fmt = eARGB8888;
  2454. *Bpp = 4;
  2455. *bpp = 32;
  2456. break;
  2457. case DISP_FORMAT_ABGR8888:
  2458. *hw_fmt = eABGR8888;
  2459. *Bpp = 4;
  2460. *bpp = 32;
  2461. break;
  2462. case DISP_FORMAT_RGBA8888:
  2463. *hw_fmt = eRGBA8888;
  2464. *Bpp = 4;
  2465. *bpp = 32;
  2466. break;
  2467. case DISP_FORMAT_BGRA8888:
  2468. /* *hw_fmt = eABGR8888; */
  2469. *hw_fmt = eBGRA8888;
  2470. *Bpp = 4;
  2471. *bpp = 32;
  2472. break;
  2473. case DISP_FORMAT_XRGB8888:
  2474. *hw_fmt = eARGB8888;
  2475. *Bpp = 4;
  2476. *bpp = 32;
  2477. break;
  2478. case DISP_FORMAT_XBGR8888:
  2479. *hw_fmt = eABGR8888;
  2480. *Bpp = 4;
  2481. *bpp = 32;
  2482. break;
  2483. case DISP_FORMAT_RGBX8888:
  2484. *hw_fmt = eRGBA8888;
  2485. *Bpp = 4;
  2486. *bpp = 32;
  2487. break;
  2488. case DISP_FORMAT_BGRX8888:
  2489. *hw_fmt = eBGRA8888;
  2490. *Bpp = 4;
  2491. *bpp = 32;
  2492. break;
  2493. case DISP_FORMAT_UYVY:
  2494. *hw_fmt = eUYVY;
  2495. *Bpp = 2;
  2496. *bpp = 16;
  2497. break;
  2498. case DISP_FORMAT_YV12:
  2499. *hw_fmt = eYV12;
  2500. *Bpp = 1;
  2501. *bpp = 8;
  2502. break;
  2503. default:
  2504. DISPERR("Invalid color format: 0x%x\n", src_fmt);
  2505. return -1;
  2506. }
  2507. return 0;
  2508. }
  2509. static int _convert_disp_input_to_ovl(OVL_CONFIG_STRUCT *dst, disp_input_config *src)
  2510. {
  2511. int ret;
  2512. unsigned int Bpp = 0;
  2513. unsigned int bpp = 0;
  2514. if (!src || !dst) {
  2515. DISP_FATAL_ERR("display", "%s src(0x%p) or dst(0x%p) is null\n",
  2516. __func__, src, dst);
  2517. return -1;
  2518. }
  2519. dst->layer = src->layer_id;
  2520. dst->isDirty = 1;
  2521. dst->buff_idx = src->next_buff_idx;
  2522. dst->layer_en = src->layer_enable;
  2523. /* if layer is disable, we just needs config above params. */
  2524. if (!src->layer_enable)
  2525. return 0;
  2526. ret = disp_fmt_to_hw_fmt(src->src_fmt, (unsigned int *)(&(dst->fmt)), (unsigned int *)(&Bpp),
  2527. (unsigned int *)(&bpp));
  2528. dst->addr = (unsigned long)src->src_phy_addr;
  2529. dst->vaddr = (unsigned long)src->src_base_addr;
  2530. dst->src_x = src->src_offset_x;
  2531. dst->src_y = src->src_offset_y;
  2532. dst->src_w = src->src_width;
  2533. dst->src_h = src->src_height;
  2534. dst->src_pitch = src->src_pitch * Bpp;
  2535. dst->dst_x = src->tgt_offset_x;
  2536. dst->dst_y = src->tgt_offset_y;
  2537. /* dst W/H should <= src W/H */
  2538. if (src->buffer_source != DISP_BUFFER_ALPHA) { /* dim layer do not care for src_width */
  2539. dst->dst_w = min(src->src_width, src->tgt_width);
  2540. dst->dst_h = min(src->src_height, src->tgt_height);
  2541. } else {
  2542. dst->dst_w = src->tgt_width;
  2543. dst->dst_h = src->tgt_height;
  2544. }
  2545. dst->keyEn = src->src_use_color_key;
  2546. dst->key = src->src_color_key;
  2547. dst->aen = src->alpha_enable;
  2548. dst->alpha = src->alpha;
  2549. dst->sur_aen = src->sur_aen;
  2550. dst->src_alpha = src->src_alpha;
  2551. dst->dst_alpha = src->dst_alpha;
  2552. #ifdef DISP_DISABLE_X_CHANNEL_ALPHA
  2553. if (DISP_FORMAT_ARGB8888 == src->src_fmt || DISP_FORMAT_ABGR8888 == src->src_fmt ||
  2554. DISP_FORMAT_RGBA8888 == src->src_fmt || DISP_FORMAT_BGRA8888 == src->src_fmt ||
  2555. src->buffer_source == DISP_BUFFER_ALPHA) {
  2556. ;/* nothing */
  2557. } else {
  2558. dst->aen = FALSE;
  2559. dst->sur_aen = FALSE;
  2560. }
  2561. #endif
  2562. dst->identity = src->identity;
  2563. dst->connected_type = src->connected_type;
  2564. dst->security = src->security;
  2565. dst->yuv_range = src->yuv_range;
  2566. if (src->buffer_source == DISP_BUFFER_ALPHA) {
  2567. dst->source = OVL_LAYER_SOURCE_RESERVED; /* dim layer, constant alpha */
  2568. } else if (src->buffer_source == DISP_BUFFER_ION || src->buffer_source == DISP_BUFFER_MVA) {
  2569. dst->source = OVL_LAYER_SOURCE_MEM; /* from memory */
  2570. } else {
  2571. DISPERR("unknown source = %d", src->buffer_source);
  2572. dst->source = OVL_LAYER_SOURCE_MEM;
  2573. }
  2574. return ret;
  2575. }
  2576. static int _convert_disp_input_to_rdma(RDMA_CONFIG_STRUCT *dst, disp_input_config *src)
  2577. {
  2578. int ret;
  2579. unsigned int Bpp = 0;
  2580. unsigned int bpp = 0;
  2581. if (!src || !dst) {
  2582. DISP_FATAL_ERR("display", "%s src(0x%p) or dst(0x%p) is null\n",
  2583. __func__, src, dst);
  2584. return -1;
  2585. }
  2586. ret = disp_fmt_to_hw_fmt(src->src_fmt, &(dst->inputFormat), &Bpp, &bpp);
  2587. dst->address = (unsigned long)src->src_phy_addr;
  2588. dst->width = src->src_width;
  2589. dst->height = src->src_height;
  2590. dst->pitch = src->src_pitch * Bpp;
  2591. return ret;
  2592. }
  2593. /* Video mode SODI CMDQ flow */
  2594. /* extern unsigned int gDumpConfigCMD; */
  2595. /* extern unsigned int gEnableOVLStatusCheck; */
  2596. #define DISP_REG_CMDQ_TOKEN_ID 0x10212060
  2597. #define DISP_REG_CMDQ_TOKEN_VALUE 0x10212064
  2598. #define DISP_MUTEX0_STREAM_EOF_ID 55
  2599. int _trigger_display_interface(int blocking, void *callback, unsigned int userdata)
  2600. {
  2601. static unsigned int cnt;
  2602. /* 4. enable SODI after config */
  2603. if (primary_display_is_video_mode() == 1)
  2604. disp_set_sodi(1, pgc->cmdq_handle_config);
  2605. #ifdef DISP_ENABLE_SODI_FOR_VIDEO_MODE
  2606. if (gPrefetchControl == 1 && cnt >= 20)
  2607. cmdqRecDisablePrefetch(pgc->cmdq_handle_config);
  2608. #endif
  2609. if (_should_wait_path_idle())
  2610. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  2611. if (_should_update_lcm())
  2612. disp_lcm_update(pgc->plcm, 0, 0, pgc->plcm->params->width,
  2613. pgc->plcm->params->height, 0);
  2614. if (_should_start_path())
  2615. dpmgr_path_start(pgc->dpmgr_handle, primary_display_cmdq_enabled());
  2616. if (_should_trigger_path())
  2617. /* trigger_loop_handle is used only for build trigger loop,
  2618. * which should always be NULL for config thread
  2619. */
  2620. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, primary_display_cmdq_enabled());
  2621. #ifndef MTK_FB_CMDQ_DISABLE
  2622. if (_should_set_cmdq_dirty()) {
  2623. _cmdq_set_config_handle_dirty();
  2624. /* disable SODI after set dirty */
  2625. if (primary_display_is_video_mode() == 0 && gDisableSODIForTriggerLoop == 1)
  2626. disp_set_sodi(0, pgc->cmdq_handle_config);
  2627. }
  2628. /* 1. disable SODI by CPU before flush CMDQ by CPU */
  2629. if (primary_display_is_video_mode() == 1)
  2630. disp_set_sodi(0, 0);
  2631. if (gDumpConfigCMD == 1) {
  2632. DISPMSG("primary_display_config, dump before flush:\n");
  2633. cmdqRecDumpCommand(pgc->cmdq_handle_config);
  2634. }
  2635. /* insert update ovl status slot command */
  2636. if (primary_display_is_video_mode() == 1) {
  2637. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_config, pgc->ovl_status_info,
  2638. 0, DISP_REG_OVL0_STATE_PA);
  2639. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_config, pgc->ovl_status_info,
  2640. 1, DISP_REG_OVL0_STATUS_PA);
  2641. }
  2642. if (_should_flush_cmdq_config_handle())
  2643. _cmdq_flush_config_handle(blocking, callback, userdata);
  2644. if (_should_reset_cmdq_config_handle()) {
  2645. _cmdq_reset_config_handle();
  2646. #ifdef DISP_ENABLE_SODI_FOR_VIDEO_MODE
  2647. /* do not have to enable/disable prefetch at power on stage. */
  2648. if (gPrefetchControl == 1 && cnt >= 20)
  2649. cmdqRecEnablePrefetch(pgc->cmdq_handle_config);
  2650. #endif
  2651. }
  2652. /* TODO: _is_decouple_mode() shuold be protected by mutex!!!!!!!!when dynamic switch decouple/directlink */
  2653. if (_should_insert_wait_frame_done_token() && (!_is_decouple_mode(pgc->session_mode))) {
  2654. /* 2. enable SODI by CMDQ before wait */
  2655. if (primary_display_is_video_mode() == 1)
  2656. disp_set_sodi(1, pgc->cmdq_handle_config);
  2657. if (primary_display_is_video_mode() == 1)
  2658. cmdqRecWaitNoClear(pgc->cmdq_handle_config, CMDQ_EVENT_DISP_RDMA0_EOF);
  2659. else
  2660. _cmdq_insert_wait_frame_done_token();
  2661. /* 3. disable AODI by CMDQ before config */
  2662. if (primary_display_is_video_mode() == 1)
  2663. disp_set_sodi(0, pgc->cmdq_handle_config);
  2664. }
  2665. if (cnt < 20)
  2666. cnt++;
  2667. else
  2668. gEnableLowPowerFeature = 1;
  2669. #endif
  2670. return 0;
  2671. }
  2672. int _trigger_ovl_to_memory(disp_path_handle disp_handle, cmdqRecHandle cmdq_handle,
  2673. fence_release_callback callback, unsigned int data, int blocking)
  2674. {
  2675. unsigned int rdma_pitch_sec;
  2676. dpmgr_wdma_path_force_power_on();
  2677. dpmgr_path_trigger(disp_handle, cmdq_handle, CMDQ_ENABLE);
  2678. cmdqRecWaitNoClear(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_EOF);
  2679. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 0, mem_config.addr);
  2680. /* rdma pitch only use bit[15..0], we use bit[31:30] to store secure information */
  2681. rdma_pitch_sec = mem_config.pitch | (mem_config.security << 30);
  2682. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 1, rdma_pitch_sec);
  2683. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 2, mem_config.fmt);
  2684. if (blocking)
  2685. cmdqRecFlush(cmdq_handle);
  2686. else
  2687. cmdqRecFlushAsyncCallback(cmdq_handle, (CmdqAsyncFlushCB) callback, data);
  2688. cmdqRecReset(cmdq_handle);
  2689. cmdqRecWait(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_EOF);
  2690. MMProfileLogEx(ddp_mmp_get_events()->ovl_trigger, MMProfileFlagPulse, 0, data);
  2691. return 0;
  2692. }
  2693. int _trigger_ovl_to_memory_mirror(disp_path_handle disp_handle, cmdqRecHandle cmdq_handle,
  2694. fence_release_callback callback, unsigned int data)
  2695. {
  2696. int layer = 0;
  2697. unsigned int rdma_pitch_sec;
  2698. dpmgr_wdma_path_force_power_on();
  2699. dpmgr_path_trigger(disp_handle, cmdq_handle, CMDQ_ENABLE);
  2700. cmdqRecWaitNoClear(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_EOF);
  2701. layer = disp_sync_get_output_timeline_id();
  2702. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->cur_config_fence, layer, mem_config.buff_idx);
  2703. layer = disp_sync_get_output_interface_timeline_id();
  2704. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->cur_config_fence, layer, mem_config.interface_idx);
  2705. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 0, (unsigned int)mem_config.addr);
  2706. /*rdma pitch only use bit[15..0], we use bit[31:30] to store secure information */
  2707. rdma_pitch_sec = mem_config.pitch | (mem_config.security << 30);
  2708. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 1, rdma_pitch_sec);
  2709. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->rdma_buff_info, 2, mem_config.fmt);
  2710. cmdqRecFlushAsyncCallback(cmdq_handle, (CmdqAsyncFlushCB) callback, data);
  2711. cmdqRecReset(cmdq_handle);
  2712. cmdqRecWait(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_EOF);
  2713. MMProfileLogEx(ddp_mmp_get_events()->ovl_trigger, MMProfileFlagPulse, 0, data);
  2714. return 0;
  2715. }
  2716. #define EEEEEEEEEEEEEEE
  2717. /******************************************************************************/
  2718. /* ESD CHECK / RECOVERY ---- BEGIN */
  2719. /******************************************************************************/
  2720. #ifdef MTK_FB_ESD_ENABLE
  2721. static struct task_struct *primary_display_esd_check_task;
  2722. #define IS_ESD_ENABLE true
  2723. #else
  2724. #define IS_ESD_ENABLE false
  2725. #endif
  2726. #ifdef MTK_FB_PULLCLK_ENABLE
  2727. static struct task_struct *primary_display_pullclk_task;
  2728. #endif
  2729. static wait_queue_head_t esd_check_task_wq; /* For Esd Check Task */
  2730. static atomic_t esd_check_task_wakeup = ATOMIC_INIT(0); /* For Esd Check Task */
  2731. static wait_queue_head_t esd_ext_te_wq; /* For Vdo Mode EXT TE Check */
  2732. static atomic_t esd_ext_te_event = ATOMIC_INIT(0); /* For Vdo Mode EXT TE Check */
  2733. static atomic_t esd_check_bycmdq = ATOMIC_INIT(0);
  2734. static inline bool _is_enable_esd_check(void)
  2735. {
  2736. return IS_ESD_ENABLE;
  2737. }
  2738. struct task_struct *primary_display_frame_update_task = NULL;
  2739. wait_queue_head_t primary_display_frame_update_wq;
  2740. atomic_t primary_display_frame_update_event = ATOMIC_INIT(0);
  2741. struct task_struct *decouple_fence_release_task = NULL;
  2742. wait_queue_head_t decouple_fence_release_wq;
  2743. atomic_t decouple_fence_release_event = ATOMIC_INIT(0);
  2744. static int eint_flag; /* For DCT Setting */
  2745. unsigned int _need_do_esd_check(void)
  2746. {
  2747. int ret = 0;
  2748. #ifdef CONFIG_OF
  2749. if ((pgc->plcm->params->dsi.esd_check_enable == 1) && (islcmconnected == 1))
  2750. ret = 1;
  2751. #else
  2752. if (pgc->plcm->params->dsi.esd_check_enable == 1)
  2753. ret = 1;
  2754. #endif
  2755. return ret;
  2756. }
  2757. unsigned int _need_register_eint(void)
  2758. {
  2759. int ret = 1;
  2760. /* 1.need do esd check */
  2761. /* 2.dsi vdo mode */
  2762. /* 3.customization_esd_check_enable = 0 */
  2763. if (_need_do_esd_check() == 0)
  2764. ret = 0;
  2765. else if (primary_display_is_video_mode() == 0)
  2766. ret = 0;
  2767. else if (pgc->plcm->params->dsi.customization_esd_check_enable == 1)
  2768. ret = 0;
  2769. return ret;
  2770. }
  2771. unsigned int _need_wait_esd_eof(void)
  2772. {
  2773. int ret = 1;
  2774. /* 1.need do esd check */
  2775. /* 2.customization_esd_check_enable = 1 */
  2776. /* 3.dsi cmd mode */
  2777. if (_need_do_esd_check() == 0)
  2778. ret = 0;
  2779. else if (pgc->plcm->params->dsi.customization_esd_check_enable == 0)
  2780. ret = 0;
  2781. else if (primary_display_is_video_mode())
  2782. ret = 0;
  2783. return ret;
  2784. }
  2785. /* For Cmd Mode Read LCM Check */
  2786. /* Config cmdq_handle_config_esd */
  2787. int _esd_check_config_handle_cmd(void)
  2788. {
  2789. int ret = 0; /* 0:success */
  2790. /* 1.reset */
  2791. cmdqRecReset(pgc->cmdq_handle_config_esd);
  2792. /* 2.write first instruction */
  2793. /* cmd mode: wait CMDQ_SYNC_TOKEN_STREAM_EOF(wait trigger thread done) */
  2794. cmdqRecWaitNoClear(pgc->cmdq_handle_config_esd, CMDQ_SYNC_TOKEN_STREAM_EOF);
  2795. /* 3.clear CMDQ_SYNC_TOKEN_ESD_EOF(trigger thread need wait this sync token) */
  2796. cmdqRecClearEventToken(pgc->cmdq_handle_config_esd, CMDQ_SYNC_TOKEN_ESD_EOF);
  2797. /* 4.write instruction(read from lcm) */
  2798. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_CHECK_READ);
  2799. /* 5.set CMDQ_SYNC_TOKE_ESD_EOF(trigger thread can work now) */
  2800. cmdqRecSetEventToken(pgc->cmdq_handle_config_esd, CMDQ_SYNC_TOKEN_ESD_EOF);
  2801. /* 6.flush instruction */
  2802. dprec_logger_start(DPREC_LOGGER_ESD_CMDQ, 0, 0);
  2803. ret = cmdqRecFlush(pgc->cmdq_handle_config_esd);
  2804. dprec_logger_done(DPREC_LOGGER_ESD_CMDQ, 0, 0);
  2805. DISPCHECK("[ESD]_esd_check_config_handle_cmd ret=%d\n", ret);
  2806. if (ret)
  2807. ret = 1;
  2808. return ret;
  2809. }
  2810. void primary_display_esd_cust_bycmdq(int enable)
  2811. {
  2812. atomic_set(&esd_check_bycmdq, enable);
  2813. }
  2814. int primary_display_esd_cust_get(void)
  2815. {
  2816. return atomic_read(&esd_check_bycmdq);
  2817. }
  2818. /* For Vdo Mode Read LCM Check */
  2819. /* Config cmdq_handle_config_esd */
  2820. /* extern unsigned int gESDEnableSODI; */
  2821. /* extern unsigned int gDumpESDCMD; */
  2822. int _esd_check_config_handle_vdo(void)
  2823. {
  2824. int ret = 0; /* 0:success , 1:fail */
  2825. primary_display_esd_cust_bycmdq(1);
  2826. _primary_path_lock(__func__);
  2827. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  2828. spm_enable_sodi(0);
  2829. #endif
  2830. /* 1.reset */
  2831. cmdqRecReset(pgc->cmdq_handle_config_esd);
  2832. /* wait stream eof first */
  2833. cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_DISP_RDMA0_EOF);
  2834. cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  2835. #ifdef DISP_DUMP_EVENT_STATUS
  2836. DISP_REG_SET_PA(pgc->cmdq_handle_config_esd, DISP_REG_CMDQ_TOKEN_ID, DISP_MUTEX0_STREAM_EOF_ID);
  2837. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_config_esd, pgc->event_status, 1, DISP_REG_CMDQ_TOKEN_VALUE);
  2838. #endif
  2839. /* stop dsi vdo mode */
  2840. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_STOP_VDO_MODE);
  2841. /* write instruction(read from lcm) */
  2842. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_CHECK_READ);
  2843. /* pull DSI clock lane */
  2844. DSI_sw_clk_trail_cmdq(0, pgc->cmdq_handle_config_esd);
  2845. DSI_manual_enter_HS(pgc->cmdq_handle_config_esd);
  2846. /* start dsi vdo mode */
  2847. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_START_VDO_MODE);
  2848. /* trigger path */
  2849. cmdqRecClearEventToken(pgc->cmdq_handle_config_esd, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  2850. if (gEnableSWTrigger == 1)
  2851. DISP_REG_SET(pgc->cmdq_handle_config_esd, DISP_REG_CONFIG_MUTEX_EN(DISP_OVL_SEPARATE_MUTEX_ID), 1);
  2852. dpmgr_path_trigger(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ENABLE);
  2853. /* flush instruction */
  2854. dprec_logger_start(DPREC_LOGGER_ESD_CMDQ, 0, 0);
  2855. if (gDumpESDCMD == 1) {
  2856. DISPMSG("esd dump before flush:\n");
  2857. cmdqRecDumpCommand(pgc->cmdq_handle_config_esd);
  2858. }
  2859. ret = cmdqRecFlush(pgc->cmdq_handle_config_esd);
  2860. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  2861. spm_enable_sodi(1);
  2862. #endif
  2863. _primary_path_unlock(__func__);
  2864. #ifdef DISP_DUMP_EVENT_STATUS
  2865. {
  2866. unsigned int i = 0;
  2867. unsigned int event_status[6];
  2868. DISPMSG("dump esd status: ");
  2869. for (i = 0; i < 6; i++)
  2870. cmdqBackupReadSlot(pgc->event_status, i, &event_status[i]);
  2871. DISPMSG("%d, %d, %d, %d, %d, %d\n",
  2872. event_status[0], event_status[1], event_status[2],
  2873. event_status[3], event_status[4], event_status[5]);
  2874. }
  2875. #endif
  2876. dprec_logger_done(DPREC_LOGGER_ESD_CMDQ, 0, 0);
  2877. DISPCHECK("[ESD]_esd_check_config_handle_vdo ret=%d\n", ret);
  2878. if (ret)
  2879. ret = 1;
  2880. primary_display_esd_cust_bycmdq(0);
  2881. return ret;
  2882. }
  2883. /* For Vdo Mode EXT TE Check */
  2884. static irqreturn_t _esd_check_ext_te_irq_handler(int irq, void *data)
  2885. {
  2886. MMProfileLogEx(ddp_mmp_get_events()->esd_vdo_eint, MMProfileFlagPulse, 0, 0);
  2887. atomic_set(&esd_ext_te_event, 1);
  2888. wake_up_interruptible(&esd_ext_te_wq);
  2889. return IRQ_HANDLED;
  2890. }
  2891. int primary_display_switch_esd_mode(int mode)
  2892. {
  2893. int ret = 0;
  2894. #ifdef GPIO_DSI_TE_PIN
  2895. int gpio_mode = 0;
  2896. #endif
  2897. DISPFUNC();
  2898. if (pgc->plcm->params->dsi.customization_esd_check_enable != 0)
  2899. return -1; /* avoid build warning. */
  2900. DISPMSG("switch esd mode to %d\n", mode);
  2901. #ifdef GPIO_DSI_TE_PIN
  2902. #ifndef CONFIG_FPGA_EARLY_PORTING
  2903. gpio_mode = mt_get_gpio_mode(GPIO_DSI_TE_PIN);
  2904. /* DISPMSG("[ESD]gpio_mode=%d\n", gpio_mode); */
  2905. #endif
  2906. #endif
  2907. if (mode == 1) {
  2908. #ifdef GPIO_DSI_TE_PIN
  2909. /*switch to vdo mode */
  2910. if (gpio_mode == GPIO_DSI_TE_PIN_M_DSI_TE) {
  2911. #endif
  2912. /* if(_need_register_eint()) */
  2913. {
  2914. /* DISPMSG("[ESD]switch video mode\n"); */
  2915. struct device_node *node = NULL;
  2916. int irq;
  2917. u32 ints[2] = { 0, 0 };
  2918. #ifdef GPIO_DSI_TE_PIN
  2919. #ifndef CONFIG_FPGA_EARLY_PORTING
  2920. /* 1.set GPIO107 eint mode */
  2921. mt_set_gpio_mode(GPIO_DSI_TE_PIN, GPIO_DSI_TE_PIN_M_GPIO);
  2922. #endif
  2923. #endif
  2924. /* 2.register eint */
  2925. node = of_find_compatible_node(NULL, NULL, "mediatek, DSI_TE_1-eint");
  2926. if (node) {
  2927. /* DISPMSG("node 0x%x\n", node); */
  2928. of_property_read_u32_array(node, "debounce", ints, ARRAY_SIZE(ints));
  2929. /* mt_gpio_set_debounce(ints[0], ints[1]); */
  2930. mt_eint_set_hw_debounce(ints[0], ints[1]);
  2931. irq = irq_of_parse_and_map(node, 0);
  2932. if (request_irq(irq, _esd_check_ext_te_irq_handler, IRQF_TRIGGER_RISING,
  2933. "DSI_TE_1-eint", NULL))
  2934. DISPERR("[ESD]EINT IRQ LINE NOT AVAILABLE!!\n");
  2935. } else {
  2936. DISPERR("[ESD][%s] can't find DSI_TE_1 eint compatible node\n", __func__);
  2937. }
  2938. }
  2939. #ifdef GPIO_DSI_TE_PIN
  2940. }
  2941. #endif
  2942. } else if (mode == 0) {
  2943. #ifdef GPIO_DSI_TE_PIN
  2944. /* switch to cmd mode */
  2945. if (gpio_mode == GPIO_DSI_TE_PIN_M_GPIO) {
  2946. #endif
  2947. struct device_node *node = NULL;
  2948. int irq;
  2949. /* DISPMSG("[ESD]switch cmd mode\n"); */
  2950. /* unregister eint */
  2951. node = of_find_compatible_node(NULL, NULL, "mediatek, DSI_TE_1-eint");
  2952. /* DISPMSG("node 0x%x\n", node); */
  2953. if (node) {
  2954. irq = irq_of_parse_and_map(node, 0);
  2955. free_irq(irq, NULL);
  2956. } else {
  2957. DISPERR("[ESD][%s] can't find DSI_TE_1 eint compatible node\n",
  2958. __func__);
  2959. }
  2960. #ifdef GPIO_DSI_TE_PIN
  2961. #ifndef CONFIG_FPGA_EARLY_PORTING
  2962. /* set GPIO107 DSI TE mode */
  2963. mt_set_gpio_mode(GPIO_DSI_TE_PIN, GPIO_DSI_TE_PIN_M_DSI_TE);
  2964. #endif
  2965. #endif
  2966. #ifdef GPIO_DSI_TE_PIN
  2967. }
  2968. #endif
  2969. }
  2970. /* DISPMSG("primary_display_switch_esd_mode end\n"); */
  2971. return ret;
  2972. }
  2973. /* ESD CHECK FUNCTION */
  2974. /* return 1: esd check fail */
  2975. /* return 0: esd check pass */
  2976. int primary_display_esd_check(void)
  2977. {
  2978. int ret = 0;
  2979. _primary_path_esd_check_lock();
  2980. dprec_logger_start(DPREC_LOGGER_ESD_CHECK, 0, 0);
  2981. MMProfileLogEx(ddp_mmp_get_events()->esd_check_t, MMProfileFlagStart, 0, 0);
  2982. DISPCHECK("[ESD]ESD check begin\n");
  2983. _primary_path_lock(__func__);
  2984. if (pgc->state == DISP_SLEPT) {
  2985. MMProfileLogEx(ddp_mmp_get_events()->esd_check_t, MMProfileFlagPulse, 1, 0);
  2986. DISPCHECK("[ESD]primary display path is slept?? -- skip esd check\n");
  2987. _primary_path_unlock(__func__);
  2988. /* goto done; */
  2989. DISPCHECK("[ESD]ESD check end\n");
  2990. MMProfileLogEx(ddp_mmp_get_events()->esd_check_t, MMProfileFlagEnd, 0, ret);
  2991. dprec_logger_done(DPREC_LOGGER_ESD_CHECK, 0, 0);
  2992. _primary_path_esd_check_unlock();
  2993. return ret;
  2994. }
  2995. _primary_path_unlock(__func__);
  2996. #ifdef MTK_DISP_IDLE_LP
  2997. _disp_primary_path_dsi_clock_on(0);
  2998. #endif
  2999. /* / Esd Check : EXT TE */
  3000. if (pgc->plcm->params->dsi.customization_esd_check_enable == 0) {
  3001. MMProfileLogEx(ddp_mmp_get_events()->esd_extte, MMProfileFlagStart, 0, 0);
  3002. if (primary_display_is_video_mode()) {
  3003. primary_display_switch_esd_mode(1);
  3004. /* use cmdq to pull DSI clk lane*/
  3005. if (primary_display_cmdq_enabled() ) {
  3006. _primary_path_lock(__func__);
  3007. /* 0.create esd check cmdq */
  3008. cmdqRecCreate(CMDQ_SCENARIO_DISP_ESD_CHECK, &(pgc->cmdq_handle_config_esd));
  3009. _primary_path_unlock(__func__);
  3010. primary_display_esd_cust_bycmdq(1);
  3011. /* 1.reset*/
  3012. cmdqRecReset(pgc->cmdq_handle_config_esd);
  3013. /* wait stream eof first */
  3014. ret = cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_DISP_RDMA0_EOF);
  3015. cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  3016. _primary_path_lock(__func__);
  3017. /* 2.stop dsi vdo mode */
  3018. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_STOP_VDO_MODE);
  3019. /* 3.pull DSI clock lane */
  3020. DSI_sw_clk_trail_cmdq(0, pgc->cmdq_handle_config_esd);
  3021. DSI_manual_enter_HS(pgc->cmdq_handle_config_esd);
  3022. /* 4.start dsi vdo mode */
  3023. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_START_VDO_MODE);
  3024. /* 5. trigger path */
  3025. cmdqRecClearEventToken(pgc->cmdq_handle_config_esd ,CMDQ_EVENT_MUTEX0_STREAM_EOF);
  3026. if (gEnableSWTrigger==1)
  3027. DISP_REG_SET(pgc->cmdq_handle_config_esd,
  3028. DISP_REG_CONFIG_MUTEX_EN(DISP_OVL_SEPARATE_MUTEX_ID), 1);
  3029. dpmgr_path_trigger(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ENABLE);
  3030. _primary_path_unlock(__func__);
  3031. cmdqRecFlush(pgc->cmdq_handle_config_esd);
  3032. primary_display_esd_cust_bycmdq(0);
  3033. cmdqRecDestroy(pgc->cmdq_handle_config_esd);
  3034. pgc->cmdq_handle_config_esd = NULL;
  3035. }
  3036. if (_need_register_eint()) {
  3037. MMProfileLogEx(ddp_mmp_get_events()->esd_extte, MMProfileFlagPulse, 1, 1);
  3038. if (wait_event_interruptible_timeout
  3039. (esd_ext_te_wq, atomic_read(&esd_ext_te_event), HZ / 2) > 0) {
  3040. ret = 0; /* esd check pass */
  3041. } else {
  3042. ret = 1; /* esd check fail */
  3043. DISPCHECK("esd check fail release fence fake\n");
  3044. primary_display_release_fence_fake();
  3045. }
  3046. atomic_set(&esd_ext_te_event, 0);
  3047. }
  3048. primary_display_switch_esd_mode(0);
  3049. } else {
  3050. MMProfileLogEx(ddp_mmp_get_events()->esd_extte, MMProfileFlagPulse, 0, 1);
  3051. if (dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC, HZ / 2) > 0) {
  3052. ret = 0; /* esd check pass */
  3053. } else {
  3054. ret = 1; /* esd check fail */
  3055. DISPCHECK("esd check fail release fence fake\n");
  3056. primary_display_release_fence_fake();
  3057. }
  3058. }
  3059. MMProfileLogEx(ddp_mmp_get_events()->esd_extte, MMProfileFlagEnd, 0, ret);
  3060. goto done;
  3061. }
  3062. /* / Esd Check : Read from lcm */
  3063. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagStart, 0, primary_display_cmdq_enabled());
  3064. if (primary_display_cmdq_enabled()) {
  3065. _primary_path_lock(__func__);
  3066. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, 0, 1);
  3067. /* 0.create esd check cmdq */
  3068. cmdqRecCreate(CMDQ_SCENARIO_DISP_ESD_CHECK, &(pgc->cmdq_handle_config_esd));
  3069. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_ALLC_SLOT);
  3070. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, 0, 2);
  3071. DISPCHECK("[ESD]ESD config thread=%p\n", pgc->cmdq_handle_config_esd);
  3072. _primary_path_unlock(__func__);
  3073. /* 1.use cmdq to read from lcm */
  3074. if (primary_display_is_video_mode())
  3075. ret = _esd_check_config_handle_vdo();
  3076. else
  3077. ret = _esd_check_config_handle_cmd();
  3078. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, primary_display_is_video_mode(), 3);
  3079. if (ret == 1) {/* cmdq fail */
  3080. /* Need set esd check eof synctoken to let trigger loop go. */
  3081. if (_need_wait_esd_eof())
  3082. cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_ESD_EOF);
  3083. /* do dsi reset */
  3084. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_DSI_RESET);
  3085. DISPCHECK("esd check fail release fence fake\n");
  3086. primary_display_release_fence_fake();
  3087. goto destroy_cmdq;
  3088. }
  3089. DISPCHECK("[ESD]ESD config thread done~\n");
  3090. /* 2.check data(*cpu check now) */
  3091. ret = dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_CHECK_CMP);
  3092. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, 0, 4);
  3093. if (ret) {
  3094. ret = 1; /* esd check fail */
  3095. DISPCHECK("esd check fail release fence fake\n");
  3096. primary_display_release_fence_fake();
  3097. }
  3098. destroy_cmdq:
  3099. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd,
  3100. CMDQ_ESD_FREE_SLOT);
  3101. /* 3.destroy esd config thread */
  3102. cmdqRecDestroy(pgc->cmdq_handle_config_esd);
  3103. pgc->cmdq_handle_config_esd = NULL;
  3104. } else { /* by cpu */
  3105. /* 0: lock path */
  3106. /* 1: stop path */
  3107. /* 2: do esd check (!!!) */
  3108. /* 3: start path */
  3109. /* 4: unlock path */
  3110. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, 0, 1);
  3111. _primary_path_lock(__func__);
  3112. /* 1: stop path */
  3113. DISPCHECK("[ESD]display cmdq trigger loop stop[begin]\n");
  3114. _cmdq_stop_trigger_loop();
  3115. DISPCHECK("[ESD]display cmdq trigger loop stop[end]\n");
  3116. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  3117. DISPCHECK("[ESD]primary display path is busy\n");
  3118. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  3119. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  3120. }
  3121. DISPCHECK("[ESD]stop dpmgr path[begin]\n");
  3122. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  3123. DISPCHECK("[ESD]stop dpmgr path[end]\n");
  3124. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  3125. DISPCHECK("[ESD]primary display path is busy after stop\n");
  3126. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  3127. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  3128. }
  3129. DISPCHECK("[ESD]reset display path[begin]\n");
  3130. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  3131. DISPCHECK("[ESD]reset display path[end]\n");
  3132. /* 2: do esd check (!!!) */
  3133. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, 0, 2);
  3134. if (primary_display_is_video_mode())
  3135. /* ret = 0; */
  3136. ret = disp_lcm_esd_check(pgc->plcm);
  3137. else
  3138. ret = disp_lcm_esd_check(pgc->plcm);
  3139. /* 3: start path */
  3140. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagPulse, primary_display_is_video_mode(), 3);
  3141. DISPCHECK("[ESD]start dpmgr path[begin]\n");
  3142. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  3143. DISPCHECK("[ESD]start dpmgr path[end]\n");
  3144. DISPCHECK("[ESD]start cmdq trigger loop[begin]\n");
  3145. _cmdq_start_trigger_loop();
  3146. DISPCHECK("[ESD]start cmdq trigger loop[end]\n");
  3147. _primary_path_unlock(__func__);
  3148. }
  3149. MMProfileLogEx(ddp_mmp_get_events()->esd_rdlcm, MMProfileFlagEnd, 0, ret);
  3150. done:
  3151. #ifdef MTK_DISP_IDLE_LP
  3152. _disp_primary_path_dsi_clock_off(0);
  3153. #endif
  3154. DISPCHECK("[ESD]ESD check end\n");
  3155. MMProfileLogEx(ddp_mmp_get_events()->esd_check_t, MMProfileFlagEnd, 0, ret);
  3156. dprec_logger_done(DPREC_LOGGER_ESD_CHECK, 0, 0);
  3157. _primary_path_esd_check_unlock();
  3158. return ret;
  3159. }
  3160. #ifdef MTK_FB_ESD_ENABLE
  3161. static int primary_display_esd_check_worker_kthread(void *data)
  3162. {
  3163. int ret = 0;
  3164. int i = 0;
  3165. int esd_try_cnt = 5; /* 20; */
  3166. int count = 0;
  3167. struct sched_param param = {.sched_priority = 87 }; /* RTPM_PRIO_FB_THREAD */
  3168. sched_setscheduler(current, SCHED_RR, &param);
  3169. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE);
  3170. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  3171. while (1) {
  3172. if (count == 0) {
  3173. count++;
  3174. msleep(3000);
  3175. }
  3176. msleep(2000); /* esd check every 2s */
  3177. ret = wait_event_interruptible(esd_check_task_wq, atomic_read(&esd_check_task_wakeup));
  3178. if (ret < 0) {
  3179. DISPCHECK("[ESD]esd check thread waked up accidently\n");
  3180. continue;
  3181. }
  3182. #ifdef DISP_SWITCH_DST_MODE
  3183. _primary_path_switch_dst_lock();
  3184. #endif
  3185. _primary_path_cmd_lock();
  3186. ret = primary_display_esd_check();
  3187. if (ret == 1) {
  3188. DISPCHECK("[ESD]esd check fail, will do esd recovery %d\n", ret);
  3189. i = esd_try_cnt;
  3190. while (i--) {
  3191. DISPCHECK("[ESD]esd recovery try:%d\n", i);
  3192. primary_display_esd_recovery();
  3193. ret = primary_display_esd_check();
  3194. if (ret == 0) {
  3195. DISPCHECK("[ESD]esd recovery success\n");
  3196. break;
  3197. }
  3198. DISPCHECK("[ESD]after esd recovery, esd check still fail\n");
  3199. if (i == 0) {
  3200. DISPCHECK("[ESD]after esd recovery %d times, esd check still fail,\n",
  3201. esd_try_cnt);
  3202. DISPCHECK("disable esd check\n");
  3203. primary_display_esd_check_enable(0);
  3204. primary_display_esd_recovery();
  3205. }
  3206. }
  3207. }
  3208. _primary_path_cmd_unlock();
  3209. #ifdef DISP_SWITCH_DST_MODE
  3210. _primary_path_switch_dst_unlock();
  3211. #endif
  3212. if (kthread_should_stop())
  3213. break;
  3214. }
  3215. return 0;
  3216. }
  3217. #endif /* MTK_FB_ESD_ENABLE */
  3218. /* ESD RECOVERY */
  3219. int primary_display_esd_recovery(void)
  3220. {
  3221. DISP_STATUS ret = DISP_STATUS_OK;
  3222. LCM_PARAMS *lcm_param = NULL;
  3223. DISPFUNC();
  3224. dprec_logger_start(DPREC_LOGGER_ESD_RECOVERY, 0, 0);
  3225. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagStart, 0, 0);
  3226. DISPCHECK("[ESD]ESD recovery begin\n");
  3227. _primary_path_lock(__func__);
  3228. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse,
  3229. primary_display_is_video_mode(), 1);
  3230. lcm_param = disp_lcm_get_params(pgc->plcm);
  3231. if (pgc->state == DISP_SLEPT) {
  3232. DISPCHECK("[ESD]esd recovery but primary display path is slept??\n");
  3233. goto done;
  3234. }
  3235. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 2);
  3236. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  3237. DISPCHECK("[ESD]primary display path is busy\n");
  3238. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  3239. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  3240. }
  3241. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 3);
  3242. DISPCHECK("[ESD]stop dpmgr path[begin]\n");
  3243. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  3244. DISPCHECK("[ESD]stop dpmgr path[end]\n");
  3245. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 4);
  3246. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  3247. DISPCHECK("[ESD]primary display path is busy after stop\n");
  3248. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  3249. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  3250. }
  3251. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 5);
  3252. DISPCHECK("[ESD]display cmdq trigger loop stop[begin]\n");
  3253. _cmdq_stop_trigger_loop();
  3254. DISPCHECK("[ESD]display cmdq trigger loop stop[end]\n");
  3255. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 6);
  3256. DISPCHECK("[ESD]reset display path[begin]\n");
  3257. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  3258. DISPCHECK("[ESD]reset display path[end]\n");
  3259. DISPCHECK("[POWER]lcm suspend[begin]\n");
  3260. disp_lcm_suspend(pgc->plcm);
  3261. DISPCHECK("[POWER]lcm suspend[end]\n");
  3262. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 7);
  3263. DISPCHECK("[ESD]dsi power reset[begin]\n");
  3264. dpmgr_path_dsi_power_off(pgc->dpmgr_handle, CMDQ_DISABLE);
  3265. dpmgr_path_dsi_power_on(pgc->dpmgr_handle, CMDQ_DISABLE);
  3266. if (!primary_display_is_video_mode())
  3267. dpmgr_path_ioctl(pgc->dpmgr_handle, NULL, DDP_DSI_ENABLE_TE, NULL);
  3268. DISPCHECK("[ESD]dsi power reset[end]\n");
  3269. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 8);
  3270. DISPCHECK("[ESD]lcm force init[begin]\n");
  3271. disp_lcm_init(pgc->plcm, 1);
  3272. DISPCHECK("[ESD]lcm force init[end]\n");
  3273. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 9);
  3274. DISPCHECK("[ESD]start dpmgr path[begin]\n");
  3275. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  3276. DISPCHECK("[ESD]start dpmgr path[end]\n");
  3277. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  3278. DISPERR("[ESD]Fatal error, we didn't trigger display path but it's already busy\n");
  3279. ret = -1;
  3280. /* goto done; */
  3281. }
  3282. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 10);
  3283. if (primary_display_is_video_mode()) {
  3284. /* for video mode, we need to force trigger here */
  3285. /* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start */
  3286. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  3287. }
  3288. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 11);
  3289. DISPCHECK("[ESD]start cmdq trigger loop[begin]\n");
  3290. _cmdq_start_trigger_loop();
  3291. DISPCHECK("[ESD]start cmdq trigger loop[end]\n");
  3292. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagPulse, 0, 12);
  3293. done:
  3294. _primary_path_unlock(__func__);
  3295. DISPCHECK("[ESD]ESD recovery end\n");
  3296. MMProfileLogEx(ddp_mmp_get_events()->esd_recovery_t, MMProfileFlagEnd, 0, 0);
  3297. dprec_logger_done(DPREC_LOGGER_ESD_RECOVERY, 0, 0);
  3298. return ret;
  3299. }
  3300. void primary_display_esd_check_enable(int enable)
  3301. {
  3302. /* check if enable ESD check mechanism first */
  3303. if (_is_enable_esd_check() != true) {
  3304. DISPCHECK("[ESD]Checking if ESD enable but ESD check mechanism is not enabled.\n");
  3305. return;
  3306. }
  3307. if (_need_do_esd_check()) {
  3308. if (_need_register_eint() && eint_flag != 2) {
  3309. DISPCHECK("[ESD]Please check DCT setting about GPIO107/EINT107\n");
  3310. return;
  3311. }
  3312. if (enable) {
  3313. DISPPRINT("[ESD]esd check thread wakeup\n");
  3314. atomic_set(&esd_check_task_wakeup, 1);
  3315. wake_up_interruptible(&esd_check_task_wq);
  3316. } else {
  3317. DISPCHECK("[ESD]esd check thread stop\n");
  3318. atomic_set(&esd_check_task_wakeup, 0);
  3319. }
  3320. }
  3321. }
  3322. /******************************************************************************/
  3323. /* ESD CHECK / RECOVERY ---- End */
  3324. /******************************************************************************/
  3325. #define EEEEEEEEEEEEEEEEEEEEEEEEEE
  3326. #ifdef MTK_FB_PULLCLK_ENABLE
  3327. static int primary_display_vdo_pullclk_worker_kthread(void *data)
  3328. {
  3329. int ret = 0;
  3330. int count = 0;
  3331. struct sched_param param = {.sched_priority = 87 }; /* RTPM_PRIO_FB_THREAD */
  3332. sched_setscheduler(current, SCHED_RR, &param);
  3333. while (1) {
  3334. if (count == 0) {
  3335. count++;
  3336. msleep(3000);
  3337. }
  3338. msleep(2000);
  3339. #ifdef DISP_SWITCH_DST_MODE
  3340. _primary_path_switch_dst_lock();
  3341. #endif
  3342. _primary_path_cmd_lock();
  3343. _primary_path_esd_check_lock();
  3344. _primary_path_lock(__func__);
  3345. if (pgc->state == DISP_SLEPT) {
  3346. _primary_path_unlock(__func__);
  3347. _primary_path_esd_check_unlock();
  3348. continue;
  3349. }
  3350. _primary_path_unlock(__func__);
  3351. #ifdef MTK_DISP_IDLE_LP
  3352. _disp_primary_path_dsi_clock_on(0);
  3353. #endif
  3354. _primary_path_lock(__func__);
  3355. cmdqRecCreate(CMDQ_SCENARIO_DISP_ESD_CHECK, &(pgc->cmdq_handle_config_esd));
  3356. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_ALLC_SLOT);
  3357. _primary_path_unlock(__func__);
  3358. #ifdef DISP_DUMP_EVENT_STATUS
  3359. DISP_REG_SET_PA(pgc->cmdq_handle_config_esd, DISP_REG_CMDQ_TOKEN_ID, DISP_MUTEX0_STREAM_EOF_ID);
  3360. cmdqRecBackupRegisterToSlot(pgc->cmdq_handle_config_esd, pgc->event_status, 1
  3361. , DISP_REG_CMDQ_TOKEN_VALUE);
  3362. #endif
  3363. primary_display_esd_cust_bycmdq(1);
  3364. _primary_path_lock(__func__);
  3365. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  3366. spm_enable_sodi(0);
  3367. #endif
  3368. /* reset*/
  3369. cmdqRecReset(pgc->cmdq_handle_config_esd);
  3370. /* wait stream eof first*/
  3371. cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_DISP_RDMA0_EOF);
  3372. cmdqRecWait(pgc->cmdq_handle_config_esd, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  3373. /*stop dsi vdo mode*/
  3374. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_STOP_VDO_MODE);
  3375. /* pull DSI clock lane */
  3376. DSI_sw_clk_trail_cmdq(0, pgc->cmdq_handle_config_esd);
  3377. DSI_manual_enter_HS(pgc->cmdq_handle_config_esd);
  3378. /* start dsi vdo mode*/
  3379. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_START_VDO_MODE);
  3380. /* trigger path*/
  3381. cmdqRecClearEventToken(pgc->cmdq_handle_config_esd, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  3382. if (gEnableSWTrigger == 1)
  3383. DISP_REG_SET(pgc->cmdq_handle_config_esd
  3384. , DISP_REG_CONFIG_MUTEX_EN(DISP_OVL_SEPARATE_MUTEX_ID), 1);
  3385. dpmgr_path_trigger(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ENABLE);
  3386. if (gDumpESDCMD == 1) {
  3387. DISPMSG("esd dump before flush:\n");
  3388. cmdqRecDumpCommand(pgc->cmdq_handle_config_esd);
  3389. }
  3390. ret = cmdqRecFlush(pgc->cmdq_handle_config_esd);
  3391. #if defined(MTK_FB_SODI_SUPPORT) && !defined(CONFIG_FPGA_EARLY_PORTING)
  3392. spm_enable_sodi(1);
  3393. #endif
  3394. _primary_path_unlock(__func__);
  3395. primary_display_esd_cust_bycmdq(0);
  3396. _primary_path_lock(__func__);
  3397. dpmgr_path_build_cmdq(pgc->dpmgr_handle, pgc->cmdq_handle_config_esd, CMDQ_ESD_FREE_SLOT);
  3398. cmdqRecDestroy(pgc->cmdq_handle_config_esd);
  3399. pgc->cmdq_handle_config_esd = NULL;
  3400. _primary_path_unlock(__func__);
  3401. #ifdef MTK_DISP_IDLE_LP
  3402. _disp_primary_path_dsi_clock_off(0);
  3403. #endif
  3404. _primary_path_esd_check_unlock();
  3405. _primary_path_cmd_unlock();
  3406. #ifdef DISP_SWITCH_DST_MODE
  3407. _primary_path_switch_dst_unlock();
  3408. #endif
  3409. if (kthread_should_stop())
  3410. break;
  3411. }
  3412. return 0;
  3413. }
  3414. #endif
  3415. static struct task_struct *primary_path_aal_task;
  3416. void disp_update_trigger_time(void)
  3417. {
  3418. last_primary_trigger_time = sched_clock();
  3419. }
  3420. /* extern unsigned int gResetOVLInAALTrigger; */
  3421. static int _disp_primary_path_check_trigger(void *data)
  3422. {
  3423. cmdqRecHandle handle = NULL;
  3424. #ifndef MTK_FB_CMDQ_DISABLE
  3425. int ret = 0;
  3426. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &handle);
  3427. #endif
  3428. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_TRIGGER);
  3429. while (1) {
  3430. dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_TRIGGER);
  3431. MMProfileLogEx(ddp_mmp_get_events()->primary_display_aalod_trigger,
  3432. MMProfileFlagPulse, 0, 0);
  3433. _primary_path_lock(__func__);
  3434. if (primary_get_state() == DISP_ALIVE) {
  3435. #ifdef MTK_DISP_IDLE_LP
  3436. last_primary_trigger_time = sched_clock();
  3437. _disp_primary_path_exit_idle(__func__, 0);
  3438. #endif
  3439. cmdqRecReset(handle);
  3440. #ifndef MTK_FB_CMDQ_DISABLE
  3441. if (primary_display_is_video_mode())
  3442. cmdqRecWaitNoClear(handle, CMDQ_EVENT_DISP_RDMA0_EOF);
  3443. else
  3444. cmdqRecWaitNoClear(handle, CMDQ_SYNC_TOKEN_STREAM_EOF);
  3445. #endif
  3446. if (gResetOVLInAALTrigger == 1) {
  3447. ovl_reset_by_cmdq(handle, DISP_MODULE_OVL0);
  3448. if (ovl_get_status() != DDP_OVL1_STATUS_SUB)
  3449. ovl_reset_by_cmdq(handle, DISP_MODULE_OVL1);
  3450. }
  3451. _cmdq_set_config_handle_dirty_mira(handle);
  3452. /* disable SODI after set dirty */
  3453. if (primary_display_is_video_mode() == 0 && gDisableSODIForTriggerLoop == 1)
  3454. disp_set_sodi(0, handle);
  3455. _cmdq_flush_config_handle_mira(handle, 0);
  3456. }
  3457. _primary_path_unlock(__func__);
  3458. if (kthread_should_stop())
  3459. break;
  3460. }
  3461. cmdqRecDestroy(handle);
  3462. return 0;
  3463. }
  3464. #define OOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOO
  3465. /* need remove */
  3466. unsigned int cmdqDdpClockOn(uint64_t engineFlag)
  3467. {
  3468. /* DISP_LOG_I("cmdqDdpClockOff\n"); */
  3469. return 0;
  3470. }
  3471. unsigned int cmdqDdpClockOff(uint64_t engineFlag)
  3472. {
  3473. /* DISP_LOG_I("cmdqDdpClockOff\n"); */
  3474. return 0;
  3475. }
  3476. unsigned int cmdqDdpDumpInfo(uint64_t engineFlag, char *pOutBuf, unsigned int bufSize)
  3477. {
  3478. DISPERR("cmdq timeout:%llu\n", engineFlag);
  3479. primary_display_diagnose();
  3480. /* DISP_LOG_I("cmdqDdpDumpInfo\n"); */
  3481. if (primary_display_is_decouple_mode()) {
  3482. ddp_dump_analysis(DISP_MODULE_OVL0);
  3483. #if defined(OVL_CASCADE_SUPPORT)
  3484. ddp_dump_analysis(DISP_MODULE_OVL1);
  3485. #endif
  3486. }
  3487. ddp_dump_analysis(DISP_MODULE_WDMA0);
  3488. /* try to set event by CPU to avoid blocking auto test such as Monkey/MTBF */
  3489. cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_STREAM_EOF);
  3490. cmdqCoreSetEvent(CMDQ_EVENT_DISP_RDMA0_EOF);
  3491. return 0;
  3492. }
  3493. unsigned int cmdqDdpResetEng(uint64_t engineFlag)
  3494. {
  3495. /* DISP_LOG_I("cmdqDdpResetEng\n"); */
  3496. return 0;
  3497. }
  3498. /* TODO: these 2 functions should be splited into another file */
  3499. unsigned int display_path_idle_cnt = 0;
  3500. #if defined(MTK_FB_SODI_SUPPORT)
  3501. static void _RDMA0_INTERNAL_IRQ_Handler(DISP_MODULE_ENUM module, unsigned int param)
  3502. {
  3503. if (!_is_decouple_mode(pgc->session_mode) && param & 0x2) {
  3504. /* RDMA Start */
  3505. display_path_idle_cnt++;
  3506. #if !defined(CONFIG_FPGA_EARLY_PORTING) && !defined(FORCE_SODI_CG_MODE)
  3507. spm_sodi_mempll_pwr_mode(1);
  3508. #endif
  3509. MMProfileLogEx(ddp_mmp_get_events()->sodi_disable, MMProfileFlagPulse, 0, 0);
  3510. }
  3511. if (param & 0x4) {
  3512. /* RDMA Done */
  3513. display_path_idle_cnt--;
  3514. if (display_path_idle_cnt == 0) {
  3515. #if !defined(CONFIG_FPGA_EARLY_PORTING) && !defined(FORCE_SODI_CG_MODE)
  3516. spm_sodi_mempll_pwr_mode(0);
  3517. #endif
  3518. MMProfileLogEx(ddp_mmp_get_events()->sodi_enable, MMProfileFlagPulse, 0, 0);
  3519. }
  3520. }
  3521. }
  3522. static void _WDMA0_INTERNAL_IRQ_Handler(DISP_MODULE_ENUM module, unsigned int param)
  3523. {
  3524. if (param & 0x1) {
  3525. /* WDMA Done */
  3526. dpmgr_wdma_path_force_power_off();
  3527. if (!primary_display_is_video_mode()) {
  3528. display_path_idle_cnt--;
  3529. if (display_path_idle_cnt == 0) {
  3530. #if !defined(CONFIG_FPGA_EARLY_PORTING) && !defined(FORCE_SODI_CG_MODE)
  3531. spm_sodi_mempll_pwr_mode(0);
  3532. #endif
  3533. }
  3534. }
  3535. }
  3536. }
  3537. static void _MUTEX_INTERNAL_IRQ_Handler(DISP_MODULE_ENUM module, unsigned int param)
  3538. {
  3539. if (param & 0x1) { /* RDMA-->DSI SOF */
  3540. display_path_idle_cnt++;
  3541. #if !defined(CONFIG_FPGA_EARLY_PORTING) && !defined(FORCE_SODI_CG_MODE)
  3542. spm_sodi_mempll_pwr_mode(1);
  3543. #endif
  3544. }
  3545. if (param & 0x2) { /* OVL->WDMA SOF */
  3546. display_path_idle_cnt++;
  3547. #if !defined(CONFIG_FPGA_EARLY_PORTING) && !defined(FORCE_SODI_CG_MODE)
  3548. spm_sodi_mempll_pwr_mode(1);
  3549. #endif
  3550. }
  3551. }
  3552. #endif
  3553. void primary_display_sodi_rule_init(void)
  3554. {
  3555. #if defined(MTK_FB_SODI_DEFEATURE)
  3556. if ((primary_display_get_width() > 800) && (primary_display_is_video_mode() == 1)) {
  3557. #ifndef CONFIG_FPGA_EARLY_PORTING
  3558. defeature_soidle_by_display();
  3559. #endif
  3560. DISPMSG("SODI defeatured!\n");
  3561. return;
  3562. }
  3563. #endif
  3564. #if defined(MTK_FB_SODI_SUPPORT)
  3565. /* if ((primary_display_mode == DECOUPLE_MODE) && primary_display_is_video_mode()) */
  3566. if (gEnableSODIControl == 0 && primary_display_is_video_mode() == 1) {
  3567. #ifndef CONFIG_FPGA_EARLY_PORTING
  3568. spm_enable_sodi(0);
  3569. #endif
  3570. DISPMSG("SODI disabled!\n");
  3571. return;
  3572. }
  3573. #ifndef CONFIG_FPGA_EARLY_PORTING
  3574. spm_enable_sodi(1);
  3575. #endif
  3576. if (primary_display_is_video_mode()) {
  3577. /* if switch to video mode, should de-register callback */
  3578. disp_unregister_module_irq_callback(DISP_MODULE_RDMA0, _RDMA0_INTERNAL_IRQ_Handler);
  3579. #ifdef WDMA_PATH_CLOCK_DYNAMIC_SWITCH
  3580. disp_register_module_irq_callback(DISP_MODULE_WDMA0, _WDMA0_INTERNAL_IRQ_Handler);
  3581. #endif
  3582. /* spm_sodi_mempll_pwr_mode(0); */
  3583. } else {
  3584. disp_register_module_irq_callback(DISP_MODULE_RDMA0, _RDMA0_INTERNAL_IRQ_Handler);
  3585. #ifdef WDMA_PATH_CLOCK_DYNAMIC_SWITCH
  3586. disp_register_module_irq_callback(DISP_MODULE_WDMA0, _WDMA0_INTERNAL_IRQ_Handler);
  3587. #endif
  3588. if (_is_decouple_mode(pgc->session_mode)) {
  3589. disp_register_module_irq_callback(DISP_MODULE_MUTEX, _MUTEX_INTERNAL_IRQ_Handler);
  3590. disp_register_module_irq_callback(DISP_MODULE_WDMA0, _WDMA0_INTERNAL_IRQ_Handler);
  3591. }
  3592. }
  3593. #endif
  3594. }
  3595. /* extern int dfo_query(const char *s, unsigned long *v); */
  3596. int primary_display_change_lcm_resolution(unsigned int width, unsigned int height)
  3597. {
  3598. if (pgc->plcm) {
  3599. DISPMSG("LCM Resolution will be changed, original: %dx%d, now: %dx%d\n",
  3600. pgc->plcm->params->width, pgc->plcm->params->height, width, height);
  3601. /* align with 4 is the minimal check, to ensure we can boot up into kernel,
  3602. * and could modify dfo setting again using meta tool
  3603. */
  3604. /* otherwise we will have a panic in lk(root cause unknown). */
  3605. if (width > pgc->plcm->params->width || height > pgc->plcm->params->height ||
  3606. width == 0 || height == 0 || width % 4 || height % 2) {
  3607. DISPERR("Invalid resolution: %dx%d\n", width, height);
  3608. return -1;
  3609. }
  3610. if (primary_display_is_video_mode()) {
  3611. DISPERR("Warning!!!Video Mode can't support multiple resolution!\n");
  3612. return -1;
  3613. }
  3614. pgc->plcm->params->width = width;
  3615. pgc->plcm->params->height = height;
  3616. return 0;
  3617. } else {
  3618. return -1;
  3619. }
  3620. }
  3621. static void primary_display_frame_update_irq_callback(DISP_MODULE_ENUM module, unsigned int param)
  3622. {
  3623. /* if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE) */
  3624. /* return; */
  3625. if (module == DISP_MODULE_RDMA0) {
  3626. if (param & 0x20) { /* rdma0 frame start */
  3627. if (pgc->session_id > 0)
  3628. update_frm_seq_info(ddp_ovl_get_cur_addr(1, 0), 0, 1, FRM_START);
  3629. }
  3630. if (param & 0x4) { /* rdma0 frame done */
  3631. atomic_set(&primary_display_frame_update_event, 1);
  3632. wake_up_interruptible(&primary_display_frame_update_wq);
  3633. }
  3634. }
  3635. if ((module == DISP_MODULE_OVL0) && (_is_decouple_mode(pgc->session_mode) == 0)) {
  3636. if (param & 0x2) { /* ov0 frame done */
  3637. atomic_set(&primary_display_frame_update_event, 1);
  3638. wake_up_interruptible(&primary_display_frame_update_wq);
  3639. }
  3640. }
  3641. #if defined(CONFIG_TRUSTONIC_TEE_SUPPORT) && defined(CONFIG_MTK_SEC_VIDEO_PATH_SUPPORT)
  3642. /* In TEE, we have to protect WDMA registers, so we can't enable WDMA interrupt */
  3643. /* here we use ovl frame done interrupt instead */
  3644. if ((module == DISP_MODULE_OVL0) && (_is_decouple_mode(pgc->session_mode) == 1)) {
  3645. if (param & 0x2) {/* OVL0 frame done */
  3646. atomic_set(&decouple_fence_release_event, 1);
  3647. wake_up_interruptible(&decouple_fence_release_wq);
  3648. }
  3649. }
  3650. #else
  3651. if ((module == DISP_MODULE_WDMA0) && (_is_decouple_mode(pgc->session_mode) == 1)) {
  3652. if (param & 0x1) {/* wdma0 frame done */
  3653. atomic_set(&decouple_fence_release_event, 1);
  3654. wake_up_interruptible(&decouple_fence_release_wq);
  3655. }
  3656. }
  3657. #endif
  3658. }
  3659. static int primary_display_frame_update_kthread(void *data)
  3660. {
  3661. struct sched_param param = {.sched_priority = 94 }; /* RTPM_PRIO_SCRN_UPDATE */
  3662. sched_setscheduler(current, SCHED_RR, &param);
  3663. for (;;) {
  3664. wait_event_interruptible(primary_display_frame_update_wq,
  3665. atomic_read(&primary_display_frame_update_event));
  3666. atomic_set(&primary_display_frame_update_event, 0);
  3667. if (pgc->session_id > 0)
  3668. update_frm_seq_info(0, 0, 0, FRM_END);
  3669. if (kthread_should_stop())
  3670. break;
  3671. }
  3672. return 0;
  3673. }
  3674. /* defined but not used */
  3675. /* static struct task_struct *if_fence_release_worker_task = NULL; */
  3676. #if 0
  3677. /**
  3678. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  3679. */
  3680. static int _if_fence_release_worker_thread(void *data)
  3681. {
  3682. unsigned int session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  3683. struct sched_param param = { .sched_priority = RTPM_PRIO_SCRN_UPDATE };
  3684. sched_setscheduler(current, SCHED_RR, &param);
  3685. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  3686. while (1) {
  3687. dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  3688. if (_is_mirror_mode(pgc->session_mode)) {
  3689. int fence_idx, subtractor, layer;
  3690. layer = disp_sync_get_output_interface_timeline_id();
  3691. cmdqBackupReadSlot(pgc->cur_config_fence, layer, &fence_idx);
  3692. cmdqBackupReadSlot(pgc->subtractor_when_free, layer, &subtractor);
  3693. mtkfb_release_fence(session_id, layer, fence_idx-1);
  3694. }
  3695. if (kthread_should_stop())
  3696. break;
  3697. }
  3698. return 0;
  3699. }
  3700. #endif
  3701. /* defined but not used. */
  3702. /* static struct task_struct *ovl2mem_fence_release_worker_task = NULL; */
  3703. #if 0
  3704. /**
  3705. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  3706. */
  3707. static int _ovl2mem_fence_release_worker_thread(void *data)
  3708. {
  3709. unsigned int session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  3710. struct sched_param param = { .sched_priority = RTPM_PRIO_SCRN_UPDATE };
  3711. sched_setscheduler(current, SCHED_RR, &param);
  3712. dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE);
  3713. while (1) {
  3714. /* it's not good to use FRAME_COMPLETE here, because when CPU read wdma addr,
  3715. * maybe it's already changed by next request,
  3716. * but luckly currently we will wait rdma frame done after wdma done(in CMDQ), so it's safe now
  3717. */
  3718. dpmgr_wait_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE);
  3719. if (_is_mirror_mode(pgc->session_mode)) {
  3720. int fence_idx, subtractor, layer;
  3721. layer = disp_sync_get_output_timeline_id();
  3722. cmdqBackupReadSlot(pgc->cur_config_fence, layer, &fence_idx);
  3723. cmdqBackupReadSlot(pgc->subtractor_when_free, layer, &subtractor);
  3724. mtkfb_release_fence(session_id, layer, fence_idx);
  3725. }
  3726. if (kthread_should_stop())
  3727. break;
  3728. }
  3729. return 0;
  3730. }
  3731. #endif
  3732. /* defined but not used. */
  3733. /* static struct task_struct *fence_release_worker_task = NULL; */
  3734. /* extern unsigned int ddp_ovl_get_cur_addr(bool rdma_mode, int layerid); */
  3735. static void _wdma_fence_release_callback(uint32_t userdata)
  3736. {
  3737. int fence_idx, layer;
  3738. layer = disp_sync_get_output_timeline_id();
  3739. cmdqBackupReadSlot(pgc->cur_config_fence, layer, &fence_idx);
  3740. mtkfb_release_fence(primary_session_id, layer, fence_idx);
  3741. MMProfileLogEx(ddp_mmp_get_events()->primary_wdma_fence_release, MMProfileFlagPulse, layer,
  3742. fence_idx);
  3743. }
  3744. static int _Interface_fence_release_callback(uint32_t userdata)
  3745. {
  3746. int layer = disp_sync_get_output_interface_timeline_id();
  3747. if (userdata > 0) {
  3748. mtkfb_release_fence(primary_session_id, layer, userdata);
  3749. MMProfileLogEx(ddp_mmp_get_events()->primary_wdma_fence_release, MMProfileFlagPulse,
  3750. layer, userdata);
  3751. }
  3752. return 0;
  3753. }
  3754. static int _ovl_ext_fence_release_callback(uint32_t userdata)
  3755. {
  3756. int i = 0;
  3757. int ret = 0;
  3758. int fence_idx, layer;
  3759. MMProfileLogEx(ddp_mmp_get_events()->session_release, MMProfileFlagStart, 1, userdata);
  3760. #ifndef MTK_FB_CMDQ_DISABLE
  3761. for (i = 0; i < PRIMARY_DISPLAY_SESSION_LAYER_COUNT; i++) {
  3762. int fence_idx = 0;
  3763. int subtractor = 0;
  3764. if (i == primary_display_get_option("ASSERT_LAYER") && is_DAL_Enabled()) {
  3765. mtkfb_release_layer_fence(ext_session_id, i);
  3766. } else {
  3767. cmdqBackupReadSlot(pgc->cur_mem_config_fence, i, &fence_idx);
  3768. cmdqBackupReadSlot(pgc->mem_subtractor_when_free, i, &subtractor);
  3769. mtkfb_release_fence(ext_session_id, i, fence_idx);
  3770. }
  3771. MMProfileLogEx(ddp_mmp_get_events()->primary_ovl_fence_release, MMProfileFlagPulse, i, fence_idx);
  3772. }
  3773. #endif
  3774. if (userdata == DISP_SESSION_MEMORY) {
  3775. #ifndef MTK_FB_CMDQ_DISABLE
  3776. layer = disp_sync_get_output_timeline_id();
  3777. cmdqBackupReadSlot(pgc->cur_mem_config_fence, layer, &fence_idx);
  3778. mtkfb_release_fence(ext_session_id, layer, fence_idx);
  3779. #endif
  3780. }
  3781. return ret;
  3782. }
  3783. static int _ovl_fence_release_callback(uint32_t userdata)
  3784. {
  3785. int i = 0;
  3786. unsigned int addr = 0;
  3787. int ret = 0;
  3788. unsigned int dsi_state[10];
  3789. unsigned int rdma_state[50];
  3790. MMProfileLogEx(ddp_mmp_get_events()->session_release, MMProfileFlagStart, 1, userdata);
  3791. /* releaes OVL1 when primary setting */
  3792. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY_RELEASED)
  3793. dpmgr_set_ovl1_status(DDP_OVL1_STATUS_SUB);
  3794. else if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY_DISABLE)
  3795. dpmgr_set_ovl1_status(DDP_OVL1_STATUS_IDLE);
  3796. #ifndef MTK_FB_CMDQ_DISABLE
  3797. for (i = 0; i < PRIMARY_DISPLAY_SESSION_LAYER_COUNT; i++) {
  3798. int fence_idx = 0;
  3799. int subtractor = 0;
  3800. if (i == primary_display_get_option("ASSERT_LAYER") && is_DAL_Enabled()) {
  3801. mtkfb_release_layer_fence(primary_session_id, i);
  3802. } else {
  3803. cmdqBackupReadSlot(pgc->cur_config_fence, i, &fence_idx);
  3804. cmdqBackupReadSlot(pgc->subtractor_when_free, i, &subtractor);
  3805. mtkfb_release_fence(primary_session_id, i, fence_idx - subtractor);
  3806. }
  3807. MMProfileLogEx(ddp_mmp_get_events()->primary_ovl_fence_release, MMProfileFlagPulse,
  3808. i, fence_idx - subtractor);
  3809. }
  3810. if (primary_display_is_video_mode() == 1) {
  3811. unsigned int ovl_status[2];
  3812. cmdqBackupReadSlot(pgc->ovl_status_info, 0, &ovl_status[0]);
  3813. cmdqBackupReadSlot(pgc->ovl_status_info, 1, &ovl_status[1]);
  3814. DISPDBG("ovl fsm state:0x%x\n", ovl_status[0]);
  3815. if ((ovl_status[1] & 0x1) != OVL_STATUS_IDLE) {
  3816. DISPERR("disp ovl status error, 0x%x, 0x%x\n", ovl_status[0],
  3817. ovl_status[1]);
  3818. /* dump cmdq cmd here */
  3819. }
  3820. }
  3821. #endif
  3822. /* backup DSI state register to slot */
  3823. if (gEnableDSIStateCheck == 1) {
  3824. for (i = 0; i < 10; i++) {
  3825. cmdqBackupReadSlot(pgc->dsi_state_info, i, &dsi_state[i]);
  3826. if ((dsi_state[i] & 0x2) == 0x2) {
  3827. DISPERR("disp DSI_STATE error, %d, 0x%x\n", i, dsi_state[i]);
  3828. cmdqBackupReadSlot(pgc->rdma_state_info, i * 5 + 0,
  3829. &rdma_state[i * 5 + 0]);
  3830. cmdqBackupReadSlot(pgc->rdma_state_info, i * 5 + 1,
  3831. &rdma_state[i * 5 + 1]);
  3832. cmdqBackupReadSlot(pgc->rdma_state_info, i * 5 + 2,
  3833. &rdma_state[i * 5 + 2]);
  3834. cmdqBackupReadSlot(pgc->rdma_state_info, i * 5 + 3,
  3835. &rdma_state[i * 5 + 3]);
  3836. cmdqBackupReadSlot(pgc->rdma_state_info, i * 5 + 4,
  3837. &rdma_state[i * 5 + 4]);
  3838. DISPERR("disp RDMA_STATE:%d, irq=0x%x, in/out=(%d,%d,%d,%d)\n",
  3839. i, rdma_state[i * 5 + 0], rdma_state[i * 5 + 1],
  3840. rdma_state[i * 5 + 2], rdma_state[i * 5 + 3], rdma_state[i * 5 + 4]);
  3841. } else {
  3842. DISPERR("disp DSI_STATE pass, %d, 0x%x\n", i, dsi_state[i]);
  3843. }
  3844. }
  3845. }
  3846. addr = ddp_ovl_get_cur_addr(!_should_config_ovl_input(), 0);
  3847. if ((_is_decouple_mode(pgc->session_mode) == 0))
  3848. update_frm_seq_info(addr, 0, 2, FRM_START);
  3849. #ifndef MTK_FB_CMDQ_DISABLE
  3850. if (userdata == 5) {
  3851. int fence_idx, subtractor, layer;
  3852. layer = disp_sync_get_output_timeline_id();
  3853. cmdqBackupReadSlot(pgc->cur_config_fence, layer, &fence_idx);
  3854. cmdqBackupReadSlot(pgc->subtractor_when_free, layer, &subtractor);
  3855. mtkfb_release_fence(primary_session_id, layer, fence_idx);
  3856. }
  3857. /* async callback, need to check if it is still decouple */
  3858. if (_is_decouple_mode(pgc->session_mode) && !_is_mirror_mode(pgc->session_mode)
  3859. && (userdata == DISP_SESSION_DECOUPLE_MODE || userdata == 5)) {
  3860. static cmdqRecHandle cmdq_handle;
  3861. unsigned int rdma_pitch_sec, rdma_fmt;
  3862. if (primary_get_state() != DISP_ALIVE)
  3863. return 0;
  3864. if (cmdq_handle == NULL)
  3865. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  3866. if (ret == 0) {
  3867. cmdqBackupReadSlot(pgc->rdma_buff_info, 1, &(rdma_pitch_sec));
  3868. rdma_pitch_sec = rdma_pitch_sec >> 30;
  3869. if (rdma_pitch_sec == DISP_NORMAL_BUFFER)
  3870. _primary_path_lock(__func__);
  3871. cmdqRecReset(cmdq_handle);
  3872. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  3873. cmdqBackupReadSlot(pgc->rdma_buff_info, 0, &addr);
  3874. decouple_rdma_config.address = addr;
  3875. /*rdma pitch only use bit[15..0], we use bit[31:30] to store secure information */
  3876. decouple_rdma_config.security = rdma_pitch_sec;
  3877. _config_rdma_input_data(&decouple_rdma_config, pgc->dpmgr_handle,
  3878. cmdq_handle);
  3879. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  3880. cmdqRecFlushAsyncCallback(cmdq_handle, NULL, 0);
  3881. cmdqBackupReadSlot(pgc->rdma_buff_info, 2, &(rdma_fmt));
  3882. decouple_rdma_config.inputFormat = rdma_fmt;
  3883. if (rdma_pitch_sec == DISP_NORMAL_BUFFER)
  3884. _primary_path_unlock(__func__);
  3885. MMProfileLogEx(ddp_mmp_get_events()->primary_rdma_config,
  3886. MMProfileFlagPulse, 0, decouple_rdma_config.address);
  3887. /* cmdqRecDestroy(cmdq_handle); */
  3888. } else {
  3889. /* ret = -1; */
  3890. DISPERR("fail to create cmdq\n");
  3891. }
  3892. }
  3893. #endif
  3894. return ret;
  3895. }
  3896. static int decouple_fence_release_kthread(void *data)
  3897. {
  3898. int interface_fence = 0;
  3899. int layer = 0;
  3900. int ret = 0;
  3901. struct sched_param param = {.sched_priority = 94 }; /* RTPM_PRIO_SCRN_UPDATE */
  3902. sched_setscheduler(current, SCHED_RR, &param);
  3903. for (;;) {
  3904. wait_event_interruptible(decouple_fence_release_wq,
  3905. atomic_read(&decouple_fence_release_event));
  3906. atomic_set(&decouple_fence_release_event, 0);
  3907. /* async callback, need to check if it is still decouple */
  3908. _primary_path_lock(__func__);
  3909. if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE
  3910. && pgc->state != DISP_SLEPT) {
  3911. static cmdqRecHandle cmdq_handle;
  3912. unsigned int rdma_pitch_sec, rdma_fmt;
  3913. if (cmdq_handle == NULL)
  3914. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  3915. if (ret == 0) {
  3916. RDMA_CONFIG_STRUCT tmpConfig = decouple_rdma_config;
  3917. cmdqRecReset(cmdq_handle);
  3918. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  3919. cmdqBackupReadSlot(pgc->rdma_buff_info, 0,
  3920. (uint32_t *)&(tmpConfig.address));
  3921. /*rdma pitch only use bit[15..0], we use bit[31:30] to store secure information */
  3922. cmdqBackupReadSlot(pgc->rdma_buff_info, 1, &(rdma_pitch_sec));
  3923. tmpConfig.pitch = rdma_pitch_sec & ~(3 << 30);
  3924. tmpConfig.security = rdma_pitch_sec >> 30;
  3925. cmdqBackupReadSlot(pgc->rdma_buff_info, 2, &(rdma_fmt));
  3926. tmpConfig.inputFormat = rdma_fmt;
  3927. _config_rdma_input_data(&tmpConfig, pgc->dpmgr_handle, cmdq_handle);
  3928. layer = disp_sync_get_output_timeline_id();
  3929. cmdqBackupReadSlot(pgc->cur_config_fence, layer, &interface_fence);
  3930. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  3931. cmdqRecFlushAsyncCallback(cmdq_handle,
  3932. (CmdqAsyncFlushCB)_Interface_fence_release_callback,
  3933. interface_fence > 1 ? interface_fence - 1 : 0);
  3934. MMProfileLogEx(ddp_mmp_get_events()->primary_rdma_config,
  3935. MMProfileFlagPulse, interface_fence,
  3936. decouple_rdma_config.address);
  3937. /* dump rdma input if enabled */
  3938. dprec_mmp_dump_rdma_layer(&tmpConfig, 0);
  3939. /* cmdqRecDestroy(cmdq_handle); */
  3940. } else {
  3941. DISPERR("fail to create cmdq\n");
  3942. }
  3943. }
  3944. _primary_path_unlock(__func__);
  3945. if (kthread_should_stop())
  3946. break;
  3947. }
  3948. return 0;
  3949. }
  3950. static int _olv_wdma_fence_release_callback(uint32_t userdata)
  3951. {
  3952. _ovl_fence_release_callback(userdata);
  3953. _wdma_fence_release_callback(userdata);
  3954. return 0;
  3955. }
  3956. #if 0
  3957. /**
  3958. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  3959. */
  3960. static int _fence_release_worker_thread(void *data)
  3961. {
  3962. int i = 0;
  3963. unsigned int addr = 0;
  3964. unsigned int session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  3965. struct sched_param param = { .sched_priority = RTPM_PRIO_SCRN_UPDATE };
  3966. sched_setscheduler(current, SCHED_RR, &param);
  3967. if (!_is_decouple_mode(pgc->session_mode))
  3968. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  3969. else
  3970. dpmgr_enable_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START);
  3971. while (1) {
  3972. if (!_is_decouple_mode(pgc->session_mode))
  3973. dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  3974. else
  3975. dpmgr_wait_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START);
  3976. if (is_hwc_enabled == 0)
  3977. continue;
  3978. /* releaes OVL1 when primary setting */
  3979. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY_RELEASED) {
  3980. ovl_set_status(DDP_OVL1_STATUS_SUB);
  3981. wake_up_interruptible(&ovl1_wait_queue);
  3982. } else if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY_DISABLE) {
  3983. ovl_set_status(DDP_OVL1_STATUS_IDLE);
  3984. wake_up_interruptible(&ovl1_wait_queue);
  3985. }
  3986. for (i = 0; i < PRIMARY_DISPLAY_SESSION_LAYER_COUNT; i++) {
  3987. int fence_idx, subtractor;
  3988. if (i == primary_display_get_option("ASSERT_LAYER") && is_DAL_Enabled()) {
  3989. mtkfb_release_layer_fence(session_id, i);
  3990. } else {
  3991. cmdqBackupReadSlot(pgc->cur_config_fence, i, &fence_idx);
  3992. cmdqBackupReadSlot(pgc->subtractor_when_free, i, &subtractor);
  3993. mtkfb_release_fence(session_id, i, fence_idx-subtractor);
  3994. }
  3995. }
  3996. addr = ddp_ovl_get_cur_addr(!_should_config_ovl_input(), 0);
  3997. if ((_is_decouple_mode(pgc->session_mode) == 0))
  3998. update_frm_seq_info(addr, 0, 2, FRM_START);
  3999. MMProfileLogEx(ddp_mmp_get_events()->session_release, MMProfileFlagEnd, 0, 0);
  4000. if (kthread_should_stop())
  4001. break;
  4002. }
  4003. return 0;
  4004. }
  4005. #endif
  4006. static struct task_struct *present_fence_release_worker_task;
  4007. static int _present_fence_release_worker_thread(void *data)
  4008. {
  4009. int ret = 0;
  4010. disp_sync_info *layer_info = NULL;
  4011. struct sched_param param = {.sched_priority = 87 }; /* RTPM_PRIO_FB_THREAD */
  4012. sched_setscheduler(current, SCHED_RR, &param);
  4013. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  4014. while (1) {
  4015. ret = dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  4016. /* release present fence in vsync callback */
  4017. {
  4018. /* extern disp_sync_info *_get_sync_info(unsigned int session_id,
  4019. unsigned int timeline_id);
  4020. */
  4021. int fence_increment = 0;
  4022. /* if session not created, do not release present fence */
  4023. if (pgc->session_id == 0) {
  4024. MMProfileLogEx(ddp_mmp_get_events()->present_fence_release,
  4025. MMProfileFlagPulse, -1, 0x4a4a4a4a);
  4026. /* DISPDBG("_get_sync_info fail in present_fence_release thread\n"); */
  4027. continue;
  4028. }
  4029. layer_info = _get_sync_info(pgc->session_id, disp_sync_get_present_timeline_id());
  4030. if (layer_info == NULL) {
  4031. MMProfileLogEx(ddp_mmp_get_events()->present_fence_release,
  4032. MMProfileFlagPulse, -1, 0x5a5a5a5a);
  4033. /* DISPERR("_get_sync_info fail in present_fence_release thread\n"); */
  4034. continue;
  4035. }
  4036. _primary_path_lock(__func__);
  4037. fence_increment = gPresentFenceIndex - layer_info->timeline->value;
  4038. if (fence_increment > 0) {
  4039. timeline_inc(layer_info->timeline, fence_increment);
  4040. MMProfileLogEx(ddp_mmp_get_events()->present_fence_release,
  4041. MMProfileFlagPulse, gPresentFenceIndex,
  4042. fence_increment);
  4043. }
  4044. _primary_path_unlock(__func__);
  4045. /* DISPPR_FENCE("RPF/%d/%d\n",
  4046. gPresentFenceIndex, gPresentFenceIndex-layer_info->timeline->value);
  4047. */
  4048. }
  4049. }
  4050. return 0;
  4051. }
  4052. int primary_display_capture_framebuffer_wdma(void *data)
  4053. {
  4054. int ret = 0;
  4055. cmdqRecHandle cmdq_handle = NULL;
  4056. disp_ddp_path_config *pconfig = NULL;
  4057. m4u_client_t *m4uClient = NULL;
  4058. unsigned int w_xres = primary_display_get_width();
  4059. unsigned int h_yres = primary_display_get_height();
  4060. unsigned int pixel_byte = primary_display_get_dc_bpp() / 8; /* bpp is either 32 or 16, can not be other value */
  4061. int buffer_size = h_yres * w_xres * pixel_byte;
  4062. int buf_idx = 0;
  4063. unsigned int mva[2] = { 0, 0 };
  4064. void *va[2] = { NULL, NULL };
  4065. unsigned int format = eRGBA8888;
  4066. struct sched_param param = {.sched_priority = 87 }; /* RTPM_PRIO_FB_THREAD */
  4067. sched_setscheduler(current, SCHED_RR, &param);
  4068. va[0] = vmalloc(buffer_size);
  4069. if (va[0] == NULL) {
  4070. DISPCHECK("wdma dump:Fail to alloc vmalloc 0\n");
  4071. ret = -1;
  4072. goto out;
  4073. }
  4074. va[1] = vmalloc(buffer_size);
  4075. if (va[1] == NULL) {
  4076. DISPCHECK("wdma dump:Fail to alloc vmalloc 1\n");
  4077. ret = -1;
  4078. goto out;
  4079. }
  4080. m4uClient = m4u_create_client();
  4081. if (m4uClient == NULL) {
  4082. DISPCHECK("wdma dump:Fail to alloc m4uClient=0x%p\n", m4uClient);
  4083. ret = -1;
  4084. goto out;
  4085. }
  4086. ret = m4u_alloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, (unsigned long)va[0], NULL, buffer_size,
  4087. M4U_PROT_READ | M4U_PROT_WRITE, 0, (unsigned int *)&mva[0]);
  4088. if (ret != 0) {
  4089. DISPCHECK("wdma dump::Fail to allocate mva 0\n");
  4090. ret = -1;
  4091. goto out;
  4092. }
  4093. ret = m4u_alloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, (unsigned long)va[1], NULL, buffer_size,
  4094. M4U_PROT_READ | M4U_PROT_WRITE, 0, (unsigned int *)&mva[1]);
  4095. if (ret != 0) {
  4096. DISPCHECK("wdma dump::Fail to allocate mva 1\n");
  4097. ret = -1;
  4098. goto out;
  4099. }
  4100. if (primary_display_cmdq_enabled()) {
  4101. /*create config thread */
  4102. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  4103. if (ret != 0) {
  4104. DISPCHECK("wdma dump:Fail to create primary cmdq handle for capture\n");
  4105. ret = -1;
  4106. goto out;
  4107. }
  4108. dpmgr_path_memout_clock(pgc->dpmgr_handle, 1);
  4109. dpmgr_enable_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  4110. cmdqRecReset(cmdq_handle);
  4111. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  4112. _primary_path_lock(__func__);
  4113. dpmgr_path_add_memout(pgc->dpmgr_handle, ENGINE_OVL0, cmdq_handle);
  4114. _primary_path_unlock(__func__);
  4115. while (primary_dump_wdma) {
  4116. ret = m4u_cache_sync(m4uClient, M4U_PORT_DISP_WDMA0, (unsigned long)va[buf_idx],
  4117. buffer_size, mva[buf_idx], M4U_CACHE_FLUSH_BY_RANGE);
  4118. _primary_path_lock(__func__);
  4119. pconfig = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  4120. pconfig->wdma_dirty = 1;
  4121. pconfig->wdma_config.dstAddress = mva[buf_idx];
  4122. pconfig->wdma_config.srcHeight = h_yres;
  4123. pconfig->wdma_config.srcWidth = w_xres;
  4124. pconfig->wdma_config.clipX = 0;
  4125. pconfig->wdma_config.clipY = 0;
  4126. pconfig->wdma_config.clipHeight = h_yres;
  4127. pconfig->wdma_config.clipWidth = w_xres;
  4128. pconfig->wdma_config.outputFormat = format;
  4129. pconfig->wdma_config.useSpecifiedAlpha = 1;
  4130. pconfig->wdma_config.alpha = 0xFF;
  4131. pconfig->wdma_config.dstPitch = w_xres * DP_COLOR_BITS_PER_PIXEL(format) / 8;
  4132. ret = dpmgr_path_config(pgc->dpmgr_handle, pconfig, cmdq_handle);
  4133. pconfig->wdma_dirty = 0;
  4134. _primary_path_unlock(__func__);
  4135. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  4136. _cmdq_flush_config_handle_mira(cmdq_handle, 0);
  4137. ret = dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START);
  4138. /* pconfig->wdma_config.dstAddress = va[buf_idx++]; */
  4139. /* DISPCHECK("capture wdma\n"); */
  4140. dprec_mmp_dump_wdma_layer(&pconfig->wdma_config, 0);
  4141. buf_idx = buf_idx % 2;
  4142. cmdqRecReset(cmdq_handle);
  4143. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  4144. }
  4145. _primary_path_lock(__func__);
  4146. dpmgr_path_remove_memout(pgc->dpmgr_handle, cmdq_handle);
  4147. _primary_path_unlock(__func__);
  4148. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  4149. /* flush remove memory to cmdq */
  4150. _cmdq_flush_config_handle_mira(cmdq_handle, 1);
  4151. DISPMSG("wdma dump: Flush remove memout\n");
  4152. dpmgr_path_memout_clock(pgc->dpmgr_handle, 0);
  4153. }
  4154. out:
  4155. cmdqRecDestroy(cmdq_handle);
  4156. if (mva[0] > 0)
  4157. m4u_dealloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, mva[0]);
  4158. if (mva[1] > 0)
  4159. m4u_dealloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, mva[1]);
  4160. if (va[0] != NULL)
  4161. vfree(va[0]);
  4162. if (va[1] != NULL)
  4163. vfree(va[1]);
  4164. if (m4uClient != 0)
  4165. m4u_destroy_client(m4uClient);
  4166. DISPMSG("wdma dump:end\n");
  4167. return ret;
  4168. }
  4169. int primary_display_switch_wdma_dump(int on)
  4170. {
  4171. if (on && (!primary_dump_wdma)) {
  4172. primary_dump_wdma = 1;
  4173. primary_display_wdma_out = kthread_create(primary_display_capture_framebuffer_wdma, NULL,
  4174. "display_wdma_out");
  4175. wake_up_process(primary_display_wdma_out);
  4176. } else {
  4177. primary_dump_wdma = 0;
  4178. }
  4179. return 0;
  4180. }
  4181. #define xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
  4182. int primary_display_set_frame_buffer_address(unsigned long va, unsigned long mva)
  4183. {
  4184. /* DISPMSG("framebuffer va %lu, mva %lu\n", va, mva); */
  4185. pgc->framebuffer_va = va;
  4186. pgc->framebuffer_mva = mva;
  4187. /*
  4188. int frame_buffer_size = ALIGN_TO(DISP_GetScreenWidth(), MTK_FB_ALIGNMENT) *
  4189. ALIGN_TO(DISP_GetScreenHeight(), MTK_FB_ALIGNMENT) * 4;
  4190. unsigned long dim_layer_va = va + 2*frame_buffer_size;
  4191. dim_layer_mva = mva + 2*frame_buffer_size;
  4192. memset_io(dim_layer_va, 0, frame_buffer_size);
  4193. */
  4194. return 0;
  4195. }
  4196. unsigned long primary_display_get_frame_buffer_mva_address(void)
  4197. {
  4198. return pgc->framebuffer_mva;
  4199. }
  4200. unsigned long primary_display_get_frame_buffer_va_address(void)
  4201. {
  4202. return pgc->framebuffer_va;
  4203. }
  4204. int primary_display_get_session_mode(void)
  4205. {
  4206. return pgc->session_mode;
  4207. }
  4208. int is_dim_layer(unsigned int long mva)
  4209. {
  4210. if (mva == dim_layer_mva)
  4211. return 1;
  4212. return 0;
  4213. }
  4214. void *primary_get_dpmgr_handle(void)
  4215. {
  4216. return pgc->dpmgr_handle;
  4217. }
  4218. unsigned long get_dim_layer_mva_addr(void)
  4219. {
  4220. if (dim_layer_mva == 0) {
  4221. int frame_buffer_size = ALIGN_TO(DISP_GetScreenWidth(),
  4222. MTK_FB_ALIGNMENT) * DISP_GetScreenHeight() * 4;
  4223. dim_layer_mva = pgc->framebuffer_mva + 2 * frame_buffer_size;
  4224. DISPMSG("init dim layer mva %lu, size %d", dim_layer_mva, frame_buffer_size);
  4225. }
  4226. return dim_layer_mva;
  4227. }
  4228. #ifndef MTK_FB_CMDQ_DISABLE
  4229. static int init_cmdq_slots(cmdqBackupSlotHandle *pSlot, int count, int init_val)
  4230. {
  4231. int i;
  4232. cmdqBackupAllocateSlot(pSlot, count);
  4233. for (i = 0; i < count; i++)
  4234. cmdqBackupWriteSlot(*pSlot, i, init_val);
  4235. return 0;
  4236. }
  4237. #endif
  4238. int primary_display_init(char *lcm_name, unsigned int lcm_fps)
  4239. {
  4240. DISP_STATUS ret = DISP_STATUS_OK;
  4241. #ifndef MTK_FB_DFO_DISABLE
  4242. unsigned int lcm_fake_width = 0;
  4243. unsigned int lcm_fake_height = 0;
  4244. #endif
  4245. LCM_PARAMS *lcm_param = NULL;
  4246. disp_ddp_path_config *data_config = NULL;
  4247. DISPPRINT("primary_display_init begin\n");
  4248. dprec_init();
  4249. dpmgr_init();
  4250. #ifndef MTK_FB_CMDQ_DISABLE
  4251. init_cmdq_slots(&(pgc->cur_config_fence), DISP_SESSION_TIMELINE_COUNT, 0);
  4252. init_cmdq_slots(&(pgc->subtractor_when_free), DISP_SESSION_TIMELINE_COUNT, 0);
  4253. init_cmdq_slots(&(pgc->cur_mem_config_fence), DISP_SESSION_TIMELINE_COUNT, 0);
  4254. init_cmdq_slots(&(pgc->mem_subtractor_when_free), DISP_SESSION_TIMELINE_COUNT, 0);
  4255. init_cmdq_slots(&(pgc->rdma_buff_info), 3, 0);
  4256. init_cmdq_slots(&(pgc->ovl_status_info), 2, 0);
  4257. init_cmdq_slots(&(pgc->dsi_state_info), 10, 0);
  4258. init_cmdq_slots(&(pgc->rdma_state_info), 50, 0);
  4259. #endif
  4260. #ifdef DISP_DUMP_EVENT_STATUS
  4261. init_cmdq_slots(&(pgc->event_status), 6, 0);
  4262. #endif
  4263. mutex_init(&(pgc->capture_lock));
  4264. mutex_init(&(pgc->lock));
  4265. mutex_init(&(pgc->cmd_lock));
  4266. mutex_init(&(pgc->vsync_lock));
  4267. mutex_init(&esd_mode_switch_lock);
  4268. #ifdef MTK_DISP_IDLE_LP
  4269. mutex_init(&idle_lock);
  4270. #endif
  4271. #ifdef DISP_SWITCH_DST_MODE
  4272. mutex_init(&(pgc->switch_dst_lock));
  4273. #endif
  4274. _primary_path_lock(__func__);
  4275. pgc->plcm = disp_lcm_probe(lcm_name, LCM_INTERFACE_NOTDEFINED);
  4276. if (pgc->plcm == NULL) {
  4277. DISPERR("disp_lcm_probe returns null\n");
  4278. ret = DISP_STATUS_ERROR;
  4279. goto done;
  4280. }
  4281. #ifndef MTK_FB_DFO_DISABLE
  4282. if ((0 == dfo_query("LCM_FAKE_WIDTH", &lcm_fake_width))
  4283. && (0 == dfo_query("LCM_FAKE_HEIGHT", &lcm_fake_height))) {
  4284. pr_debug("[DFO] LCM_FAKE_WIDTH=%d, LCM_FAKE_HEIGHT=%d\n", lcm_fake_width,
  4285. lcm_fake_height);
  4286. if (lcm_fake_width && lcm_fake_height) {
  4287. if (DISP_STATUS_OK != primary_display_change_lcm_resolution(lcm_fake_width, lcm_fake_height))
  4288. DISPMSG("[DISP\DFO]WARNING!!! Change LCM Resolution FAILED!!!\n");
  4289. }
  4290. }
  4291. #endif
  4292. lcm_param = disp_lcm_get_params(pgc->plcm);
  4293. if (lcm_param == NULL) {
  4294. DISPERR("get lcm params FAILED\n");
  4295. ret = DISP_STATUS_ERROR;
  4296. goto done;
  4297. }
  4298. #ifndef MTK_FB_CMDQ_DISABLE
  4299. ret = cmdqCoreRegisterCB(CMDQ_GROUP_DISP, (CmdqClockOnCB)cmdqDdpClockOn, (CmdqDumpInfoCB)cmdqDdpDumpInfo,
  4300. (CmdqResetEngCB)cmdqDdpResetEng, (CmdqClockOffCB)cmdqDdpClockOff);
  4301. if (ret) {
  4302. DISPERR("cmdqCoreRegisterCB failed, ret=%d\n", ret);
  4303. ret = DISP_STATUS_ERROR;
  4304. goto done;
  4305. }
  4306. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &(pgc->cmdq_handle_config));
  4307. if (ret) {
  4308. DISPCHECK("cmdqRecCreate FAIL, ret=%d\n", ret);
  4309. ret = DISP_STATUS_ERROR;
  4310. goto done;
  4311. } else {
  4312. DISPCHECK("cmdqRecCreate SUCCESS, g_cmdq_handle=0x%p\n", pgc->cmdq_handle_config);
  4313. }
  4314. /*create ovl2mem path cmdq handle */
  4315. ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_COLOR, &(pgc->cmdq_handle_ovl1to2_config));
  4316. if (ret != 0) {
  4317. DISPERR("cmdqRecCreate FAIL, ret=%d\n", ret);
  4318. return -1;
  4319. }
  4320. primary_display_use_cmdq = CMDQ_ENABLE;
  4321. #else
  4322. primary_display_use_cmdq = CMDQ_DISABLE;
  4323. #endif
  4324. /* debug for bus hang issue (need to remove) */
  4325. ddp_dump_analysis(DISP_MODULE_CONFIG);
  4326. if (primary_display_mode == DIRECT_LINK_MODE) {
  4327. __build_path_direct_link();
  4328. pgc->session_mode = DISP_SESSION_DIRECT_LINK_MODE;
  4329. DISPCHECK("primary display is DIRECT LINK MODE\n");
  4330. } else if (primary_display_mode == DECOUPLE_MODE) {
  4331. __build_path_decouple();
  4332. pgc->session_mode = DISP_SESSION_DECOUPLE_MODE;
  4333. DISPCHECK("primary display is DECOUPLE MODE\n");
  4334. } else if (primary_display_mode == SINGLE_LAYER_MODE) {
  4335. __build_path_single_layer();
  4336. DISPCHECK("primary display is SINGLE LAYER MODE\n");
  4337. } else if (primary_display_mode == DEBUG_RDMA1_DSI0_MODE) {
  4338. __build_path_debug_rdma1_dsi0();
  4339. DISPCHECK("primary display is DEBUG RDMA1 DSI0 MODE\n");
  4340. } else {
  4341. DISPCHECK("primary display mode is WRONG\n");
  4342. }
  4343. #ifndef MTK_FB_CMDQ_DISABLE
  4344. primary_display_use_cmdq = CMDQ_ENABLE;
  4345. #else
  4346. primary_display_use_cmdq = CMDQ_DISABLE;
  4347. #endif
  4348. /* dpmgr_path_init(pgc->dpmgr_handle, CMDQ_DISABLE); */
  4349. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  4350. /* dpmgr_path_init(pgc->dpmgr_handle, CMDQ_DISABLE); */
  4351. /* use fake timer to generate vsync signal for cmd mode w/o LCM(originally using LCM TE Signal as VSYNC) */
  4352. /* so we don't need to modify display driver's behavior. */
  4353. if (disp_helper_get_option
  4354. (DISP_HELPER_OPTION_NO_LCM_FOR_LOW_POWER_MEASUREMENT)) {
  4355. /* only for low power measurement */
  4356. DISPCHECK("WARNING!!!!!! FORCE NO LCM MODE!!!\n");
  4357. islcmconnected = 0;
  4358. /* no need to change video mode vsync behavior */
  4359. if (!primary_display_is_video_mode()) {
  4360. _init_vsync_fake_monitor(lcm_fps);
  4361. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC,
  4362. DDP_IRQ_UNKNOWN);
  4363. }
  4364. }
  4365. #ifdef CONFIG_FPGA_EARLY_PORTING
  4366. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  4367. #endif
  4368. if (primary_display_use_cmdq == CMDQ_ENABLE) {
  4369. _cmdq_build_trigger_loop();
  4370. _cmdq_start_trigger_loop();
  4371. _cmdq_reset_config_handle();
  4372. _cmdq_insert_wait_frame_done_token();
  4373. }
  4374. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  4375. #ifdef OVL_CASCADE_SUPPORT
  4376. if (ovl_get_status() == DDP_OVL1_STATUS_IDLE || ovl_get_status() == DDP_OVL1_STATUS_PRIMARY) {
  4377. if (primary_display_mode == DECOUPLE_MODE)
  4378. dpmgr_path_enable_cascade(pgc->ovl2mem_path_handle,
  4379. pgc->cmdq_handle_config);
  4380. else
  4381. dpmgr_path_enable_cascade(pgc->dpmgr_handle, pgc->cmdq_handle_config);
  4382. }
  4383. #endif
  4384. memcpy(&(data_config->dispif_config), lcm_param, sizeof(LCM_PARAMS));
  4385. data_config->dst_w = lcm_param->width;
  4386. data_config->dst_h = lcm_param->height;
  4387. if (lcm_param->type == LCM_TYPE_DSI) {
  4388. if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB888)
  4389. data_config->lcm_bpp = 24;
  4390. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB565)
  4391. data_config->lcm_bpp = 16;
  4392. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB666)
  4393. data_config->lcm_bpp = 18;
  4394. } else if (lcm_param->type == LCM_TYPE_DPI) {
  4395. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB888)
  4396. data_config->lcm_bpp = 24;
  4397. else if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB565)
  4398. data_config->lcm_bpp = 16;
  4399. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB666)
  4400. data_config->lcm_bpp = 18;
  4401. }
  4402. data_config->fps = lcm_fps;
  4403. data_config->dst_dirty = 1;
  4404. #ifdef CONFIG_FPGA_EARLY_PORTING
  4405. data_config->ovl_dirty = 1;
  4406. #endif
  4407. rdma_set_target_line(DISP_MODULE_RDMA0, primary_display_get_height() * 1 / 2,
  4408. pgc->cmdq_handle_config);
  4409. rdma_set_target_line(DISP_MODULE_RDMA0, primary_display_get_height() * 1 / 2, NULL);
  4410. if (primary_display_use_cmdq == CMDQ_ENABLE) {
  4411. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, pgc->cmdq_handle_config);
  4412. /* should we set dirty here???????? */
  4413. _cmdq_flush_config_handle(0, NULL, 0);
  4414. _cmdq_reset_config_handle();
  4415. _cmdq_insert_wait_frame_done_token();
  4416. } else {
  4417. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, NULL);
  4418. }
  4419. {
  4420. #ifdef MTK_NO_DISP_IN_LK
  4421. ret = disp_lcm_init(pgc->plcm, 1);
  4422. #else
  4423. ret = disp_lcm_init(pgc->plcm, 0);
  4424. #endif
  4425. }
  4426. #ifndef MTK_NO_DISP_IN_LK
  4427. if (_is_decouple_mode(pgc->session_mode))
  4428. #endif
  4429. /* this should remove? for video mode when LK has start path */
  4430. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  4431. #ifdef MTK_FB_PULLCLK_ENABLE
  4432. primary_display_pullclk_task = kthread_create(primary_display_vdo_pullclk_worker_kthread,
  4433. NULL, "display_vdo_pullclk");
  4434. wake_up_process(primary_display_pullclk_task);
  4435. #endif
  4436. #ifdef MTK_FB_ESD_ENABLE
  4437. primary_display_esd_check_task = kthread_create(primary_display_esd_check_worker_kthread,
  4438. NULL, "display_esd_check");
  4439. init_waitqueue_head(&esd_ext_te_wq);
  4440. init_waitqueue_head(&esd_check_task_wq);
  4441. if (_need_do_esd_check())
  4442. wake_up_process(primary_display_esd_check_task);
  4443. #if 0
  4444. if (_need_do_esd_check())
  4445. primary_display_esd_check_enable(1);
  4446. #endif
  4447. if (_need_register_eint()) {
  4448. struct device_node *node = NULL;
  4449. int irq;
  4450. u32 ints[2] = { 0, 0 };
  4451. #ifdef GPIO_DSI_TE_PIN
  4452. #ifndef CONFIG_FPGA_EARLY_PORTING
  4453. /* 1.set GPIO107 eint mode */
  4454. mt_set_gpio_mode(GPIO_DSI_TE_PIN, GPIO_DSI_TE_PIN_M_GPIO);
  4455. #endif
  4456. eint_flag++;
  4457. #endif
  4458. #ifndef CONFIG_MTK_LEGACY
  4459. eint_flag++;
  4460. #endif
  4461. /* 2.register eint */
  4462. node = of_find_compatible_node(NULL, NULL, "mediatek, DSI_TE_1-eint");
  4463. if (node) {
  4464. of_property_read_u32_array(node, "debounce", ints, ARRAY_SIZE(ints));
  4465. /* FIXME: find definatition of mt_gpio_set_debounce */
  4466. /* mt_gpio_set_debounce(ints[0], ints[1]); */
  4467. mt_eint_set_hw_debounce(ints[0], ints[1]);
  4468. irq = irq_of_parse_and_map(node, 0);
  4469. if (request_irq(irq, _esd_check_ext_te_irq_handler, IRQF_TRIGGER_RISING, "DSI_TE_1-eint", NULL))
  4470. DISPCHECK("[ESD]EINT IRQ LINE NOT AVAILABLE!!\n");
  4471. else
  4472. eint_flag++;
  4473. } else {
  4474. DISPCHECK("[ESD][%s] can't find DSI_TE_1 eint compatible node\n", __func__);
  4475. }
  4476. }
  4477. if (_need_do_esd_check())
  4478. primary_display_esd_check_enable(1);
  4479. #endif
  4480. #ifdef DISP_SWITCH_DST_MODE
  4481. primary_display_switch_dst_mode_task = kthread_create(_disp_primary_path_switch_dst_mode_thread, NULL,
  4482. "display_switch_dst_mode");
  4483. wake_up_process(primary_display_switch_dst_mode_task);
  4484. #endif
  4485. primary_path_aal_task = kthread_create(_disp_primary_path_check_trigger, NULL, "display_check_aal");
  4486. wake_up_process(primary_path_aal_task);
  4487. #if 0 /* Zaikuo: disable it when HWC not enable to reduce error log */
  4488. fence_release_worker_task =
  4489. kthread_create(_fence_release_worker_thread, NULL, "fence_worker");
  4490. wake_up_process(fence_release_worker_task);
  4491. if (_is_decouple_mode(pgc->session_mode)) {
  4492. if_fence_release_worker_task =
  4493. kthread_create(_if_fence_release_worker_thread, NULL, "if_fence_worker");
  4494. wake_up_process(if_fence_release_worker_task);
  4495. ovl2mem_fence_release_worker_task =
  4496. kthread_create(_ovl2mem_fence_release_worker_thread, NULL,
  4497. "ovl2mem_fence_worker");
  4498. wake_up_process(ovl2mem_fence_release_worker_task);
  4499. }
  4500. #endif
  4501. present_fence_release_worker_task = kthread_create(_present_fence_release_worker_thread, NULL,
  4502. "present_fence_worker");
  4503. wake_up_process(present_fence_release_worker_task);
  4504. if (primary_display_frame_update_task == NULL) {
  4505. init_waitqueue_head(&primary_display_frame_update_wq);
  4506. disp_register_module_irq_callback(DISP_MODULE_RDMA0, primary_display_frame_update_irq_callback);
  4507. disp_register_module_irq_callback(DISP_MODULE_OVL0, primary_display_frame_update_irq_callback);
  4508. disp_register_module_irq_callback(DISP_MODULE_WDMA0, primary_display_frame_update_irq_callback);
  4509. primary_display_frame_update_task = kthread_create(primary_display_frame_update_kthread, NULL,
  4510. "frame_update_worker");
  4511. wake_up_process(primary_display_frame_update_task);
  4512. init_waitqueue_head(&decouple_fence_release_wq);
  4513. decouple_fence_release_task = kthread_create(decouple_fence_release_kthread, NULL,
  4514. "decouple_fence_release_worker");
  4515. wake_up_process(decouple_fence_release_task);
  4516. }
  4517. /* primary_display_use_cmdq = CMDQ_ENABLE; */
  4518. /* this will be set to always enable cmdq later */
  4519. if (primary_display_is_video_mode()) {
  4520. #ifdef DISP_SWITCH_DST_MODE
  4521. primary_display_cur_dst_mode = 1; /* video mode */
  4522. primary_display_def_dst_mode = 1; /* default mode is video mode */
  4523. #endif
  4524. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC, DDP_IRQ_RDMA0_DONE);
  4525. } else {
  4526. }
  4527. #ifndef MTKFB_NO_M4U
  4528. {
  4529. M4U_PORT_STRUCT sPort;
  4530. sPort.ePortID = M4U_PORT_DISP_WDMA0;
  4531. sPort.Virtuality = primary_display_use_m4u;
  4532. sPort.Security = 0;
  4533. sPort.Distance = 1;
  4534. sPort.Direction = 0;
  4535. ret = m4u_config_port(&sPort);
  4536. if (ret != 0) {
  4537. DISPCHECK("config M4U Port %s to %s FAIL(ret=%d)\n",
  4538. ddp_get_module_name(DISP_MODULE_WDMA0),
  4539. primary_display_use_m4u ? "virtual" : "physical", ret);
  4540. return -1;
  4541. }
  4542. }
  4543. #endif
  4544. pgc->lcm_fps = lcm_fps;
  4545. if (lcm_fps > 6000) /* FIXME: if fps bigger than 60, support 8 layer? */
  4546. pgc->max_layer = OVL_LAYER_NUM;
  4547. else
  4548. pgc->max_layer = OVL_LAYER_NUM;
  4549. pgc->state = DISP_ALIVE;
  4550. #ifdef CONFIG_TRUSTONIC_TRUSTED_UI
  4551. disp_switch_data.name = "disp";
  4552. disp_switch_data.index = 0;
  4553. disp_switch_data.state = DISP_ALIVE;
  4554. ret = switch_dev_register(&disp_switch_data);
  4555. #endif
  4556. primary_display_sodi_rule_init();
  4557. #ifdef MTK_DISP_IDLE_LP
  4558. init_waitqueue_head(&idle_detect_wq);
  4559. primary_display_idle_detect_task = kthread_create(_disp_primary_path_idle_detect_thread, NULL,
  4560. "display_idle_detect");
  4561. wake_up_process(primary_display_idle_detect_task);
  4562. #endif
  4563. pgc->dc_type = DISP_OUTPUT_DECOUPLE;
  4564. if ((mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_P_PLUS) ||
  4565. (mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D2_M_PLUS))
  4566. register_mmclk_switch_cb(primary_display_switch_mmsys_clk, _switch_mmsys_clk);
  4567. done:
  4568. /* disable OVL TF in video mode, cause cmdq/sodi log is not ready, avoid too much TF issue */
  4569. {
  4570. /* extern unsigned int gDisableOVLTF; */
  4571. if (gDisableOVLTF == 1 && primary_display_is_video_mode() == 1) {
  4572. /* extern void disp_m4u_tf_disable(void); */
  4573. disp_m4u_tf_disable();
  4574. }
  4575. }
  4576. #ifndef WDMA_PATH_CLOCK_DYNAMIC_SWITCH
  4577. #ifndef CONFIG_FPGA_EARLY_PORTING
  4578. enable_soidle_by_bit(MT_CG_DISP0_DISP_WDMA0);
  4579. #endif
  4580. #endif
  4581. _primary_path_unlock(__func__);
  4582. /* DISPCHECK("primary_display_init end\n"); */
  4583. return ret;
  4584. }
  4585. int primary_display_deinit(void)
  4586. {
  4587. _primary_path_lock(__func__);
  4588. _cmdq_stop_trigger_loop();
  4589. if(!pgc->dpmgr_handle){
  4590. _primary_path_unlock(__func__);
  4591. return 0;
  4592. }else
  4593. dpmgr_path_deinit(pgc->dpmgr_handle, CMDQ_DISABLE);
  4594. #ifndef CONFIG_MTK_CLKMGR
  4595. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  4596. #endif
  4597. _primary_path_unlock(__func__);
  4598. return 0;
  4599. }
  4600. /* register rdma done event */
  4601. int primary_display_wait_for_idle(void)
  4602. {
  4603. DISP_STATUS ret = DISP_STATUS_OK;
  4604. DISPFUNC();
  4605. _primary_path_lock(__func__);
  4606. /* Nothing to do ??? */
  4607. _primary_path_unlock(__func__);
  4608. return ret;
  4609. }
  4610. int primary_display_wait_for_dump(void)
  4611. {
  4612. return 0; /* avoid build warning. */
  4613. }
  4614. int primary_display_release_fence_fake(void)
  4615. {
  4616. unsigned int layer_en = 0;
  4617. unsigned int addr = 0;
  4618. unsigned int fence_idx = -1;
  4619. unsigned int session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  4620. int i = 0;
  4621. DISPFUNC();
  4622. for (i = 0; i < PRIMARY_DISPLAY_SESSION_LAYER_COUNT; i++) {
  4623. if (i == primary_display_get_option("ASSERT_LAYER") && is_DAL_Enabled()) {
  4624. mtkfb_release_layer_fence(session_id, 3);
  4625. } else {
  4626. disp_sync_get_cached_layer_info(session_id, i, &layer_en,
  4627. (unsigned long *)&addr, &fence_idx);
  4628. if (fence_idx < 0) {
  4629. if (fence_idx == -1) {
  4630. DISPPR_ERROR("find fence idx for layer %d,addr 0x%08x fail,\n", i, 0);
  4631. DISPPR_ERROR("unregistered addr%d\n", fence_idx);
  4632. } else if (fence_idx == -2) {
  4633. ;
  4634. } else {
  4635. DISPPR_ERROR("find fence idx for layer %d,addr 0x%08x fail,reason unknown%d\n",
  4636. i, 0, fence_idx);
  4637. }
  4638. } else {
  4639. if (layer_en)
  4640. mtkfb_release_fence(session_id, i, fence_idx - 1);
  4641. else
  4642. mtkfb_release_fence(session_id, i, fence_idx);
  4643. }
  4644. }
  4645. }
  4646. return 0; /* avoid build warning */
  4647. }
  4648. int primary_display_wait_for_vsync(void *config)
  4649. {
  4650. disp_session_vsync_config *c = (disp_session_vsync_config *) config;
  4651. int ret = 0;
  4652. /* kick idle manager here to ensure sodi is disabled when screen update begin(not 100% ensure) */
  4653. primary_display_idlemgr_kick((char *)__func__);
  4654. _primary_path_vsync_lock();
  4655. if (pgc->state == DISP_SLEPT) {
  4656. DISPCHECK("VSYNC DISP_SLEPT Return\n");
  4657. _primary_path_vsync_unlock();
  4658. return 0;
  4659. }
  4660. #ifdef MTK_DISP_IDLE_LP
  4661. _disp_primary_path_dsi_clock_on(0);
  4662. #endif
  4663. if (!islcmconnected) {
  4664. DISPCHECK("lcm not connect, use fake vsync\n");
  4665. /* msleep(16); */
  4666. usleep_range(16000, 17000);
  4667. goto done;
  4668. }
  4669. ret = dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  4670. if (pgc->vsync_drop)
  4671. ret = dpmgr_wait_event(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC);
  4672. if (ret != 0)
  4673. DISPCHECK("vsync signaled by unknown signal ret=%d\n", ret);
  4674. c->vsync_ts = get_current_time_us();
  4675. c->vsync_cnt++;
  4676. done:
  4677. #ifdef MTK_DISP_IDLE_LP
  4678. _disp_primary_path_dsi_clock_off(0);
  4679. #endif
  4680. _primary_path_vsync_unlock();
  4681. return 0;
  4682. }
  4683. unsigned int primary_display_get_ticket(void)
  4684. {
  4685. return dprec_get_vsync_count();
  4686. }
  4687. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  4688. int primary_suspend_release_ovl_fence(disp_session_input_config *session_input)
  4689. {
  4690. unsigned int session = (unsigned int)((DISP_SESSION_PRIMARY)<<16 | (0));
  4691. unsigned int i = 0;
  4692. unsigned int layer = 0;
  4693. for (i = 0; i < session_input->config_layer_num; i++) {
  4694. unsigned int cur_fence;
  4695. disp_input_config *input_cfg = &session_input->config[i];
  4696. layer = input_cfg->layer_id;
  4697. cur_fence = input_cfg->next_buff_idx;
  4698. if (cur_fence != -1)
  4699. mtkfb_release_fence(session, i, cur_fence);
  4700. }
  4701. return 0;
  4702. }
  4703. int primary_suspend_clr_ovl_config(void)
  4704. {
  4705. disp_ddp_path_config *data_config = NULL;
  4706. disp_path_handle handle = NULL;
  4707. OVL_CONFIG_STRUCT ovl_config[OVL_LAYER_NUM];
  4708. int i = 0;
  4709. DISPFUNC();
  4710. if (0 == g_suspend_flag)
  4711. return -1;
  4712. if (_is_decouple_mode(pgc->session_mode) == 0)
  4713. handle = pgc->dpmgr_handle;
  4714. else
  4715. handle = pgc->ovl2mem_path_handle;
  4716. data_config = dpmgr_path_get_last_config(handle);
  4717. memset((void *)&(data_config->ovl_config), 0, sizeof(ovl_config));
  4718. last_primary_config = *data_config;
  4719. for (i = 0; i < OVL_LAYER_NUM; i++)
  4720. ovl_layer_switch(DISP_MODULE_OVL0, i, 0, NULL);
  4721. DISP_REG_SET(NULL, DISP_REG_OVL_SRC_CON, 0x0);
  4722. return 0;
  4723. }
  4724. #endif
  4725. int primary_suspend_release_fence(void)
  4726. {
  4727. unsigned int session = (unsigned int)((DISP_SESSION_PRIMARY) << 16 | (0));
  4728. unsigned int i = 0;
  4729. for (i = 0; i < HW_OVERLAY_COUNT; i++) {
  4730. DISPMSG("mtkfb_release_layer_fence session=0x%x,layerid=%d\n", session, i);
  4731. mtkfb_release_layer_fence(session, i);
  4732. }
  4733. return 0;
  4734. }
  4735. static void primary_display_suspend_wait_tui(void)
  4736. {
  4737. while (primary_get_state() == DISP_BLANK) {
  4738. _primary_path_vsync_unlock();
  4739. _primary_path_unlock(__func__);
  4740. _primary_path_esd_check_unlock();
  4741. disp_sw_mutex_unlock(&(pgc->capture_lock));
  4742. #ifdef CONFIG_TRUSTONIC_TRUSTED_UI
  4743. switch_set_state(&disp_switch_data, DISP_SLEPT);
  4744. #endif
  4745. primary_display_wait_state(DISP_ALIVE, MAX_SCHEDULE_TIMEOUT);
  4746. disp_sw_mutex_lock(&(pgc->capture_lock));
  4747. _primary_path_esd_check_lock();
  4748. _primary_path_lock(__func__);
  4749. _primary_path_vsync_lock();
  4750. DISPCHECK("primary_display_suspend wait tui done stat=%d\n", primary_get_state());
  4751. }
  4752. }
  4753. int primary_display_suspend(void)
  4754. {
  4755. DISP_STATUS ret = DISP_STATUS_OK;
  4756. DISPCHECK("primary_display_suspend begin\n");
  4757. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagStart, 0, 0);
  4758. #ifdef DISP_SWITCH_DST_MODE
  4759. primary_display_switch_dst_mode(primary_display_def_dst_mode);
  4760. #endif
  4761. disp_sw_mutex_lock(&(pgc->capture_lock));
  4762. _primary_path_esd_check_lock();
  4763. _primary_path_lock(__func__);
  4764. _primary_path_vsync_lock();
  4765. DISPCHECK("wait tui finish[begin]\n");
  4766. primary_display_suspend_wait_tui();
  4767. DISPCHECK("wait tui finish[end]\n");
  4768. if (pgc->state == DISP_SLEPT) {
  4769. DISPCHECK("primary display path is already sleep, skip\n");
  4770. goto done;
  4771. }
  4772. #ifdef MTK_DISP_IDLE_LP
  4773. _disp_primary_path_exit_idle(__func__, 0);
  4774. #endif
  4775. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 1);
  4776. /* msleep(16); */ /* wait last frame done */
  4777. usleep_range(16000, 17000);
  4778. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  4779. int event_ret = 0;
  4780. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 1, 2);
  4781. event_ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  4782. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 2, 2);
  4783. DISPCHECK("[POWER]primary display path is busy now, wait frame done, event_ret=%d\n", event_ret);
  4784. if (event_ret <= 0) {
  4785. DISPERR("wait frame done in suspend timeout\n");
  4786. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 3, 2);
  4787. primary_display_diagnose();
  4788. ret = -1;
  4789. }
  4790. }
  4791. /* for decouple mode */
  4792. if (_is_decouple_mode(pgc->session_mode)) {
  4793. if (dpmgr_path_is_busy(pgc->ovl2mem_path_handle))
  4794. dpmgr_wait_event_timeout(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_COMPLETE, HZ);
  4795. /* xuecheng, BAD WROKAROUND for decouple mode */
  4796. msleep(30);
  4797. }
  4798. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 2);
  4799. DISPCHECK("[POWER]display cmdq trigger loop stop[begin]\n");
  4800. _cmdq_stop_trigger_loop();
  4801. DISPCHECK("[POWER]display cmdq trigger loop stop[end]\n");
  4802. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 3);
  4803. #ifdef CONFIG_LCM_SEND_CMD_IN_VIDEO
  4804. DISPCHECK("[POWER]lcm suspend[begin]\n");
  4805. disp_lcm_suspend(pgc->plcm);
  4806. DISPCHECK("[POWER]lcm suspend[end]\n");
  4807. #endif
  4808. DISPCHECK("[POWER]primary display path stop[begin]\n");
  4809. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  4810. DISPCHECK("[POWER]primary display path stop[end]\n");
  4811. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 4);
  4812. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  4813. primary_suspend_clr_ovl_config();
  4814. #endif
  4815. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  4816. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 1, 4);
  4817. DISPERR("[POWER]stop display path failed, still busy\n");
  4818. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  4819. ret = -1;
  4820. /* even path is busy(stop fail), we still need to continue power off other module/devices */
  4821. /* goto done; */
  4822. }
  4823. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 5);
  4824. /* remove OVL1 from display if there is more than one session, */
  4825. /* because the other session may use OVL1 in suspend mode */
  4826. #if 0
  4827. {
  4828. /* extern int disp_get_session_number(void); */
  4829. if (ovl_get_status() != DDP_OVL1_STATUS_SUB && disp_get_session_number() > 1) {
  4830. unsigned int dp_handle;
  4831. unsigned int cmdq_handle;
  4832. DISPMSG("disable cascade before suspend!\n");
  4833. dpmgr_path_get_handle(&dp_handle, &cmdq_handle);
  4834. dpmgr_path_disable_cascade(dp_handle, CMDQ_DISABLE);
  4835. if (ovl_get_status() == DDP_OVL1_STATUS_SUB_REQUESTING) {
  4836. ovl_set_status(DDP_OVL1_STATUS_SUB);
  4837. wake_up_interruptible(&ovl1_wait_queue);
  4838. } else {
  4839. ovl_set_status(DDP_OVL1_STATUS_IDLE);
  4840. }
  4841. _cmdq_build_trigger_loop();
  4842. }
  4843. }
  4844. #else
  4845. DISPMSG("disable cascade before suspend!\n");
  4846. if (_is_decouple_mode(pgc->session_mode) == 0)
  4847. dpmgr_path_disable_cascade(pgc->dpmgr_handle, CMDQ_DISABLE);
  4848. else
  4849. dpmgr_path_disable_cascade(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  4850. if (ovl_get_status() == DDP_OVL1_STATUS_SUB_REQUESTING)
  4851. dpmgr_set_ovl1_status(DDP_OVL1_STATUS_SUB);
  4852. else if (ovl_get_status() != DDP_OVL1_STATUS_SUB)
  4853. dpmgr_set_ovl1_status(DDP_OVL1_STATUS_IDLE);
  4854. #endif
  4855. #ifndef CONFIG_LCM_SEND_CMD_IN_VIDEO
  4856. DISPCHECK("[POWER]lcm suspend[begin]\n");
  4857. disp_lcm_suspend(pgc->plcm);
  4858. DISPCHECK("[POWER]lcm suspend[end]\n");
  4859. #endif
  4860. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 6);
  4861. DISPCHECK("[POWER]primary display path Release Fence[begin]\n");
  4862. primary_suspend_release_fence();
  4863. DISPCHECK("[POWER]primary display path Release Fence[end]\n");
  4864. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 7);
  4865. DISPCHECK("[POWER]dpmanager path power off[begin]\n");
  4866. dpmgr_path_power_off(pgc->dpmgr_handle, CMDQ_DISABLE);
  4867. DISPCHECK("[POWER]dpmanager path power off[end]\n");
  4868. if (_is_decouple_mode(pgc->session_mode)) {
  4869. dpmgr_path_power_off(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  4870. } else if (is_mmdvfs_supported() && mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D1_PLUS) {
  4871. DISPMSG("set MMDVFS low\n");
  4872. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_LOW); /* Vote to LPM mode */
  4873. }
  4874. #ifndef CONFIG_MTK_CLKMGR
  4875. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  4876. #endif
  4877. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagPulse, 0, 8);
  4878. display_path_idle_cnt = 0;
  4879. pgc->state = DISP_SLEPT;
  4880. done:
  4881. _primary_path_vsync_unlock();
  4882. _primary_path_unlock(__func__);
  4883. _primary_path_esd_check_unlock();
  4884. disp_sw_mutex_unlock(&(pgc->capture_lock));
  4885. #ifndef DISP_NO_AEE
  4886. aee_kernel_wdt_kick_Powkey_api("mtkfb_early_suspend", WDT_SETBY_Display);
  4887. #endif
  4888. primary_trigger_cnt = 0;
  4889. MMProfileLogEx(ddp_mmp_get_events()->primary_suspend, MMProfileFlagEnd, 0, 0);
  4890. DISPCHECK("primary_display_suspend end\n");
  4891. return ret;
  4892. }
  4893. int primary_display_get_lcm_index(void)
  4894. {
  4895. int index = 0;
  4896. DISPFUNC();
  4897. if (pgc->plcm == NULL) {
  4898. DISPERR("lcm handle is null\n");
  4899. return 0;
  4900. }
  4901. index = pgc->plcm->index;
  4902. DISPMSG("lcm index = %d\n", index);
  4903. return index;
  4904. }
  4905. int primary_display_resume(void)
  4906. {
  4907. DISP_STATUS ret = DISP_STATUS_OK;
  4908. DISPFUNC();
  4909. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagStart, 0, 0);
  4910. _primary_path_lock(__func__);
  4911. if (pgc->state == DISP_ALIVE) {
  4912. DISPCHECK("primary display path is already resume, skip\n");
  4913. goto done;
  4914. }
  4915. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 1);
  4916. #ifdef CONFIG_MTK_CLKMGR
  4917. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  4918. dpmgr_reset_module_handle(pgc->dpmgr_handle);
  4919. #endif
  4920. /* For CCF, we move the power on control to noirq_restore in mtkfb */
  4921. DISPCHECK("dpmanager path power on[begin]\n");
  4922. dpmgr_path_power_on(pgc->dpmgr_handle, CMDQ_DISABLE);
  4923. DISPCHECK("dpmanager path power on[end]\n");
  4924. if (_is_decouple_mode(pgc->session_mode) && !pgc->force_on_wdma_path)
  4925. dpmgr_path_power_on(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  4926. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 2);
  4927. #endif
  4928. #ifndef CONFIG_MTK_CLKMGR
  4929. ddp_clk_prepare(DISP_MTCMOS_CLK);
  4930. #endif
  4931. if (is_ipoh_bootup) {
  4932. DISPCHECK("[primary display path] leave primary_display_resume -- IPOH\n");
  4933. DISPCHECK("ESD check start[begin]\n");
  4934. primary_display_esd_check_enable(1);
  4935. DISPCHECK("ESD check start[end]\n");
  4936. is_ipoh_bootup = false;
  4937. DISPCHECK("[POWER]start cmdq[begin]--IPOH\n");
  4938. _cmdq_start_trigger_loop();
  4939. DISPCHECK("[POWER]start cmdq[end]--IPOH\n");
  4940. pgc->state = DISP_ALIVE;
  4941. goto done;
  4942. }
  4943. #ifndef CONFIG_MTK_CLKMGR
  4944. DISPCHECK("dpmanager path power on[begin]\n");
  4945. dpmgr_path_power_on(pgc->dpmgr_handle, CMDQ_DISABLE);
  4946. DISPCHECK("dpmanager path power on[end]\n");
  4947. if (_is_decouple_mode(pgc->session_mode) && !pgc->force_on_wdma_path)
  4948. dpmgr_path_power_on(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  4949. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 2);
  4950. #endif
  4951. if (primary_display_is_video_mode()) {
  4952. DISPCHECK("mutex0 clear[begin]\n");
  4953. ddp_mutex_clear(0, NULL);
  4954. DISPCHECK("mutex0 clear[end]\n");
  4955. }
  4956. DISPCHECK("[POWER]dpmanager re-init[begin]\n");
  4957. {
  4958. LCM_PARAMS *lcm_param = NULL;
  4959. disp_ddp_path_config *data_config = NULL;
  4960. dpmgr_path_connect(pgc->dpmgr_handle, CMDQ_DISABLE);
  4961. if (_is_decouple_mode(pgc->session_mode))
  4962. dpmgr_path_connect(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  4963. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 1, 2);
  4964. lcm_param = disp_lcm_get_params(pgc->plcm);
  4965. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  4966. data_config->dst_w = lcm_param->width;
  4967. data_config->dst_h = lcm_param->height;
  4968. if (lcm_param->type == LCM_TYPE_DSI) {
  4969. if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB888)
  4970. data_config->lcm_bpp = 24;
  4971. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB565)
  4972. data_config->lcm_bpp = 16;
  4973. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB666)
  4974. data_config->lcm_bpp = 18;
  4975. } else if (lcm_param->type == LCM_TYPE_DPI) {
  4976. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB888)
  4977. data_config->lcm_bpp = 24;
  4978. else if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB565)
  4979. data_config->lcm_bpp = 16;
  4980. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB666)
  4981. data_config->lcm_bpp = 18;
  4982. }
  4983. data_config->fps = pgc->lcm_fps;
  4984. data_config->dst_dirty = 1;
  4985. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, NULL);
  4986. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 2, 2);
  4987. if (_is_decouple_mode(pgc->session_mode)) {
  4988. data_config = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  4989. data_config->fps = pgc->lcm_fps;
  4990. data_config->dst_dirty = 1;
  4991. data_config->wdma_dirty = 1;
  4992. ret = dpmgr_path_config(pgc->ovl2mem_path_handle, data_config, NULL);
  4993. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 2,
  4994. 2);
  4995. }
  4996. data_config->dst_dirty = 0;
  4997. }
  4998. DISPCHECK("[POWER]dpmanager re-init[end]\n");
  4999. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 3);
  5000. DISPCHECK("[POWER]lcm resume[begin]\n");
  5001. disp_lcm_resume(pgc->plcm);
  5002. DISPCHECK("[POWER]lcm resume[end]\n");
  5003. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 4);
  5004. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  5005. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 1, 4);
  5006. DISPERR("[POWER]Fatal error, we didn't start display path but it's already busy\n");
  5007. ret = -1;
  5008. /* goto done; */
  5009. }
  5010. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 5);
  5011. DISPCHECK("[POWER]dpmgr path start[begin]\n");
  5012. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  5013. if (_is_decouple_mode(pgc->session_mode))
  5014. dpmgr_path_start(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  5015. DISPCHECK("[POWER]dpmgr path start[end]\n");
  5016. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 6);
  5017. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  5018. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 1, 6);
  5019. DISPERR
  5020. ("[POWER]Fatal error, we didn't trigger display path but it's already busy\n");
  5021. ret = -1;
  5022. /* goto done; */
  5023. }
  5024. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 7);
  5025. if (primary_display_is_video_mode()) {
  5026. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 1, 7);
  5027. /* for video mode, we need to force trigger here */
  5028. /* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start */
  5029. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  5030. /* insert a wait token to make sure first config after resume will config to HW when HW idle */
  5031. _cmdq_insert_wait_frame_done_token();
  5032. }
  5033. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 8);
  5034. DISPCHECK("[POWER]start cmdq[begin]\n");
  5035. _cmdq_start_trigger_loop();
  5036. DISPCHECK("[POWER]start cmdq[end]\n");
  5037. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagPulse, 0, 9);
  5038. if (!primary_display_is_video_mode()) {
  5039. /*refresh black picture of ovl bg */
  5040. DISPCHECK("[POWER]triggger cmdq[begin]\n");
  5041. _trigger_display_interface(0, NULL, 0);
  5042. DISPCHECK("[POWER]triggger cmdq[end]\n");
  5043. }
  5044. DISPCHECK("[POWER]wait frame done[begin]\n");
  5045. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ / 10);
  5046. DISPCHECK("[POWER]wait frame done[end]\n");
  5047. /* reinit fake timer for debug, we can enable option then press powerkey to enable thsi feature. */
  5048. /* use fake timer to generate vsync signal for cmd mode w/o LCM(originally using LCM TE Signal as VSYNC) */
  5049. /* so we don't need to modify display driver's behavior. */
  5050. if (disp_helper_get_option
  5051. (DISP_HELPER_OPTION_NO_LCM_FOR_LOW_POWER_MEASUREMENT)) {
  5052. /* only for low power measurement */
  5053. DISPCHECK("WARNING!!!!!! FORCE NO LCM MODE!!!\n");
  5054. islcmconnected = 0;
  5055. /* no need to change video mode vsync behavior */
  5056. if (!primary_display_is_video_mode()) {
  5057. _init_vsync_fake_monitor(pgc->lcm_fps);
  5058. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC,
  5059. DDP_IRQ_UNKNOWN);
  5060. }
  5061. }
  5062. if (is_mmdvfs_supported() && mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D1_PLUS &&
  5063. primary_display_get_width() > 800) {
  5064. DISPMSG("set MMDVFS high\n");
  5065. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_HIGH); /* Enter HPM mode */
  5066. }
  5067. #if 0
  5068. DISPCHECK("[POWER]wakeup aal/od trigger process[begin]\n");
  5069. wake_up_process(primary_path_aal_task);
  5070. DISPCHECK("[POWER]wakeup aal/od trigger process[end]\n");
  5071. #endif
  5072. pgc->state = DISP_ALIVE;
  5073. pgc->force_on_wdma_path = 0;
  5074. #ifdef CONFIG_TRUSTONIC_TRUSTED_UI
  5075. switch_set_state(&disp_switch_data, DISP_ALIVE);
  5076. #endif
  5077. done:
  5078. _primary_path_unlock(__func__);
  5079. /* primary_display_diagnose(); */
  5080. #ifndef DISP_NO_AEE
  5081. aee_kernel_wdt_kick_Powkey_api("mtkfb_late_resume", WDT_SETBY_Display);
  5082. #endif
  5083. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagEnd, 0, 0);
  5084. ddp_dump_analysis(DISP_MODULE_OVL0);
  5085. #if defined(OVL_CASCADE_SUPPORT)
  5086. ddp_dump_analysis(DISP_MODULE_OVL1);
  5087. #endif
  5088. ddp_dump_analysis(DISP_MODULE_RDMA0);
  5089. return 0;
  5090. }
  5091. int primary_display_resume_ovl2mem(void)
  5092. {
  5093. DISP_STATUS ret = DISP_STATUS_OK;
  5094. DISPFUNC();
  5095. #ifdef CONFIG_MTK_CLKMGR
  5096. if (_is_decouple_mode(pgc->session_mode))
  5097. dpmgr_path_power_on(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  5098. #endif
  5099. #ifndef CONFIG_MTK_CLKMGR
  5100. ddp_clk_prepare(DISP_MTCMOS_CLK);
  5101. #endif
  5102. #ifndef CONFIG_MTK_CLKMGR
  5103. if (_is_decouple_mode(pgc->session_mode))
  5104. dpmgr_path_power_on(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  5105. #endif
  5106. DISPCHECK("[POWER]dpmanager re-init[begin]\n");
  5107. {
  5108. LCM_PARAMS *lcm_param = NULL;
  5109. disp_ddp_path_config *data_config = NULL;
  5110. if (_is_decouple_mode(pgc->session_mode))
  5111. dpmgr_path_connect(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  5112. lcm_param = disp_lcm_get_params(pgc->plcm);
  5113. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  5114. data_config->dst_w = lcm_param->width;
  5115. data_config->dst_h = lcm_param->height;
  5116. if (lcm_param->type == LCM_TYPE_DSI) {
  5117. if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB888)
  5118. data_config->lcm_bpp = 24;
  5119. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB565)
  5120. data_config->lcm_bpp = 16;
  5121. else if (lcm_param->dsi.data_format.format == LCM_DSI_FORMAT_RGB666)
  5122. data_config->lcm_bpp = 18;
  5123. } else if (lcm_param->type == LCM_TYPE_DPI) {
  5124. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB888)
  5125. data_config->lcm_bpp = 24;
  5126. else if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB565)
  5127. data_config->lcm_bpp = 16;
  5128. if (lcm_param->dpi.format == LCM_DPI_FORMAT_RGB666)
  5129. data_config->lcm_bpp = 18;
  5130. }
  5131. data_config->fps = pgc->lcm_fps;
  5132. data_config->dst_dirty = 1;
  5133. if (_is_decouple_mode(pgc->session_mode)) {
  5134. data_config = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  5135. data_config->fps = pgc->lcm_fps;
  5136. data_config->dst_dirty = 1;
  5137. data_config->wdma_dirty = 1;
  5138. ret = dpmgr_path_config(pgc->ovl2mem_path_handle, data_config, NULL);
  5139. }
  5140. }
  5141. DISPCHECK("[POWER]ovl2mem path start[begin]\n");
  5142. if (_is_decouple_mode(pgc->session_mode))
  5143. dpmgr_path_start(pgc->ovl2mem_path_handle, CMDQ_DISABLE);
  5144. DISPCHECK("[POWER]ovl2mem path start[end]\n");
  5145. DISPCHECK("[POWER]start cmdq[begin]\n");
  5146. _cmdq_start_trigger_loop();
  5147. DISPCHECK("[POWER]start cmdq[end]\n");
  5148. if (is_mmdvfs_supported() && mmdvfs_get_mmdvfs_profile() == MMDVFS_PROFILE_D1_PLUS &&
  5149. primary_display_get_width() > 800) {
  5150. DISPMSG("set MMDVFS high\n");
  5151. mmdvfs_set_step(MMDVFS_SCEN_DISP, MMDVFS_VOLTAGE_HIGH); /* Enter HPM mode */
  5152. }
  5153. /* primary_display_diagnose(); */
  5154. #ifndef DISP_NO_AEE
  5155. aee_kernel_wdt_kick_Powkey_api("mtkfb_late_resume", WDT_SETBY_Display);
  5156. #endif
  5157. MMProfileLogEx(ddp_mmp_get_events()->primary_resume, MMProfileFlagEnd, 0, 0);
  5158. ddp_dump_analysis(DISP_MODULE_OVL0);
  5159. pgc->force_on_wdma_path = 1;
  5160. return 0;
  5161. }
  5162. int primary_display_ipoh_restore(void)
  5163. {
  5164. DISPMSG("primary_display_ipoh_restore In\n");
  5165. DISPCHECK("ESD check stop[begin]\n");
  5166. primary_display_esd_check_enable(0);
  5167. DISPCHECK("ESD check stop[end]\n");
  5168. if (NULL != pgc->cmdq_handle_trigger) {
  5169. struct TaskStruct *pTask = pgc->cmdq_handle_trigger->pRunningTask;
  5170. if (NULL != pTask) {
  5171. DISPCHECK("[Primary_display]display cmdq trigger loop stop[begin]\n");
  5172. _cmdq_stop_trigger_loop();
  5173. DISPCHECK("[Primary_display]display cmdq trigger loop stop[end]\n");
  5174. }
  5175. }
  5176. DISPMSG("primary_display_ipoh_restore Out\n");
  5177. return 0;
  5178. }
  5179. int primary_display_ipoh_recover(void)
  5180. {
  5181. DISPMSG("%s In\n", __func__);
  5182. _cmdq_start_trigger_loop();
  5183. DISPMSG("%s Out\n", __func__);
  5184. return 0;
  5185. }
  5186. int primary_display_start(void)
  5187. {
  5188. DISP_STATUS ret = DISP_STATUS_OK;
  5189. DISPFUNC();
  5190. _primary_path_lock(__func__);
  5191. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  5192. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  5193. DISPCHECK("Fatal error, we didn't trigger display path but it's already busy\n");
  5194. ret = -1;
  5195. goto done;
  5196. }
  5197. done:
  5198. _primary_path_unlock(__func__);
  5199. return ret;
  5200. }
  5201. int primary_display_stop(void)
  5202. {
  5203. DISP_STATUS ret = DISP_STATUS_OK;
  5204. DISPFUNC();
  5205. _primary_path_lock(__func__);
  5206. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  5207. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  5208. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  5209. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  5210. DISPCHECK("stop display path failed, still busy\n");
  5211. ret = -1;
  5212. goto done;
  5213. }
  5214. done:
  5215. _primary_path_unlock(__func__);
  5216. return ret;
  5217. }
  5218. static int primary_display_remove_output(void *callback, unsigned int userdata)
  5219. {
  5220. int ret = 0;
  5221. static cmdqRecHandle cmdq_handle;
  5222. static cmdqRecHandle cmdq_wait_handle;
  5223. /*create config thread */
  5224. if (cmdq_handle == NULL)
  5225. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  5226. if (ret == 0) {
  5227. /* capture thread wait wdma sof */
  5228. ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_MIRROR_MODE, &cmdq_wait_handle);
  5229. if (ret == 0) {
  5230. cmdqRecReset(cmdq_wait_handle);
  5231. cmdqRecWait(cmdq_wait_handle, CMDQ_EVENT_DISP_WDMA0_SOF);
  5232. cmdqRecFlush(cmdq_wait_handle);
  5233. /* cmdqRecDestroy(cmdq_wait_handle); */
  5234. } else {
  5235. DISPERR("fail to create wait handle\n");
  5236. }
  5237. cmdqRecReset(cmdq_handle);
  5238. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  5239. /* update output fence */
  5240. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->cur_config_fence,
  5241. disp_sync_get_output_timeline_id(), mem_config.buff_idx);
  5242. dpmgr_path_remove_memout(pgc->dpmgr_handle, cmdq_handle);
  5243. cmdqRecClearEventToken(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_SOF);
  5244. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  5245. cmdqRecFlushAsyncCallback(cmdq_handle, callback, 0);
  5246. pgc->need_trigger_ovl1to2 = 0;
  5247. /* cmdqRecDestroy(cmdq_handle); */
  5248. } else {
  5249. ret = -1;
  5250. DISPERR("fail to remove memout out\n");
  5251. }
  5252. return ret;
  5253. }
  5254. static bool is_multipass_trigger;
  5255. int primary_display_merge_session_cmd(disp_session_config *config)
  5256. {
  5257. disp_session_input_config *session_input;
  5258. disp_mem_output_config *primary_output;
  5259. unsigned int output_type = config->dc_type;
  5260. if (DISP_OUTPUT_UNKNOWN == output_type)
  5261. output_type = DISP_OUTPUT_DECOUPLE;
  5262. pgc->dc_type = output_type;
  5263. #ifdef CONFIG_ALL_IN_TRIGGER_STAGE
  5264. mutex_lock(&session_config_mutex);
  5265. session_input = &cached_session_input[config->type - 1];
  5266. cached_session_input[config->type - 1] = captured_session_input[config->type - 1];
  5267. cached_session_input[config->type - 1].session_id = config->session_id;
  5268. if (output_type == DISP_OUTPUT_MEMORY || is_multipass_trigger)
  5269. cached_session_output[config->type - 1] = captured_session_output[config->type - 1];
  5270. mutex_unlock(&session_config_mutex);
  5271. session_input->config_layer_num = OVL_LAYER_NUM;
  5272. if (isAEEEnabled) {
  5273. int aee_layer = primary_display_get_option("ASSERT_LAYER");
  5274. /*
  5275. For multipass mode, the AEE layer can only be enabled at final pass when AEE enabled.
  5276. Otherwise the AEE layer would be composed several time.
  5277. */
  5278. session_input->setter = SESSION_USER_AEE;
  5279. if (config->type == DISP_SESSION_PRIMARY) {
  5280. if (output_type == DISP_OUTPUT_MEMORY) {
  5281. session_input->config[aee_layer].layer_enable = 0;
  5282. DISPMSG("set the aee layer:%d en to 0\n",
  5283. session_input->config[aee_layer].layer_id);
  5284. } else {
  5285. session_input->config[aee_layer].layer_enable = 1;
  5286. }
  5287. }
  5288. }
  5289. primary_display_config_input_multiple(session_input);
  5290. if (output_type == DISP_OUTPUT_MEMORY || pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE ||
  5291. config->type == DISP_SESSION_MEMORY) {
  5292. mutex_lock(&session_config_mutex);
  5293. cached_session_output[config->type - 1] = captured_session_output[config->type - 1];
  5294. mutex_unlock(&session_config_mutex);
  5295. primary_output = &cached_session_output[config->type - 1];
  5296. primary_display_config_output(primary_output, config->session_id);
  5297. }
  5298. #endif
  5299. return 0;
  5300. }
  5301. void primary_display_update_present_fence(unsigned int fence_idx)
  5302. {
  5303. gPresentFenceIndex = fence_idx;
  5304. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  5305. if (pgc->state == DISP_SLEPT)
  5306. primary_suspend_release_present_fence();
  5307. #endif
  5308. }
  5309. static int config_wdma_output(disp_path_handle disp_handle, cmdqRecHandle cmdq_handle,
  5310. disp_mem_output_config *output, int is_multipass);
  5311. int primary_display_trigger(int blocking, void *callback, unsigned int userdata)
  5312. {
  5313. int ret = 0;
  5314. last_primary_trigger_time = sched_clock();
  5315. #ifdef DISP_SWITCH_DST_MODE
  5316. if (is_switched_dst_mode) {
  5317. primary_display_switch_dst_mode(1); /* swith to vdo mode if trigger disp */
  5318. is_switched_dst_mode = false;
  5319. }
  5320. #endif
  5321. if (gTriggerDispMode > 0) {
  5322. primary_display_release_fence_fake();
  5323. return ret;
  5324. }
  5325. primary_trigger_cnt++;
  5326. _primary_path_lock(__func__);
  5327. if (pgc->state == DISP_SLEPT) {
  5328. DISPMSG("%s, skip because primary dipslay is sleep\n", __func__);
  5329. goto done;
  5330. }
  5331. if (blocking)
  5332. DISPMSG("%s, change blocking to non blocking trigger\n", __func__);
  5333. _primary_path_set_dvfsHPM(true, 0);
  5334. #ifdef MTK_DISP_IDLE_LP
  5335. _disp_primary_path_exit_idle(__func__, 0);
  5336. #endif
  5337. dprec_logger_start(DPREC_LOGGER_PRIMARY_TRIGGER, pgc->session_mode, pgc->dc_type);
  5338. if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  5339. _trigger_display_interface(blocking, _ovl_fence_release_callback,
  5340. DISP_SESSION_DIRECT_LINK_MODE);
  5341. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE) {
  5342. _trigger_display_interface(0, _ovl_fence_release_callback,
  5343. DISP_SESSION_DIRECT_LINK_MIRROR_MODE);
  5344. if (pgc->need_trigger_ovl1to2 == 0) {
  5345. DISPPR_ERROR("There is no output config when directlink mirror!!\n");
  5346. } else {
  5347. primary_display_remove_output(_wdma_fence_release_callback,
  5348. DISP_SESSION_DIRECT_LINK_MIRROR_MODE);
  5349. pgc->need_trigger_ovl1to2 = 0;
  5350. }
  5351. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  5352. if (pgc->need_trigger_dcMirror_out == 0) {
  5353. if (cached_session_output[DISP_SESSION_PRIMARY - 1].security == DISP_SECURE_BUFFER)
  5354. config_wdma_output(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config,
  5355. &cached_session_output[DISP_SESSION_PRIMARY - 1], 0);
  5356. DISPPR_ERROR("There is no output config when decouple mirror!!\n");
  5357. } else {
  5358. #ifdef CONFIG_ALL_IN_TRIGGER_STAGE
  5359. is_output_buffer_set = 0;
  5360. #endif
  5361. pgc->need_trigger_dcMirror_out = 0;
  5362. _trigger_ovl_to_memory_mirror(pgc->ovl2mem_path_handle,
  5363. pgc->cmdq_handle_ovl1to2_config,
  5364. (fence_release_callback)
  5365. _olv_wdma_fence_release_callback,
  5366. DISP_SESSION_DECOUPLE_MIRROR_MODE);
  5367. }
  5368. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE) {
  5369. if (pgc->dc_type == DISP_OUTPUT_DECOUPLE) {
  5370. uint32_t writing_mva = 0;
  5371. if (primary_display_is_secure_path(DISP_SESSION_PRIMARY)) {
  5372. pgc->session_buf_id++;
  5373. pgc->session_buf_id %= DISP_INTERNAL_BUFFER_COUNT;
  5374. writing_mva = pgc->session_buf[pgc->session_buf_id];
  5375. mem_config.security = DISP_SECURE_BUFFER;
  5376. decouple_wdma_config.security = DISP_SECURE_BUFFER;
  5377. } else {
  5378. pgc->dc_buf_id++;
  5379. pgc->dc_buf_id %= DISP_INTERNAL_BUFFER_COUNT;
  5380. writing_mva = pgc->dc_buf[pgc->dc_buf_id];
  5381. mem_config.security = DISP_NORMAL_BUFFER;
  5382. decouple_wdma_config.security = DISP_NORMAL_BUFFER;
  5383. }
  5384. decouple_wdma_config.dstAddress = writing_mva;
  5385. mem_config.addr = writing_mva;
  5386. mem_config.fmt = decouple_wdma_config.outputFormat;
  5387. mem_config.pitch = decouple_wdma_config.dstPitch;
  5388. _config_wdma_output(&decouple_wdma_config, pgc->ovl2mem_path_handle,
  5389. pgc->cmdq_handle_ovl1to2_config);
  5390. MMProfileLogEx(ddp_mmp_get_events()->primary_wdma_config,
  5391. MMProfileFlagPulse, pgc->dc_buf_id, writing_mva);
  5392. DISPMSG("%s, config wdma address 0x%lx\n", __func__,
  5393. decouple_wdma_config.dstAddress);
  5394. /*
  5395. * At the last round of multipass, the session mode would be configured as DECOUPLE MODE.
  5396. * Need to set data as DISP_SESSION_DECOUPLE_MULTIPASS_MODE to _ovl_fence_release_callback.
  5397. * Then the multipass output buffer fence would be release in _ovl_fence_release_callback.
  5398. */
  5399. if (is_multipass_trigger) {
  5400. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config,
  5401. (fence_release_callback)_ovl_fence_release_callback, 5, 0);
  5402. is_multipass_trigger = 0;
  5403. } else {
  5404. if (primary_display_is_secure_path(DISP_SESSION_PRIMARY)) {
  5405. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle,
  5406. pgc->cmdq_handle_ovl1to2_config,
  5407. (fence_release_callback)_ovl_fence_release_callback,
  5408. DISP_SESSION_DECOUPLE_MODE, 1);
  5409. _ovl_fence_release_callback(DISP_SESSION_DECOUPLE_MODE);
  5410. } else {
  5411. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle,
  5412. pgc->cmdq_handle_ovl1to2_config,
  5413. (fence_release_callback)_ovl_fence_release_callback,
  5414. DISP_SESSION_DECOUPLE_MODE, 0);
  5415. }
  5416. }
  5417. } else if (pgc->dc_type == DISP_OUTPUT_MEMORY) {
  5418. is_multipass_trigger = 1;
  5419. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config, NULL, 0, 0);
  5420. }
  5421. }
  5422. dprec_logger_done(DPREC_LOGGER_PRIMARY_TRIGGER, 0, 0);
  5423. done:
  5424. _primary_path_unlock(__func__);
  5425. /* FIXME: find aee_kernel_Powerkey_is_press definitation */
  5426. #ifndef DISP_NO_AEE
  5427. if ((primary_trigger_cnt > PRIMARY_DISPLAY_TRIGGER_CNT) && aee_kernel_Powerkey_is_press()) {
  5428. aee_kernel_wdt_kick_Powkey_api("primary_display_trigger", WDT_SETBY_Display);
  5429. primary_trigger_cnt = 0;
  5430. }
  5431. #endif
  5432. if (pgc->session_id > 0)
  5433. update_frm_seq_info(0, 0, 0, FRM_TRIGGER);
  5434. return ret;
  5435. }
  5436. int primary_display_memory_trigger(int blocking, void *callback, unsigned int userdata)
  5437. {
  5438. int ret = 0;
  5439. _primary_path_lock(__func__);
  5440. if (pgc->state == DISP_SLEPT) {
  5441. DISPMSG("%s, primary dipslay is sleep\n", __func__);
  5442. if (pgc->force_on_wdma_path == 0)
  5443. primary_display_resume_ovl2mem();
  5444. }
  5445. if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE || pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  5446. if (primary_display_is_secure_path(DISP_SESSION_MEMORY)) {
  5447. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config,
  5448. (fence_release_callback)_ovl_ext_fence_release_callback,
  5449. DISP_SESSION_MEMORY, 1);
  5450. _ovl_ext_fence_release_callback(DISP_SESSION_MEMORY);
  5451. } else {
  5452. _trigger_ovl_to_memory(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config,
  5453. (fence_release_callback)_ovl_ext_fence_release_callback,
  5454. DISP_SESSION_MEMORY, 0);
  5455. }
  5456. } else {
  5457. DISPMSG("Not support memory trigger using session_mode:%d\n", pgc->session_mode);
  5458. }
  5459. _primary_path_unlock(__func__);
  5460. return ret;
  5461. }
  5462. static int primary_display_ovl2mem_callback(unsigned int userdata)
  5463. {
  5464. disp_ddp_path_config *data_config = NULL;
  5465. unsigned int session_id = 0;
  5466. int fence_idx = userdata;
  5467. WDMA0_FRAME_START_FLAG = 0;
  5468. session_id = MAKE_DISP_SESSION(DISP_SESSION_PRIMARY, 0);
  5469. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  5470. if (data_config) {
  5471. WDMA_CONFIG_STRUCT wdma_layer;
  5472. wdma_layer.dstAddress = mtkfb_query_buf_mva(session_id, 4, fence_idx);
  5473. wdma_layer.outputFormat = data_config->wdma_config.outputFormat;
  5474. wdma_layer.srcWidth = primary_display_get_width();
  5475. wdma_layer.srcHeight = primary_display_get_height();
  5476. wdma_layer.dstPitch = data_config->wdma_config.dstPitch;
  5477. dprec_mmp_dump_wdma_layer(&wdma_layer, 0);
  5478. }
  5479. if (fence_idx > 0)
  5480. mtkfb_release_fence(session_id, EXTERNAL_DISPLAY_SESSION_LAYER_COUNT, fence_idx);
  5481. #ifdef OVL_CASCADE_SUPPORT
  5482. if (fence_idx < NEW_BUF_IDX && ALL_LAYER_DISABLE_STEP == 1) {
  5483. DISPMSG("primary and memout does not match!!\n");
  5484. cmdqCoreSetEvent(CMDQ_EVENT_DISP_WDMA0_SOF);
  5485. if (_should_set_cmdq_dirty())
  5486. _cmdq_set_config_handle_dirty_mira(pgc->cmdq_handle_ovl1to2_config);
  5487. if ((fence_idx + 1) == NEW_BUF_IDX)
  5488. ALL_LAYER_DISABLE_STEP = 0;
  5489. }
  5490. #endif
  5491. DISPMSG("mem_out release fence idx:0x%x\n", fence_idx);
  5492. return 0;
  5493. }
  5494. /* extern unsigned int gDumpMemoutCmdq; */
  5495. int primary_display_mem_out_trigger(int blocking, void *callback, unsigned int userdata)
  5496. {
  5497. int ret = 0;
  5498. /* DISPFUNC(); */
  5499. if (pgc->state == DISP_SLEPT || !_is_mirror_mode(pgc->session_mode)) {
  5500. DISPMSG("mem out trigger is already slept or is not mirror mode(%d)\n",
  5501. pgc->session_mode);
  5502. return 0;
  5503. }
  5504. /* /dprec_logger_start(DPREC_LOGGER_PRIMARY_TRIGGER, 0, 0); */
  5505. /* if(blocking) */
  5506. _primary_path_lock(__func__);
  5507. if (pgc->need_trigger_ovl1to2 == 0)
  5508. goto done;
  5509. NEW_BUF_IDX = userdata;
  5510. ALL_LAYER_DISABLE_STEP = 0;
  5511. if (_should_wait_path_idle())
  5512. dpmgr_wait_event_timeout(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_DONE,
  5513. HZ * 1);
  5514. if (_should_trigger_path())
  5515. ;/* /dpmgr_path_trigger(pgc->dpmgr_handle, NULL, primary_display_cmdq_enabled()); */
  5516. if (_should_set_cmdq_dirty())
  5517. _cmdq_set_config_handle_dirty_mira(pgc->cmdq_handle_ovl1to2_config);
  5518. if (gDumpMemoutCmdq == 1) {
  5519. DISPMSG("primary_display_mem_out_trigger, dump before flush 1:\n");
  5520. cmdqRecDumpCommand(pgc->cmdq_handle_ovl1to2_config);
  5521. }
  5522. if (_should_flush_cmdq_config_handle())
  5523. _cmdq_flush_config_handle_mira(pgc->cmdq_handle_ovl1to2_config, 0);
  5524. if (_should_reset_cmdq_config_handle())
  5525. cmdqRecReset(pgc->cmdq_handle_ovl1to2_config);
  5526. cmdqRecWait(pgc->cmdq_handle_ovl1to2_config, CMDQ_EVENT_DISP_WDMA0_SOF);
  5527. WDMA0_FRAME_START_FLAG = 1;
  5528. _cmdq_insert_wait_frame_done_token_mira(pgc->cmdq_handle_ovl1to2_config);
  5529. dpmgr_path_remove_memout(pgc->ovl2mem_path_handle, pgc->cmdq_handle_ovl1to2_config);
  5530. if (gDumpMemoutCmdq == 1) {
  5531. DISPMSG("primary_display_mem_out_trigger, dump before flush 2:\n");
  5532. cmdqRecDumpCommand(pgc->cmdq_handle_ovl1to2_config);
  5533. }
  5534. if (_should_flush_cmdq_config_handle())
  5535. /* _cmdq_flush_config_handle_mira(pgc->cmdq_handle_ovl1to2_config, 0); */
  5536. cmdqRecFlushAsyncCallback(pgc->cmdq_handle_ovl1to2_config,
  5537. (CmdqAsyncFlushCB)primary_display_ovl2mem_callback,
  5538. userdata);
  5539. if (_should_reset_cmdq_config_handle())
  5540. cmdqRecReset(pgc->cmdq_handle_ovl1to2_config);
  5541. /* _cmdq_insert_wait_frame_done_token_mira(pgc->cmdq_handle_ovl1to2_config); */
  5542. done:
  5543. pgc->need_trigger_ovl1to2 = 0;
  5544. _primary_path_unlock(__func__);
  5545. /* dprec_logger_done(DPREC_LOGGER_PRIMARY_TRIGGER, 0, 0); */
  5546. return ret;
  5547. }
  5548. static int config_wdma_output(disp_path_handle disp_handle,
  5549. cmdqRecHandle cmdq_handle,
  5550. disp_mem_output_config *output, int is_multipass)
  5551. {
  5552. disp_ddp_path_config *pconfig = NULL;
  5553. ASSERT(output != NULL);
  5554. pconfig = dpmgr_path_get_last_config(disp_handle);
  5555. pconfig->wdma_config.dstAddress = output->addr;
  5556. pconfig->wdma_config.srcHeight = output->h;
  5557. pconfig->wdma_config.srcWidth = output->w;
  5558. pconfig->wdma_config.clipX = output->x;
  5559. pconfig->wdma_config.clipY = output->y;
  5560. pconfig->wdma_config.clipHeight = output->h;
  5561. pconfig->wdma_config.clipWidth = output->w;
  5562. pconfig->wdma_config.outputFormat = output->fmt;
  5563. pconfig->wdma_config.alpha = 0xFF;
  5564. pconfig->wdma_config.dstPitch = output->pitch;
  5565. pconfig->wdma_config.security = output->security;
  5566. pconfig->wdma_dirty = 1;
  5567. pconfig->dst_dirty = 1;
  5568. pconfig->dst_h = output->h;
  5569. pconfig->dst_w = output->w;
  5570. if (is_multipass)
  5571. pconfig->wdma_config.useSpecifiedAlpha = 0;
  5572. else
  5573. pconfig->wdma_config.useSpecifiedAlpha = 1;
  5574. return dpmgr_path_config(disp_handle, pconfig, cmdq_handle);
  5575. }
  5576. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  5577. int primary_suspend_outputbuf_fence_release(void)
  5578. {
  5579. int layer = 0;
  5580. layer = disp_sync_get_output_timeline_id();
  5581. mtkfb_release_layer_fence(primary_session_id, layer);
  5582. /* release rdma buffer */
  5583. layer = disp_sync_get_output_interface_timeline_id();
  5584. mtkfb_release_layer_fence(primary_session_id, layer);
  5585. return 0;
  5586. }
  5587. #endif
  5588. int primary_display_config_output(disp_mem_output_config *output, unsigned int session_id)
  5589. {
  5590. int ret = 0;
  5591. disp_path_handle disp_handle;
  5592. cmdqRecHandle cmdq_handle = NULL;
  5593. DISPFUNC();
  5594. _primary_path_lock(__func__);
  5595. if (pgc->state == DISP_SLEPT && DISP_SESSION_TYPE(session_id) != DISP_SESSION_MEMORY) {
  5596. DISPMSG("mem out is already slept or mode wrong(%d)\n", pgc->session_mode);
  5597. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  5598. primary_suspend_outputbuf_fence_release();
  5599. #endif
  5600. goto done;
  5601. }
  5602. #if !defined(OVL_MULTIPASS_SUPPORT) && !defined(OVL_TIME_SHARING)
  5603. if (!_is_mirror_mode(pgc->session_mode)) {
  5604. DISPERR("should not config output if not mirror mode!!\n");
  5605. goto done;
  5606. }
  5607. #endif
  5608. if (_is_decouple_mode(pgc->session_mode)) {
  5609. disp_handle = pgc->ovl2mem_path_handle;
  5610. cmdq_handle = pgc->cmdq_handle_ovl1to2_config;
  5611. } else {
  5612. disp_handle = pgc->dpmgr_handle;
  5613. cmdq_handle = pgc->cmdq_handle_config;
  5614. }
  5615. if (_is_decouple_mode(pgc->session_mode)) {
  5616. #ifndef CONFIG_ALL_IN_TRIGGER_STAGE
  5617. if (_is_mirror_mode(pgc->session_mode))
  5618. pgc->need_trigger_dcMirror_out = 1;
  5619. #else
  5620. if (_is_mirror_mode(pgc->session_mode) && is_output_buffer_set)
  5621. pgc->need_trigger_dcMirror_out = 1;
  5622. #endif
  5623. } else {
  5624. /* direct link mirror mode should add memout first */
  5625. dpmgr_path_add_memout(pgc->dpmgr_handle, ENGINE_OVL0, cmdq_handle);
  5626. pgc->need_trigger_ovl1to2 = 1;
  5627. }
  5628. if (_is_decouple_mode(pgc->session_mode) && !_is_mirror_mode(pgc->session_mode))
  5629. /* Multipass Trigger */
  5630. ret = config_wdma_output(disp_handle, cmdq_handle, output, 1);
  5631. else
  5632. ret = config_wdma_output(disp_handle, cmdq_handle, output, 0);
  5633. if ((pgc->session_id > 0) && _is_decouple_mode(pgc->session_mode))
  5634. update_frm_seq_info(output->addr, 0, mtkfb_query_frm_seq_by_addr(pgc->session_id, 0, 0), FRM_CONFIG);
  5635. if (DISP_SESSION_TYPE(session_id) == DISP_SESSION_PRIMARY) {
  5636. mem_config = *output;
  5637. } else {
  5638. int layer;
  5639. layer = disp_sync_get_output_timeline_id();
  5640. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->cur_mem_config_fence, layer, output->buff_idx);
  5641. layer = disp_sync_get_output_interface_timeline_id();
  5642. cmdqRecBackupUpdateSlot(cmdq_handle, pgc->cur_mem_config_fence, layer, output->interface_idx);
  5643. }
  5644. MMProfileLogEx(ddp_mmp_get_events()->primary_wdma_config, MMProfileFlagPulse,
  5645. output->buff_idx, (unsigned int)output->addr);
  5646. done:
  5647. _primary_path_unlock(__func__);
  5648. /* dprec_logger_done(DPREC_LOGGER_PRIMARY_CONFIG, output->src_x, output->src_y); */
  5649. return ret;
  5650. }
  5651. #if 0
  5652. /**
  5653. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  5654. */
  5655. static int _config_interface_input(primary_disp_input_config *input)
  5656. {
  5657. int ret = 0;
  5658. void *cmdq_handle = NULL;
  5659. disp_ddp_path_config *data_config;
  5660. /* all dirty should be cleared in dpmgr_path_get_last_config() */
  5661. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  5662. dprec_logger_start(DPREC_LOGGER_PRIMARY_CONFIG, input->layer|(input->layer_en<<16), input->addr);
  5663. ret = _convert_disp_input_to_rdma(&(data_config->rdma_config), (disp_input_config *)input);
  5664. data_config->rdma_dirty = 1;
  5665. if (_should_wait_path_idle()) {
  5666. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  5667. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ*1);
  5668. }
  5669. if (primary_display_cmdq_enabled())
  5670. cmdq_handle = pgc->cmdq_handle_config;
  5671. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, cmdq_handle);
  5672. dprec_logger_done(DPREC_LOGGER_PRIMARY_CONFIG, input->src_x, input->src_y);
  5673. return ret;
  5674. }
  5675. #endif
  5676. static void update_debug_fps_meter(disp_ddp_path_config *data_config)
  5677. {
  5678. int i, dst_id = 0;
  5679. for (i = 0; i < HW_OVERLAY_COUNT; i++) {
  5680. if (data_config->ovl_config[i].layer_en && data_config->ovl_config[i].dst_x == 0 &&
  5681. data_config->ovl_config[i].dst_y == 0)
  5682. dst_id = i;
  5683. }
  5684. _debug_fps_meter(data_config->ovl_config[dst_id].addr,
  5685. data_config->ovl_config[dst_id].vaddr,
  5686. data_config->ovl_config[dst_id].dst_w,
  5687. data_config->ovl_config[dst_id].dst_h,
  5688. data_config->ovl_config[dst_id].src_pitch, 0x00000000, dst_id,
  5689. data_config->ovl_config[dst_id].buff_idx);
  5690. }
  5691. static int _config_ovl_input(disp_session_input_config *session_input,
  5692. disp_path_handle disp_handle, cmdqRecHandle cmdq_handle)
  5693. {
  5694. int ret = 0, i = 0, layer = 0;
  5695. disp_ddp_path_config *data_config = NULL;
  5696. int max_layer_id_configed = 0;
  5697. #ifndef MTK_FB_CMDQ_DISABLE
  5698. int force_disable_ovl1 = 0;
  5699. #endif
  5700. cmdqBackupSlotHandle *p_cur_config_fence;
  5701. cmdqBackupSlotHandle *p_subtractor_when_free;
  5702. LCM_PARAMS *lcm_param = NULL;
  5703. /* create new data_config for ovl input */
  5704. data_config = dpmgr_path_get_last_config(disp_handle);
  5705. if (DISP_SESSION_TYPE(session_input->session_id) == DISP_SESSION_MEMORY) {
  5706. p_cur_config_fence = &(pgc->cur_mem_config_fence);
  5707. p_subtractor_when_free = &(pgc->mem_subtractor_when_free);
  5708. data_config->is_memory = true;
  5709. } else {
  5710. lcm_param = disp_lcm_get_params(pgc->plcm);
  5711. p_cur_config_fence = &(pgc->cur_config_fence);
  5712. p_subtractor_when_free = &(pgc->subtractor_when_free);
  5713. data_config->dst_h = lcm_param->height;
  5714. data_config->dst_w = lcm_param->width;
  5715. data_config->is_memory = false;
  5716. }
  5717. for (i = 0; i < session_input->config_layer_num; i++) {
  5718. disp_input_config *input_cfg = &session_input->config[i];
  5719. OVL_CONFIG_STRUCT *ovl_cfg;
  5720. layer = input_cfg->layer_id;
  5721. /*security issue*/
  5722. if (layer >= OVL_LAYER_NUM)
  5723. continue;
  5724. ovl_cfg = &(data_config->ovl_config[layer]);
  5725. if (session_input->setter != SESSION_USER_AEE) {
  5726. if (isAEEEnabled && layer == primary_display_get_option("ASSERT_LAYER")) {
  5727. DISPMSG("skip AEE layer %d\n", layer);
  5728. continue;
  5729. }
  5730. } else {
  5731. DISPMSG("set AEE layer %d\n", layer);
  5732. }
  5733. _convert_disp_input_to_ovl(ovl_cfg, input_cfg);
  5734. if (ovl_cfg->layer_en)
  5735. _debug_pattern(ovl_cfg->addr, ovl_cfg->vaddr, ovl_cfg->dst_w,
  5736. ovl_cfg->dst_h, ovl_cfg->src_pitch, 0x00000000,
  5737. ovl_cfg->layer, ovl_cfg->buff_idx);
  5738. dprec_logger_start(DPREC_LOGGER_PRIMARY_CONFIG, ovl_cfg->layer | (ovl_cfg->layer_en << 16),
  5739. ovl_cfg->addr);
  5740. dprec_logger_done(DPREC_LOGGER_PRIMARY_CONFIG, ovl_cfg->src_x, ovl_cfg->src_y);
  5741. dprec_mmp_dump_ovl_layer(ovl_cfg, layer, 1);
  5742. if ((ovl_cfg->layer == 0) && (!_is_decouple_mode(pgc->session_mode)))
  5743. update_frm_seq_info(ovl_cfg->addr, ovl_cfg->src_x * 4 + ovl_cfg->src_y * ovl_cfg->src_pitch,
  5744. mtkfb_query_frm_seq_by_addr(pgc->session_id, 0, 0), FRM_CONFIG);
  5745. if (max_layer_id_configed < layer)
  5746. max_layer_id_configed = layer;
  5747. data_config->ovl_dirty = 1;
  5748. #if defined(CONFIG_FOR_SOURCE_PQ)
  5749. data_config->dst_dirty = 1;
  5750. #endif
  5751. #if defined(OVL_TIME_SHARING)
  5752. data_config->roi_dirty = 1;
  5753. #endif
  5754. }
  5755. #ifdef OVL_CASCADE_SUPPORT
  5756. if (ovl_get_status() == DDP_OVL1_STATUS_SUB_REQUESTING) {
  5757. /* disable ovl layer 4~8 to free ovl1 */
  5758. if (max_layer_id_configed < OVL_LAYER_NUM_PER_OVL - is_DAL_Enabled()) {
  5759. for (i = OVL_LAYER_NUM_PER_OVL; i < OVL_LAYER_NUM; i++)
  5760. data_config->ovl_config[i].layer_en = 0;
  5761. force_disable_ovl1 = 1;
  5762. DISPMSG("cascade: HWC set %d layers, force disable OVL1 layers\n",
  5763. max_layer_id_configed);
  5764. } else {
  5765. DISPMSG("cascade: try to split ovl1 fail: HWC set %d layers\n",
  5766. max_layer_id_configed);
  5767. }
  5768. }
  5769. #endif
  5770. if (_should_wait_path_idle())
  5771. dpmgr_wait_event_timeout(disp_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  5772. /* should we use cmdq_hand_config? need to check */
  5773. ret = dpmgr_path_config(disp_handle, data_config, cmdq_handle);
  5774. #ifndef MTK_FB_CMDQ_DISABLE
  5775. /* write fence_id/enable to DRAM using cmdq,
  5776. * it will be used when release fence (put these after config registers done)
  5777. */
  5778. for (i = 0; i < session_input->config_layer_num; i++) {
  5779. unsigned int last_fence, cur_fence;
  5780. disp_input_config *input_cfg = &session_input->config[i];
  5781. layer = input_cfg->layer_id;
  5782. cmdqBackupReadSlot(*p_cur_config_fence, layer, &last_fence);
  5783. cur_fence = input_cfg->next_buff_idx;
  5784. if (cur_fence != -1 && cur_fence > last_fence)
  5785. cmdqRecBackupUpdateSlot(cmdq_handle, *p_cur_config_fence, layer, cur_fence);
  5786. /* for dim_layer/disable_layer/no_fence_layer, just release all fences configured */
  5787. /* for other layers, release current_fence-1 */
  5788. if (input_cfg->buffer_source == DISP_BUFFER_ALPHA || input_cfg->layer_enable == 0 ||
  5789. cur_fence == -1 || DISP_SESSION_TYPE(session_input->session_id) == DISP_SESSION_MEMORY)
  5790. cmdqRecBackupUpdateSlot(cmdq_handle, *p_subtractor_when_free, layer, 0);
  5791. else
  5792. cmdqRecBackupUpdateSlot(cmdq_handle, *p_subtractor_when_free, layer, 1);
  5793. }
  5794. if (force_disable_ovl1) {
  5795. for (layer = OVL_LAYER_NUM_PER_OVL; layer < OVL_LAYER_NUM; layer++)
  5796. /* will release all fences */
  5797. cmdqRecBackupUpdateSlot(cmdq_handle, *p_subtractor_when_free, layer, 0);
  5798. }
  5799. #endif
  5800. update_debug_fps_meter(data_config);
  5801. if (DISP_SESSION_TYPE(session_input->session_id) == DISP_SESSION_PRIMARY) {
  5802. last_primary_config = *data_config;
  5803. is_hwc_update_frame = 1;
  5804. }
  5805. return ret;
  5806. }
  5807. #if 0
  5808. /**
  5809. * Defined but not used, avoid build warning. (2015.1.30 Rynn Wu)
  5810. */
  5811. static int _config_ovl_output(disp_mem_output_config *output)
  5812. {
  5813. int ret = 0;
  5814. disp_ddp_path_config *data_config = NULL;
  5815. disp_path_handle *handle = NULL;
  5816. data_config = dpmgr_path_get_last_config(pgc->ovl2mem_path_handle);
  5817. data_config->wdma_dirty = 1;
  5818. if (_should_wait_path_idle())
  5819. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ*1);
  5820. handle = pgc->ovl2mem_path_handle;
  5821. ret = dpmgr_path_config(pgc->ovl2mem_path_handle, data_config, pgc->cmdq_handle_config);
  5822. return ret;
  5823. }
  5824. #endif
  5825. static int _config_rdma_input(disp_session_input_config *session_input, disp_path_handle *handle)
  5826. {
  5827. int ret;
  5828. disp_ddp_path_config *data_config = NULL;
  5829. /* all dirty should be cleared in dpmgr_path_get_last_config() */
  5830. data_config = dpmgr_path_get_last_config(handle);
  5831. data_config->dst_dirty = 0;
  5832. data_config->ovl_dirty = 0;
  5833. data_config->rdma_dirty = 0;
  5834. data_config->wdma_dirty = 0;
  5835. ret =
  5836. _convert_disp_input_to_rdma(&(data_config->rdma_config),
  5837. (disp_input_config *) session_input);
  5838. data_config->rdma_dirty = 1;
  5839. if (_should_wait_path_idle())
  5840. dpmgr_wait_event_timeout(handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  5841. ret =
  5842. dpmgr_path_config(handle, data_config,
  5843. primary_display_cmdq_enabled() ? pgc->cmdq_handle_config : NULL);
  5844. return ret;
  5845. }
  5846. int primary_display_config_input_multiple(disp_session_input_config *session_input)
  5847. {
  5848. int ret = 0;
  5849. disp_path_handle disp_handle;
  5850. cmdqRecHandle cmdq_handle;
  5851. if (gTriggerDispMode > 0)
  5852. return 0;
  5853. _primary_path_lock(__func__);
  5854. if (primary_get_state() == DISP_SLEPT && DISP_SESSION_TYPE(session_input->session_id) != DISP_SESSION_MEMORY) {
  5855. DISPMSG("%s, skip because primary dipslay is sleep\n", __func__);
  5856. #ifdef CONFIG_SINGLE_PANEL_OUTPUT
  5857. primary_suspend_release_ovl_fence(session_input);
  5858. #endif
  5859. goto done;
  5860. }
  5861. #ifdef MTK_DISP_IDLE_LP
  5862. /* call this in trigger is enough, do not have to call this in config */
  5863. _disp_primary_path_exit_idle(__func__, 0);
  5864. #endif
  5865. if (_is_decouple_mode(pgc->session_mode)) {
  5866. disp_handle = pgc->ovl2mem_path_handle;
  5867. cmdq_handle = pgc->cmdq_handle_ovl1to2_config;
  5868. } else {
  5869. disp_handle = pgc->dpmgr_handle;
  5870. cmdq_handle = pgc->cmdq_handle_config;
  5871. }
  5872. if (_should_config_ovl_input())
  5873. _config_ovl_input(session_input, disp_handle, cmdq_handle);
  5874. else
  5875. _config_rdma_input(session_input, disp_handle);
  5876. done:
  5877. _primary_path_unlock(__func__);
  5878. return ret;
  5879. }
  5880. int primary_display_config_interface_input(primary_disp_input_config *input)
  5881. {
  5882. int ret = 0;
  5883. void *cmdq_handle = NULL;
  5884. disp_ddp_path_config *data_config;
  5885. /* all dirty should be cleared in dpmgr_path_get_last_config() */
  5886. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  5887. dprec_logger_start(DPREC_LOGGER_PRIMARY_CONFIG, input->layer | (input->layer_en << 16),
  5888. input->addr);
  5889. _primary_path_lock(__func__);
  5890. if (pgc->state == DISP_SLEPT) {
  5891. DISPMSG("%s, skip because primary dipslay is sleep\n", __func__);
  5892. goto done;
  5893. }
  5894. ret = _convert_disp_input_to_rdma(&(data_config->rdma_config), (disp_input_config *) input);
  5895. data_config->rdma_dirty = 1;
  5896. if (_should_wait_path_idle()) {
  5897. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  5898. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE,
  5899. HZ * 1);
  5900. }
  5901. if (primary_display_cmdq_enabled())
  5902. cmdq_handle = pgc->cmdq_handle_config;
  5903. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, cmdq_handle);
  5904. /* this is used for decouple mode, to indicate whether we need to trigger ovl */
  5905. pgc->need_trigger_overlay = 1;
  5906. dprec_logger_done(DPREC_LOGGER_PRIMARY_CONFIG, input->src_x, input->src_y);
  5907. done:
  5908. _primary_path_unlock(__func__);
  5909. return ret;
  5910. }
  5911. static int Panel_Master_primary_display_config_dsi(const char *name, uint32_t config_value)
  5912. {
  5913. int ret = 0;
  5914. disp_ddp_path_config *data_config;
  5915. /* all dirty should be cleared in dpmgr_path_get_last_config() */
  5916. data_config = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  5917. /* modify below for config dsi */
  5918. if (!strcmp(name, "PM_CLK")) {
  5919. pr_debug("Pmaster_config_dsi: PM_CLK:%d\n", config_value);
  5920. data_config->dispif_config.dsi.PLL_CLOCK = config_value;
  5921. } else if (!strcmp(name, "PM_SSC")) {
  5922. data_config->dispif_config.dsi.ssc_range = config_value;
  5923. }
  5924. pr_debug("Pmaster_config_dsi: will Run path_config()\n");
  5925. ret = dpmgr_path_config(pgc->dpmgr_handle, data_config, NULL);
  5926. return ret;
  5927. }
  5928. int primary_display_user_cmd(unsigned int cmd, unsigned long arg)
  5929. {
  5930. int ret = 0;
  5931. cmdqRecHandle handle = NULL;
  5932. int cmdqsize = 0;
  5933. MMProfileLogEx(ddp_mmp_get_events()->primary_display_cmd, MMProfileFlagStart,
  5934. (unsigned long)handle, 0);
  5935. #ifndef MTK_FB_CMDQ_DISABLE
  5936. #if 0 /* CONFIG_FOR_SOURCE_PQ */
  5937. if (primary_display_is_decouple_mode()) {
  5938. ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_COLOR, &handle);
  5939. cmdqRecReset(handle);
  5940. cmdqRecWait(handle, CMDQ_EVENT_DISP_WDMA0_EOF);
  5941. } else {
  5942. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &handle);
  5943. cmdqRecReset(handle);
  5944. _cmdq_insert_wait_frame_done_token_mira(handle);
  5945. }
  5946. #else
  5947. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &handle);
  5948. cmdqRecReset(handle);
  5949. if (primary_display_is_video_mode())
  5950. cmdqRecWaitNoClear(handle, CMDQ_EVENT_DISP_RDMA0_EOF);
  5951. else
  5952. cmdqRecWaitNoClear(handle, CMDQ_SYNC_TOKEN_STREAM_EOF);
  5953. #endif
  5954. cmdqsize = cmdqRecGetInstructionCount(handle);
  5955. #ifdef MTK_DISP_IDLE_LP
  5956. /* will write register in dpmgr_path_user_cmd, need to exit idle */
  5957. last_primary_trigger_time = sched_clock();
  5958. _disp_primary_path_exit_idle(__func__, 1);
  5959. #endif
  5960. #ifdef CONFIG_FOR_SOURCE_PQ
  5961. if (primary_display_is_decouple_mode())
  5962. ret = dpmgr_path_user_cmd(pgc->ovl2mem_path_handle, cmd, arg, handle);
  5963. else
  5964. ret = dpmgr_path_user_cmd(pgc->dpmgr_handle, cmd, arg, handle);
  5965. #else
  5966. ret = dpmgr_path_user_cmd(pgc->dpmgr_handle, cmd, arg, handle);
  5967. #endif
  5968. if (handle) {
  5969. if (cmdqRecGetInstructionCount(handle) > cmdqsize) {
  5970. _primary_path_lock(__func__);
  5971. if (pgc->state == DISP_ALIVE) {
  5972. #if 0 /* CONFIG_FOR_SOURCE_PQ */
  5973. if (primary_display_is_decouple_mode()) {
  5974. dpmgr_path_trigger(pgc->ovl2mem_path_handle, handle,
  5975. CMDQ_ENABLE);
  5976. cmdqRecFlushAsync(handle);
  5977. } else {
  5978. _cmdq_set_config_handle_dirty_mira(handle);
  5979. /* use non-blocking flush here to avoid primary path is locked for too long */
  5980. _cmdq_flush_config_handle_mira(handle, 0);
  5981. }
  5982. #else
  5983. _cmdq_set_config_handle_dirty_mira(handle);
  5984. /* use non-blocking flush here to avoid primary path is locked for too long */
  5985. _cmdq_flush_config_handle_mira(handle, 0);
  5986. #endif
  5987. }
  5988. _primary_path_unlock(__func__);
  5989. }
  5990. cmdqRecDestroy(handle);
  5991. }
  5992. #endif
  5993. MMProfileLogEx(ddp_mmp_get_events()->primary_display_cmd, MMProfileFlagEnd,
  5994. (unsigned long)handle, cmdqsize);
  5995. return ret;
  5996. }
  5997. int init_ext_decouple_buffers(void)
  5998. {
  5999. int ret = 0;
  6000. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  6001. _primary_path_lock(__func__);
  6002. if (pgc->state == DISP_SLEPT) {
  6003. ret = -1;
  6004. } else {
  6005. if (decouple_buffer_info[0] == NULL)
  6006. ret = allocate_idle_lp_dc_buffer();
  6007. }
  6008. _primary_path_unlock(__func__);
  6009. #endif
  6010. return ret;
  6011. }
  6012. int deinit_ext_decouple_buffers(void)
  6013. {
  6014. int ret = 0;
  6015. #if defined(CONFIG_MTK_GMO_RAM_OPTIMIZE)
  6016. _primary_path_lock(__func__);
  6017. ret = release_idle_lp_dc_buffer(0);
  6018. _primary_path_unlock(__func__);
  6019. #endif
  6020. return ret;
  6021. }
  6022. int primary_display_switch_mode(int sess_mode, unsigned int session, int force)
  6023. {
  6024. DISPDBG("primary_display_switch_mode sess_mode %d, session 0x%x\n", sess_mode, session);
  6025. _primary_path_lock(__func__);
  6026. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagStart,
  6027. pgc->session_mode, sess_mode);
  6028. if (primary_get_state() == DISP_BLANK)
  6029. sess_mode = DISP_SESSION_DECOUPLE_MODE;
  6030. if (pgc->session_mode == sess_mode)
  6031. goto done;
  6032. if (pgc->state == DISP_SLEPT) {
  6033. DISPMSG("primary display switch from %s to %s in suspend state!!!\n",
  6034. session_mode_spy(pgc->session_mode), session_mode_spy(sess_mode));
  6035. #ifdef OVL_TIME_SHARING
  6036. /*
  6037. Allow system to switch mode between Deouple and Decouple Mirror when time-sharing support.
  6038. Since system may use extension path to output data even when the primay path has suspended.
  6039. */
  6040. if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE
  6041. && sess_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  6042. pgc->session_mode = sess_mode;
  6043. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6044. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE
  6045. && sess_mode == DISP_SESSION_DECOUPLE_MODE) {
  6046. pgc->session_mode = sess_mode;
  6047. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6048. }
  6049. #endif
  6050. goto done;
  6051. }
  6052. DISPMSG("primary display will switch from %s to %s\n", session_mode_spy(pgc->session_mode),
  6053. session_mode_spy(sess_mode));
  6054. if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE
  6055. && sess_mode == DISP_SESSION_DECOUPLE_MODE) {
  6056. /* dl to dc */
  6057. _DL_switch_to_DC_fast();
  6058. pgc->session_mode = sess_mode;
  6059. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6060. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6061. pgc->session_mode, sess_mode);
  6062. /* primary_display_diagnose(); */
  6063. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE
  6064. && sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6065. /* dc to dl */
  6066. _DC_switch_to_DL_fast();
  6067. pgc->session_mode = sess_mode;
  6068. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6069. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6070. pgc->session_mode, sess_mode);
  6071. /* primary_display_diagnose(); */
  6072. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE
  6073. && sess_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE) {
  6074. /* dl to dl mirror */
  6075. /* cmdqRecCreate(CMDQ_SCENARIO_DISP_SCREEN_CAPTURE, &pgc->cmdq_handle_dl_mirror); */
  6076. /* cmdqRecReset(pgc->cmdq_handle_dl_mirror); */
  6077. pgc->session_mode = sess_mode;
  6078. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6079. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6080. pgc->session_mode, sess_mode);
  6081. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE
  6082. && sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6083. /*dl mirror to dl */
  6084. /* cmdqRecDestroy(pgc->cmdq_handle_dl_mirror); */
  6085. /* pgc->cmdq_handle_dl_mirror = NULL; */
  6086. pgc->session_mode = sess_mode;
  6087. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6088. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6089. pgc->session_mode, sess_mode);
  6090. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE
  6091. && sess_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  6092. /* dl to dc mirror mirror */
  6093. #ifdef CONFIG_MTK_GMO_RAM_OPTIMIZE
  6094. if (decouple_buffer_info[0] == NULL)
  6095. allocate_idle_lp_dc_buffer();
  6096. #endif
  6097. _DL_switch_to_DC_fast();
  6098. pgc->session_mode = sess_mode;
  6099. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6100. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6101. pgc->session_mode, sess_mode);
  6102. /* primary_display_diagnose(); */
  6103. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE
  6104. && sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6105. /*dc mirror to dl */
  6106. _DC_switch_to_DL_fast();
  6107. pgc->session_mode = sess_mode;
  6108. #ifdef CONFIG_MTK_GMO_RAM_OPTIMIZE
  6109. release_idle_lp_dc_buffer(0);
  6110. #endif
  6111. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6112. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6113. pgc->session_mode, sess_mode);
  6114. /* primary_display_diagnose(); */
  6115. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE
  6116. && sess_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  6117. /*dc to dc mirror */
  6118. pgc->session_mode = sess_mode;
  6119. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6120. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6121. pgc->session_mode, sess_mode);
  6122. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE
  6123. && sess_mode == DISP_SESSION_DECOUPLE_MODE) {
  6124. /*dc mirror to dc */
  6125. pgc->session_mode = sess_mode;
  6126. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6127. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6128. pgc->session_mode, sess_mode);
  6129. } else {
  6130. DISPERR("invalid mode switch from %s to %s\n", session_mode_spy(pgc->session_mode),
  6131. session_mode_spy(sess_mode));
  6132. }
  6133. done:
  6134. _primary_path_unlock(__func__);
  6135. pgc->session_id = session;
  6136. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagEnd,
  6137. pgc->session_mode, sess_mode);
  6138. return 0; /* avoid build warning. */
  6139. }
  6140. #ifdef MTK_DISP_IDLE_LP
  6141. int primary_display_switch_mode_nolock(int sess_mode, unsigned int session, int force)
  6142. {
  6143. DISPCHECK("primary_display_switch_mode sess_mode %d, session 0x%x\n", sess_mode, session);
  6144. /* if(!force && _is_decouple_mode(pgc->session_mode)) */
  6145. /* return 0; */
  6146. /* _primary_path_lock(__func__); */
  6147. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagStart,
  6148. pgc->session_mode, sess_mode);
  6149. if (pgc->session_mode == sess_mode)
  6150. goto done;
  6151. DISPMSG("primary display will switch from %s to %s\n", session_mode_spy(pgc->session_mode),
  6152. session_mode_spy(sess_mode));
  6153. if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE && sess_mode == DISP_SESSION_DECOUPLE_MODE) {
  6154. /* signal frame start event to switch logic in fence_release_worker_thread */
  6155. /* dpmgr_signal_event(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_START); */
  6156. _DL_switch_to_DC_fast();
  6157. pgc->session_mode = sess_mode;
  6158. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6159. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6160. pgc->session_mode, sess_mode);
  6161. /* primary_display_diagnose(); */
  6162. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MODE && sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6163. /* signal frame start event to switch logic in fence_release_worker_thread */
  6164. /* dpmgr_signal_event(pgc->ovl2mem_path_handle, DISP_PATH_EVENT_FRAME_START); */
  6165. _DC_switch_to_DL_fast();
  6166. pgc->session_mode = sess_mode;
  6167. DISPMSG("* primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6168. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6169. pgc->session_mode, sess_mode);
  6170. /* primary_display_diagnose(); */
  6171. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE &&
  6172. sess_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE) {
  6173. /*need delay switch to output */
  6174. pgc->session_mode = sess_mode;
  6175. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MIRROR_MODE
  6176. && sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6177. /* xxx */
  6178. pgc->session_mode = sess_mode;
  6179. } else if (pgc->session_mode == DISP_SESSION_DIRECT_LINK_MODE &&
  6180. sess_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE) {
  6181. /*need delay switch to output */
  6182. _DL_switch_to_DC_fast();
  6183. pgc->session_mode = sess_mode;
  6184. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6185. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6186. pgc->session_mode, sess_mode);
  6187. /* primary_display_diagnose(); */
  6188. /* pgc->session_delay_mode = sess_mode; */
  6189. } else if (pgc->session_mode == DISP_SESSION_DECOUPLE_MIRROR_MODE &&
  6190. sess_mode == DISP_SESSION_DIRECT_LINK_MODE) {
  6191. _DC_switch_to_DL_fast();
  6192. pgc->session_mode = sess_mode;
  6193. DISPMSG("primary display is %s mode now\n", session_mode_spy(pgc->session_mode));
  6194. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagPulse,
  6195. pgc->session_mode, sess_mode);
  6196. /* primary_display_diagnose(); */
  6197. } else {
  6198. DISPERR("invalid mode switch from %s to %s\n", session_mode_spy(pgc->session_mode),
  6199. session_mode_spy(sess_mode));
  6200. }
  6201. done:
  6202. /* _primary_path_unlock(__func__); */
  6203. pgc->session_id = session;
  6204. MMProfileLogEx(ddp_mmp_get_events()->primary_switch_mode, MMProfileFlagEnd,
  6205. pgc->session_mode, sess_mode);
  6206. return 0; /* avoid build warning. */
  6207. }
  6208. #endif
  6209. int primary_display_is_alive(void)
  6210. {
  6211. unsigned int temp = 0;
  6212. /* DISPFUNC(); */
  6213. _primary_path_lock(__func__);
  6214. if (pgc->state == DISP_ALIVE)
  6215. temp = 1;
  6216. _primary_path_unlock(__func__);
  6217. return temp;
  6218. }
  6219. int primary_display_is_sleepd_nolock(void)
  6220. {
  6221. unsigned int temp = 0;
  6222. /* DISPFUNC(); */
  6223. if (pgc->state == DISP_SLEPT)
  6224. temp = 1;
  6225. return temp;
  6226. }
  6227. int primary_display_is_sleepd(void)
  6228. {
  6229. unsigned int temp = 0;
  6230. /* DISPFUNC(); */
  6231. _primary_path_lock(__func__);
  6232. if (pgc->state == DISP_SLEPT)
  6233. temp = 1;
  6234. _primary_path_unlock(__func__);
  6235. return temp;
  6236. }
  6237. int primary_display_get_width(void)
  6238. {
  6239. if (pgc->plcm == NULL) {
  6240. DISPERR("lcm handle is null\n");
  6241. return 0;
  6242. }
  6243. if (pgc->plcm->params)
  6244. return pgc->plcm->params->width;
  6245. DISPERR("lcm_params is null!\n");
  6246. return 0;
  6247. }
  6248. int primary_display_get_height(void)
  6249. {
  6250. if (pgc->plcm == NULL) {
  6251. DISPERR("lcm handle is null\n");
  6252. return 0;
  6253. }
  6254. if (pgc->plcm->params)
  6255. return pgc->plcm->params->height;
  6256. DISPERR("lcm_params is null!\n");
  6257. return 0;
  6258. }
  6259. int primary_display_get_original_width(void)
  6260. {
  6261. if (pgc->plcm == NULL) {
  6262. DISPERR("lcm handle is null\n");
  6263. return 0;
  6264. }
  6265. if (pgc->plcm->params)
  6266. return pgc->plcm->lcm_original_width;
  6267. DISPERR("lcm_params is null!\n");
  6268. return 0;
  6269. }
  6270. int primary_display_get_original_height(void)
  6271. {
  6272. if (pgc->plcm == NULL) {
  6273. DISPERR("lcm handle is null\n");
  6274. return 0;
  6275. }
  6276. if (pgc->plcm->params)
  6277. return pgc->plcm->lcm_original_height;
  6278. DISPERR("lcm_params is null!\n");
  6279. return 0;
  6280. }
  6281. int primary_display_get_bpp(void)
  6282. {
  6283. return 32;
  6284. }
  6285. int primary_display_get_dc_bpp(void)
  6286. {
  6287. return 24;
  6288. }
  6289. void primary_display_set_max_layer(int maxlayer)
  6290. {
  6291. pgc->max_layer = maxlayer;
  6292. }
  6293. int primary_display_get_info(void *info)
  6294. {
  6295. #if 1
  6296. /* DISPFUNC(); */
  6297. disp_session_info *dispif_info = (disp_session_info *) info;
  6298. LCM_PARAMS *lcm_param = disp_lcm_get_params(pgc->plcm);
  6299. if (lcm_param == NULL) {
  6300. DISPCHECK("lcm_param is null\n");
  6301. return -1;
  6302. }
  6303. memset((void *)dispif_info, 0, sizeof(disp_session_info));
  6304. #ifdef OVL_CASCADE_SUPPORT
  6305. if (is_DAL_Enabled() && pgc->max_layer == OVL_LAYER_NUM) {
  6306. /* OVL1 is used by mem session */
  6307. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY || ovl_get_status() == DDP_OVL1_STATUS_IDLE)
  6308. dispif_info->maxLayerNum = pgc->max_layer - 1;
  6309. else
  6310. dispif_info->maxLayerNum = pgc->max_layer - 1 - 4;
  6311. } else {
  6312. /* OVL1 is used by mem session */
  6313. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY || ovl_get_status() == DDP_OVL1_STATUS_IDLE)
  6314. dispif_info->maxLayerNum = pgc->max_layer;
  6315. else
  6316. dispif_info->maxLayerNum = pgc->max_layer - 4;
  6317. }
  6318. #else
  6319. if (is_DAL_Enabled() && pgc->max_layer == OVL_LAYER_NUM)
  6320. dispif_info->maxLayerNum = pgc->max_layer - 1;
  6321. else
  6322. dispif_info->maxLayerNum = pgc->max_layer;
  6323. #endif
  6324. /* DISPDBG("available layer num=%d\n", dispif_info->maxLayerNum); */
  6325. switch (lcm_param->type) {
  6326. case LCM_TYPE_DBI:
  6327. {
  6328. dispif_info->displayType = DISP_IF_TYPE_DBI;
  6329. dispif_info->displayMode = DISP_IF_MODE_COMMAND;
  6330. dispif_info->isHwVsyncAvailable = 1;
  6331. /* DISPMSG("DISP Info: DBI, CMD Mode, HW Vsync enable\n"); */
  6332. break;
  6333. }
  6334. case LCM_TYPE_DPI:
  6335. {
  6336. dispif_info->displayType = DISP_IF_TYPE_DPI;
  6337. dispif_info->displayMode = DISP_IF_MODE_VIDEO;
  6338. dispif_info->isHwVsyncAvailable = 1;
  6339. /* DISPMSG("DISP Info: DPI, VDO Mode, HW Vsync enable\n"); */
  6340. break;
  6341. }
  6342. case LCM_TYPE_DSI:
  6343. {
  6344. dispif_info->displayType = DISP_IF_TYPE_DSI0;
  6345. if (lcm_param->dsi.mode == CMD_MODE) {
  6346. dispif_info->displayMode = DISP_IF_MODE_COMMAND;
  6347. dispif_info->isHwVsyncAvailable = 1;
  6348. /* DISPMSG("DISP Info: DSI, CMD Mode, HW Vsync enable\n"); */
  6349. } else {
  6350. dispif_info->displayMode = DISP_IF_MODE_VIDEO;
  6351. dispif_info->isHwVsyncAvailable = 1;
  6352. /* DISPMSG("DISP Info: DSI, VDO Mode, HW Vsync enable\n"); */
  6353. }
  6354. break;
  6355. }
  6356. default:
  6357. break;
  6358. }
  6359. dispif_info->displayFormat = DISP_IF_FORMAT_RGB888;
  6360. dispif_info->displayWidth = primary_display_get_width();
  6361. dispif_info->displayHeight = primary_display_get_height();
  6362. dispif_info->physicalWidth = DISP_GetActiveWidth();
  6363. dispif_info->physicalHeight = DISP_GetActiveHeight();
  6364. dispif_info->vsyncFPS = pgc->lcm_fps;
  6365. dispif_info->isConnected = 1;
  6366. #ifdef ROME_TODO
  6367. #error
  6368. {
  6369. LCM_PARAMS lcm_params_temp;
  6370. memset((void *)&lcm_params_temp, 0, sizeof(lcm_params_temp));
  6371. if (lcm_drv) {
  6372. lcm_drv->get_params(&lcm_params_temp);
  6373. dispif_info->lcmOriginalWidth = lcm_params_temp.width;
  6374. dispif_info->lcmOriginalHeight = lcm_params_temp.height;
  6375. DISPMSG("DISP Info: LCM Panel Original Resolution(For DFO Only): %d x %d\n",
  6376. dispif_info->lcmOriginalWidth, dispif_info->lcmOriginalHeight);
  6377. } else {
  6378. DISPMSG("DISP Info: Fatal Error!!, lcm_drv is null\n");
  6379. }
  6380. }
  6381. #endif
  6382. #endif
  6383. return 0; /* avoid build warning. */
  6384. }
  6385. int primary_display_get_pages(void)
  6386. {
  6387. return 3;
  6388. }
  6389. int primary_display_is_video_mode(void)
  6390. {
  6391. /* TODO: we should store the video/cmd mode in runtime, because we will support cmd/vdo dynamic switch */
  6392. return disp_lcm_is_video_mode(pgc->plcm);
  6393. }
  6394. int primary_display_is_decouple_mode(void)
  6395. {
  6396. return _is_decouple_mode(pgc->session_mode);
  6397. }
  6398. int primary_display_is_mirror_mode(void)
  6399. {
  6400. return _is_mirror_mode(pgc->session_mode);
  6401. }
  6402. int primary_display_is_ovl1to2_handle(cmdqRecHandle *handle)
  6403. {
  6404. if (handle && (handle == (cmdqRecHandle *)pgc->cmdq_handle_ovl1to2_config))
  6405. return 1;
  6406. else
  6407. return 0;
  6408. }
  6409. int primary_display_diagnose(void)
  6410. {
  6411. int ret = 0;
  6412. if (pgc->dpmgr_handle != NULL)
  6413. dpmgr_check_status(pgc->dpmgr_handle);
  6414. if (_is_decouple_mode(pgc->session_mode) && (pgc->ovl2mem_path_handle != NULL))
  6415. dpmgr_check_status(pgc->ovl2mem_path_handle);
  6416. primary_display_check_path(NULL, 0);
  6417. return ret;
  6418. }
  6419. CMDQ_SWITCH primary_display_cmdq_enabled(void)
  6420. {
  6421. return primary_display_use_cmdq;
  6422. }
  6423. int primary_display_switch_cmdq_cpu(CMDQ_SWITCH use_cmdq)
  6424. {
  6425. _primary_path_lock(__func__);
  6426. primary_display_use_cmdq = use_cmdq;
  6427. DISPCHECK("display driver use %s to config register now\n",
  6428. (use_cmdq == CMDQ_ENABLE) ? "CMDQ" : "CPU");
  6429. _primary_path_unlock(__func__);
  6430. return primary_display_use_cmdq;
  6431. }
  6432. int primary_display_manual_lock(void)
  6433. {
  6434. _primary_path_lock(__func__);
  6435. return 0; /* avoid build warning */
  6436. }
  6437. int primary_display_manual_unlock(void)
  6438. {
  6439. _primary_path_unlock(__func__);
  6440. return 0;
  6441. }
  6442. void primary_display_reset(void)
  6443. {
  6444. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  6445. }
  6446. unsigned int primary_display_get_fps(void)
  6447. {
  6448. unsigned int fps = 0;
  6449. _primary_path_lock(__func__);
  6450. fps = pgc->lcm_fps;
  6451. _primary_path_unlock(__func__);
  6452. return fps;
  6453. }
  6454. int primary_display_force_set_vsync_fps(unsigned int fps)
  6455. {
  6456. int ret = 0;
  6457. DISPMSG("force set fps to %d\n", fps);
  6458. _primary_path_lock(__func__);
  6459. if (fps == pgc->lcm_fps) {
  6460. pgc->vsync_drop = 0;
  6461. ret = 0;
  6462. } else if (fps == 30) {
  6463. pgc->vsync_drop = 1;
  6464. ret = 0;
  6465. } else {
  6466. ret = -1;
  6467. }
  6468. _primary_path_unlock(__func__);
  6469. return ret;
  6470. }
  6471. int primary_display_insert_session_buf(disp_session_buf_info *session_buf_info)
  6472. {
  6473. int i = 0;
  6474. unsigned int *session_buf;
  6475. session_buf = pgc->session_buf;
  6476. pgc->session_buf_id = 0;
  6477. for (i = 0; i < DISP_INTERNAL_BUFFER_COUNT; i++) {
  6478. session_buf[i] = session_buf_info->buf_hnd[i];
  6479. DISPMSG("%s buf_hnd[%d] = 0x%x\n", __func__, i, session_buf_info->buf_hnd[i]);
  6480. }
  6481. return 0;
  6482. }
  6483. int primary_display_enable_path_cg(int enable)
  6484. {
  6485. int ret = 0;
  6486. #ifdef ENABLE_CLK_MGR
  6487. DISPMSG("%s primary display's path cg\n", enable ? "enable" : "disable");
  6488. _primary_path_lock(__func__);
  6489. if (enable) {
  6490. #ifdef CONFIG_MTK_CLKMGR
  6491. ret += disable_clock(MT_CG_DISP1_DSI_ENGINE, "DSI0");
  6492. ret += disable_clock(MT_CG_DISP1_DSI_DIGITAL, "DSI0");
  6493. ret += disable_clock(MT_CG_DISP0_DISP_OVL0, "DDP");
  6494. ret += disable_clock(MT_CG_DISP0_DISP_COLOR, "DDP");
  6495. ret += disable_clock(MT_CG_DISP0_DISP_CCORR, "DDP");
  6496. ret += disable_clock(MT_CG_DISP0_DISP_AAL, "DDP");
  6497. ret += disable_clock(MT_CG_DISP0_DISP_GAMMA, "DDP");
  6498. ret += disable_clock(MT_CG_DISP0_DISP_DITHER, "DDP");
  6499. ret += disable_clock(MT_CG_DISP0_DISP_RDMA0, "DDP");
  6500. ret += disable_clock(MT_CG_PERI_DISP_PWM, "PWM");
  6501. /* ret += disable_clock(MT_CG_DISP0_MUTEX_32K , "Debug"); */
  6502. ret += disable_clock(MT_CG_DISP0_SMI_LARB0, "Debug");
  6503. ret += disable_clock(MT_CG_DISP0_SMI_COMMON, "Debug");
  6504. /* ret += disable_clock(MT_CG_DISP0_MUTEX_32K , "Debug2"); */
  6505. ret += disable_clock(MT_CG_DISP0_SMI_LARB0, "Debug2");
  6506. ret += disable_clock(MT_CG_DISP0_SMI_COMMON, "Debug2");
  6507. #else
  6508. ddp_clk_disable(DISP1_DSI_ENGINE);
  6509. ddp_clk_disable(DISP1_DSI_DIGITAL);
  6510. ddp_clk_disable(DISP0_DISP_OVL0);
  6511. ddp_clk_disable(DISP0_DISP_COLOR);
  6512. ddp_clk_disable(DISP0_DISP_CCORR);
  6513. ddp_clk_disable(DISP0_DISP_AAL);
  6514. ddp_clk_disable(DISP0_DISP_GAMMA);
  6515. ddp_clk_disable(DISP0_DISP_DITHER);
  6516. ddp_clk_disable(DISP0_DISP_RDMA0);
  6517. ddp_clk_disable(DISP_PWM);
  6518. /*ddp_clk_disable(DISP0_MUTEX_32K); */
  6519. ddp_clk_disable(DISP0_SMI_LARB0);
  6520. ddp_clk_disable(DISP0_SMI_COMMON);
  6521. /*ddp_clk_disable(DISP0_MUTEX_32K); */
  6522. ddp_clk_disable(DISP0_SMI_LARB0);
  6523. ddp_clk_disable(DISP0_SMI_COMMON);
  6524. ddp_clk_disable(DISP_MTCMOS_CLK);
  6525. ddp_clk_unprepare(DISP_MTCMOS_CLK);
  6526. #endif
  6527. } else {
  6528. #ifdef CONFIG_MTK_CLKMGR
  6529. ret += enable_clock(MT_CG_DISP1_DSI_ENGINE, "DSI0");
  6530. ret += enable_clock(MT_CG_DISP1_DSI_DIGITAL, "DSI0");
  6531. ret += enable_clock(MT_CG_DISP0_DISP_OVL0, "DDP");
  6532. ret += enable_clock(MT_CG_DISP0_DISP_COLOR, "DDP");
  6533. ret += enable_clock(MT_CG_DISP0_DISP_CCORR, "DDP");
  6534. ret += enable_clock(MT_CG_DISP0_DISP_AAL, "DDP");
  6535. ret += enable_clock(MT_CG_DISP0_DISP_GAMMA, "DDP");
  6536. ret += enable_clock(MT_CG_DISP0_DISP_DITHER, "DDP");
  6537. ret += enable_clock(MT_CG_DISP0_DISP_RDMA0, "DDP");
  6538. ret += enable_clock(MT_CG_PERI_DISP_PWM, "PWM");
  6539. /* ret += enable_clock(MT_CG_DISP0_MUTEX_32K , "Debug"); */
  6540. ret += enable_clock(MT_CG_DISP0_SMI_LARB0, "Debug");
  6541. ret += enable_clock(MT_CG_DISP0_SMI_COMMON, "Debug");
  6542. /* ret += enable_clock(MT_CG_DISP0_MUTEX_32K , "Debug2"); */
  6543. ret += enable_clock(MT_CG_DISP0_SMI_LARB0, "Debug2");
  6544. ret += enable_clock(MT_CG_DISP0_SMI_COMMON, "Debug2");
  6545. #else
  6546. ddp_clk_prepare(DISP_MTCMOS_CLK);
  6547. ret += ddp_clk_enable(DISP_MTCMOS_CLK);
  6548. ret += ddp_clk_enable(DISP1_DSI_ENGINE);
  6549. ret += ddp_clk_enable(DISP1_DSI_DIGITAL);
  6550. ret += ddp_clk_enable(DISP0_DISP_OVL0);
  6551. ret += ddp_clk_enable(DISP0_DISP_COLOR);
  6552. ret += ddp_clk_enable(DISP0_DISP_CCORR);
  6553. ret += ddp_clk_enable(DISP0_DISP_AAL);
  6554. ret += ddp_clk_enable(DISP0_DISP_GAMMA);
  6555. ret += ddp_clk_enable(DISP0_DISP_DITHER);
  6556. ret += ddp_clk_enable(DISP0_DISP_RDMA0);
  6557. ret += ddp_clk_enable(DISP_PWM);
  6558. /*ret += ddp_clk_enable(DISP0_MUTEX_32K); */
  6559. ret += ddp_clk_enable(DISP0_SMI_LARB0);
  6560. ret += ddp_clk_enable(DISP0_SMI_COMMON);
  6561. /*ret += ddp_clk_enable(DISP0_MUTEX_32K); */
  6562. ret += ddp_clk_enable(DISP0_SMI_LARB0);
  6563. ret += ddp_clk_enable(DISP0_SMI_COMMON);
  6564. #endif
  6565. }
  6566. _primary_path_unlock(__func__);
  6567. #endif
  6568. return ret;
  6569. }
  6570. int _set_backlight_by_cmdq(unsigned int level)
  6571. {
  6572. int ret = 0;
  6573. cmdqRecHandle cmdq_handle_backlight = NULL;
  6574. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 1);
  6575. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle_backlight);
  6576. DISPCHECK("primary backlight, handle=%p\n", cmdq_handle_backlight);
  6577. if (ret != 0) {
  6578. DISPCHECK("fail to create primary cmdq handle for backlight\n");
  6579. return -1;
  6580. }
  6581. if (primary_display_is_video_mode()) {
  6582. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 2);
  6583. cmdqRecReset(cmdq_handle_backlight);
  6584. dpmgr_path_ioctl(pgc->dpmgr_handle, cmdq_handle_backlight, DDP_BACK_LIGHT,
  6585. (unsigned long *)&level);
  6586. _cmdq_flush_config_handle_mira(cmdq_handle_backlight, 1);
  6587. DISPCHECK("[BL]_set_backlight_by_cmdq ret=%d\n", ret);
  6588. } else {
  6589. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 3);
  6590. cmdqRecReset(cmdq_handle_backlight);
  6591. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle_backlight);
  6592. cmdqRecClearEventToken(cmdq_handle_backlight, CMDQ_SYNC_TOKEN_CABC_EOF);
  6593. dpmgr_path_ioctl(pgc->dpmgr_handle, cmdq_handle_backlight, DDP_BACK_LIGHT,
  6594. (unsigned long *)&level);
  6595. cmdqRecSetEventToken(cmdq_handle_backlight, CMDQ_SYNC_TOKEN_CABC_EOF);
  6596. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 4);
  6597. _cmdq_flush_config_handle_mira(cmdq_handle_backlight, 1);
  6598. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 6);
  6599. DISPCHECK("[BL]_set_backlight_by_cmdq ret=%d\n", ret);
  6600. }
  6601. cmdqRecDestroy(cmdq_handle_backlight);
  6602. cmdq_handle_backlight = NULL;
  6603. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 1, 5);
  6604. return ret;
  6605. }
  6606. int _set_backlight_by_cpu(unsigned int level)
  6607. {
  6608. int ret = 0;
  6609. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 0, 1);
  6610. if (primary_display_is_video_mode()) {
  6611. disp_lcm_set_backlight(pgc->plcm, level);
  6612. } else {
  6613. DISPCHECK("[BL]display cmdq trigger loop stop[begin]\n");
  6614. _cmdq_stop_trigger_loop();
  6615. DISPCHECK("[BL]display cmdq trigger loop stop[end]\n");
  6616. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  6617. DISPCHECK("[BL]primary display path is busy\n");
  6618. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  6619. DISPCHECK("[BL]wait frame done ret:%d\n", ret);
  6620. }
  6621. DISPCHECK("[BL]stop dpmgr path[begin]\n");
  6622. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  6623. DISPCHECK("[BL]stop dpmgr path[end]\n");
  6624. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  6625. DISPCHECK("[BL]primary display path is busy after stop\n");
  6626. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE,
  6627. HZ * 1);
  6628. DISPCHECK("[BL]wait frame done ret:%d\n", ret);
  6629. }
  6630. DISPCHECK("[BL]reset display path[begin]\n");
  6631. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  6632. DISPCHECK("[BL]reset display path[end]\n");
  6633. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 0, 2);
  6634. disp_lcm_set_backlight(pgc->plcm, level);
  6635. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 0, 3);
  6636. DISPCHECK("[BL]start dpmgr path[begin]\n");
  6637. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  6638. DISPCHECK("[BL]start dpmgr path[end]\n");
  6639. DISPCHECK("[BL]start cmdq trigger loop[begin]\n");
  6640. _cmdq_start_trigger_loop();
  6641. DISPCHECK("[BL]start cmdq trigger loop[end]\n");
  6642. }
  6643. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagPulse, 0, 7);
  6644. return ret;
  6645. }
  6646. int primary_display_setbacklight(unsigned int level)
  6647. {
  6648. int ret = 0;
  6649. static unsigned int last_level;
  6650. DISPFUNC();
  6651. if (last_level == level)
  6652. return 0;
  6653. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagStart, 0, 0);
  6654. #ifdef DISP_SWITCH_DST_MODE
  6655. _primary_path_switch_dst_lock();
  6656. #endif
  6657. _primary_path_cmd_lock();
  6658. _primary_path_lock(__func__);
  6659. if (pgc->state == DISP_SLEPT) {
  6660. DISPCHECK("Sleep State set backlight invald\n");
  6661. } else {
  6662. disp_update_trigger_time();
  6663. if (primary_display_cmdq_enabled()) {
  6664. if (primary_display_is_video_mode()) {
  6665. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl,
  6666. MMProfileFlagPulse, 0, 7);
  6667. disp_lcm_set_backlight(pgc->plcm, level);
  6668. } else {
  6669. #ifdef MTK_DISP_IDLE_LP
  6670. /* CMD mode need to exit top clock off idle mode */
  6671. _disp_primary_path_exit_idle("primary_display_setbacklight", 0);
  6672. #endif
  6673. _set_backlight_by_cmdq(level);
  6674. }
  6675. } else {
  6676. #ifdef MTK_DISP_IDLE_LP
  6677. if (primary_display_is_video_mode() == 0)
  6678. _disp_primary_path_exit_idle("primary_display_setbacklight", 0);
  6679. #endif
  6680. _set_backlight_by_cpu(level);
  6681. }
  6682. last_level = level;
  6683. }
  6684. #ifndef CONFIG_FPGA_EARLY_PORTING
  6685. #ifdef GPIO_LCM_LED_EN
  6686. if (0 == level)
  6687. mt_set_gpio_out(GPIO_LCM_LED_EN, GPIO_OUT_ZERO);
  6688. else
  6689. mt_set_gpio_out(GPIO_LCM_LED_EN, GPIO_OUT_ONE);
  6690. #endif
  6691. #endif
  6692. #if 0 /* check writed success? for test after CABC */
  6693. {
  6694. /* extern uint32_t DSI_dcs_read_lcm_reg_v2(DISP_MODULE_ENUM module, cmdqRecHandle cmdq,
  6695. uint8_t cmd, uint8_t *buffer, uint8_t buffer_size);
  6696. */
  6697. uint8_t buffer[2];
  6698. if (primary_display_is_video_mode())
  6699. dpmgr_path_ioctl(pgc->dpmgr_handle, NULL, DDP_STOP_VIDEO_MODE, NULL);
  6700. DSI_dcs_read_lcm_reg_v2(DISP_MODULE_DSI0, NULL, 0x51, buffer, 1);
  6701. pr_debug("[CABC check result 0x51 = 0x%x,0x%x]\n", buffer[0], buffer[1]);
  6702. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  6703. if (primary_display_is_video_mode()) {
  6704. /* for video mode, we need to force trigger here */
  6705. /* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start */
  6706. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  6707. }
  6708. }
  6709. #endif
  6710. _primary_path_unlock(__func__);
  6711. _primary_path_cmd_unlock();
  6712. #ifdef DISP_SWITCH_DST_MODE
  6713. _primary_path_switch_dst_unlock();
  6714. #endif
  6715. MMProfileLogEx(ddp_mmp_get_events()->primary_set_bl, MMProfileFlagEnd, 0, 0);
  6716. return ret;
  6717. }
  6718. int primary_display_set_cmd(int *lcm_cmd, unsigned int cmd_num)
  6719. {
  6720. int ret = 0;
  6721. DISPFUNC();
  6722. _primary_path_cmd_lock();
  6723. _primary_path_lock(__func__);
  6724. if (pgc->state == DISP_SLEPT) {
  6725. DISPCHECK("Sleep State set cmd invald\n");
  6726. } else {
  6727. if (primary_display_cmdq_enabled()) { /* cmdq */
  6728. int ret = 0;
  6729. cmdqRecHandle cmdq_handle_cmd = NULL;
  6730. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle_cmd);
  6731. DISPCHECK("primary set cmd, handle=%p\n", cmdq_handle_cmd);
  6732. if (ret != 0) {
  6733. DISPCHECK("fail to create primary cmdq handle for cmd\n");
  6734. return -1;
  6735. }
  6736. if (primary_display_is_video_mode()) {
  6737. /* video mode */
  6738. cmdqRecReset(cmdq_handle_cmd);
  6739. disp_lcm_set_cmd(pgc->plcm, (void *)cmdq_handle_cmd, lcm_cmd,
  6740. cmd_num);
  6741. _cmdq_flush_config_handle_mira(cmdq_handle_cmd, 1);
  6742. DISPCHECK("[Display]_set_cmd_by_cmdq ret=%d\n", ret);
  6743. } else {
  6744. /* cmd mode */
  6745. cmdqRecReset(cmdq_handle_cmd);
  6746. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle_cmd);
  6747. cmdqRecClearEventToken(cmdq_handle_cmd, CMDQ_SYNC_TOKEN_CABC_EOF);
  6748. disp_lcm_set_cmd(pgc->plcm, (void *)cmdq_handle_cmd, lcm_cmd,
  6749. cmd_num);
  6750. cmdqRecSetEventToken(cmdq_handle_cmd, CMDQ_SYNC_TOKEN_CABC_EOF);
  6751. _cmdq_flush_config_handle_mira(cmdq_handle_cmd, 1);
  6752. DISPCHECK("[Display]_set_cmd_by_cmdq ret=%d\n", ret);
  6753. }
  6754. cmdqRecDestroy(cmdq_handle_cmd);
  6755. cmdq_handle_cmd = NULL;
  6756. }
  6757. }
  6758. _primary_path_unlock(__func__);
  6759. _primary_path_cmd_unlock();
  6760. return ret;
  6761. }
  6762. #define LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL
  6763. /***********************/
  6764. /*****Legacy DISP API*****/
  6765. /***********************/
  6766. uint32_t DISP_GetScreenWidth(void)
  6767. {
  6768. return primary_display_get_width();
  6769. }
  6770. uint32_t DISP_GetScreenHeight(void)
  6771. {
  6772. return primary_display_get_height();
  6773. }
  6774. uint32_t DISP_GetActiveHeight(void)
  6775. {
  6776. if (pgc->plcm == NULL) {
  6777. DISPERR("lcm handle is null\n");
  6778. return 0;
  6779. }
  6780. if (pgc->plcm->params)
  6781. return pgc->plcm->params->physical_height;
  6782. DISPERR("lcm_params is null!\n");
  6783. return 0;
  6784. }
  6785. uint32_t DISP_GetActiveWidth(void)
  6786. {
  6787. if (pgc->plcm == NULL) {
  6788. DISPERR("lcm handle is null\n");
  6789. return 0;
  6790. }
  6791. if (pgc->plcm->params)
  6792. return pgc->plcm->params->physical_width;
  6793. DISPERR("lcm_params is null!\n");
  6794. return 0;
  6795. }
  6796. LCM_PARAMS *DISP_GetLcmPara(void)
  6797. {
  6798. if (pgc->plcm == NULL) {
  6799. DISPERR("lcm handle is null\n");
  6800. return NULL;
  6801. }
  6802. if (pgc->plcm->params)
  6803. return pgc->plcm->params;
  6804. else
  6805. return NULL;
  6806. }
  6807. LCM_DRIVER *DISP_GetLcmDrv(void)
  6808. {
  6809. if (pgc->plcm == NULL) {
  6810. DISPERR("lcm handle is null\n");
  6811. return NULL;
  6812. }
  6813. if (pgc->plcm->drv)
  6814. return pgc->plcm->drv;
  6815. else
  6816. return NULL;
  6817. }
  6818. int primary_display_capture_framebuffer_decouple(unsigned long pbuf, unsigned int format)
  6819. {
  6820. unsigned int i = 0;
  6821. int ret = 0;
  6822. disp_ddp_path_config *pconfig = NULL;
  6823. m4u_client_t *m4uClient = NULL;
  6824. unsigned int mva = 0;
  6825. unsigned long va = 0;
  6826. unsigned int mapped_size = 0;
  6827. unsigned int w_xres = primary_display_get_width();
  6828. unsigned int h_yres = primary_display_get_height();
  6829. unsigned int pixel_byte = primary_display_get_dc_bpp() / 8;/* bpp is either 32 or 16, can not be other value */
  6830. unsigned int pitch = 0;
  6831. int buffer_size = h_yres * w_xres * pixel_byte;
  6832. m4uClient = m4u_create_client();
  6833. if (m4uClient == NULL) {
  6834. DISPCHECK("primary capture:Fail to alloc m4uClient=0x%p\n", m4uClient);
  6835. ret = -1;
  6836. goto out;
  6837. }
  6838. /* mva = pgc->dc_buf[pgc->dc_buf_id]; */
  6839. pconfig = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  6840. if (_is_decouple_mode(pgc->session_mode) && !_is_mirror_mode(pgc->session_mode)) {
  6841. w_xres = decouple_wdma_config.srcWidth;
  6842. h_yres = decouple_wdma_config.srcHeight;
  6843. pitch = decouple_wdma_config.dstPitch;
  6844. mva = decouple_wdma_config.dstAddress;
  6845. } else {
  6846. w_xres = pconfig->wdma_config.srcWidth;
  6847. h_yres = pconfig->wdma_config.srcHeight;
  6848. pitch = pconfig->wdma_config.dstPitch;
  6849. mva = pconfig->wdma_config.dstAddress;
  6850. }
  6851. buffer_size = h_yres * pitch;
  6852. ASSERT((pitch / 4) >= w_xres);
  6853. /* dpmgr_get_input_address(pgc->dpmgr_handle,&mva); */
  6854. m4u_mva_map_kernel(mva, buffer_size, &va, &mapped_size);
  6855. if (!va) {
  6856. DISPERR("map mva 0x%08x failed\n", mva);
  6857. goto out;
  6858. }
  6859. DISPMSG("map 0x%08x with %d bytes to 0x%08lx with %d bytes\n", mva, buffer_size, va,
  6860. mapped_size);
  6861. ret =
  6862. m4u_cache_sync(m4uClient, M4U_PORT_DISP_WDMA0, va, buffer_size, mva,
  6863. M4U_CACHE_FLUSH_ALL);
  6864. #if 1
  6865. {
  6866. unsigned int j = 0;
  6867. unsigned char tem_va[4];
  6868. unsigned long src_va = va;
  6869. for (i = 0; i < h_yres; i++) {
  6870. for (j = 0; j < w_xres; j++) {
  6871. memcpy((void *)tem_va, (void *)src_va, 4);
  6872. *(unsigned long *)(pbuf + (i * w_xres + j) * 4) = 0xFF000000 | (tem_va[0] << 16) |
  6873. (tem_va[1] << 8) | (tem_va[2]);
  6874. src_va += 4;
  6875. }
  6876. src_va += (pitch - w_xres * 4);
  6877. }
  6878. }
  6879. #else
  6880. memcpy(pbuf, va, mapped_size);
  6881. #endif
  6882. out:
  6883. if (mapped_size)
  6884. m4u_mva_unmap_kernel(mva, mapped_size, va);
  6885. if (m4uClient != NULL)
  6886. m4u_destroy_client(m4uClient);
  6887. DISPMSG("primary capture: end\n");
  6888. return ret;
  6889. }
  6890. int primary_display_capture_framebuffer_ovl(unsigned long pbuf, unsigned int format)
  6891. {
  6892. int ret = 0;
  6893. cmdqRecHandle cmdq_handle = NULL;
  6894. cmdqRecHandle cmdq_wait_handle = NULL;
  6895. disp_ddp_path_config *pconfig = NULL;
  6896. m4u_client_t *m4uClient = NULL;
  6897. unsigned int mva = 0;
  6898. unsigned int w_xres = primary_display_get_width();
  6899. unsigned int h_yres = primary_display_get_height();
  6900. unsigned int pixel_byte = primary_display_get_dc_bpp() / 8; /* bpp is either 32 or 16, can not be other value */
  6901. int buffer_size = h_yres * w_xres * pixel_byte;
  6902. DISPMSG("primary capture: begin\n");
  6903. disp_sw_mutex_lock(&(pgc->capture_lock));
  6904. _primary_path_lock(__func__);
  6905. if (pgc->state == DISP_SLEPT) {
  6906. /*memset_io((void *)pbuf, 0, buffer_size);*/
  6907. DISPMSG("primary capture:fill black for sleep\n");
  6908. goto out;
  6909. }
  6910. if (!primary_display_cmdq_enabled()) {
  6911. /*memset_io((void *)pbuf, 0, buffer_size);*/
  6912. DISPMSG("primary capture:fill black to cmdq disable\n");
  6913. goto out;
  6914. }
  6915. if (_is_decouple_mode(pgc->session_mode) || _is_mirror_mode(pgc->session_mode)) {
  6916. primary_display_capture_framebuffer_decouple(pbuf, format);
  6917. /*memset_io((void *)pbuf, 0, buffer_size);*/
  6918. DISPMSG("primary capture: fill black for decouple & mirror mode End\n");
  6919. goto out;
  6920. }
  6921. m4uClient = m4u_create_client();
  6922. if (m4uClient == NULL) {
  6923. DISPCHECK("primary capture:Fail to alloc m4uClient=0x%p\n", m4uClient);
  6924. ret = -1;
  6925. goto out;
  6926. }
  6927. ret = m4u_alloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, pbuf, NULL, buffer_size,
  6928. M4U_PROT_READ | M4U_PROT_WRITE, 0, (unsigned int *)&mva);
  6929. if (ret != 0) {
  6930. DISPCHECK("primary capture:Fail to allocate mva\n");
  6931. ret = -1;
  6932. goto out;
  6933. }
  6934. ret = m4u_cache_sync(m4uClient, M4U_PORT_DISP_WDMA0, pbuf, buffer_size, mva,
  6935. M4U_CACHE_FLUSH_BY_RANGE);
  6936. if (ret != 0) {
  6937. DISPCHECK("primary capture:Fail to cach sync\n");
  6938. ret = -1;
  6939. goto out;
  6940. }
  6941. if (primary_display_cmdq_enabled()) {
  6942. /* create config thread */
  6943. ret = cmdqRecCreate(CMDQ_SCENARIO_PRIMARY_DISP, &cmdq_handle);
  6944. if (ret != 0) {
  6945. DISPCHECK("primary capture:Fail to create primary cmdq handle for capture\n");
  6946. ret = -1;
  6947. goto out;
  6948. }
  6949. cmdqRecReset(cmdq_handle);
  6950. /* create wait thread */
  6951. ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_SCREEN_CAPTURE, &cmdq_wait_handle);
  6952. if (ret != 0) {
  6953. DISPCHECK("primary capture:Fail to create primary cmdq wait handle for capture\n");
  6954. ret = -1;
  6955. goto out;
  6956. }
  6957. cmdqRecReset(cmdq_wait_handle);
  6958. /* configure config thread */
  6959. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  6960. dpmgr_path_memout_clock(pgc->dpmgr_handle, 1);
  6961. pconfig = dpmgr_path_get_last_config(pgc->dpmgr_handle);
  6962. pconfig->wdma_dirty = 1;
  6963. pconfig->wdma_config.dstAddress = mva;
  6964. pconfig->wdma_config.srcHeight = h_yres;
  6965. pconfig->wdma_config.srcWidth = w_xres;
  6966. pconfig->wdma_config.clipX = 0;
  6967. pconfig->wdma_config.clipY = 0;
  6968. pconfig->wdma_config.clipHeight = h_yres;
  6969. pconfig->wdma_config.clipWidth = w_xres;
  6970. pconfig->wdma_config.outputFormat = format;
  6971. pconfig->wdma_config.useSpecifiedAlpha = 1;
  6972. pconfig->wdma_config.alpha = 0xFF;
  6973. pconfig->wdma_config.dstPitch = w_xres * DP_COLOR_BITS_PER_PIXEL(format) / 8;
  6974. dpmgr_path_add_memout(pgc->dpmgr_handle, ENGINE_OVL0, cmdq_handle);
  6975. #if 0 /* do not have to call enable_cascade here, dpmgr_path_add_memout() will add ovl1 if necessary */
  6976. if (ovl_get_status() == DDP_OVL1_STATUS_IDLE || ovl_get_status() == DDP_OVL1_STATUS_PRIMARY) {
  6977. DISPMSG("dpmgr_path_enable_cascade called!\n");
  6978. dpmgr_path_enable_cascade(pgc->dpmgr_handle, cmdq_handle);
  6979. if (ovl_get_status() != DDP_OVL1_STATUS_PRIMARY)
  6980. pconfig->ovl_dirty = 1;
  6981. }
  6982. #endif
  6983. ret = dpmgr_path_config(pgc->dpmgr_handle, pconfig, cmdq_handle);
  6984. pconfig->wdma_dirty = 0;
  6985. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  6986. _cmdq_flush_config_handle_mira(cmdq_handle, 0);
  6987. DISPMSG("primary capture:Flush add memout mva(0x%x)\n", mva);
  6988. /* primary_display_diagnose(); */
  6989. /* wait wdma0 sof */
  6990. cmdqRecWait(cmdq_wait_handle, CMDQ_EVENT_DISP_WDMA0_SOF);
  6991. cmdqRecFlush(cmdq_wait_handle);
  6992. DISPMSG("primary capture:Flush wait wdma sof\n");
  6993. cmdqRecReset(cmdq_handle);
  6994. _cmdq_insert_wait_frame_done_token_mira(cmdq_handle);
  6995. dpmgr_path_remove_memout(pgc->dpmgr_handle, cmdq_handle);
  6996. cmdqRecClearEventToken(cmdq_handle, CMDQ_EVENT_DISP_WDMA0_SOF);
  6997. _cmdq_set_config_handle_dirty_mira(cmdq_handle);
  6998. /* flush remove memory to cmdq */
  6999. _cmdq_flush_config_handle_mira(cmdq_handle, 1);
  7000. DISPMSG("primary capture: Flush remove memout\n");
  7001. dpmgr_path_memout_clock(pgc->dpmgr_handle, 0);
  7002. }
  7003. out:
  7004. cmdqRecDestroy(cmdq_handle);
  7005. cmdqRecDestroy(cmdq_wait_handle);
  7006. if (mva > 0)
  7007. m4u_dealloc_mva(m4uClient, M4U_PORT_DISP_WDMA0, mva);
  7008. if (m4uClient != 0)
  7009. m4u_destroy_client(m4uClient);
  7010. _primary_path_unlock(__func__);
  7011. disp_sw_mutex_unlock(&(pgc->capture_lock));
  7012. DISPMSG("primary capture: end\n");
  7013. return ret;
  7014. }
  7015. int primary_display_capture_framebuffer(unsigned long pbuf)
  7016. {
  7017. #if 1
  7018. unsigned int fb_layer_id = primary_display_get_option("FB_LAYER");
  7019. unsigned int w_xres = primary_display_get_width();
  7020. unsigned int h_yres = primary_display_get_height();
  7021. /* bpp is either 32 or 16, can not be other value */
  7022. unsigned int pixel_bpp = primary_display_get_dc_bpp() / 8;
  7023. unsigned int w_fb = ALIGN_TO(w_xres, MTK_FB_ALIGNMENT);
  7024. unsigned int fbsize = w_fb * h_yres * pixel_bpp; /* frame buffer size */
  7025. unsigned long fbaddress = dpmgr_path_get_last_config(pgc->dpmgr_handle)->ovl_config[fb_layer_id].addr;
  7026. unsigned long fbv = 0;
  7027. unsigned int i = 0;
  7028. unsigned long ttt = 0;
  7029. DISPMSG("w_res=%d, h_yres=%d, pixel_bpp=%d, w_fb=%d, fbsize=%d, fbaddress=0x%lx\n",
  7030. w_xres, h_yres, pixel_bpp, w_fb, fbsize, fbaddress);
  7031. fbv = (unsigned long)ioremap(fbaddress, fbsize);
  7032. DISPMSG("w_xres = %d, h_yres = %d, w_fb = %d, pixel_bpp = %d, fbsize = %d, fbaddress = 0x%lx\n",
  7033. w_xres, h_yres, w_fb, pixel_bpp, fbsize, fbaddress);
  7034. if (!fbv) {
  7035. DISPMSG("[FB Driver], Unable to allocate memory for frame buffer: address=0x%lx, size=0x%08x\n",
  7036. fbaddress, fbsize);
  7037. return -1;
  7038. }
  7039. ttt = get_current_time_us();
  7040. for (i = 0; i < h_yres; i++) {
  7041. /* DISPMSG("i=%d, dst=0x%08x, src=%08x\n",
  7042. i, (pbuf + i * w_xres * pixel_bpp), (fbv + i * w_fb * pixel_bpp));
  7043. */
  7044. memcpy((void *)(pbuf + i * w_xres * pixel_bpp),
  7045. (void *)(fbv + i * w_fb * pixel_bpp), w_xres * pixel_bpp);
  7046. }
  7047. DISPMSG("capture framebuffer cost %ld us\n", get_current_time_us() - ttt);
  7048. iounmap((void *)fbv);
  7049. #endif
  7050. return -1;
  7051. }
  7052. uint32_t DISP_GetPanelBPP(void)
  7053. {
  7054. #if 0
  7055. PANEL_COLOR_FORMAT fmt;
  7056. disp_drv_init_context();
  7057. if (disp_if_drv->get_panel_color_format == NULL)
  7058. return DISP_STATUS_NOT_IMPLEMENTED;
  7059. fmt = disp_if_drv->get_panel_color_format();
  7060. switch (fmt) {
  7061. case PANEL_COLOR_FORMAT_RGB332:
  7062. return 8;
  7063. case PANEL_COLOR_FORMAT_RGB444:
  7064. return 12;
  7065. case PANEL_COLOR_FORMAT_RGB565:
  7066. return 16;
  7067. case PANEL_COLOR_FORMAT_RGB666:
  7068. return 18;
  7069. case PANEL_COLOR_FORMAT_RGB888:
  7070. return 24;
  7071. default:
  7072. return 0;
  7073. }
  7074. #else
  7075. return 0; /* avoid build warning */
  7076. #endif
  7077. }
  7078. static uint32_t disp_fb_bpp = 32;
  7079. static uint32_t disp_fb_pages = 3;
  7080. uint32_t DISP_GetScreenBpp(void)
  7081. {
  7082. return disp_fb_bpp;
  7083. }
  7084. uint32_t DISP_GetPages(void)
  7085. {
  7086. return disp_fb_pages;
  7087. }
  7088. uint32_t DISP_GetFBRamSize(void)
  7089. {
  7090. return ALIGN_TO(DISP_GetScreenWidth(), MTK_FB_ALIGNMENT) * DISP_GetScreenHeight() *
  7091. ((DISP_GetScreenBpp() + 7) >> 3) * DISP_GetPages();
  7092. }
  7093. uint32_t DISP_GetVRamSize(void)
  7094. {
  7095. #if 0
  7096. /* Use a local static variable to cache the calculated vram size */
  7097. /* */
  7098. static uint32_t vramSize;
  7099. if (0 == vramSize) {
  7100. disp_drv_init_context();
  7101. /* /get framebuffer size */
  7102. vramSize = DISP_GetFBRamSize();
  7103. /* /get DXI working buffer size */
  7104. vramSize += disp_if_drv->get_working_buffer_size();
  7105. /* get assertion layer buffer size */
  7106. vramSize += DAL_GetLayerSize();
  7107. /* Align vramSize to 1MB */
  7108. /* */
  7109. vramSize = ALIGN_TO_POW_OF_2(vramSize, 0x100000);
  7110. DISP_LOG("DISP_GetVRamSize: %u bytes\n", vramSize);
  7111. }
  7112. return vramSize;
  7113. #else
  7114. return 0; /* avoid build warning */
  7115. #endif
  7116. }
  7117. uint32_t DISP_GetVRamSizeBoot(char *cmdline)
  7118. {
  7119. #ifdef CONFIG_OF
  7120. /* extern unsigned int vramsize; */
  7121. /* extern int _parse_tag_videolfb(void); */
  7122. _parse_tag_videolfb();
  7123. if (vramsize == 0)
  7124. vramsize = 0x3000000;
  7125. /* not necessary */
  7126. /* DISPCHECK("[DT]display vram size = 0x%08x|%d\n", vramsize, vramsize); */
  7127. return vramsize;
  7128. #else
  7129. int ret = 0;
  7130. char *p = NULL;
  7131. uint32_t vramSize = 0;
  7132. DISPMSG("%s, cmdline=%s\n", __func__, cmdline);
  7133. p = strstr(cmdline, "vram=");
  7134. if (p == NULL) {
  7135. vramSize = 0x3000000;
  7136. DISPERR("[FB driver]can not get vram size from lk\n");
  7137. } else {
  7138. p += 5;
  7139. ret = kstrtol(p, 10, &vramSize);
  7140. if (ret)
  7141. pr_err("DISP/%s: errno %d\n", __func__, ret);
  7142. if (0 == vramSize)
  7143. vramSize = 0x3000000;
  7144. }
  7145. DISPCHECK("display vram size = 0x%08x|%d\n", vramSize, vramSize);
  7146. return vramSize;
  7147. #endif
  7148. }
  7149. struct sg_table table;
  7150. int disp_hal_allocate_framebuffer(phys_addr_t pa_start, phys_addr_t pa_end, unsigned long *va,
  7151. unsigned long *mva)
  7152. {
  7153. #ifndef MTKFB_NO_M4U
  7154. int ret = 0;
  7155. #endif
  7156. *va = (unsigned long)ioremap_nocache(pa_start, pa_end - pa_start + 1);
  7157. /* printk("disphal_allocate_fb, pa=%p, va=0x%lx\n", &pa_start, *va); */
  7158. /* if (_get_init_setting("M4U")) */
  7159. /* xuecheng, m4u not enabled now */
  7160. #ifndef MTKFB_NO_M4U
  7161. if (1) {
  7162. m4u_client_t *client;
  7163. struct sg_table *sg_table = &table;
  7164. sg_alloc_table(sg_table, 1, GFP_KERNEL);
  7165. sg_dma_address(sg_table->sgl) = pa_start;
  7166. sg_dma_len(sg_table->sgl) = (pa_end - pa_start + 1);
  7167. client = m4u_create_client();
  7168. if (IS_ERR_OR_NULL(client))
  7169. DISPMSG("create client fail!\n");
  7170. *mva = pa_start & 0xffffffffULL;
  7171. ret = m4u_alloc_mva(client, M4U_PORT_DISP_OVL0, 0, sg_table, (pa_end - pa_start + 1),
  7172. M4U_PROT_READ | M4U_PROT_WRITE, M4U_FLAGS_FIX_MVA,
  7173. (unsigned int *)mva);
  7174. /* m4u_alloc_mva(M4U_PORT_DISP_OVL0, pa_start, (pa_end - pa_start + 1), 0, 0, mva); */
  7175. if (ret)
  7176. DISPMSG("m4u_alloc_mva returns fail: %d\n", ret);
  7177. /* printk("[DISPHAL] FB MVA is 0x%lx PA is %p\n", *mva, &pa_start);*/
  7178. } else
  7179. #endif
  7180. {
  7181. *mva = pa_start & 0xffffffffULL;
  7182. }
  7183. return 0;
  7184. }
  7185. int primary_display_remap_irq_event_map(void)
  7186. {
  7187. return 0; /* avoid build warning. */
  7188. }
  7189. unsigned int primary_display_get_option(const char *option)
  7190. {
  7191. if (!strcmp(option, "FB_LAYER"))
  7192. return 0;
  7193. if (!strcmp(option, "ASSERT_LAYER")) {
  7194. #ifdef OVL_CASCADE_SUPPORT
  7195. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY
  7196. || ovl_get_status() == DDP_OVL1_STATUS_IDLE)
  7197. return OVL_LAYER_NUM - 1;
  7198. else
  7199. return OVL_LAYER_NUM - 4 - 1;
  7200. #else
  7201. return OVL_LAYER_NUM - 1;
  7202. #endif
  7203. }
  7204. if (!strcmp(option, "M4U_ENABLE"))
  7205. return 1;
  7206. ASSERT(0);
  7207. return 0; /* avoid build warning */
  7208. }
  7209. int primary_display_get_debug_info(char *buf)
  7210. {
  7211. /* resolution */
  7212. /* cmd/video mode */
  7213. /* display path */
  7214. /* dsi data rate/lane number/state */
  7215. /* primary path trigger count */
  7216. /* frame done count */
  7217. /* suspend/resume count */
  7218. /* current fps 10s/5s/1s */
  7219. /* error count and message */
  7220. /* current state of each module on the path */
  7221. return 0; /* avoid build warning */
  7222. }
  7223. #include "ddp_reg.h"
  7224. #define IS_READY(x) ((x)?"READY\t":"Not READY")
  7225. #define IS_VALID(x) ((x)?"VALID\t":"Not VALID")
  7226. #define READY_BIT0(x) ((DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a0) & (1 << x)))
  7227. #define VALID_BIT0(x) ((DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a4) & (1 << x)))
  7228. #define READY_BIT1(x) ((DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a8) & (1 << x)))
  7229. #define VALID_BIT1(x) ((DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8ac) & (1 << x)))
  7230. int primary_display_check_path(char *stringbuf, int buf_len)
  7231. {
  7232. return 0; /* status will print in config dump. */
  7233. #if 0
  7234. int len = 0;
  7235. DISPMSG("primary_display_check_path() check signal status:\n");
  7236. if (stringbuf) {
  7237. len +=
  7238. scnprintf(stringbuf + len, buf_len - len,
  7239. "|--------------------------------------------------------------------------------------|\n");
  7240. len +=
  7241. scnprintf(stringbuf + len, buf_len - len,
  7242. "READY0=0x%08x, READY1=0x%08x, VALID0=0x%08x, VALID1=0x%08x\n",
  7243. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a0),
  7244. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a4),
  7245. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a8),
  7246. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8ac));
  7247. len +=
  7248. scnprintf(stringbuf + len, buf_len - len, "OVL0\t\t\t%s\t%s\n",
  7249. IS_READY(READY_BIT0(DDP_SIGNAL_OVL0__OVL0_MOUT)),
  7250. IS_VALID(READY_BIT0(DDP_SIGNAL_OVL0__OVL0_MOUT)));
  7251. len +=
  7252. scnprintf(stringbuf + len, buf_len - len, "OVL0_MOUT:\t\t%s\t%s\n",
  7253. IS_READY(READY_BIT1(DDP_SIGNAL_OVL0_MOUT0__COLOR_SIN1)),
  7254. IS_VALID(READY_BIT1(DDP_SIGNAL_OVL0_MOUT0__COLOR_SIN1)));
  7255. len +=
  7256. scnprintf(stringbuf + len, buf_len - len, "COLOR0_SEL:\t\t%s\t%s\n",
  7257. IS_READY(READY_BIT0(DDP_SIGNAL_COLOR_SEL__COLOR)),
  7258. IS_VALID(READY_BIT0(DDP_SIGNAL_COLOR_SEL__COLOR)));
  7259. len +=
  7260. scnprintf(stringbuf + len, buf_len - len, "COLOR0:\t\t\t%s\t%s\n",
  7261. IS_READY(READY_BIT0(DDP_SIGNAL_COLOR__CCORR)),
  7262. IS_VALID(READY_BIT0(DDP_SIGNAL_COLOR__CCORR)));
  7263. len +=
  7264. scnprintf(stringbuf + len, buf_len - len, "CCORR:\t\t%s\t%s\n",
  7265. IS_READY(READY_BIT0(DDP_SIGNAL_CCORR__AAL)),
  7266. IS_VALID(READY_BIT0(DDP_SIGNAL_CCORR__AAL)));
  7267. len +=
  7268. scnprintf(stringbuf + len, buf_len - len, "AAL0:\t\t\t%s\t%s\n",
  7269. IS_READY(READY_BIT0(DDP_SIGNAL_AAL__GAMMA)),
  7270. IS_VALID(READY_BIT0(DDP_SIGNAL_AAL__GAMMA)));
  7271. len +=
  7272. scnprintf(stringbuf + len, buf_len - len, "GAMMA:\t\t\t%s\t%s\n",
  7273. IS_READY(READY_BIT1(DDP_SIGNAL_GAMMA__DITHER)),
  7274. IS_VALID(READY_BIT1(DDP_SIGNAL_GAMMA__DITHER)));
  7275. len +=
  7276. scnprintf(stringbuf + len, buf_len - len, "DITHER:\t\t\t%s\t%s\n",
  7277. IS_READY(READY_BIT1(DDP_SIGNAL_DITHER__DITHER_MOUT)),
  7278. IS_VALID(READY_BIT1(DDP_SIGNAL_DITHER__DITHER_MOUT)));
  7279. len +=
  7280. scnprintf(stringbuf + len, buf_len - len, "DITHER_MOUT:\t\t%s\t%s\n",
  7281. IS_READY(READY_BIT1(DDP_SIGNAL_DITHER_MOUT0__RDMA0)),
  7282. IS_VALID(READY_BIT1(DDP_SIGNAL_DITHER_MOUT0__RDMA0)));
  7283. len +=
  7284. scnprintf(stringbuf + len, buf_len - len, "RDMA0:\t\t\t%s\t%s\n",
  7285. IS_READY(READY_BIT1(DDP_SIGNAL_RDMA0__RDMA0_SOUT)),
  7286. IS_VALID(READY_BIT1(DDP_SIGNAL_RDMA0__RDMA0_SOUT)));
  7287. len +=
  7288. scnprintf(stringbuf + len, buf_len - len, "RDMA0_SOUT:\t\t%s\t%s\n",
  7289. IS_READY(READY_BIT1(DDP_SIGNAL_RDMA0_SOUT0__UFOE_SIN0)),
  7290. IS_VALID(READY_BIT1(DDP_SIGNAL_RDMA0_SOUT0__UFOE_SIN0)));
  7291. len +=
  7292. scnprintf(stringbuf + len, buf_len - len, "UFOE_SEL:\t\t%s\t%s\n",
  7293. IS_READY(READY_BIT0(DDP_SIGNAL_UFOE_SEL__UFOE)),
  7294. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE_SEL__UFOE)));
  7295. len +=
  7296. scnprintf(stringbuf + len, buf_len - len, "UFOE:\t\t\t%s\t%s\n",
  7297. IS_READY(READY_BIT0(DDP_SIGNAL_UFOE__UFOE_MOUT)),
  7298. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE__UFOE_MOUT)));
  7299. len +=
  7300. scnprintf(stringbuf + len, buf_len - len, "UFOE_MOUT:\t\t%s\t%s\n",
  7301. IS_READY(READY_BIT0(DDP_SIGNAL_UFOE_MOUT0__DSI0_SIN0)),
  7302. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE_MOUT0__DSI0_SIN0)));
  7303. len +=
  7304. scnprintf(stringbuf + len, buf_len - len, "DSI0_SEL:\t\t%s\t%s\n",
  7305. IS_READY(READY_BIT1(DDP_SIGNAL_DIS0_SEL__DSI0)),
  7306. IS_VALID(READY_BIT1(DDP_SIGNAL_DIS0_SEL__DSI0)));
  7307. } else {
  7308. DISPMSG
  7309. ("|--------------------------------------------------------------------------------------|\n");
  7310. DISPMSG("READY0=0x%08x, READY1=0x%08x, VALID0=0x%08x, VALID1=0x%08x\n",
  7311. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a0),
  7312. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a4),
  7313. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8a8),
  7314. DISP_REG_GET(DISPSYS_CONFIG_BASE + 0x8ac));
  7315. DISPMSG("OVL0\t\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_OVL0__OVL0_MOUT)),
  7316. IS_VALID(READY_BIT0(DDP_SIGNAL_OVL0__OVL0_MOUT)));
  7317. DISPMSG("OVL0_MOUT:\t\t%s\t%s\n",
  7318. IS_READY(READY_BIT1(DDP_SIGNAL_OVL0_MOUT0__COLOR_SIN1)),
  7319. IS_VALID(READY_BIT1(DDP_SIGNAL_OVL0_MOUT0__COLOR_SIN1)));
  7320. DISPMSG("COLOR0_SEL:\t\t%s\t%s\n",
  7321. IS_READY(READY_BIT0(DDP_SIGNAL_COLOR_SEL__COLOR)),
  7322. IS_VALID(READY_BIT0(DDP_SIGNAL_COLOR_SEL__COLOR)));
  7323. DISPMSG("COLOR0:\t\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_COLOR__CCORR)),
  7324. IS_VALID(READY_BIT0(DDP_SIGNAL_COLOR__CCORR)));
  7325. DISPMSG("CCORR:\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_CCORR__AAL)),
  7326. IS_VALID(READY_BIT0(DDP_SIGNAL_CCORR__AAL)));
  7327. DISPMSG("AAL0:\t\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_AAL__GAMMA)),
  7328. IS_VALID(READY_BIT0(DDP_SIGNAL_AAL__GAMMA)));
  7329. DISPMSG("GAMMA:\t\t\t%s\t%s\n", IS_READY(READY_BIT1(DDP_SIGNAL_GAMMA__DITHER)),
  7330. IS_VALID(READY_BIT1(DDP_SIGNAL_GAMMA__DITHER)));
  7331. DISPMSG("DITHER:\t\t\t%s\t%s\n",
  7332. IS_READY(READY_BIT1(DDP_SIGNAL_DITHER__DITHER_MOUT)),
  7333. IS_VALID(READY_BIT1(DDP_SIGNAL_DITHER__DITHER_MOUT)));
  7334. DISPMSG("DITHER_MOUT:\t\t%s\t%s\n",
  7335. IS_READY(READY_BIT1(DDP_SIGNAL_DITHER_MOUT0__RDMA0)),
  7336. IS_VALID(READY_BIT1(DDP_SIGNAL_DITHER_MOUT0__RDMA0)));
  7337. DISPMSG("RDMA0:\t\t\t%s\t%s\n", IS_READY(READY_BIT1(DDP_SIGNAL_RDMA0__RDMA0_SOUT)),
  7338. IS_VALID(READY_BIT1(DDP_SIGNAL_RDMA0__RDMA0_SOUT)));
  7339. DISPMSG("RDMA0_SOUT:\t\t%s\t%s\n",
  7340. IS_READY(READY_BIT1(DDP_SIGNAL_RDMA0_SOUT0__UFOE_SIN0)),
  7341. IS_VALID(READY_BIT1(DDP_SIGNAL_RDMA0_SOUT0__UFOE_SIN0)));
  7342. DISPMSG("UFOE_SEL:\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_UFOE_SEL__UFOE)),
  7343. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE_SEL__UFOE)));
  7344. DISPMSG("UFOE:\t\t\t%s\t%s\n", IS_READY(READY_BIT0(DDP_SIGNAL_UFOE__UFOE_MOUT)),
  7345. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE__UFOE_MOUT)));
  7346. DISPMSG("UFOE_MOUT:\t\t%s\t%s\n",
  7347. IS_READY(READY_BIT0(DDP_SIGNAL_UFOE_MOUT0__DSI0_SIN0)),
  7348. IS_VALID(READY_BIT0(DDP_SIGNAL_UFOE_MOUT0__DSI0_SIN0)));
  7349. DISPMSG("DSI0_SEL:\t\t%s\t%s\n", IS_READY(READY_BIT1(DDP_SIGNAL_DIS0_SEL__DSI0)),
  7350. IS_VALID(READY_BIT1(DDP_SIGNAL_DIS0_SEL__DSI0)));
  7351. }
  7352. return len;
  7353. #endif
  7354. }
  7355. int primary_display_lcm_ATA(void)
  7356. {
  7357. DISP_STATUS ret = DISP_STATUS_OK;
  7358. DISPFUNC();
  7359. _primary_path_lock(__func__);
  7360. if (pgc->state == 0) {
  7361. DISPCHECK("ATA_LCM, primary display path is already sleep, skip\n");
  7362. goto done;
  7363. }
  7364. DISPCHECK("[ATA_LCM]primary display path stop[begin]\n");
  7365. if (primary_display_is_video_mode())
  7366. dpmgr_path_ioctl(pgc->dpmgr_handle, NULL, DDP_STOP_VIDEO_MODE, NULL);
  7367. DISPCHECK("[ATA_LCM]primary display path stop[end]\n");
  7368. ret = disp_lcm_ATA(pgc->plcm);
  7369. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7370. /* for video mode, we need to force trigger here
  7371. * for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start
  7372. */
  7373. if (primary_display_is_video_mode())
  7374. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7375. done:
  7376. _primary_path_unlock(__func__);
  7377. return ret;
  7378. }
  7379. int fbconfig_get_esd_check_test(uint32_t dsi_id, uint32_t cmd, uint8_t *buffer, uint32_t num)
  7380. {
  7381. int ret = 0;
  7382. /* extern int fbconfig_get_esd_check(DSI_INDEX dsi_id, uint32_t cmd, uint8_t *buffer, uint32_t num); */
  7383. _primary_path_lock(__func__);
  7384. if (pgc->state == DISP_SLEPT) {
  7385. DISPCHECK("[ESD]primary display path is slept?? -- skip esd check\n");
  7386. _primary_path_unlock(__func__);
  7387. goto done;
  7388. }
  7389. /* primary_display_esd_check_enable(0); */
  7390. /* / 1: stop path */
  7391. _cmdq_stop_trigger_loop();
  7392. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7393. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  7394. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  7395. DISPCHECK("[ESD]stop dpmgr path[end]\n");
  7396. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7397. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  7398. /* dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE); */
  7399. ret = fbconfig_get_esd_check(dsi_id, cmd, buffer, num);
  7400. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7401. DISPCHECK("[ESD]start dpmgr path[end]\n");
  7402. if (primary_display_is_video_mode()) {
  7403. /* for video mode, we need to force trigger here */
  7404. /* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start */
  7405. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7406. }
  7407. _cmdq_start_trigger_loop();
  7408. DISPCHECK("[ESD]start cmdq trigger loop[end]\n");
  7409. _primary_path_unlock(__func__);
  7410. done:
  7411. return ret;
  7412. }
  7413. int Panel_Master_dsi_config_entry(const char *name, void *config_value)
  7414. {
  7415. int ret = 0;
  7416. int force_trigger_path = 0;
  7417. uint32_t *config_dsi = (uint32_t *)config_value;
  7418. LCM_PARAMS *lcm_param = NULL;
  7419. LCM_DRIVER *pLcm_drv = NULL;
  7420. int esd_check_backup = 0;
  7421. DISPFUNC();
  7422. pLcm_drv = DISP_GetLcmDrv();
  7423. esd_check_backup = atomic_read(&esd_check_task_wakeup);
  7424. if (!strcmp(name, "DRIVER_IC_RESET") || !strcmp(name, "PM_DDIC_CONFIG")) {
  7425. primary_display_esd_check_enable(0);
  7426. msleep(2500);
  7427. }
  7428. _primary_path_lock(__func__);
  7429. lcm_param = disp_lcm_get_params(pgc->plcm);
  7430. if (pgc->state == DISP_SLEPT) {
  7431. DISPERR("[Pmaster]Panel_Master: primary display path is slept??\n");
  7432. goto done;
  7433. }
  7434. /* / Esd Check : Read from lcm */
  7435. /* / the following code is to */
  7436. /* / 0: lock path */
  7437. /* / 1: stop path */
  7438. /* / 2: do esd check (!!!) */
  7439. /* / 3: start path */
  7440. /* / 4: unlock path */
  7441. /* / 1: stop path */
  7442. _cmdq_stop_trigger_loop();
  7443. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7444. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  7445. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  7446. DISPCHECK("[ESD]stop dpmgr path[end]\n");
  7447. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7448. DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
  7449. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  7450. if ((!strcmp(name, "PM_CLK")) || (!strcmp(name, "PM_SSC")))
  7451. Panel_Master_primary_display_config_dsi(name, *config_dsi);
  7452. else if (!strcmp(name, "PM_DDIC_CONFIG")) {
  7453. Panel_Master_DDIC_config();
  7454. force_trigger_path = 1;
  7455. } else if (!strcmp(name, "DRIVER_IC_RESET")) {
  7456. if (pLcm_drv && pLcm_drv->init_power)
  7457. pLcm_drv->init_power();
  7458. if (pLcm_drv)
  7459. pLcm_drv->init();
  7460. else
  7461. ret = -1;
  7462. force_trigger_path = 1;
  7463. }
  7464. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7465. if (primary_display_is_video_mode()) {
  7466. /* for video mode, we need to force trigger here */
  7467. /* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW when trigger loop start */
  7468. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7469. force_trigger_path = 0;
  7470. }
  7471. _cmdq_start_trigger_loop();
  7472. DISPCHECK("[Pmaster]start cmdq trigger loop\n");
  7473. done:
  7474. _primary_path_unlock(__func__);
  7475. if (force_trigger_path) {/* command mode only */
  7476. primary_display_trigger(0, NULL, 0);
  7477. DISPCHECK("[Pmaster]force trigger display path\r\n");
  7478. }
  7479. atomic_set(&esd_check_task_wakeup, esd_check_backup);
  7480. return ret;
  7481. }
  7482. /*
  7483. mode: 0, switch to cmd mode; 1, switch to vdo mode
  7484. */
  7485. int primary_display_switch_dst_mode(int mode)
  7486. {
  7487. DISP_STATUS ret = DISP_STATUS_ERROR;
  7488. #ifdef DISP_SWITCH_DST_MODE
  7489. void *lcm_cmd = NULL;
  7490. DISPFUNC();
  7491. _primary_path_switch_dst_lock();
  7492. disp_sw_mutex_lock(&(pgc->capture_lock));
  7493. if (pgc->plcm->params->type != LCM_TYPE_DSI) {
  7494. pr_debug("[primary_display_switch_dst_mode] Error, only support DSI IF\n");
  7495. goto done;
  7496. }
  7497. if (pgc->state == DISP_SLEPT) {
  7498. DISPCHECK
  7499. ("[primary_display_switch_dst_mode], primary display path is already sleep, skip\n");
  7500. goto done;
  7501. }
  7502. if (mode == primary_display_cur_dst_mode) {
  7503. DISPCHECK("[primary_display_switch_dst_mode]not need switch, cur_mode:%d, switch_mode:%d\n",
  7504. primary_display_cur_dst_mode, mode);
  7505. goto done;
  7506. }
  7507. /* DISPCHECK("[primary_display_switch_mode]need switch, cur_mode:%d, switch_mode:%d\n",
  7508. primary_display_cur_dst_mode, mode);
  7509. */
  7510. lcm_cmd = disp_lcm_switch_mode(pgc->plcm, mode);
  7511. if (lcm_cmd == NULL) {
  7512. DISPCHECK("[primary_display_switch_dst_mode]get lcm cmd fail %d, %d\n",
  7513. primary_display_cur_dst_mode, mode);
  7514. goto done;
  7515. } else {
  7516. int temp_mode = 0;
  7517. if (0 != dpmgr_path_ioctl(pgc->dpmgr_handle, pgc->cmdq_handle_config,
  7518. DDP_SWITCH_LCM_MODE, lcm_cmd)) {
  7519. pr_debug("switch lcm mode fail, return directly\n");
  7520. goto done;
  7521. }
  7522. _primary_path_lock(__func__);
  7523. temp_mode = (int)(pgc->plcm->params->dsi.mode);
  7524. pgc->plcm->params->dsi.mode = pgc->plcm->params->dsi.switch_mode;
  7525. pgc->plcm->params->dsi.switch_mode = temp_mode;
  7526. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  7527. if (0 != dpmgr_path_ioctl(pgc->dpmgr_handle, pgc->cmdq_handle_config,
  7528. DDP_SWITCH_DSI_MODE, lcm_cmd)) {
  7529. pr_debug("switch dsi mode fail, return directly\n");
  7530. _primary_path_unlock(__func__);
  7531. goto done;
  7532. }
  7533. }
  7534. primary_display_sodi_rule_init();
  7535. _cmdq_stop_trigger_loop();
  7536. _cmdq_build_trigger_loop();
  7537. _cmdq_start_trigger_loop();
  7538. _cmdq_reset_config_handle(); /* must do this */
  7539. _cmdq_insert_wait_frame_done_token();
  7540. primary_display_cur_dst_mode = mode;
  7541. if (primary_display_is_video_mode())
  7542. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC,
  7543. DDP_IRQ_RDMA0_DONE);
  7544. else
  7545. dpmgr_map_event_to_irq(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC,
  7546. DDP_IRQ_DSI0_EXT_TE);
  7547. _primary_path_unlock(__func__);
  7548. ret = DISP_STATUS_OK;
  7549. done:
  7550. /* dprec_handle_option(0x0); */
  7551. disp_sw_mutex_unlock(&(pgc->capture_lock));
  7552. _primary_path_switch_dst_unlock();
  7553. #else
  7554. pr_debug("[ERROR: primary_display_switch_dst_mode]this function not enable in disp driver\n");
  7555. #endif
  7556. return ret;
  7557. }
  7558. /* warm reset ovl0 and ovl1 by CMDQ */
  7559. void primary_display_reset_ovl_by_cmdq(unsigned int force)
  7560. {
  7561. cmdqRecWaitNoClear(pgc->cmdq_handle_config, CMDQ_EVENT_MUTEX0_STREAM_EOF);
  7562. if (force == 0) { /* warm reset ovl */
  7563. DISPMSG("warm reset ovl\n");
  7564. /* reset ovl0 */
  7565. DISP_REG_SET(pgc->cmdq_handle_config, DISP_REG_OVL_RST, 0x1);
  7566. DISP_REG_SET(pgc->cmdq_handle_config, DISP_REG_OVL_RST, 0x0);
  7567. cmdqRecPoll(pgc->cmdq_handle_config, disp_addr_convert(DISP_REG_OVL_STA), 0, 0x1);
  7568. /* reset ovl1 */
  7569. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY) {
  7570. DISP_REG_SET(pgc->cmdq_handle_config,
  7571. DISP_REG_OVL_RST + DISP_OVL_INDEX_OFFSET, 0x1);
  7572. DISP_REG_SET(pgc->cmdq_handle_config,
  7573. DISP_REG_OVL_RST + DISP_OVL_INDEX_OFFSET, 0x0);
  7574. cmdqRecPoll(pgc->cmdq_handle_config,
  7575. disp_addr_convert(DISP_REG_OVL_STA + DISP_OVL_INDEX_OFFSET), 0, 0x1);
  7576. }
  7577. } else if (force == 1) { /* force reset ovl */
  7578. DISPMSG("force reset ovl\n");
  7579. /* reset ovl0 */
  7580. DISP_REG_SET_FIELD(pgc->cmdq_handle_config, RST_FLD_FORCE_RST, DISP_REG_OVL_RST, 0x1);
  7581. DISP_REG_SET_FIELD(pgc->cmdq_handle_config, RST_FLD_FORCE_RST, DISP_REG_OVL_RST, 0x0);
  7582. cmdqRecPoll(pgc->cmdq_handle_config, disp_addr_convert(DISP_REG_OVL_STA), 0, 0x1);
  7583. /* reset ovl1 */
  7584. if (ovl_get_status() == DDP_OVL1_STATUS_PRIMARY) {
  7585. DISP_REG_SET_FIELD(pgc->cmdq_handle_config, RST_FLD_FORCE_RST,
  7586. DISP_REG_OVL_RST + DISP_OVL_INDEX_OFFSET, 0x1);
  7587. DISP_REG_SET_FIELD(pgc->cmdq_handle_config, RST_FLD_FORCE_RST,
  7588. DISP_REG_OVL_RST + DISP_OVL_INDEX_OFFSET, 0x0);
  7589. cmdqRecPoll(pgc->cmdq_handle_config,
  7590. disp_addr_convert(DISP_REG_OVL_STA + DISP_OVL_INDEX_OFFSET), 0,
  7591. 0x1);
  7592. }
  7593. }
  7594. }
  7595. /* extern void DSI_ForceConfig(int forceconfig); */
  7596. /* extern int DSI_set_roi(int x, int y); */
  7597. /* extern int DSI_check_roi(void); */
  7598. /* extern atomic_t ESDCheck_byCPU; */
  7599. static int width_array[] = { 2560, 1440, 1920, 1280, 1200, 800, 960, 640 };
  7600. static int heigh_array[] = { 1440, 2560, 1200, 800, 1920, 1280, 640, 960 };
  7601. static int array_id[] = { 6, 2, 7, 4, 3, 0, 5, 1 };
  7602. LCM_PARAMS *lcm_param2 = NULL;
  7603. disp_ddp_path_config data_config2;
  7604. int primary_display_te_test(void)
  7605. {
  7606. int ret = 0;
  7607. int try_cnt = 3;
  7608. int time_interval = 0;
  7609. int time_interval_max = 0;
  7610. long long time_te = 0;
  7611. long long time_framedone = 0;
  7612. pr_debug("display_test te begin\n");
  7613. if (primary_display_is_video_mode()) {
  7614. pr_debug("Video Mode No TE\n");
  7615. return ret;
  7616. }
  7617. while (try_cnt >= 0) {
  7618. try_cnt--;
  7619. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_IF_VSYNC, HZ * 1);
  7620. time_te = sched_clock();
  7621. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7622. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  7623. time_framedone = sched_clock();
  7624. time_interval = (int)(time_framedone - time_te);
  7625. time_interval = time_interval / 1000;
  7626. if (time_interval > time_interval_max)
  7627. time_interval_max = time_interval;
  7628. }
  7629. if (time_interval_max > 20000)
  7630. ret = 0;
  7631. else
  7632. ret = -1;
  7633. if (ret >= 0)
  7634. pr_debug("[display_test_result]==>Force On TE Open!(%d)\n", time_interval_max);
  7635. else
  7636. pr_debug("[display_test_result]==>Force On TE Closed!(%d)\n", time_interval_max);
  7637. pr_debug("display_test te end\n");
  7638. return ret;
  7639. }
  7640. #if 0
  7641. int primary_display_fps_test(void)
  7642. {
  7643. int ret = 0;
  7644. unsigned int w_backup = 0;
  7645. unsigned int h_backup = 0;
  7646. LCM_DSI_MODE_CON dsi_mode_backup = primary_display_is_video_mode();
  7647. memset((void *)&data_config2, 0, sizeof(data_config2));
  7648. lcm_param2 = NULL;
  7649. memcpy((void *)&data_config2, (void *)dpmgr_path_get_last_config(pgc->dpmgr_handle),
  7650. sizeof(disp_ddp_path_config));
  7651. w_backup = data_config2.dst_w;
  7652. h_backup = data_config2.dst_h;
  7653. DISPCHECK("[display_test]w_backup %d h_backup %d dsi_mode_backup %d\n", w_backup, h_backup,
  7654. dsi_mode_backup);
  7655. /* for dsi config */
  7656. DSI_ForceConfig(1);
  7657. DISPCHECK("[display_test]FPS config[begin]\n");
  7658. lcm_param2 = disp_lcm_get_params(pgc->plcm);
  7659. lcm_param2->dsi.mode = SYNC_PULSE_VDO_MODE;
  7660. lcm_param2->dsi.vertical_active_line = 1280;
  7661. lcm_param2->dsi.horizontal_active_pixel = 360;
  7662. data_config2.dst_w = 360;
  7663. data_config2.dst_h = 1280;
  7664. data_config2.dispif_config.dsi.vertical_active_line = 1280;
  7665. data_config2.dispif_config.dsi.horizontal_active_pixel = 360;
  7666. data_config2.dispif_config.dsi.mode = SYNC_PULSE_VDO_MODE;
  7667. data_config2.dst_dirty = 1;
  7668. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  7669. DISPCHECK("[display_test]==>FPS set vdo mode done, is_vdo_mode:%d\n",
  7670. primary_display_is_video_mode());
  7671. dpmgr_path_connect(pgc->dpmgr_handle, CMDQ_DISABLE);
  7672. dpmgr_path_config(pgc->dpmgr_handle, &data_config2, CMDQ_DISABLE);
  7673. data_config2.dst_dirty = 0;
  7674. DISPCHECK("[display_test]FPS config[end]\n");
  7675. DISPCHECK("[display_test]Start dpmgr path[begin]\n");
  7676. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7677. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7678. DISPERR("[display_test]==>Fatal error, we didn't trigger display path but it's already busy\n");
  7679. DISPCHECK("[display_test]Start dpmgr path[end]\n");
  7680. DISPCHECK("[display_test]Trigger dpmgr path[begin]\n");
  7681. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7682. DISPCHECK("[display_test]Trigger dpmgr path[end]\n");
  7683. /* check fps bonding: rdma frame end interval < 12ms */
  7684. disp_record_rdma_end_interval(1);
  7685. /* loop 50 times to get max rdma end interval */
  7686. int i = 50;
  7687. while (i--)
  7688. dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  7689. unsigned long long min_interval = disp_get_rdma_min_interval();
  7690. unsigned long long max_interval = disp_get_rdma_max_interval();
  7691. disp_record_rdma_end_interval(0);
  7692. DISPCHECK("[display_test]Check RDMA frame end interval:%lld[end]\n", min_interval);
  7693. if (min_interval < 12 * 1000000) {
  7694. DISPCHECK("[display_test_result]=>0.No limit\n");
  7695. } else {
  7696. DISPCHECK("[display_test_result]=>1.limit max_interval %lld\n", max_interval);
  7697. if (max_interval < 13 * 1000000)
  7698. DISPCHECK("[display_test_result]=>2.naughty enable\n");
  7699. }
  7700. DISPCHECK("[display_test]Stop dpmgr path[begin]\n");
  7701. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  7702. DISPCHECK("[display_test]Stop dpmgr path[end]\n");
  7703. DISPCHECK("[display_test]Restore path config[begin]\n");
  7704. lcm_param2 = disp_lcm_get_params(pgc->plcm);
  7705. lcm_param2->dsi.mode = dsi_mode_backup;
  7706. lcm_param2->dsi.vertical_active_line = h_backup;
  7707. lcm_param2->dsi.horizontal_active_pixel = w_backup;
  7708. data_config2.dispif_config.dsi.vertical_active_line = h_backup;
  7709. data_config2.dispif_config.dsi.horizontal_active_pixel = w_backup;
  7710. data_config2.dispif_config.dsi.mode = dsi_mode_backup;
  7711. data_config2.dst_w = w_backup;
  7712. data_config2.dst_h = h_backup;
  7713. data_config2.dst_dirty = 1;
  7714. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  7715. DISPCHECK("[display_test]==>Restore mode done, is_vdo_mode:%d\n",
  7716. primary_display_is_video_mode());
  7717. DISPCHECK("[display_test]==>Restore resolution done, w=%d, h=%d\n", w_backup, h_backup);
  7718. dpmgr_path_connect(pgc->dpmgr_handle, CMDQ_DISABLE);
  7719. dpmgr_path_config(pgc->dpmgr_handle, &data_config2, CMDQ_DISABLE);
  7720. data_config2.dst_dirty = 0;
  7721. DSI_ForceConfig(0);
  7722. DISPCHECK("[display_test]Restore path config[end]\n");
  7723. return ret;
  7724. }
  7725. #endif
  7726. int primary_display_roi_test(int x, int y)
  7727. {
  7728. int ret = 0;
  7729. pr_debug("display_test roi begin\n");
  7730. pr_debug("display_test roi set roi %d, %d\n", x, y);
  7731. DSI_set_roi(x, y);
  7732. msleep(50);
  7733. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7734. msleep(50);
  7735. pr_debug("display_test DSI_check_roi\n");
  7736. ret = DSI_check_roi();
  7737. msleep(20);
  7738. if (ret == 0)
  7739. pr_debug("[display_test_result]==>DSI_ROI limit!\n");
  7740. else
  7741. pr_debug("[display_test_result]==>DSI_ROI Normal!\n");
  7742. pr_debug("display_test set roi %d, %d\n", 0, 0);
  7743. DSI_set_roi(0, 0);
  7744. msleep(20);
  7745. DISPCHECK("display_test end\n");
  7746. return ret;
  7747. }
  7748. int primary_display_resolution_test(void)
  7749. {
  7750. int ret = 0;
  7751. int i = 0;
  7752. unsigned int w_backup = 0;
  7753. unsigned int h_backup = 0;
  7754. int dst_width = 0;
  7755. int dst_heigh = 0;
  7756. LCM_DSI_MODE_CON dsi_mode_backup = primary_display_is_video_mode();
  7757. memset((void *)&data_config2, 0, sizeof(data_config2));
  7758. lcm_param2 = NULL;
  7759. memcpy((void *)&data_config2, (void *)dpmgr_path_get_last_config(pgc->dpmgr_handle),
  7760. sizeof(disp_ddp_path_config));
  7761. w_backup = data_config2.dst_w;
  7762. h_backup = data_config2.dst_h;
  7763. DISPCHECK("[display_test resolution]w_backup %d h_backup %d dsi_mode_backup %d\n",
  7764. w_backup, h_backup, dsi_mode_backup);
  7765. /* for dsi config */
  7766. DSI_ForceConfig(1);
  7767. for (i = 0; i < sizeof(width_array) / sizeof(int); i++) {
  7768. dst_width = width_array[i];
  7769. dst_heigh = heigh_array[i];
  7770. DISPCHECK("[display_test resolution] width %d, heigh %d\n", dst_width, dst_heigh);
  7771. lcm_param2 = disp_lcm_get_params(pgc->plcm);
  7772. lcm_param2->dsi.mode = CMD_MODE;
  7773. lcm_param2->dsi.horizontal_active_pixel = dst_width;
  7774. lcm_param2->dsi.vertical_active_line = dst_heigh;
  7775. data_config2.dispif_config.dsi.mode = CMD_MODE;
  7776. data_config2.dispif_config.dsi.horizontal_active_pixel = dst_width;
  7777. data_config2.dispif_config.dsi.vertical_active_line = dst_heigh;
  7778. data_config2.dst_w = dst_width;
  7779. data_config2.dst_h = dst_heigh;
  7780. data_config2.ovl_config[0].layer = 0;
  7781. data_config2.ovl_config[0].layer_en = 0;
  7782. data_config2.ovl_config[1].layer = 1;
  7783. data_config2.ovl_config[1].layer_en = 0;
  7784. data_config2.ovl_config[2].layer = 2;
  7785. data_config2.ovl_config[2].layer_en = 0;
  7786. data_config2.ovl_config[3].layer = 3;
  7787. data_config2.ovl_config[3].layer_en = 0;
  7788. data_config2.dst_dirty = 1;
  7789. data_config2.ovl_dirty = 1;
  7790. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  7791. dpmgr_path_config(pgc->dpmgr_handle, &data_config2, CMDQ_DISABLE);
  7792. data_config2.dst_dirty = 0;
  7793. data_config2.ovl_dirty = 0;
  7794. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7795. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7796. DISPERR("[display_test]==>Fatal error, we didn't trigger display path but it's already busy\n");
  7797. dpmgr_path_trigger(pgc->dpmgr_handle, NULL, CMDQ_DISABLE);
  7798. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  7799. if (ret > 0) {
  7800. if (!dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  7801. if (i == 0)
  7802. DISPCHECK("[display_test resolution] display_result 0x%x unlimited!\n",
  7803. array_id[i]);
  7804. else if (i == 1)
  7805. DISPCHECK("[display_test resolution] display_result 0x%x unlimited (W<H)\n",
  7806. array_id[i]);
  7807. else
  7808. DISPCHECK("[display_test resolution] display_result 0x%x(%d x %d)\n",
  7809. array_id[i], dst_width, dst_heigh);
  7810. break;
  7811. }
  7812. }
  7813. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  7814. }
  7815. dpmgr_path_stop(pgc->dpmgr_handle, CMDQ_DISABLE);
  7816. lcm_param2 = disp_lcm_get_params(pgc->plcm);
  7817. lcm_param2->dsi.mode = dsi_mode_backup;
  7818. lcm_param2->dsi.vertical_active_line = h_backup;
  7819. lcm_param2->dsi.horizontal_active_pixel = w_backup;
  7820. data_config2.dispif_config.dsi.vertical_active_line = h_backup;
  7821. data_config2.dispif_config.dsi.horizontal_active_pixel = w_backup;
  7822. data_config2.dispif_config.dsi.mode = dsi_mode_backup;
  7823. data_config2.dst_w = w_backup;
  7824. data_config2.dst_h = h_backup;
  7825. data_config2.dst_dirty = 1;
  7826. dpmgr_path_set_video_mode(pgc->dpmgr_handle, primary_display_is_video_mode());
  7827. dpmgr_path_connect(pgc->dpmgr_handle, CMDQ_DISABLE);
  7828. dpmgr_path_config(pgc->dpmgr_handle, &data_config2, CMDQ_DISABLE);
  7829. data_config2.dst_dirty = 0;
  7830. DSI_ForceConfig(0);
  7831. return ret;
  7832. }
  7833. int primary_display_check_test(void)
  7834. {
  7835. int ret = 0;
  7836. int esd_backup = 0;
  7837. DISPCHECK("[display_test]Display test[Start]\n");
  7838. _primary_path_lock(__func__);
  7839. /* disable esd check */
  7840. if (atomic_read(&esd_check_task_wakeup)) {
  7841. esd_backup = 1;
  7842. primary_display_esd_check_enable(0);
  7843. msleep(2000);
  7844. DISPCHECK("[display_test]Disable esd check end\n");
  7845. }
  7846. /* if suspend => return */
  7847. if (pgc->state == DISP_SLEPT) {
  7848. DISPCHECK("[display_test_result]======================================\n");
  7849. DISPCHECK("[display_test_result]==>Test Fail : primary display path is slept\n");
  7850. DISPCHECK("[display_test_result]======================================\n");
  7851. goto done;
  7852. }
  7853. /* stop trigger loop */
  7854. DISPCHECK("[display_test]Stop trigger loop[begin]\n");
  7855. _cmdq_stop_trigger_loop();
  7856. atomic_set(&ESDCheck_byCPU, 1);
  7857. if (dpmgr_path_is_busy(pgc->dpmgr_handle)) {
  7858. DISPCHECK("[display_test]==>primary display path is busy\n");
  7859. ret = dpmgr_wait_event_timeout(pgc->dpmgr_handle, DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
  7860. if (ret <= 0)
  7861. dpmgr_path_reset(pgc->dpmgr_handle, CMDQ_DISABLE);
  7862. DISPCHECK("[display_test]==>wait frame done ret:%d\n", ret);
  7863. }
  7864. DISPCHECK("[display_test]Stop trigger loop[end]\n");
  7865. /* test rdma res after reset */
  7866. /* primary_display_rdma_res_test(); */
  7867. /* test force te */
  7868. primary_display_te_test();
  7869. /* test roi */
  7870. primary_display_roi_test(30, 30);
  7871. /* test resolution test */
  7872. primary_display_resolution_test();
  7873. /* mutex fps */
  7874. /* primary_display_fps_test(); */
  7875. DISPCHECK("[display_test]start dpmgr path[begin]\n");
  7876. dpmgr_path_start(pgc->dpmgr_handle, CMDQ_DISABLE);
  7877. if (dpmgr_path_is_busy(pgc->dpmgr_handle))
  7878. DISPERR("[display_test]==>Fatal error, we didn't trigger display path but it's already busy\n");
  7879. DISPCHECK("[display_test]start dpmgr path[end]\n");
  7880. DISPCHECK("[display_test]Start trigger loop[begin]\n");
  7881. _cmdq_start_trigger_loop();
  7882. DISPCHECK("[display_test]Start trigger loop[end]\n");
  7883. atomic_set(&ESDCheck_byCPU, 0);
  7884. done:
  7885. /* restore esd */
  7886. if (esd_backup == 1) {
  7887. primary_display_esd_check_enable(1);
  7888. DISPCHECK("[display_test]Restore esd check\n");
  7889. }
  7890. /* unlock path */
  7891. _primary_path_unlock(__func__);
  7892. DISPCHECK("[display_test]Display test[End]\n");
  7893. return ret;
  7894. }
  7895. static DISP_POWER_STATE tui_power_stat_backup;
  7896. static int tui_session_mode_backup;
  7897. int display_enter_tui(void)
  7898. {
  7899. msleep(500);
  7900. DISPMSG("TDDP: %s\n", __func__);
  7901. MMProfileLogEx(ddp_mmp_get_events()->tui, MMProfileFlagStart, 0, 0);
  7902. _primary_path_lock(__func__);
  7903. if (primary_get_state() != DISP_ALIVE) {
  7904. DISPERR("Can't enter tui: current_stat=%d is not alive\n", primary_get_state());
  7905. goto err0;
  7906. }
  7907. tui_power_stat_backup = primary_set_state(DISP_BLANK);
  7908. if (primary_display_is_mirror_mode()) {
  7909. DISPERR("Can't enter tui: current_mode=%s\n", session_mode_spy(pgc->session_mode));
  7910. goto err1;
  7911. }
  7912. #ifdef MTK_DISP_IDLE_LP
  7913. _disp_primary_path_exit_idle(__func__, 0);
  7914. #endif
  7915. tui_session_mode_backup = pgc->session_mode;
  7916. primary_display_switch_mode_nolock(DISP_SESSION_DECOUPLE_MODE, pgc->session_id, 0);
  7917. MMProfileLogEx(ddp_mmp_get_events()->tui, MMProfileFlagPulse, 0, 1);
  7918. _primary_path_unlock(__func__);
  7919. return 0;
  7920. err1:
  7921. primary_set_state(tui_power_stat_backup);
  7922. err0:
  7923. MMProfileLogEx(ddp_mmp_get_events()->tui, MMProfileFlagEnd, 0, 0);
  7924. _primary_path_unlock(__func__);
  7925. return -1;
  7926. }
  7927. int display_exit_tui(void)
  7928. {
  7929. MMProfileLogEx(ddp_mmp_get_events()->tui, MMProfileFlagPulse, 1, 1);
  7930. _primary_path_lock(__func__);
  7931. primary_set_state(tui_power_stat_backup);
  7932. /* trigger rdma to display last normal buffer
  7933. _decouple_update_rdma_config_nolock();*/
  7934. /* workaround: wait until this frame triggered to lcm */
  7935. msleep(32);
  7936. primary_display_switch_mode_nolock(tui_session_mode_backup, pgc->session_id, 0);
  7937. _primary_path_unlock(__func__);
  7938. MMProfileLogEx(ddp_mmp_get_events()->tui, MMProfileFlagEnd, 0, 0);
  7939. DISPMSG("TDDP: %s\n", __func__);
  7940. return 0;
  7941. }