123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338 |
- From 18383d506cc9489eb5e55bc3f7f481e82d5e33c5 Mon Sep 17 00:00:00 2001
- From: Vincent Sanders <vincent.sanders@collabora.co.uk>
- Date: Wed, 30 Jan 2013 12:45:18 +0000
- Subject: [PATCH] bcm2835: add v4l2 camera device
- - Supports raw YUV capture, preview, JPEG and H264.
- - Uses videobuf2 for data transfer, using dma_buf.
- - Uses 3.6.10 timestamping
- - Camera power based on use
- - Uses immutable input mode on video encoder
- Signed-off-by: Daniel Stone <daniels@collabora.com>
- Signed-off-by: Luke Diamand <luked@broadcom.com>
- V4L2: Fixes from 6by9
- V4L2: Fix EV values. Add manual shutter speed control
- V4L2 EV values should be in units of 1/1000. Corrected.
- Add support for V4L2_CID_EXPOSURE_ABSOLUTE which should
- give manual shutter control. Requires manual exposure mode
- to be selected first.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Correct JPEG Q-factor range
- Should be 1-100, not 0-100
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix issue of driver jamming if STREAMON failed.
- Fix issue where the driver was left in a partially enabled
- state if STREAMON failed, and would then reject many IOCTLs
- as it thought it was streaming.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix ISO controls.
- Driver was passing the index to the GPU, and not the desired
- ISO value.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add flicker avoidance controls
- Add support for V4L2_CID_POWER_LINE_FREQUENCY to set flicker
- avoidance frequencies.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add support for frame rate control.
- Add support for frame rate (or time per frame as V4L2
- inverts it) control via s_parm.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Improve G_FBUF handling so we pass conformance
- Return some sane numbers for get framebuffer so that
- we pass conformance.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix information advertised through g_vidfmt
- Width and height were being stored based on incorrect
- values.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add support for inline H264 headers
- Add support for V4L2_CID_MPEG_VIDEO_REPEAT_SEQ_HEADER
- to control H264 inline headers.
- Requires firmware fix to work correctly, otherwise format
- has to be set to H264 before this parameter is set.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix JPEG timestamp issue
- JPEG images were coming through from the GPU with timestamp
- of 0. Detect this and give current system time instead
- of some invalid value.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix issue when switching down JPEG resolution.
- JPEG buffer size calculation is based on input resolution.
- Input resolution was being configured after output port
- format. Caused failures if switching from one JPEG resolution
- to a smaller one.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Enable MJPEG encoding
- Requires GPU firmware update to support MJPEG encoder.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Correct flag settings for compressed formats
- Set flags field correctly on enum_fmt_vid_cap for compressed
- image formats.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: H264 profile & level ctrls, FPS control and auto exp pri
- Several control handling updates.
- H264 profile and level controls.
- Timeperframe/FPS reworked to add V4L2_CID_EXPOSURE_AUTO_PRIORITY to
- select whether AE is allowed to override the framerate specified.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Correct BGR24 to RGB24 in format table
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add additional pixel formats. Correct colourspace
- Adds the other flavours of YUYV, and NV12.
- Corrects the overlay advertised colourspace.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Drop logging msg from info to debug
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Initial pass at scene modes.
- Only supports exposure mode and metering modes.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add manual white balance control.
- Adds support for V4L2_CID_RED_BALANCE and
- V4L2_CID_BLUE_BALANCE. Only has an effect if
- V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE has
- V4L2_WHITE_BALANCE_MANUAL selected.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- config: Enable V4L / MMAL driver
- V4L2: Increase the MMAL timeout to 3sec
- MJPEG codec flush is now taking longer and results
- in a kernel panic if the driver has stopped waiting for
- the result when it finally completes.
- Increase the timeout value from 1 to 3secs.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add support for setting H264_I_PERIOD
- Adds support for the parameter V4L2_CID_MPEG_VIDEO_H264_I_PERIOD
- to set the frequency with which I frames are produced.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Enable GPU function for removing padding from images.
- GPU can now support arbitrary strides, although may require
- additional processing to achieve it. Enable this feature
- so that the images delivered are the size requested.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add support for V4L2_PIX_FMT_BGR32
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Set the colourspace to avoid odd YUV-RGB conversions
- Removes the amiguity from the conversion routines and stops
- them dropping back to the SD vs HD choice of coeffs.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Make video/still threshold a run-time param
- Move the define for at what resolution the driver
- switches from a video mode capture to a stills mode
- capture to module parameters.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Fix incorrect pool sizing
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add option to disable enum_framesizes.
- Gstreamer's handling of a driver that advertises
- V4L2_FRMSIZE_TYPE_STEPWISE to define the supported
- resolutions is broken. See bug
- https://bugzilla.gnome.org/show_bug.cgi?id=726521
- Optional parameter of gst_v4l2src_is_broken added.
- If non-zero, the driver claims not to support that
- ioctl, and gstreamer should be happy again (it
- guesses a set of defaults for itself).
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Add support for more image formats
- Adds YVU420 (YV12), YVU420SP (NV21), and BGR888.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- V4L2: Extend range for V4L2_CID_MPEG_VIDEO_H264_I_PERIOD
- Request to extend the range from the fairly arbitrary
- 1000 frames (33 seconds at 30fps). Extend out to the
- max range supported (int32 value).
- Also allow 0, which is handled by the codec as only
- send an I-frame on the first frame and never again.
- There may be an exception if it detects a significant
- scene change, but there's no easy way around that.
- Signed-off-by: Dave Stevenson <dsteve@broadcom.com>
- bcm2835-camera: stop_streaming now has a void return
- BCM2835-V4L2: Fix compliance test failures
- VIDIOC_TRY_FMT and VIDIOC_S_FMT tests were faling due
- to reporting V4L2_COLORSPACE_JPEG when the colour
- format wasn't V4L2_PIX_FMT_JPEG.
- Now reports V4L2_COLORSPACE_SMPTE170M for YUV formats.
- bcm2835 camera planar/packed stride length
- Added a field to the mmal_fmt struct used to compute the bytes per line
- when using a particular format. This results in the correct stride being
- calculated even when the format is planar.
- Signed-off-by: Garrett Wilson <g@floft.net>
- bcm2835: camera: check for scene not being found
- static analysis by cppcheck detected some potential NULL pointer
- dereference issues:
- [drivers/media/platform/bcm2835/controls.c:854]: (error) Possible null
- pointer dereference: scene
- (and lines 858, 859 too)
- it is possible that scene is not found because of an invalue ctrl->val
- and is therefore NULL and hence causing a null pointer dereference.
- Signed-off-by: Colin Ian King <colin.king@canonical.com>
- bcm2835: memcpy port data to m rather than rmsg
- static analysis by cppcheck detected a memcpy to rmsg which is
- not actually initialized at that point. The memcpy should be copying
- to variable m instead.
- Signed-off-by: Colin Ian King <colin.king@canonical.com>
- BCM2835-V4L2: Return buffers to videobuf2 on shutdown
- https://github.com/raspberrypi/linux/issues/817
- Fixes the kernel warning from videobuf2 as buffers
- are now returned as they are being flushed on
- stop_streaming.
- squash: Fixup bcm2835-camera for changes in kernel 4.4 api
- ---
- Documentation/video4linux/bcm2835-v4l2.txt | 60 +
- drivers/media/platform/Kconfig | 2 +
- drivers/media/platform/Makefile | 2 +
- drivers/media/platform/bcm2835/Kconfig | 25 +
- drivers/media/platform/bcm2835/Makefile | 5 +
- drivers/media/platform/bcm2835/bcm2835-camera.c | 1844 +++++++++++++++++++++
- drivers/media/platform/bcm2835/bcm2835-camera.h | 126 ++
- drivers/media/platform/bcm2835/controls.c | 1324 +++++++++++++++
- drivers/media/platform/bcm2835/mmal-common.h | 53 +
- drivers/media/platform/bcm2835/mmal-encodings.h | 127 ++
- drivers/media/platform/bcm2835/mmal-msg-common.h | 50 +
- drivers/media/platform/bcm2835/mmal-msg-format.h | 81 +
- drivers/media/platform/bcm2835/mmal-msg-port.h | 107 ++
- drivers/media/platform/bcm2835/mmal-msg.h | 404 +++++
- drivers/media/platform/bcm2835/mmal-parameters.h | 656 ++++++++
- drivers/media/platform/bcm2835/mmal-vchiq.c | 1916 ++++++++++++++++++++++
- drivers/media/platform/bcm2835/mmal-vchiq.h | 178 ++
- 17 files changed, 6960 insertions(+)
- create mode 100644 Documentation/video4linux/bcm2835-v4l2.txt
- create mode 100644 drivers/media/platform/bcm2835/Kconfig
- create mode 100644 drivers/media/platform/bcm2835/Makefile
- create mode 100644 drivers/media/platform/bcm2835/bcm2835-camera.c
- create mode 100644 drivers/media/platform/bcm2835/bcm2835-camera.h
- create mode 100644 drivers/media/platform/bcm2835/controls.c
- create mode 100644 drivers/media/platform/bcm2835/mmal-common.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-encodings.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-msg-common.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-msg-format.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-msg-port.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-msg.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-parameters.h
- create mode 100644 drivers/media/platform/bcm2835/mmal-vchiq.c
- create mode 100644 drivers/media/platform/bcm2835/mmal-vchiq.h
- --- /dev/null
- +++ b/Documentation/video4linux/bcm2835-v4l2.txt
- @@ -0,0 +1,60 @@
- +
- +BCM2835 (aka Raspberry Pi) V4L2 driver
- +======================================
- +
- +1. Copyright
- +============
- +
- +Copyright © 2013 Raspberry Pi (Trading) Ltd.
- +
- +2. License
- +==========
- +
- +This program is free software; you can redistribute it and/or modify
- +it under the terms of the GNU General Public License as published by
- +the Free Software Foundation; either version 2 of the License, or
- +(at your option) any later version.
- +
- +This program is distributed in the hope that it will be useful,
- +but WITHOUT ANY WARRANTY; without even the implied warranty of
- +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- +GNU General Public License for more details.
- +
- +You should have received a copy of the GNU General Public License
- +along with this program; if not, write to the Free Software
- +Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- +
- +3. Quick Start
- +==============
- +
- +You need a version 1.0 or later of v4l2-ctl, available from:
- + git://git.linuxtv.org/v4l-utils.git
- +
- +$ sudo modprobe bcm2835-v4l2
- +
- +Turn on the overlay:
- +
- +$ v4l2-ctl --overlay=1
- +
- +Turn off the overlay:
- +
- +$ v4l2-ctl --overlay=0
- +
- +Set the capture format for video:
- +
- +$ v4l2-ctl --set-fmt-video=width=1920,height=1088,pixelformat=4
- +
- +(Note: 1088 not 1080).
- +
- +Capture:
- +
- +$ v4l2-ctl --stream-mmap=3 --stream-count=100 --stream-to=somefile.h264
- +
- +Stills capture:
- +
- +$ v4l2-ctl --set-fmt-video=width=2592,height=1944,pixelformat=3
- +$ v4l2-ctl --stream-mmap=3 --stream-count=1 --stream-to=somefile.jpg
- +
- +List of available formats:
- +
- +$ v4l2-ctl --list-formats
- --- a/drivers/media/platform/Kconfig
- +++ b/drivers/media/platform/Kconfig
- @@ -11,6 +11,8 @@ menuconfig V4L_PLATFORM_DRIVERS
-
- if V4L_PLATFORM_DRIVERS
-
- +source "drivers/media/platform/bcm2835/Kconfig"
- +
- source "drivers/media/platform/marvell-ccic/Kconfig"
-
- config VIDEO_VIA_CAMERA
- --- a/drivers/media/platform/Makefile
- +++ b/drivers/media/platform/Makefile
- @@ -2,6 +2,8 @@
- # Makefile for the video capture/playback device drivers.
- #
-
- +obj-$(CONFIG_VIDEO_BCM2835) += bcm2835/
- +
- obj-$(CONFIG_VIDEO_TIMBERDALE) += timblogiw.o
- obj-$(CONFIG_VIDEO_M32R_AR_M64278) += arv.o
-
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/Kconfig
- @@ -0,0 +1,25 @@
- +# Broadcom VideoCore IV v4l2 camera support
- +
- +config VIDEO_BCM2835
- + bool "Broadcom BCM2835 camera interface driver"
- + depends on VIDEO_V4L2 && (ARCH_BCM2708 || ARCH_BCM2709 || ARCH_BCM2835)
- + ---help---
- + Say Y here to enable camera host interface devices for
- + Broadcom BCM2835 SoC. This operates over the VCHIQ interface
- + to a service running on VideoCore.
- +
- +
- +if VIDEO_BCM2835
- +
- +config VIDEO_BCM2835_MMAL
- + tristate "Broadcom BM2835 MMAL camera interface driver"
- + depends on BCM2708_VCHIQ
- + select VIDEOBUF2_VMALLOC
- + ---help---
- + This is a V4L2 driver for the Broadcom BCM2835 MMAL camera host interface
- +
- + To compile this driver as a module, choose M here: the
- + module will be called bcm2835-v4l2.o
- +
- +
- +endif # VIDEO_BM2835
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/Makefile
- @@ -0,0 +1,5 @@
- +bcm2835-v4l2-objs := bcm2835-camera.o controls.o mmal-vchiq.o
- +
- +obj-$(CONFIG_VIDEO_BCM2835_MMAL) += bcm2835-v4l2.o
- +
- +ccflags-$(CONFIG_VIDEO_BCM2835) += -Idrivers/misc/vc04_services -Idrivers/misc/vc04_services/interface/vcos/linuxkernel -D__VCCOREVER__=0x04000000
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/bcm2835-camera.c
- @@ -0,0 +1,1844 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +#include <linux/errno.h>
- +#include <linux/kernel.h>
- +#include <linux/module.h>
- +#include <linux/slab.h>
- +#include <media/videobuf2-vmalloc.h>
- +#include <media/videobuf2-dma-contig.h>
- +#include <media/v4l2-device.h>
- +#include <media/v4l2-ioctl.h>
- +#include <media/v4l2-ctrls.h>
- +#include <media/v4l2-fh.h>
- +#include <media/v4l2-event.h>
- +#include <media/v4l2-common.h>
- +#include <linux/delay.h>
- +
- +#include "mmal-common.h"
- +#include "mmal-encodings.h"
- +#include "mmal-vchiq.h"
- +#include "mmal-msg.h"
- +#include "mmal-parameters.h"
- +#include "bcm2835-camera.h"
- +
- +#define BM2835_MMAL_VERSION "0.0.2"
- +#define BM2835_MMAL_MODULE_NAME "bcm2835-v4l2"
- +#define MIN_WIDTH 16
- +#define MIN_HEIGHT 16
- +#define MAX_WIDTH 2592
- +#define MAX_HEIGHT 1944
- +#define MIN_BUFFER_SIZE (80*1024)
- +
- +#define MAX_VIDEO_MODE_WIDTH 1280
- +#define MAX_VIDEO_MODE_HEIGHT 720
- +
- +MODULE_DESCRIPTION("Broadcom 2835 MMAL video capture");
- +MODULE_AUTHOR("Vincent Sanders");
- +MODULE_LICENSE("GPL");
- +MODULE_VERSION(BM2835_MMAL_VERSION);
- +
- +int bcm2835_v4l2_debug;
- +module_param_named(debug, bcm2835_v4l2_debug, int, 0644);
- +MODULE_PARM_DESC(bcm2835_v4l2_debug, "Debug level 0-2");
- +
- +int max_video_width = MAX_VIDEO_MODE_WIDTH;
- +int max_video_height = MAX_VIDEO_MODE_HEIGHT;
- +module_param(max_video_width, int, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
- +MODULE_PARM_DESC(max_video_width, "Threshold for video mode");
- +module_param(max_video_height, int, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
- +MODULE_PARM_DESC(max_video_height, "Threshold for video mode");
- +
- +/* Gstreamer bug https://bugzilla.gnome.org/show_bug.cgi?id=726521
- + * v4l2src does bad (and actually wrong) things when the vidioc_enum_framesizes
- + * function says type V4L2_FRMSIZE_TYPE_STEPWISE, which we do by default.
- + * It's happier if we just don't say anything at all, when it then
- + * sets up a load of defaults that it thinks might work.
- + * If gst_v4l2src_is_broken is non-zero, then we remove the function from
- + * our function table list (actually switch to an alternate set, but same
- + * result).
- + */
- +int gst_v4l2src_is_broken = 0;
- +module_param(gst_v4l2src_is_broken, int, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
- +MODULE_PARM_DESC(gst_v4l2src_is_broken, "If non-zero, enable workaround for Gstreamer");
- +
- +static struct bm2835_mmal_dev *gdev; /* global device data */
- +
- +#define FPS_MIN 1
- +#define FPS_MAX 90
- +
- +/* timeperframe: min/max and default */
- +static const struct v4l2_fract
- + tpf_min = {.numerator = 1, .denominator = FPS_MAX},
- + tpf_max = {.numerator = 1, .denominator = FPS_MIN},
- + tpf_default = {.numerator = 1000, .denominator = 30000};
- +
- +/* video formats */
- +static struct mmal_fmt formats[] = {
- + {
- + .name = "4:2:0, planar, YUV",
- + .fourcc = V4L2_PIX_FMT_YUV420,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_I420,
- + .depth = 12,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 1,
- + },
- + {
- + .name = "4:2:2, packed, YUYV",
- + .fourcc = V4L2_PIX_FMT_YUYV,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_YUYV,
- + .depth = 16,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 2,
- + },
- + {
- + .name = "RGB24 (LE)",
- + .fourcc = V4L2_PIX_FMT_RGB24,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_BGR24,
- + .depth = 24,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 3,
- + },
- + {
- + .name = "JPEG",
- + .fourcc = V4L2_PIX_FMT_JPEG,
- + .flags = V4L2_FMT_FLAG_COMPRESSED,
- + .mmal = MMAL_ENCODING_JPEG,
- + .depth = 8,
- + .mmal_component = MMAL_COMPONENT_IMAGE_ENCODE,
- + .ybbp = 0,
- + },
- + {
- + .name = "H264",
- + .fourcc = V4L2_PIX_FMT_H264,
- + .flags = V4L2_FMT_FLAG_COMPRESSED,
- + .mmal = MMAL_ENCODING_H264,
- + .depth = 8,
- + .mmal_component = MMAL_COMPONENT_VIDEO_ENCODE,
- + .ybbp = 0,
- + },
- + {
- + .name = "MJPEG",
- + .fourcc = V4L2_PIX_FMT_MJPEG,
- + .flags = V4L2_FMT_FLAG_COMPRESSED,
- + .mmal = MMAL_ENCODING_MJPEG,
- + .depth = 8,
- + .mmal_component = MMAL_COMPONENT_VIDEO_ENCODE,
- + .ybbp = 0,
- + },
- + {
- + .name = "4:2:2, packed, YVYU",
- + .fourcc = V4L2_PIX_FMT_YVYU,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_YVYU,
- + .depth = 16,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 2,
- + },
- + {
- + .name = "4:2:2, packed, VYUY",
- + .fourcc = V4L2_PIX_FMT_VYUY,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_VYUY,
- + .depth = 16,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 2,
- + },
- + {
- + .name = "4:2:2, packed, UYVY",
- + .fourcc = V4L2_PIX_FMT_UYVY,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_UYVY,
- + .depth = 16,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 2,
- + },
- + {
- + .name = "4:2:0, planar, NV12",
- + .fourcc = V4L2_PIX_FMT_NV12,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_NV12,
- + .depth = 12,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 1,
- + },
- + {
- + .name = "RGB24 (BE)",
- + .fourcc = V4L2_PIX_FMT_BGR24,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_RGB24,
- + .depth = 24,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 3,
- + },
- + {
- + .name = "4:2:0, planar, YVU",
- + .fourcc = V4L2_PIX_FMT_YVU420,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_YV12,
- + .depth = 12,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 1,
- + },
- + {
- + .name = "4:2:0, planar, NV21",
- + .fourcc = V4L2_PIX_FMT_NV21,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_NV21,
- + .depth = 12,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 1,
- + },
- + {
- + .name = "RGB32 (BE)",
- + .fourcc = V4L2_PIX_FMT_BGR32,
- + .flags = 0,
- + .mmal = MMAL_ENCODING_BGRA,
- + .depth = 32,
- + .mmal_component = MMAL_COMPONENT_CAMERA,
- + .ybbp = 4,
- + },
- +};
- +
- +static struct mmal_fmt *get_format(struct v4l2_format *f)
- +{
- + struct mmal_fmt *fmt;
- + unsigned int k;
- +
- + for (k = 0; k < ARRAY_SIZE(formats); k++) {
- + fmt = &formats[k];
- + if (fmt->fourcc == f->fmt.pix.pixelformat)
- + break;
- + }
- +
- + if (k == ARRAY_SIZE(formats))
- + return NULL;
- +
- + return &formats[k];
- +}
- +
- +/* ------------------------------------------------------------------
- + Videobuf queue operations
- + ------------------------------------------------------------------*/
- +
- +static int queue_setup(struct vb2_queue *vq, const void *parg,
- + unsigned int *nbuffers, unsigned int *nplanes,
- + unsigned int sizes[], void *alloc_ctxs[])
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vq);
- + unsigned long size;
- +
- + /* refuse queue setup if port is not configured */
- + if (dev->capture.port == NULL) {
- + v4l2_err(&dev->v4l2_dev,
- + "%s: capture port not configured\n", __func__);
- + return -EINVAL;
- + }
- +
- + size = dev->capture.port->current_buffer.size;
- + if (size == 0) {
- + v4l2_err(&dev->v4l2_dev,
- + "%s: capture port buffer size is zero\n", __func__);
- + return -EINVAL;
- + }
- +
- + if (*nbuffers < (dev->capture.port->current_buffer.num + 2))
- + *nbuffers = (dev->capture.port->current_buffer.num + 2);
- +
- + *nplanes = 1;
- +
- + sizes[0] = size;
- +
- + /*
- + * videobuf2-vmalloc allocator is context-less so no need to set
- + * alloc_ctxs array.
- + */
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "%s: dev:%p\n",
- + __func__, dev);
- +
- + return 0;
- +}
- +
- +static int buffer_prepare(struct vb2_buffer *vb)
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
- + unsigned long size;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "%s: dev:%p\n",
- + __func__, dev);
- +
- + BUG_ON(dev->capture.port == NULL);
- + BUG_ON(dev->capture.fmt == NULL);
- +
- + size = dev->capture.stride * dev->capture.height;
- + if (vb2_plane_size(vb, 0) < size) {
- + v4l2_err(&dev->v4l2_dev,
- + "%s data will not fit into plane (%lu < %lu)\n",
- + __func__, vb2_plane_size(vb, 0), size);
- + return -EINVAL;
- + }
- +
- + return 0;
- +}
- +
- +static inline bool is_capturing(struct bm2835_mmal_dev *dev)
- +{
- + return dev->capture.camera_port ==
- + &dev->
- + component[MMAL_COMPONENT_CAMERA]->output[MMAL_CAMERA_PORT_CAPTURE];
- +}
- +
- +static void buffer_cb(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + int status,
- + struct mmal_buffer *buf,
- + unsigned long length, u32 mmal_flags, s64 dts, s64 pts)
- +{
- + struct bm2835_mmal_dev *dev = port->cb_ctx;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: status:%d, buf:%p, length:%lu, flags %u, pts %lld\n",
- + __func__, status, buf, length, mmal_flags, pts);
- +
- + if (status != 0) {
- + /* error in transfer */
- + if (buf != NULL) {
- + /* there was a buffer with the error so return it */
- + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR);
- + }
- + return;
- + } else if (length == 0) {
- + /* stream ended */
- + if (buf != NULL) {
- + /* this should only ever happen if the port is
- + * disabled and there are buffers still queued
- + */
- + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR);
- + pr_debug("Empty buffer");
- + } else if (dev->capture.frame_count) {
- + /* grab another frame */
- + if (is_capturing(dev)) {
- + pr_debug("Grab another frame");
- + vchiq_mmal_port_parameter_set(
- + instance,
- + dev->capture.
- + camera_port,
- + MMAL_PARAMETER_CAPTURE,
- + &dev->capture.
- + frame_count,
- + sizeof(dev->capture.frame_count));
- + }
- + } else {
- + /* signal frame completion */
- + complete(&dev->capture.frame_cmplt);
- + }
- + } else {
- + if (dev->capture.frame_count) {
- + if (dev->capture.vc_start_timestamp != -1 &&
- + pts != 0) {
- + s64 runtime_us = pts -
- + dev->capture.vc_start_timestamp;
- + u32 div = 0;
- + u32 rem = 0;
- +
- + div =
- + div_u64_rem(runtime_us, USEC_PER_SEC, &rem);
- + buf->vb.timestamp.tv_sec =
- + dev->capture.kernel_start_ts.tv_sec - 1 +
- + div;
- + buf->vb.timestamp.tv_usec =
- + dev->capture.kernel_start_ts.tv_usec + rem;
- +
- + if (buf->vb.timestamp.tv_usec >=
- + USEC_PER_SEC) {
- + buf->vb.timestamp.tv_sec++;
- + buf->vb.timestamp.tv_usec -=
- + USEC_PER_SEC;
- + }
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Convert start time %d.%06d and %llu "
- + "with offset %llu to %d.%06d\n",
- + (int)dev->capture.kernel_start_ts.
- + tv_sec,
- + (int)dev->capture.kernel_start_ts.
- + tv_usec,
- + dev->capture.vc_start_timestamp, pts,
- + (int)buf->vb.timestamp.tv_sec,
- + (int)buf->vb.timestamp.
- + tv_usec);
- + } else {
- + v4l2_get_timestamp(&buf->vb.timestamp);
- + }
- +
- + vb2_set_plane_payload(&buf->vb.vb2_buf, 0, length);
- + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_DONE);
- +
- + if (mmal_flags & MMAL_BUFFER_HEADER_FLAG_EOS &&
- + is_capturing(dev)) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Grab another frame as buffer has EOS");
- + vchiq_mmal_port_parameter_set(
- + instance,
- + dev->capture.
- + camera_port,
- + MMAL_PARAMETER_CAPTURE,
- + &dev->capture.
- + frame_count,
- + sizeof(dev->capture.frame_count));
- + }
- + } else {
- + /* signal frame completion */
- + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR);
- + complete(&dev->capture.frame_cmplt);
- + }
- + }
- +}
- +
- +static int enable_camera(struct bm2835_mmal_dev *dev)
- +{
- + int ret;
- + if (!dev->camera_use_count) {
- + ret = vchiq_mmal_component_enable(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_CAMERA]);
- + if (ret < 0) {
- + v4l2_err(&dev->v4l2_dev,
- + "Failed enabling camera, ret %d\n", ret);
- + return -EINVAL;
- + }
- + }
- + dev->camera_use_count++;
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev, "enabled camera (refcount %d)\n",
- + dev->camera_use_count);
- + return 0;
- +}
- +
- +static int disable_camera(struct bm2835_mmal_dev *dev)
- +{
- + int ret;
- + if (!dev->camera_use_count) {
- + v4l2_err(&dev->v4l2_dev,
- + "Disabled the camera when already disabled\n");
- + return -EINVAL;
- + }
- + dev->camera_use_count--;
- + if (!dev->camera_use_count) {
- + unsigned int i = 0xFFFFFFFF;
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Disabling camera\n");
- + ret =
- + vchiq_mmal_component_disable(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_CAMERA]);
- + if (ret < 0) {
- + v4l2_err(&dev->v4l2_dev,
- + "Failed disabling camera, ret %d\n", ret);
- + return -EINVAL;
- + }
- + vchiq_mmal_port_parameter_set(
- + dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->control,
- + MMAL_PARAMETER_CAMERA_NUM, &i,
- + sizeof(i));
- + }
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Camera refcount now %d\n", dev->camera_use_count);
- + return 0;
- +}
- +
- +static void buffer_queue(struct vb2_buffer *vb)
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
- + struct vb2_v4l2_buffer *vb2 = to_vb2_v4l2_buffer(vb);
- + struct mmal_buffer *buf = container_of(vb2, struct mmal_buffer, vb);
- + int ret;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: dev:%p buf:%p\n", __func__, dev, buf);
- +
- + buf->buffer = vb2_plane_vaddr(&buf->vb.vb2_buf, 0);
- + buf->buffer_size = vb2_plane_size(&buf->vb.vb2_buf, 0);
- +
- + ret = vchiq_mmal_submit_buffer(dev->instance, dev->capture.port, buf);
- + if (ret < 0)
- + v4l2_err(&dev->v4l2_dev, "%s: error submitting buffer\n",
- + __func__);
- +}
- +
- +static int start_streaming(struct vb2_queue *vq, unsigned int count)
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vq);
- + int ret;
- + int parameter_size;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "%s: dev:%p\n",
- + __func__, dev);
- +
- + /* ensure a format has actually been set */
- + if (dev->capture.port == NULL)
- + return -EINVAL;
- +
- + if (enable_camera(dev) < 0) {
- + v4l2_err(&dev->v4l2_dev, "Failed to enable camera\n");
- + return -EINVAL;
- + }
- +
- + /*init_completion(&dev->capture.frame_cmplt); */
- +
- + /* enable frame capture */
- + dev->capture.frame_count = 1;
- +
- + /* if the preview is not already running, wait for a few frames for AGC
- + * to settle down.
- + */
- + if (!dev->component[MMAL_COMPONENT_PREVIEW]->enabled)
- + msleep(300);
- +
- + /* enable the connection from camera to encoder (if applicable) */
- + if (dev->capture.camera_port != dev->capture.port
- + && dev->capture.camera_port) {
- + ret = vchiq_mmal_port_enable(dev->instance,
- + dev->capture.camera_port, NULL);
- + if (ret) {
- + v4l2_err(&dev->v4l2_dev,
- + "Failed to enable encode tunnel - error %d\n",
- + ret);
- + return -1;
- + }
- + }
- +
- + /* Get VC timestamp at this point in time */
- + parameter_size = sizeof(dev->capture.vc_start_timestamp);
- + if (vchiq_mmal_port_parameter_get(dev->instance,
- + dev->capture.camera_port,
- + MMAL_PARAMETER_SYSTEM_TIME,
- + &dev->capture.vc_start_timestamp,
- + ¶meter_size)) {
- + v4l2_err(&dev->v4l2_dev,
- + "Failed to get VC start time - update your VC f/w\n");
- +
- + /* Flag to indicate just to rely on kernel timestamps */
- + dev->capture.vc_start_timestamp = -1;
- + } else
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Start time %lld size %d\n",
- + dev->capture.vc_start_timestamp, parameter_size);
- +
- + v4l2_get_timestamp(&dev->capture.kernel_start_ts);
- +
- + /* enable the camera port */
- + dev->capture.port->cb_ctx = dev;
- + ret =
- + vchiq_mmal_port_enable(dev->instance, dev->capture.port, buffer_cb);
- + if (ret) {
- + v4l2_err(&dev->v4l2_dev,
- + "Failed to enable capture port - error %d. "
- + "Disabling camera port again\n", ret);
- +
- + vchiq_mmal_port_disable(dev->instance,
- + dev->capture.camera_port);
- + if (disable_camera(dev) < 0) {
- + v4l2_err(&dev->v4l2_dev, "Failed to disable camera\n");
- + return -EINVAL;
- + }
- + return -1;
- + }
- +
- + /* capture the first frame */
- + vchiq_mmal_port_parameter_set(dev->instance,
- + dev->capture.camera_port,
- + MMAL_PARAMETER_CAPTURE,
- + &dev->capture.frame_count,
- + sizeof(dev->capture.frame_count));
- + return 0;
- +}
- +
- +/* abort streaming and wait for last buffer */
- +static void stop_streaming(struct vb2_queue *vq)
- +{
- + int ret;
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vq);
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "%s: dev:%p\n",
- + __func__, dev);
- +
- + init_completion(&dev->capture.frame_cmplt);
- + dev->capture.frame_count = 0;
- +
- + /* ensure a format has actually been set */
- + if (dev->capture.port == NULL) {
- + v4l2_err(&dev->v4l2_dev,
- + "no capture port - stream not started?\n");
- + return;
- + }
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "stopping capturing\n");
- +
- + /* stop capturing frames */
- + vchiq_mmal_port_parameter_set(dev->instance,
- + dev->capture.camera_port,
- + MMAL_PARAMETER_CAPTURE,
- + &dev->capture.frame_count,
- + sizeof(dev->capture.frame_count));
- +
- + /* wait for last frame to complete */
- + ret = wait_for_completion_timeout(&dev->capture.frame_cmplt, HZ);
- + if (ret <= 0)
- + v4l2_err(&dev->v4l2_dev,
- + "error %d waiting for frame completion\n", ret);
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "disabling connection\n");
- +
- + /* disable the connection from camera to encoder */
- + ret = vchiq_mmal_port_disable(dev->instance, dev->capture.camera_port);
- + if (!ret && dev->capture.camera_port != dev->capture.port) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "disabling port\n");
- + ret = vchiq_mmal_port_disable(dev->instance, dev->capture.port);
- + } else if (dev->capture.camera_port != dev->capture.port) {
- + v4l2_err(&dev->v4l2_dev, "port_disable failed, error %d\n",
- + ret);
- + }
- +
- + if (disable_camera(dev) < 0)
- + v4l2_err(&dev->v4l2_dev, "Failed to disable camera\n");
- +}
- +
- +static void bm2835_mmal_lock(struct vb2_queue *vq)
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vq);
- + mutex_lock(&dev->mutex);
- +}
- +
- +static void bm2835_mmal_unlock(struct vb2_queue *vq)
- +{
- + struct bm2835_mmal_dev *dev = vb2_get_drv_priv(vq);
- + mutex_unlock(&dev->mutex);
- +}
- +
- +static struct vb2_ops bm2835_mmal_video_qops = {
- + .queue_setup = queue_setup,
- + .buf_prepare = buffer_prepare,
- + .buf_queue = buffer_queue,
- + .start_streaming = start_streaming,
- + .stop_streaming = stop_streaming,
- + .wait_prepare = bm2835_mmal_unlock,
- + .wait_finish = bm2835_mmal_lock,
- +};
- +
- +/* ------------------------------------------------------------------
- + IOCTL operations
- + ------------------------------------------------------------------*/
- +
- +/* overlay ioctl */
- +static int vidioc_enum_fmt_vid_overlay(struct file *file, void *priv,
- + struct v4l2_fmtdesc *f)
- +{
- + struct mmal_fmt *fmt;
- +
- + if (f->index >= ARRAY_SIZE(formats))
- + return -EINVAL;
- +
- + fmt = &formats[f->index];
- +
- + strlcpy(f->description, fmt->name, sizeof(f->description));
- + f->pixelformat = fmt->fourcc;
- + f->flags = fmt->flags;
- +
- + return 0;
- +}
- +
- +static int vidioc_g_fmt_vid_overlay(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- +
- + f->fmt.win = dev->overlay;
- +
- + return 0;
- +}
- +
- +static int vidioc_try_fmt_vid_overlay(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + /* Only support one format so get the current one. */
- + vidioc_g_fmt_vid_overlay(file, priv, f);
- +
- + /* todo: allow the size and/or offset to be changed. */
- + return 0;
- +}
- +
- +static int vidioc_s_fmt_vid_overlay(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- +
- + vidioc_try_fmt_vid_overlay(file, priv, f);
- +
- + dev->overlay = f->fmt.win;
- +
- + /* todo: program the preview port parameters */
- + return 0;
- +}
- +
- +static int vidioc_overlay(struct file *file, void *f, unsigned int on)
- +{
- + int ret;
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + struct vchiq_mmal_port *src;
- + struct vchiq_mmal_port *dst;
- + struct mmal_parameter_displayregion prev_config = {
- + .set = MMAL_DISPLAY_SET_LAYER | MMAL_DISPLAY_SET_ALPHA |
- + MMAL_DISPLAY_SET_DEST_RECT | MMAL_DISPLAY_SET_FULLSCREEN,
- + .layer = PREVIEW_LAYER,
- + .alpha = 255,
- + .fullscreen = 0,
- + .dest_rect = {
- + .x = dev->overlay.w.left,
- + .y = dev->overlay.w.top,
- + .width = dev->overlay.w.width,
- + .height = dev->overlay.w.height,
- + },
- + };
- +
- + if ((on && dev->component[MMAL_COMPONENT_PREVIEW]->enabled) ||
- + (!on && !dev->component[MMAL_COMPONENT_PREVIEW]->enabled))
- + return 0; /* already in requested state */
- +
- + src =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_PREVIEW];
- +
- + if (!on) {
- + /* disconnect preview ports and disable component */
- + ret = vchiq_mmal_port_disable(dev->instance, src);
- + if (!ret)
- + ret =
- + vchiq_mmal_port_connect_tunnel(dev->instance, src,
- + NULL);
- + if (ret >= 0)
- + ret = vchiq_mmal_component_disable(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_PREVIEW]);
- +
- + disable_camera(dev);
- + return ret;
- + }
- +
- + /* set preview port format and connect it to output */
- + dst = &dev->component[MMAL_COMPONENT_PREVIEW]->input[0];
- +
- + ret = vchiq_mmal_port_set_format(dev->instance, src);
- + if (ret < 0)
- + goto error;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, dst,
- + MMAL_PARAMETER_DISPLAYREGION,
- + &prev_config, sizeof(prev_config));
- + if (ret < 0)
- + goto error;
- +
- + if (enable_camera(dev) < 0)
- + goto error;
- +
- + ret = vchiq_mmal_component_enable(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_PREVIEW]);
- + if (ret < 0)
- + goto error;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev, "connecting %p to %p\n",
- + src, dst);
- + ret = vchiq_mmal_port_connect_tunnel(dev->instance, src, dst);
- + if (!ret)
- + ret = vchiq_mmal_port_enable(dev->instance, src, NULL);
- +error:
- + return ret;
- +}
- +
- +static int vidioc_g_fbuf(struct file *file, void *fh,
- + struct v4l2_framebuffer *a)
- +{
- + /* The video overlay must stay within the framebuffer and can't be
- + positioned independently. */
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + struct vchiq_mmal_port *preview_port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_PREVIEW];
- + a->flags = V4L2_FBUF_FLAG_OVERLAY;
- + a->fmt.width = preview_port->es.video.width;
- + a->fmt.height = preview_port->es.video.height;
- + a->fmt.pixelformat = V4L2_PIX_FMT_YUV420;
- + a->fmt.bytesperline = preview_port->es.video.width;
- + a->fmt.sizeimage = (preview_port->es.video.width *
- + preview_port->es.video.height * 3)>>1;
- + a->fmt.colorspace = V4L2_COLORSPACE_SMPTE170M;
- +
- + return 0;
- +}
- +
- +/* input ioctls */
- +static int vidioc_enum_input(struct file *file, void *priv,
- + struct v4l2_input *inp)
- +{
- + /* only a single camera input */
- + if (inp->index != 0)
- + return -EINVAL;
- +
- + inp->type = V4L2_INPUT_TYPE_CAMERA;
- + sprintf(inp->name, "Camera %u", inp->index);
- + return 0;
- +}
- +
- +static int vidioc_g_input(struct file *file, void *priv, unsigned int *i)
- +{
- + *i = 0;
- + return 0;
- +}
- +
- +static int vidioc_s_input(struct file *file, void *priv, unsigned int i)
- +{
- + if (i != 0)
- + return -EINVAL;
- +
- + return 0;
- +}
- +
- +/* capture ioctls */
- +static int vidioc_querycap(struct file *file, void *priv,
- + struct v4l2_capability *cap)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + u32 major;
- + u32 minor;
- +
- + vchiq_mmal_version(dev->instance, &major, &minor);
- +
- + strcpy(cap->driver, "bm2835 mmal");
- + snprintf(cap->card, sizeof(cap->card), "mmal service %d.%d",
- + major, minor);
- +
- + snprintf(cap->bus_info, sizeof(cap->bus_info),
- + "platform:%s", dev->v4l2_dev.name);
- + cap->device_caps = V4L2_CAP_VIDEO_CAPTURE | V4L2_CAP_VIDEO_OVERLAY |
- + V4L2_CAP_STREAMING | V4L2_CAP_READWRITE;
- + cap->capabilities = cap->device_caps | V4L2_CAP_DEVICE_CAPS;
- +
- + return 0;
- +}
- +
- +static int vidioc_enum_fmt_vid_cap(struct file *file, void *priv,
- + struct v4l2_fmtdesc *f)
- +{
- + struct mmal_fmt *fmt;
- +
- + if (f->index >= ARRAY_SIZE(formats))
- + return -EINVAL;
- +
- + fmt = &formats[f->index];
- +
- + strlcpy(f->description, fmt->name, sizeof(f->description));
- + f->pixelformat = fmt->fourcc;
- + f->flags = fmt->flags;
- +
- + return 0;
- +}
- +
- +static int vidioc_g_fmt_vid_cap(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- +
- + f->fmt.pix.width = dev->capture.width;
- + f->fmt.pix.height = dev->capture.height;
- + f->fmt.pix.field = V4L2_FIELD_NONE;
- + f->fmt.pix.pixelformat = dev->capture.fmt->fourcc;
- + f->fmt.pix.bytesperline = dev->capture.stride;
- + f->fmt.pix.sizeimage = dev->capture.buffersize;
- +
- + if (dev->capture.fmt->fourcc == V4L2_PIX_FMT_RGB24)
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_SRGB;
- + else if (dev->capture.fmt->fourcc == V4L2_PIX_FMT_JPEG)
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_JPEG;
- + else
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_SMPTE170M;
- + f->fmt.pix.priv = 0;
- +
- + v4l2_dump_pix_format(1, bcm2835_v4l2_debug, &dev->v4l2_dev, &f->fmt.pix,
- + __func__);
- + return 0;
- +}
- +
- +static int vidioc_try_fmt_vid_cap(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + struct mmal_fmt *mfmt;
- +
- + mfmt = get_format(f);
- + if (!mfmt) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Fourcc format (0x%08x) unknown.\n",
- + f->fmt.pix.pixelformat);
- + f->fmt.pix.pixelformat = formats[0].fourcc;
- + mfmt = get_format(f);
- + }
- +
- + f->fmt.pix.field = V4L2_FIELD_NONE;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Clipping/aligning %dx%d format %08X\n",
- + f->fmt.pix.width, f->fmt.pix.height, f->fmt.pix.pixelformat);
- +
- + v4l_bound_align_image(&f->fmt.pix.width, MIN_WIDTH, MAX_WIDTH, 1,
- + &f->fmt.pix.height, MIN_HEIGHT, MAX_HEIGHT, 1, 0);
- + f->fmt.pix.bytesperline = f->fmt.pix.width * mfmt->ybbp;
- +
- + /* Image buffer has to be padded to allow for alignment, even though
- + * we then remove that padding before delivering the buffer.
- + */
- + f->fmt.pix.sizeimage = ((f->fmt.pix.height+15)&~15) *
- + (((f->fmt.pix.width+31)&~31) * mfmt->depth) >> 3;
- +
- + if ((mfmt->flags & V4L2_FMT_FLAG_COMPRESSED) &&
- + f->fmt.pix.sizeimage < MIN_BUFFER_SIZE)
- + f->fmt.pix.sizeimage = MIN_BUFFER_SIZE;
- +
- + if (f->fmt.pix.pixelformat == V4L2_PIX_FMT_RGB24)
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_SRGB;
- + else if (f->fmt.pix.pixelformat == V4L2_PIX_FMT_JPEG)
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_JPEG;
- + else
- + f->fmt.pix.colorspace = V4L2_COLORSPACE_SMPTE170M;
- + f->fmt.pix.priv = 0;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Now %dx%d format %08X\n",
- + f->fmt.pix.width, f->fmt.pix.height, f->fmt.pix.pixelformat);
- +
- + v4l2_dump_pix_format(1, bcm2835_v4l2_debug, &dev->v4l2_dev, &f->fmt.pix,
- + __func__);
- + return 0;
- +}
- +
- +static int mmal_setup_components(struct bm2835_mmal_dev *dev,
- + struct v4l2_format *f)
- +{
- + int ret;
- + struct vchiq_mmal_port *port = NULL, *camera_port = NULL;
- + struct vchiq_mmal_component *encode_component = NULL;
- + struct mmal_fmt *mfmt = get_format(f);
- +
- + BUG_ON(!mfmt);
- +
- + if (dev->capture.encode_component) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "vid_cap - disconnect previous tunnel\n");
- +
- + /* Disconnect any previous connection */
- + vchiq_mmal_port_connect_tunnel(dev->instance,
- + dev->capture.camera_port, NULL);
- + dev->capture.camera_port = NULL;
- + ret = vchiq_mmal_component_disable(dev->instance,
- + dev->capture.
- + encode_component);
- + if (ret)
- + v4l2_err(&dev->v4l2_dev,
- + "Failed to disable encode component %d\n",
- + ret);
- +
- + dev->capture.encode_component = NULL;
- + }
- + /* format dependant port setup */
- + switch (mfmt->mmal_component) {
- + case MMAL_COMPONENT_CAMERA:
- + /* Make a further decision on port based on resolution */
- + if (f->fmt.pix.width <= max_video_width
- + && f->fmt.pix.height <= max_video_height)
- + camera_port = port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO];
- + else
- + camera_port = port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_CAPTURE];
- + break;
- + case MMAL_COMPONENT_IMAGE_ENCODE:
- + encode_component = dev->component[MMAL_COMPONENT_IMAGE_ENCODE];
- + port = &dev->component[MMAL_COMPONENT_IMAGE_ENCODE]->output[0];
- + camera_port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_CAPTURE];
- + break;
- + case MMAL_COMPONENT_VIDEO_ENCODE:
- + encode_component = dev->component[MMAL_COMPONENT_VIDEO_ENCODE];
- + port = &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0];
- + camera_port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO];
- + break;
- + default:
- + break;
- + }
- +
- + if (!port)
- + return -EINVAL;
- +
- + if (encode_component)
- + camera_port->format.encoding = MMAL_ENCODING_OPAQUE;
- + else
- + camera_port->format.encoding = mfmt->mmal;
- +
- + camera_port->format.encoding_variant = 0;
- + camera_port->es.video.width = f->fmt.pix.width;
- + camera_port->es.video.height = f->fmt.pix.height;
- + camera_port->es.video.crop.x = 0;
- + camera_port->es.video.crop.y = 0;
- + camera_port->es.video.crop.width = f->fmt.pix.width;
- + camera_port->es.video.crop.height = f->fmt.pix.height;
- + camera_port->es.video.frame_rate.num = 0;
- + camera_port->es.video.frame_rate.den = 1;
- + camera_port->es.video.color_space = MMAL_COLOR_SPACE_JPEG_JFIF;
- +
- + ret = vchiq_mmal_port_set_format(dev->instance, camera_port);
- +
- + if (!ret
- + && camera_port ==
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO]) {
- + bool overlay_enabled =
- + !!dev->component[MMAL_COMPONENT_PREVIEW]->enabled;
- + struct vchiq_mmal_port *preview_port =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_PREVIEW];
- + /* Preview and encode ports need to match on resolution */
- + if (overlay_enabled) {
- + /* Need to disable the overlay before we can update
- + * the resolution
- + */
- + ret =
- + vchiq_mmal_port_disable(dev->instance,
- + preview_port);
- + if (!ret)
- + ret =
- + vchiq_mmal_port_connect_tunnel(
- + dev->instance,
- + preview_port,
- + NULL);
- + }
- + preview_port->es.video.width = f->fmt.pix.width;
- + preview_port->es.video.height = f->fmt.pix.height;
- + preview_port->es.video.crop.x = 0;
- + preview_port->es.video.crop.y = 0;
- + preview_port->es.video.crop.width = f->fmt.pix.width;
- + preview_port->es.video.crop.height = f->fmt.pix.height;
- + preview_port->es.video.frame_rate.num =
- + dev->capture.timeperframe.denominator;
- + preview_port->es.video.frame_rate.den =
- + dev->capture.timeperframe.numerator;
- + ret = vchiq_mmal_port_set_format(dev->instance, preview_port);
- + if (overlay_enabled) {
- + ret = vchiq_mmal_port_connect_tunnel(
- + dev->instance,
- + preview_port,
- + &dev->component[MMAL_COMPONENT_PREVIEW]->input[0]);
- + if (!ret)
- + ret = vchiq_mmal_port_enable(dev->instance,
- + preview_port,
- + NULL);
- + }
- + }
- +
- + if (ret) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s failed to set format %dx%d %08X\n", __func__,
- + f->fmt.pix.width, f->fmt.pix.height,
- + f->fmt.pix.pixelformat);
- + /* ensure capture is not going to be tried */
- + dev->capture.port = NULL;
- + } else {
- + if (encode_component) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "vid_cap - set up encode comp\n");
- +
- + /* configure buffering */
- + camera_port->current_buffer.size =
- + camera_port->recommended_buffer.size;
- + camera_port->current_buffer.num =
- + camera_port->recommended_buffer.num;
- +
- + ret =
- + vchiq_mmal_port_connect_tunnel(
- + dev->instance,
- + camera_port,
- + &encode_component->input[0]);
- + if (ret) {
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "%s failed to create connection\n",
- + __func__);
- + /* ensure capture is not going to be tried */
- + dev->capture.port = NULL;
- + } else {
- + port->es.video.width = f->fmt.pix.width;
- + port->es.video.height = f->fmt.pix.height;
- + port->es.video.crop.x = 0;
- + port->es.video.crop.y = 0;
- + port->es.video.crop.width = f->fmt.pix.width;
- + port->es.video.crop.height = f->fmt.pix.height;
- + port->es.video.frame_rate.num =
- + dev->capture.timeperframe.denominator;
- + port->es.video.frame_rate.den =
- + dev->capture.timeperframe.numerator;
- +
- + port->format.encoding = mfmt->mmal;
- + port->format.encoding_variant = 0;
- + /* Set any encoding specific parameters */
- + switch (mfmt->mmal_component) {
- + case MMAL_COMPONENT_VIDEO_ENCODE:
- + port->format.bitrate =
- + dev->capture.encode_bitrate;
- + break;
- + case MMAL_COMPONENT_IMAGE_ENCODE:
- + /* Could set EXIF parameters here */
- + break;
- + default:
- + break;
- + }
- + ret = vchiq_mmal_port_set_format(dev->instance,
- + port);
- + if (ret)
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "%s failed to set format %dx%d fmt %08X\n",
- + __func__,
- + f->fmt.pix.width,
- + f->fmt.pix.height,
- + f->fmt.pix.pixelformat
- + );
- + }
- +
- + if (!ret) {
- + ret = vchiq_mmal_component_enable(
- + dev->instance,
- + encode_component);
- + if (ret) {
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "%s Failed to enable encode components\n",
- + __func__);
- + }
- + }
- + if (!ret) {
- + /* configure buffering */
- + port->current_buffer.num = 1;
- + port->current_buffer.size =
- + f->fmt.pix.sizeimage;
- + if (port->format.encoding ==
- + MMAL_ENCODING_JPEG) {
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "JPG - buf size now %d was %d\n",
- + f->fmt.pix.sizeimage,
- + port->current_buffer.size);
- + port->current_buffer.size =
- + (f->fmt.pix.sizeimage <
- + (100 << 10))
- + ? (100 << 10) : f->fmt.pix.
- + sizeimage;
- + }
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "vid_cap - cur_buf.size set to %d\n",
- + f->fmt.pix.sizeimage);
- + port->current_buffer.alignment = 0;
- + }
- + } else {
- + /* configure buffering */
- + camera_port->current_buffer.num = 1;
- + camera_port->current_buffer.size = f->fmt.pix.sizeimage;
- + camera_port->current_buffer.alignment = 0;
- + }
- +
- + if (!ret) {
- + dev->capture.fmt = mfmt;
- + dev->capture.stride = f->fmt.pix.bytesperline;
- + dev->capture.width = camera_port->es.video.crop.width;
- + dev->capture.height = camera_port->es.video.crop.height;
- + dev->capture.buffersize = port->current_buffer.size;
- +
- + /* select port for capture */
- + dev->capture.port = port;
- + dev->capture.camera_port = camera_port;
- + dev->capture.encode_component = encode_component;
- + v4l2_dbg(1, bcm2835_v4l2_debug,
- + &dev->v4l2_dev,
- + "Set dev->capture.fmt %08X, %dx%d, stride %d, size %d",
- + port->format.encoding,
- + dev->capture.width, dev->capture.height,
- + dev->capture.stride, dev->capture.buffersize);
- + }
- + }
- +
- + /* todo: Need to convert the vchiq/mmal error into a v4l2 error. */
- + return ret;
- +}
- +
- +static int vidioc_s_fmt_vid_cap(struct file *file, void *priv,
- + struct v4l2_format *f)
- +{
- + int ret;
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + struct mmal_fmt *mfmt;
- +
- + /* try the format to set valid parameters */
- + ret = vidioc_try_fmt_vid_cap(file, priv, f);
- + if (ret) {
- + v4l2_err(&dev->v4l2_dev,
- + "vid_cap - vidioc_try_fmt_vid_cap failed\n");
- + return ret;
- + }
- +
- + /* if a capture is running refuse to set format */
- + if (vb2_is_busy(&dev->capture.vb_vidq)) {
- + v4l2_info(&dev->v4l2_dev, "%s device busy\n", __func__);
- + return -EBUSY;
- + }
- +
- + /* If the format is unsupported v4l2 says we should switch to
- + * a supported one and not return an error. */
- + mfmt = get_format(f);
- + if (!mfmt) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Fourcc format (0x%08x) unknown.\n",
- + f->fmt.pix.pixelformat);
- + f->fmt.pix.pixelformat = formats[0].fourcc;
- + mfmt = get_format(f);
- + }
- +
- + ret = mmal_setup_components(dev, f);
- + if (ret != 0) {
- + v4l2_err(&dev->v4l2_dev,
- + "%s: failed to setup mmal components: %d\n",
- + __func__, ret);
- + ret = -EINVAL;
- + }
- +
- + return ret;
- +}
- +
- +int vidioc_enum_framesizes(struct file *file, void *fh,
- + struct v4l2_frmsizeenum *fsize)
- +{
- + static const struct v4l2_frmsize_stepwise sizes = {
- + MIN_WIDTH, MAX_WIDTH, 2,
- + MIN_HEIGHT, MAX_HEIGHT, 2
- + };
- + int i;
- +
- + if (fsize->index)
- + return -EINVAL;
- + for (i = 0; i < ARRAY_SIZE(formats); i++)
- + if (formats[i].fourcc == fsize->pixel_format)
- + break;
- + if (i == ARRAY_SIZE(formats))
- + return -EINVAL;
- + fsize->type = V4L2_FRMSIZE_TYPE_STEPWISE;
- + fsize->stepwise = sizes;
- + return 0;
- +}
- +
- +/* timeperframe is arbitrary and continous */
- +static int vidioc_enum_frameintervals(struct file *file, void *priv,
- + struct v4l2_frmivalenum *fival)
- +{
- + int i;
- +
- + if (fival->index)
- + return -EINVAL;
- +
- + for (i = 0; i < ARRAY_SIZE(formats); i++)
- + if (formats[i].fourcc == fival->pixel_format)
- + break;
- + if (i == ARRAY_SIZE(formats))
- + return -EINVAL;
- +
- + /* regarding width & height - we support any within range */
- + if (fival->width < MIN_WIDTH || fival->width > MAX_WIDTH ||
- + fival->height < MIN_HEIGHT || fival->height > MAX_HEIGHT)
- + return -EINVAL;
- +
- + fival->type = V4L2_FRMIVAL_TYPE_CONTINUOUS;
- +
- + /* fill in stepwise (step=1.0 is requred by V4L2 spec) */
- + fival->stepwise.min = tpf_min;
- + fival->stepwise.max = tpf_max;
- + fival->stepwise.step = (struct v4l2_fract) {1, 1};
- +
- + return 0;
- +}
- +
- +static int vidioc_g_parm(struct file *file, void *priv,
- + struct v4l2_streamparm *parm)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- +
- + if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
- + return -EINVAL;
- +
- + parm->parm.capture.capability = V4L2_CAP_TIMEPERFRAME;
- + parm->parm.capture.timeperframe = dev->capture.timeperframe;
- + parm->parm.capture.readbuffers = 1;
- + return 0;
- +}
- +
- +#define FRACT_CMP(a, OP, b) \
- + ((u64)(a).numerator * (b).denominator OP \
- + (u64)(b).numerator * (a).denominator)
- +
- +static int vidioc_s_parm(struct file *file, void *priv,
- + struct v4l2_streamparm *parm)
- +{
- + struct bm2835_mmal_dev *dev = video_drvdata(file);
- + struct v4l2_fract tpf;
- + struct mmal_parameter_rational fps_param;
- +
- + if (parm->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
- + return -EINVAL;
- +
- + tpf = parm->parm.capture.timeperframe;
- +
- + /* tpf: {*, 0} resets timing; clip to [min, max]*/
- + tpf = tpf.denominator ? tpf : tpf_default;
- + tpf = FRACT_CMP(tpf, <, tpf_min) ? tpf_min : tpf;
- + tpf = FRACT_CMP(tpf, >, tpf_max) ? tpf_max : tpf;
- +
- + dev->capture.timeperframe = tpf;
- + parm->parm.capture.timeperframe = tpf;
- + parm->parm.capture.readbuffers = 1;
- +
- + fps_param.num = 0; /* Select variable fps, and then use
- + * FPS_RANGE to select the actual limits.
- + */
- + fps_param.den = 1;
- + set_framerate_params(dev);
- +
- + return 0;
- +}
- +
- +static const struct v4l2_ioctl_ops camera0_ioctl_ops = {
- + /* overlay */
- + .vidioc_enum_fmt_vid_overlay = vidioc_enum_fmt_vid_overlay,
- + .vidioc_g_fmt_vid_overlay = vidioc_g_fmt_vid_overlay,
- + .vidioc_try_fmt_vid_overlay = vidioc_try_fmt_vid_overlay,
- + .vidioc_s_fmt_vid_overlay = vidioc_s_fmt_vid_overlay,
- + .vidioc_overlay = vidioc_overlay,
- + .vidioc_g_fbuf = vidioc_g_fbuf,
- +
- + /* inputs */
- + .vidioc_enum_input = vidioc_enum_input,
- + .vidioc_g_input = vidioc_g_input,
- + .vidioc_s_input = vidioc_s_input,
- +
- + /* capture */
- + .vidioc_querycap = vidioc_querycap,
- + .vidioc_enum_fmt_vid_cap = vidioc_enum_fmt_vid_cap,
- + .vidioc_g_fmt_vid_cap = vidioc_g_fmt_vid_cap,
- + .vidioc_try_fmt_vid_cap = vidioc_try_fmt_vid_cap,
- + .vidioc_s_fmt_vid_cap = vidioc_s_fmt_vid_cap,
- +
- + /* buffer management */
- + .vidioc_reqbufs = vb2_ioctl_reqbufs,
- + .vidioc_create_bufs = vb2_ioctl_create_bufs,
- + .vidioc_prepare_buf = vb2_ioctl_prepare_buf,
- + .vidioc_querybuf = vb2_ioctl_querybuf,
- + .vidioc_qbuf = vb2_ioctl_qbuf,
- + .vidioc_dqbuf = vb2_ioctl_dqbuf,
- + .vidioc_enum_framesizes = vidioc_enum_framesizes,
- + .vidioc_enum_frameintervals = vidioc_enum_frameintervals,
- + .vidioc_g_parm = vidioc_g_parm,
- + .vidioc_s_parm = vidioc_s_parm,
- + .vidioc_streamon = vb2_ioctl_streamon,
- + .vidioc_streamoff = vb2_ioctl_streamoff,
- +
- + .vidioc_log_status = v4l2_ctrl_log_status,
- + .vidioc_subscribe_event = v4l2_ctrl_subscribe_event,
- + .vidioc_unsubscribe_event = v4l2_event_unsubscribe,
- +};
- +
- +static const struct v4l2_ioctl_ops camera0_ioctl_ops_gstreamer = {
- + /* overlay */
- + .vidioc_enum_fmt_vid_overlay = vidioc_enum_fmt_vid_overlay,
- + .vidioc_g_fmt_vid_overlay = vidioc_g_fmt_vid_overlay,
- + .vidioc_try_fmt_vid_overlay = vidioc_try_fmt_vid_overlay,
- + .vidioc_s_fmt_vid_overlay = vidioc_s_fmt_vid_overlay,
- + .vidioc_overlay = vidioc_overlay,
- + .vidioc_g_fbuf = vidioc_g_fbuf,
- +
- + /* inputs */
- + .vidioc_enum_input = vidioc_enum_input,
- + .vidioc_g_input = vidioc_g_input,
- + .vidioc_s_input = vidioc_s_input,
- +
- + /* capture */
- + .vidioc_querycap = vidioc_querycap,
- + .vidioc_enum_fmt_vid_cap = vidioc_enum_fmt_vid_cap,
- + .vidioc_g_fmt_vid_cap = vidioc_g_fmt_vid_cap,
- + .vidioc_try_fmt_vid_cap = vidioc_try_fmt_vid_cap,
- + .vidioc_s_fmt_vid_cap = vidioc_s_fmt_vid_cap,
- +
- + /* buffer management */
- + .vidioc_reqbufs = vb2_ioctl_reqbufs,
- + .vidioc_create_bufs = vb2_ioctl_create_bufs,
- + .vidioc_prepare_buf = vb2_ioctl_prepare_buf,
- + .vidioc_querybuf = vb2_ioctl_querybuf,
- + .vidioc_qbuf = vb2_ioctl_qbuf,
- + .vidioc_dqbuf = vb2_ioctl_dqbuf,
- + /* Remove this function ptr to fix gstreamer bug
- + .vidioc_enum_framesizes = vidioc_enum_framesizes, */
- + .vidioc_enum_frameintervals = vidioc_enum_frameintervals,
- + .vidioc_g_parm = vidioc_g_parm,
- + .vidioc_s_parm = vidioc_s_parm,
- + .vidioc_streamon = vb2_ioctl_streamon,
- + .vidioc_streamoff = vb2_ioctl_streamoff,
- +
- + .vidioc_log_status = v4l2_ctrl_log_status,
- + .vidioc_subscribe_event = v4l2_ctrl_subscribe_event,
- + .vidioc_unsubscribe_event = v4l2_event_unsubscribe,
- +};
- +
- +/* ------------------------------------------------------------------
- + Driver init/finalise
- + ------------------------------------------------------------------*/
- +
- +static const struct v4l2_file_operations camera0_fops = {
- + .owner = THIS_MODULE,
- + .open = v4l2_fh_open,
- + .release = vb2_fop_release,
- + .read = vb2_fop_read,
- + .poll = vb2_fop_poll,
- + .unlocked_ioctl = video_ioctl2, /* V4L2 ioctl handler */
- + .mmap = vb2_fop_mmap,
- +};
- +
- +static struct video_device vdev_template = {
- + .name = "camera0",
- + .fops = &camera0_fops,
- + .ioctl_ops = &camera0_ioctl_ops,
- + .release = video_device_release_empty,
- +};
- +
- +static int set_camera_parameters(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *camera)
- +{
- + int ret;
- + struct mmal_parameter_camera_config cam_config = {
- + .max_stills_w = MAX_WIDTH,
- + .max_stills_h = MAX_HEIGHT,
- + .stills_yuv422 = 1,
- + .one_shot_stills = 1,
- + .max_preview_video_w = (max_video_width > 1920) ?
- + max_video_width : 1920,
- + .max_preview_video_h = (max_video_height > 1088) ?
- + max_video_height : 1088,
- + .num_preview_video_frames = 3,
- + .stills_capture_circular_buffer_height = 0,
- + .fast_preview_resume = 0,
- + .use_stc_timestamp = MMAL_PARAM_TIMESTAMP_MODE_RAW_STC
- + };
- +
- + ret = vchiq_mmal_port_parameter_set(instance, &camera->control,
- + MMAL_PARAMETER_CAMERA_CONFIG,
- + &cam_config, sizeof(cam_config));
- + return ret;
- +}
- +
- +/* MMAL instance and component init */
- +static int __init mmal_init(struct bm2835_mmal_dev *dev)
- +{
- + int ret;
- + struct mmal_es_format *format;
- + u32 bool_true = 1;
- +
- + ret = vchiq_mmal_init(&dev->instance);
- + if (ret < 0)
- + return ret;
- +
- + /* get the camera component ready */
- + ret = vchiq_mmal_component_init(dev->instance, "ril.camera",
- + &dev->component[MMAL_COMPONENT_CAMERA]);
- + if (ret < 0)
- + goto unreg_mmal;
- +
- + if (dev->component[MMAL_COMPONENT_CAMERA]->outputs <
- + MMAL_CAMERA_PORT_COUNT) {
- + ret = -EINVAL;
- + goto unreg_camera;
- + }
- +
- + ret = set_camera_parameters(dev->instance,
- + dev->component[MMAL_COMPONENT_CAMERA]);
- + if (ret < 0)
- + goto unreg_camera;
- +
- + format =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_PREVIEW].format;
- +
- + format->encoding = MMAL_ENCODING_OPAQUE;
- + format->encoding_variant = MMAL_ENCODING_I420;
- +
- + format->es->video.width = 1024;
- + format->es->video.height = 768;
- + format->es->video.crop.x = 0;
- + format->es->video.crop.y = 0;
- + format->es->video.crop.width = 1024;
- + format->es->video.crop.height = 768;
- + format->es->video.frame_rate.num = 0; /* Rely on fps_range */
- + format->es->video.frame_rate.den = 1;
- +
- + format =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO].format;
- +
- + format->encoding = MMAL_ENCODING_OPAQUE;
- + format->encoding_variant = MMAL_ENCODING_I420;
- +
- + format->es->video.width = 1024;
- + format->es->video.height = 768;
- + format->es->video.crop.x = 0;
- + format->es->video.crop.y = 0;
- + format->es->video.crop.width = 1024;
- + format->es->video.crop.height = 768;
- + format->es->video.frame_rate.num = 0; /* Rely on fps_range */
- + format->es->video.frame_rate.den = 1;
- +
- + vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO],
- + MMAL_PARAMETER_NO_IMAGE_PADDING,
- + &bool_true, sizeof(bool_true));
- +
- + format =
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_CAPTURE].format;
- +
- + format->encoding = MMAL_ENCODING_OPAQUE;
- +
- + format->es->video.width = 2592;
- + format->es->video.height = 1944;
- + format->es->video.crop.x = 0;
- + format->es->video.crop.y = 0;
- + format->es->video.crop.width = 2592;
- + format->es->video.crop.height = 1944;
- + format->es->video.frame_rate.num = 0; /* Rely on fps_range */
- + format->es->video.frame_rate.den = 1;
- +
- + dev->capture.width = format->es->video.width;
- + dev->capture.height = format->es->video.height;
- + dev->capture.fmt = &formats[0];
- + dev->capture.encode_component = NULL;
- + dev->capture.timeperframe = tpf_default;
- + dev->capture.enc_profile = V4L2_MPEG_VIDEO_H264_PROFILE_HIGH;
- + dev->capture.enc_level = V4L2_MPEG_VIDEO_H264_LEVEL_4_0;
- +
- + vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_CAPTURE],
- + MMAL_PARAMETER_NO_IMAGE_PADDING,
- + &bool_true, sizeof(bool_true));
- +
- + /* get the preview component ready */
- + ret = vchiq_mmal_component_init(
- + dev->instance, "ril.video_render",
- + &dev->component[MMAL_COMPONENT_PREVIEW]);
- + if (ret < 0)
- + goto unreg_camera;
- +
- + if (dev->component[MMAL_COMPONENT_PREVIEW]->inputs < 1) {
- + ret = -EINVAL;
- + pr_debug("too few input ports %d needed %d\n",
- + dev->component[MMAL_COMPONENT_PREVIEW]->inputs, 1);
- + goto unreg_preview;
- + }
- +
- + /* get the image encoder component ready */
- + ret = vchiq_mmal_component_init(
- + dev->instance, "ril.image_encode",
- + &dev->component[MMAL_COMPONENT_IMAGE_ENCODE]);
- + if (ret < 0)
- + goto unreg_preview;
- +
- + if (dev->component[MMAL_COMPONENT_IMAGE_ENCODE]->inputs < 1) {
- + ret = -EINVAL;
- + v4l2_err(&dev->v4l2_dev, "too few input ports %d needed %d\n",
- + dev->component[MMAL_COMPONENT_IMAGE_ENCODE]->inputs,
- + 1);
- + goto unreg_image_encoder;
- + }
- +
- + /* get the video encoder component ready */
- + ret = vchiq_mmal_component_init(dev->instance, "ril.video_encode",
- + &dev->
- + component[MMAL_COMPONENT_VIDEO_ENCODE]);
- + if (ret < 0)
- + goto unreg_image_encoder;
- +
- + if (dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->inputs < 1) {
- + ret = -EINVAL;
- + v4l2_err(&dev->v4l2_dev, "too few input ports %d needed %d\n",
- + dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->inputs,
- + 1);
- + goto unreg_vid_encoder;
- + }
- +
- + {
- + struct vchiq_mmal_port *encoder_port =
- + &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0];
- + encoder_port->format.encoding = MMAL_ENCODING_H264;
- + ret = vchiq_mmal_port_set_format(dev->instance,
- + encoder_port);
- + }
- +
- + {
- + unsigned int enable = 1;
- + vchiq_mmal_port_parameter_set(
- + dev->instance,
- + &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->control,
- + MMAL_PARAMETER_VIDEO_IMMUTABLE_INPUT,
- + &enable, sizeof(enable));
- +
- + vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->control,
- + MMAL_PARAMETER_MINIMISE_FRAGMENTATION,
- + &enable,
- + sizeof(enable));
- + }
- + ret = bm2835_mmal_set_all_camera_controls(dev);
- + if (ret < 0)
- + goto unreg_vid_encoder;
- +
- + return 0;
- +
- +unreg_vid_encoder:
- + pr_err("Cleanup: Destroy video encoder\n");
- + vchiq_mmal_component_finalise(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_VIDEO_ENCODE]);
- +
- +unreg_image_encoder:
- + pr_err("Cleanup: Destroy image encoder\n");
- + vchiq_mmal_component_finalise(
- + dev->instance,
- + dev->component[MMAL_COMPONENT_IMAGE_ENCODE]);
- +
- +unreg_preview:
- + pr_err("Cleanup: Destroy video render\n");
- + vchiq_mmal_component_finalise(dev->instance,
- + dev->component[MMAL_COMPONENT_PREVIEW]);
- +
- +unreg_camera:
- + pr_err("Cleanup: Destroy camera\n");
- + vchiq_mmal_component_finalise(dev->instance,
- + dev->component[MMAL_COMPONENT_CAMERA]);
- +
- +unreg_mmal:
- + vchiq_mmal_finalise(dev->instance);
- + return ret;
- +}
- +
- +static int __init bm2835_mmal_init_device(struct bm2835_mmal_dev *dev,
- + struct video_device *vfd)
- +{
- + int ret;
- +
- + *vfd = vdev_template;
- + if (gst_v4l2src_is_broken) {
- + v4l2_info(&dev->v4l2_dev,
- + "Work-around for gstreamer issue is active.\n");
- + vfd->ioctl_ops = &camera0_ioctl_ops_gstreamer;
- + }
- +
- + vfd->v4l2_dev = &dev->v4l2_dev;
- +
- + vfd->lock = &dev->mutex;
- +
- + vfd->queue = &dev->capture.vb_vidq;
- +
- + /* video device needs to be able to access instance data */
- + video_set_drvdata(vfd, dev);
- +
- + ret = video_register_device(vfd, VFL_TYPE_GRABBER, -1);
- + if (ret < 0)
- + return ret;
- +
- + v4l2_info(vfd->v4l2_dev,
- + "V4L2 device registered as %s - stills mode > %dx%d\n",
- + video_device_node_name(vfd), max_video_width, max_video_height);
- +
- + return 0;
- +}
- +
- +static struct v4l2_format default_v4l2_format = {
- + .fmt.pix.pixelformat = V4L2_PIX_FMT_JPEG,
- + .fmt.pix.width = 1024,
- + .fmt.pix.bytesperline = 1024,
- + .fmt.pix.height = 768,
- + .fmt.pix.sizeimage = 1024*768,
- +};
- +
- +static int __init bm2835_mmal_init(void)
- +{
- + int ret;
- + struct bm2835_mmal_dev *dev;
- + struct vb2_queue *q;
- +
- + dev = kzalloc(sizeof(*gdev), GFP_KERNEL);
- + if (!dev)
- + return -ENOMEM;
- +
- + /* setup device defaults */
- + dev->overlay.w.left = 150;
- + dev->overlay.w.top = 50;
- + dev->overlay.w.width = 1024;
- + dev->overlay.w.height = 768;
- + dev->overlay.clipcount = 0;
- + dev->overlay.field = V4L2_FIELD_NONE;
- +
- + dev->capture.fmt = &formats[3]; /* JPEG */
- +
- + /* v4l device registration */
- + snprintf(dev->v4l2_dev.name, sizeof(dev->v4l2_dev.name),
- + "%s", BM2835_MMAL_MODULE_NAME);
- + ret = v4l2_device_register(NULL, &dev->v4l2_dev);
- + if (ret)
- + goto free_dev;
- +
- + /* setup v4l controls */
- + ret = bm2835_mmal_init_controls(dev, &dev->ctrl_handler);
- + if (ret < 0)
- + goto unreg_dev;
- + dev->v4l2_dev.ctrl_handler = &dev->ctrl_handler;
- +
- + /* mmal init */
- + ret = mmal_init(dev);
- + if (ret < 0)
- + goto unreg_dev;
- +
- + /* initialize queue */
- + q = &dev->capture.vb_vidq;
- + memset(q, 0, sizeof(*q));
- + q->type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
- + q->io_modes = VB2_MMAP | VB2_USERPTR | VB2_READ;
- + q->drv_priv = dev;
- + q->buf_struct_size = sizeof(struct mmal_buffer);
- + q->ops = &bm2835_mmal_video_qops;
- + q->mem_ops = &vb2_vmalloc_memops;
- + q->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_MONOTONIC;
- + ret = vb2_queue_init(q);
- + if (ret < 0)
- + goto unreg_dev;
- +
- + /* v4l2 core mutex used to protect all fops and v4l2 ioctls. */
- + mutex_init(&dev->mutex);
- +
- + /* initialise video devices */
- + ret = bm2835_mmal_init_device(dev, &dev->vdev);
- + if (ret < 0)
- + goto unreg_dev;
- +
- + /* Really want to call vidioc_s_fmt_vid_cap with the default
- + * format, but currently the APIs don't join up.
- + */
- + ret = mmal_setup_components(dev, &default_v4l2_format);
- + if (ret < 0) {
- + v4l2_err(&dev->v4l2_dev,
- + "%s: could not setup components\n", __func__);
- + goto unreg_dev;
- + }
- +
- + v4l2_info(&dev->v4l2_dev,
- + "Broadcom 2835 MMAL video capture ver %s loaded.\n",
- + BM2835_MMAL_VERSION);
- +
- + gdev = dev;
- + return 0;
- +
- +unreg_dev:
- + v4l2_ctrl_handler_free(&dev->ctrl_handler);
- + v4l2_device_unregister(&dev->v4l2_dev);
- +
- +free_dev:
- + kfree(dev);
- +
- + v4l2_err(&dev->v4l2_dev,
- + "%s: error %d while loading driver\n",
- + BM2835_MMAL_MODULE_NAME, ret);
- +
- + return ret;
- +}
- +
- +static void __exit bm2835_mmal_exit(void)
- +{
- + if (!gdev)
- + return;
- +
- + v4l2_info(&gdev->v4l2_dev, "unregistering %s\n",
- + video_device_node_name(&gdev->vdev));
- +
- + video_unregister_device(&gdev->vdev);
- +
- + if (gdev->capture.encode_component) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &gdev->v4l2_dev,
- + "mmal_exit - disconnect tunnel\n");
- + vchiq_mmal_port_connect_tunnel(gdev->instance,
- + gdev->capture.camera_port, NULL);
- + vchiq_mmal_component_disable(gdev->instance,
- + gdev->capture.encode_component);
- + }
- + vchiq_mmal_component_disable(gdev->instance,
- + gdev->component[MMAL_COMPONENT_CAMERA]);
- +
- + vchiq_mmal_component_finalise(gdev->instance,
- + gdev->
- + component[MMAL_COMPONENT_VIDEO_ENCODE]);
- +
- + vchiq_mmal_component_finalise(gdev->instance,
- + gdev->
- + component[MMAL_COMPONENT_IMAGE_ENCODE]);
- +
- + vchiq_mmal_component_finalise(gdev->instance,
- + gdev->component[MMAL_COMPONENT_PREVIEW]);
- +
- + vchiq_mmal_component_finalise(gdev->instance,
- + gdev->component[MMAL_COMPONENT_CAMERA]);
- +
- + vchiq_mmal_finalise(gdev->instance);
- +
- + v4l2_ctrl_handler_free(&gdev->ctrl_handler);
- +
- + v4l2_device_unregister(&gdev->v4l2_dev);
- +
- + kfree(gdev);
- +}
- +
- +module_init(bm2835_mmal_init);
- +module_exit(bm2835_mmal_exit);
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/bcm2835-camera.h
- @@ -0,0 +1,126 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + *
- + * core driver device
- + */
- +
- +#define V4L2_CTRL_COUNT 28 /* number of v4l controls */
- +
- +enum {
- + MMAL_COMPONENT_CAMERA = 0,
- + MMAL_COMPONENT_PREVIEW,
- + MMAL_COMPONENT_IMAGE_ENCODE,
- + MMAL_COMPONENT_VIDEO_ENCODE,
- + MMAL_COMPONENT_COUNT
- +};
- +
- +enum {
- + MMAL_CAMERA_PORT_PREVIEW = 0,
- + MMAL_CAMERA_PORT_VIDEO,
- + MMAL_CAMERA_PORT_CAPTURE,
- + MMAL_CAMERA_PORT_COUNT
- +};
- +
- +#define PREVIEW_LAYER 2
- +
- +extern int bcm2835_v4l2_debug;
- +
- +struct bm2835_mmal_dev {
- + /* v4l2 devices */
- + struct v4l2_device v4l2_dev;
- + struct video_device vdev;
- + struct mutex mutex;
- +
- + /* controls */
- + struct v4l2_ctrl_handler ctrl_handler;
- + struct v4l2_ctrl *ctrls[V4L2_CTRL_COUNT];
- + enum v4l2_scene_mode scene_mode;
- + struct mmal_colourfx colourfx;
- + int hflip;
- + int vflip;
- + int red_gain;
- + int blue_gain;
- + enum mmal_parameter_exposuremode exposure_mode_user;
- + enum v4l2_exposure_auto_type exposure_mode_v4l2_user;
- + /* active exposure mode may differ if selected via a scene mode */
- + enum mmal_parameter_exposuremode exposure_mode_active;
- + enum mmal_parameter_exposuremeteringmode metering_mode;
- + unsigned int manual_shutter_speed;
- + bool exp_auto_priority;
- +
- + /* allocated mmal instance and components */
- + struct vchiq_mmal_instance *instance;
- + struct vchiq_mmal_component *component[MMAL_COMPONENT_COUNT];
- + int camera_use_count;
- +
- + struct v4l2_window overlay;
- +
- + struct {
- + unsigned int width; /* width */
- + unsigned int height; /* height */
- + unsigned int stride; /* stride */
- + unsigned int buffersize; /* buffer size with padding */
- + struct mmal_fmt *fmt;
- + struct v4l2_fract timeperframe;
- +
- + /* H264 encode bitrate */
- + int encode_bitrate;
- + /* H264 bitrate mode. CBR/VBR */
- + int encode_bitrate_mode;
- + /* H264 profile */
- + enum v4l2_mpeg_video_h264_profile enc_profile;
- + /* H264 level */
- + enum v4l2_mpeg_video_h264_level enc_level;
- + /* JPEG Q-factor */
- + int q_factor;
- +
- + struct vb2_queue vb_vidq;
- +
- + /* VC start timestamp for streaming */
- + s64 vc_start_timestamp;
- + /* Kernel start timestamp for streaming */
- + struct timeval kernel_start_ts;
- +
- + struct vchiq_mmal_port *port; /* port being used for capture */
- + /* camera port being used for capture */
- + struct vchiq_mmal_port *camera_port;
- + /* component being used for encode */
- + struct vchiq_mmal_component *encode_component;
- + /* number of frames remaining which driver should capture */
- + unsigned int frame_count;
- + /* last frame completion */
- + struct completion frame_cmplt;
- +
- + } capture;
- +
- +};
- +
- +int bm2835_mmal_init_controls(
- + struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl_handler *hdl);
- +
- +int bm2835_mmal_set_all_camera_controls(struct bm2835_mmal_dev *dev);
- +int set_framerate_params(struct bm2835_mmal_dev *dev);
- +
- +/* Debug helpers */
- +
- +#define v4l2_dump_pix_format(level, debug, dev, pix_fmt, desc) \
- +{ \
- + v4l2_dbg(level, debug, dev, \
- +"%s: w %u h %u field %u pfmt 0x%x bpl %u sz_img %u colorspace 0x%x priv %u\n", \
- + desc == NULL ? "" : desc, \
- + (pix_fmt)->width, (pix_fmt)->height, (pix_fmt)->field, \
- + (pix_fmt)->pixelformat, (pix_fmt)->bytesperline, \
- + (pix_fmt)->sizeimage, (pix_fmt)->colorspace, (pix_fmt)->priv); \
- +}
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/controls.c
- @@ -0,0 +1,1324 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +#include <linux/errno.h>
- +#include <linux/kernel.h>
- +#include <linux/module.h>
- +#include <linux/slab.h>
- +#include <media/videobuf2-vmalloc.h>
- +#include <media/v4l2-device.h>
- +#include <media/v4l2-ioctl.h>
- +#include <media/v4l2-ctrls.h>
- +#include <media/v4l2-fh.h>
- +#include <media/v4l2-event.h>
- +#include <media/v4l2-common.h>
- +
- +#include "mmal-common.h"
- +#include "mmal-vchiq.h"
- +#include "mmal-parameters.h"
- +#include "bcm2835-camera.h"
- +
- +/* The supported V4L2_CID_AUTO_EXPOSURE_BIAS values are from -4.0 to +4.0.
- + * MMAL values are in 1/6th increments so the MMAL range is -24 to +24.
- + * V4L2 docs say value "is expressed in terms of EV, drivers should interpret
- + * the values as 0.001 EV units, where the value 1000 stands for +1 EV."
- + * V4L2 is limited to a max of 32 values in a menu, so count in 1/3rds from
- + * -4 to +4
- + */
- +static const s64 ev_bias_qmenu[] = {
- + -4000, -3667, -3333,
- + -3000, -2667, -2333,
- + -2000, -1667, -1333,
- + -1000, -667, -333,
- + 0, 333, 667,
- + 1000, 1333, 1667,
- + 2000, 2333, 2667,
- + 3000, 3333, 3667,
- + 4000
- +};
- +
- +/* Supported ISO values
- + * ISOO = auto ISO
- + */
- +static const s64 iso_qmenu[] = {
- + 0, 100, 200, 400, 800,
- +};
- +
- +static const s64 mains_freq_qmenu[] = {
- + V4L2_CID_POWER_LINE_FREQUENCY_DISABLED,
- + V4L2_CID_POWER_LINE_FREQUENCY_50HZ,
- + V4L2_CID_POWER_LINE_FREQUENCY_60HZ,
- + V4L2_CID_POWER_LINE_FREQUENCY_AUTO
- +};
- +
- +/* Supported video encode modes */
- +static const s64 bitrate_mode_qmenu[] = {
- + (s64)V4L2_MPEG_VIDEO_BITRATE_MODE_VBR,
- + (s64)V4L2_MPEG_VIDEO_BITRATE_MODE_CBR,
- +};
- +
- +enum bm2835_mmal_ctrl_type {
- + MMAL_CONTROL_TYPE_STD,
- + MMAL_CONTROL_TYPE_STD_MENU,
- + MMAL_CONTROL_TYPE_INT_MENU,
- + MMAL_CONTROL_TYPE_CLUSTER, /* special cluster entry */
- +};
- +
- +struct bm2835_mmal_v4l2_ctrl;
- +
- +typedef int(bm2835_mmal_v4l2_ctrl_cb)(
- + struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl);
- +
- +struct bm2835_mmal_v4l2_ctrl {
- + u32 id; /* v4l2 control identifier */
- + enum bm2835_mmal_ctrl_type type;
- + /* control minimum value or
- + * mask for MMAL_CONTROL_TYPE_STD_MENU */
- + s32 min;
- + s32 max; /* maximum value of control */
- + s32 def; /* default value of control */
- + s32 step; /* step size of the control */
- + const s64 *imenu; /* integer menu array */
- + u32 mmal_id; /* mmal parameter id */
- + bm2835_mmal_v4l2_ctrl_cb *setter;
- + bool ignore_errors;
- +};
- +
- +struct v4l2_to_mmal_effects_setting {
- + u32 v4l2_effect;
- + u32 mmal_effect;
- + s32 col_fx_enable;
- + s32 col_fx_fixed_cbcr;
- + u32 u;
- + u32 v;
- + u32 num_effect_params;
- + u32 effect_params[MMAL_MAX_IMAGEFX_PARAMETERS];
- +};
- +
- +static const struct v4l2_to_mmal_effects_setting
- + v4l2_to_mmal_effects_values[] = {
- + { V4L2_COLORFX_NONE, MMAL_PARAM_IMAGEFX_NONE,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_BW, MMAL_PARAM_IMAGEFX_NONE,
- + 1, 0, 128, 128, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SEPIA, MMAL_PARAM_IMAGEFX_NONE,
- + 1, 0, 87, 151, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_NEGATIVE, MMAL_PARAM_IMAGEFX_NEGATIVE,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_EMBOSS, MMAL_PARAM_IMAGEFX_EMBOSS,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SKETCH, MMAL_PARAM_IMAGEFX_SKETCH,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SKY_BLUE, MMAL_PARAM_IMAGEFX_PASTEL,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_GRASS_GREEN, MMAL_PARAM_IMAGEFX_WATERCOLOUR,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SKIN_WHITEN, MMAL_PARAM_IMAGEFX_WASHEDOUT,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_VIVID, MMAL_PARAM_IMAGEFX_SATURATION,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_AQUA, MMAL_PARAM_IMAGEFX_NONE,
- + 1, 0, 171, 121, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_ART_FREEZE, MMAL_PARAM_IMAGEFX_HATCH,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SILHOUETTE, MMAL_PARAM_IMAGEFX_FILM,
- + 0, 0, 0, 0, 0, {0, 0, 0, 0, 0} },
- + { V4L2_COLORFX_SOLARIZATION, MMAL_PARAM_IMAGEFX_SOLARIZE,
- + 0, 0, 0, 0, 5, {1, 128, 160, 160, 48} },
- + { V4L2_COLORFX_ANTIQUE, MMAL_PARAM_IMAGEFX_COLOURBALANCE,
- + 0, 0, 0, 0, 3, {108, 274, 238, 0, 0} },
- + { V4L2_COLORFX_SET_CBCR, MMAL_PARAM_IMAGEFX_NONE,
- + 1, 1, 0, 0, 0, {0, 0, 0, 0, 0} }
- +};
- +
- +struct v4l2_mmal_scene_config {
- + enum v4l2_scene_mode v4l2_scene;
- + enum mmal_parameter_exposuremode exposure_mode;
- + enum mmal_parameter_exposuremeteringmode metering_mode;
- +};
- +
- +static const struct v4l2_mmal_scene_config scene_configs[] = {
- + /* V4L2_SCENE_MODE_NONE automatically added */
- + {
- + V4L2_SCENE_MODE_NIGHT,
- + MMAL_PARAM_EXPOSUREMODE_NIGHT,
- + MMAL_PARAM_EXPOSUREMETERINGMODE_AVERAGE
- + },
- + {
- + V4L2_SCENE_MODE_SPORTS,
- + MMAL_PARAM_EXPOSUREMODE_SPORTS,
- + MMAL_PARAM_EXPOSUREMETERINGMODE_AVERAGE
- + },
- +};
- +
- +/* control handlers*/
- +
- +static int ctrl_set_rational(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + struct mmal_parameter_rational rational_value;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + rational_value.num = ctrl->val;
- + rational_value.den = 100;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &rational_value,
- + sizeof(rational_value));
- +}
- +
- +static int ctrl_set_value(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + u32_value = ctrl->val;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_value_menu(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *control;
- +
- + if (ctrl->val > mmal_ctrl->max || ctrl->val < mmal_ctrl->min)
- + return 1;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + u32_value = mmal_ctrl->imenu[ctrl->val];
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_value_ev(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + s32 s32_value;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + s32_value = (ctrl->val-12)*2; /* Convert from index to 1/6ths */
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &s32_value, sizeof(s32_value));
- +}
- +
- +static int ctrl_set_rotate(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret;
- + u32 u32_value;
- + struct vchiq_mmal_component *camera;
- +
- + camera = dev->component[MMAL_COMPONENT_CAMERA];
- +
- + u32_value = ((ctrl->val % 360) / 90) * 90;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[0],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- + if (ret < 0)
- + return ret;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[1],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- + if (ret < 0)
- + return ret;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[2],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +
- + return ret;
- +}
- +
- +static int ctrl_set_flip(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret;
- + u32 u32_value;
- + struct vchiq_mmal_component *camera;
- +
- + if (ctrl->id == V4L2_CID_HFLIP)
- + dev->hflip = ctrl->val;
- + else
- + dev->vflip = ctrl->val;
- +
- + camera = dev->component[MMAL_COMPONENT_CAMERA];
- +
- + if (dev->hflip && dev->vflip)
- + u32_value = MMAL_PARAM_MIRROR_BOTH;
- + else if (dev->hflip)
- + u32_value = MMAL_PARAM_MIRROR_HORIZONTAL;
- + else if (dev->vflip)
- + u32_value = MMAL_PARAM_MIRROR_VERTICAL;
- + else
- + u32_value = MMAL_PARAM_MIRROR_NONE;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[0],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- + if (ret < 0)
- + return ret;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[1],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- + if (ret < 0)
- + return ret;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, &camera->output[2],
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +
- + return ret;
- +
- +}
- +
- +static int ctrl_set_exposure(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + enum mmal_parameter_exposuremode exp_mode = dev->exposure_mode_user;
- + u32 shutter_speed = 0;
- + struct vchiq_mmal_port *control;
- + int ret = 0;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + if (mmal_ctrl->mmal_id == MMAL_PARAMETER_SHUTTER_SPEED) {
- + /* V4L2 is in 100usec increments.
- + * MMAL is 1usec.
- + */
- + dev->manual_shutter_speed = ctrl->val * 100;
- + } else if (mmal_ctrl->mmal_id == MMAL_PARAMETER_EXPOSURE_MODE) {
- + switch (ctrl->val) {
- + case V4L2_EXPOSURE_AUTO:
- + exp_mode = MMAL_PARAM_EXPOSUREMODE_AUTO;
- + break;
- +
- + case V4L2_EXPOSURE_MANUAL:
- + exp_mode = MMAL_PARAM_EXPOSUREMODE_OFF;
- + break;
- + }
- + dev->exposure_mode_user = exp_mode;
- + dev->exposure_mode_v4l2_user = ctrl->val;
- + } else if (mmal_ctrl->id == V4L2_CID_EXPOSURE_AUTO_PRIORITY) {
- + dev->exp_auto_priority = ctrl->val;
- + }
- +
- + if (dev->scene_mode == V4L2_SCENE_MODE_NONE) {
- + if (exp_mode == MMAL_PARAM_EXPOSUREMODE_OFF)
- + shutter_speed = dev->manual_shutter_speed;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_SHUTTER_SPEED,
- + &shutter_speed,
- + sizeof(shutter_speed));
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_EXPOSURE_MODE,
- + &exp_mode,
- + sizeof(u32));
- + dev->exposure_mode_active = exp_mode;
- + }
- + /* exposure_dynamic_framerate (V4L2_CID_EXPOSURE_AUTO_PRIORITY) should
- + * always apply irrespective of scene mode.
- + */
- + ret += set_framerate_params(dev);
- +
- + return ret;
- +}
- +
- +static int ctrl_set_metering_mode(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + switch (ctrl->val) {
- + case V4L2_EXPOSURE_METERING_AVERAGE:
- + dev->metering_mode = MMAL_PARAM_EXPOSUREMETERINGMODE_AVERAGE;
- + break;
- +
- + case V4L2_EXPOSURE_METERING_CENTER_WEIGHTED:
- + dev->metering_mode = MMAL_PARAM_EXPOSUREMETERINGMODE_BACKLIT;
- + break;
- +
- + case V4L2_EXPOSURE_METERING_SPOT:
- + dev->metering_mode = MMAL_PARAM_EXPOSUREMETERINGMODE_SPOT;
- + break;
- +
- + /* todo matrix weighting not added to Linux API till 3.9
- + case V4L2_EXPOSURE_METERING_MATRIX:
- + dev->metering_mode = MMAL_PARAM_EXPOSUREMETERINGMODE_MATRIX;
- + break;
- + */
- +
- + }
- +
- + if (dev->scene_mode == V4L2_SCENE_MODE_NONE) {
- + struct vchiq_mmal_port *control;
- + u32 u32_value = dev->metering_mode;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- + } else
- + return 0;
- +}
- +
- +static int ctrl_set_flicker_avoidance(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + switch (ctrl->val) {
- + case V4L2_CID_POWER_LINE_FREQUENCY_DISABLED:
- + u32_value = MMAL_PARAM_FLICKERAVOID_OFF;
- + break;
- + case V4L2_CID_POWER_LINE_FREQUENCY_50HZ:
- + u32_value = MMAL_PARAM_FLICKERAVOID_50HZ;
- + break;
- + case V4L2_CID_POWER_LINE_FREQUENCY_60HZ:
- + u32_value = MMAL_PARAM_FLICKERAVOID_60HZ;
- + break;
- + case V4L2_CID_POWER_LINE_FREQUENCY_AUTO:
- + u32_value = MMAL_PARAM_FLICKERAVOID_AUTO;
- + break;
- + }
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_awb_mode(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + switch (ctrl->val) {
- + case V4L2_WHITE_BALANCE_MANUAL:
- + u32_value = MMAL_PARAM_AWBMODE_OFF;
- + break;
- +
- + case V4L2_WHITE_BALANCE_AUTO:
- + u32_value = MMAL_PARAM_AWBMODE_AUTO;
- + break;
- +
- + case V4L2_WHITE_BALANCE_INCANDESCENT:
- + u32_value = MMAL_PARAM_AWBMODE_INCANDESCENT;
- + break;
- +
- + case V4L2_WHITE_BALANCE_FLUORESCENT:
- + u32_value = MMAL_PARAM_AWBMODE_FLUORESCENT;
- + break;
- +
- + case V4L2_WHITE_BALANCE_FLUORESCENT_H:
- + u32_value = MMAL_PARAM_AWBMODE_TUNGSTEN;
- + break;
- +
- + case V4L2_WHITE_BALANCE_HORIZON:
- + u32_value = MMAL_PARAM_AWBMODE_HORIZON;
- + break;
- +
- + case V4L2_WHITE_BALANCE_DAYLIGHT:
- + u32_value = MMAL_PARAM_AWBMODE_SUNLIGHT;
- + break;
- +
- + case V4L2_WHITE_BALANCE_FLASH:
- + u32_value = MMAL_PARAM_AWBMODE_FLASH;
- + break;
- +
- + case V4L2_WHITE_BALANCE_CLOUDY:
- + u32_value = MMAL_PARAM_AWBMODE_CLOUDY;
- + break;
- +
- + case V4L2_WHITE_BALANCE_SHADE:
- + u32_value = MMAL_PARAM_AWBMODE_SHADE;
- + break;
- +
- + }
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_awb_gains(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + struct vchiq_mmal_port *control;
- + struct mmal_parameter_awbgains gains;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + if (ctrl->id == V4L2_CID_RED_BALANCE)
- + dev->red_gain = ctrl->val;
- + else if (ctrl->id == V4L2_CID_BLUE_BALANCE)
- + dev->blue_gain = ctrl->val;
- +
- + gains.r_gain.num = dev->red_gain;
- + gains.b_gain.num = dev->blue_gain;
- + gains.r_gain.den = gains.b_gain.den = 1000;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, control,
- + mmal_ctrl->mmal_id,
- + &gains, sizeof(gains));
- +}
- +
- +static int ctrl_set_image_effect(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret = -EINVAL;
- + int i, j;
- + struct vchiq_mmal_port *control;
- + struct mmal_parameter_imagefx_parameters imagefx;
- +
- + for (i = 0; i < ARRAY_SIZE(v4l2_to_mmal_effects_values); i++) {
- + if (ctrl->val == v4l2_to_mmal_effects_values[i].v4l2_effect) {
- +
- + imagefx.effect =
- + v4l2_to_mmal_effects_values[i].mmal_effect;
- + imagefx.num_effect_params =
- + v4l2_to_mmal_effects_values[i].num_effect_params;
- +
- + if (imagefx.num_effect_params > MMAL_MAX_IMAGEFX_PARAMETERS)
- + imagefx.num_effect_params = MMAL_MAX_IMAGEFX_PARAMETERS;
- +
- + for (j = 0; j < imagefx.num_effect_params; j++)
- + imagefx.effect_parameter[j] =
- + v4l2_to_mmal_effects_values[i].effect_params[j];
- +
- + dev->colourfx.enable =
- + v4l2_to_mmal_effects_values[i].col_fx_enable;
- + if (!v4l2_to_mmal_effects_values[i].col_fx_fixed_cbcr) {
- + dev->colourfx.u =
- + v4l2_to_mmal_effects_values[i].u;
- + dev->colourfx.v =
- + v4l2_to_mmal_effects_values[i].v;
- + }
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + ret = vchiq_mmal_port_parameter_set(
- + dev->instance, control,
- + MMAL_PARAMETER_IMAGE_EFFECT_PARAMETERS,
- + &imagefx, sizeof(imagefx));
- + if (ret)
- + goto exit;
- +
- + ret = vchiq_mmal_port_parameter_set(
- + dev->instance, control,
- + MMAL_PARAMETER_COLOUR_EFFECT,
- + &dev->colourfx, sizeof(dev->colourfx));
- + }
- + }
- +
- +exit:
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "mmal_ctrl:%p ctrl id:0x%x ctrl val:%d imagefx:0x%x color_effect:%s u:%d v:%d ret %d(%d)\n",
- + mmal_ctrl, ctrl->id, ctrl->val, imagefx.effect,
- + dev->colourfx.enable ? "true" : "false",
- + dev->colourfx.u, dev->colourfx.v,
- + ret, (ret == 0 ? 0 : -EINVAL));
- + return (ret == 0 ? 0 : EINVAL);
- +}
- +
- +static int ctrl_set_colfx(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret = -EINVAL;
- + struct vchiq_mmal_port *control;
- +
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + dev->colourfx.enable = (ctrl->val & 0xff00) >> 8;
- + dev->colourfx.enable = ctrl->val & 0xff;
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, control,
- + MMAL_PARAMETER_COLOUR_EFFECT,
- + &dev->colourfx, sizeof(dev->colourfx));
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: After: mmal_ctrl:%p ctrl id:0x%x ctrl val:%d ret %d(%d)\n",
- + __func__, mmal_ctrl, ctrl->id, ctrl->val, ret,
- + (ret == 0 ? 0 : -EINVAL));
- + return (ret == 0 ? 0 : EINVAL);
- +}
- +
- +static int ctrl_set_bitrate(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret;
- + struct vchiq_mmal_port *encoder_out;
- +
- + dev->capture.encode_bitrate = ctrl->val;
- +
- + encoder_out = &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0];
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, encoder_out,
- + mmal_ctrl->mmal_id,
- + &ctrl->val, sizeof(ctrl->val));
- + ret = 0;
- + return ret;
- +}
- +
- +static int ctrl_set_bitrate_mode(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 bitrate_mode;
- + struct vchiq_mmal_port *encoder_out;
- +
- + encoder_out = &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0];
- +
- + dev->capture.encode_bitrate_mode = ctrl->val;
- + switch (ctrl->val) {
- + default:
- + case V4L2_MPEG_VIDEO_BITRATE_MODE_VBR:
- + bitrate_mode = MMAL_VIDEO_RATECONTROL_VARIABLE;
- + break;
- + case V4L2_MPEG_VIDEO_BITRATE_MODE_CBR:
- + bitrate_mode = MMAL_VIDEO_RATECONTROL_CONSTANT;
- + break;
- + }
- +
- + vchiq_mmal_port_parameter_set(dev->instance, encoder_out,
- + mmal_ctrl->mmal_id,
- + &bitrate_mode,
- + sizeof(bitrate_mode));
- + return 0;
- +}
- +
- +static int ctrl_set_image_encode_output(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *jpeg_out;
- +
- + jpeg_out = &dev->component[MMAL_COMPONENT_IMAGE_ENCODE]->output[0];
- +
- + u32_value = ctrl->val;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, jpeg_out,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_video_encode_param_output(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + u32 u32_value;
- + struct vchiq_mmal_port *vid_enc_ctl;
- +
- + vid_enc_ctl = &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0];
- +
- + u32_value = ctrl->val;
- +
- + return vchiq_mmal_port_parameter_set(dev->instance, vid_enc_ctl,
- + mmal_ctrl->mmal_id,
- + &u32_value, sizeof(u32_value));
- +}
- +
- +static int ctrl_set_video_encode_profile_level(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + struct mmal_parameter_video_profile param;
- + int ret = 0;
- +
- + if (ctrl->id == V4L2_CID_MPEG_VIDEO_H264_PROFILE) {
- + switch (ctrl->val) {
- + case V4L2_MPEG_VIDEO_H264_PROFILE_BASELINE:
- + case V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE:
- + case V4L2_MPEG_VIDEO_H264_PROFILE_MAIN:
- + case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH:
- + dev->capture.enc_profile = ctrl->val;
- + break;
- + default:
- + ret = -EINVAL;
- + break;
- + }
- + } else if (ctrl->id == V4L2_CID_MPEG_VIDEO_H264_LEVEL) {
- + switch (ctrl->val) {
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_0:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1B:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_1:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_2:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_3:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_0:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_1:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_2:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_0:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_1:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_2:
- + case V4L2_MPEG_VIDEO_H264_LEVEL_4_0:
- + dev->capture.enc_level = ctrl->val;
- + break;
- + default:
- + ret = -EINVAL;
- + break;
- + }
- + }
- +
- + if (!ret) {
- + switch (dev->capture.enc_profile) {
- + case V4L2_MPEG_VIDEO_H264_PROFILE_BASELINE:
- + param.profile = MMAL_VIDEO_PROFILE_H264_BASELINE;
- + break;
- + case V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE:
- + param.profile =
- + MMAL_VIDEO_PROFILE_H264_CONSTRAINED_BASELINE;
- + break;
- + case V4L2_MPEG_VIDEO_H264_PROFILE_MAIN:
- + param.profile = MMAL_VIDEO_PROFILE_H264_MAIN;
- + break;
- + case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH:
- + param.profile = MMAL_VIDEO_PROFILE_H264_HIGH;
- + break;
- + default:
- + /* Should never get here */
- + break;
- + }
- +
- + switch (dev->capture.enc_level) {
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_0:
- + param.level = MMAL_VIDEO_LEVEL_H264_1;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1B:
- + param.level = MMAL_VIDEO_LEVEL_H264_1b;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_1:
- + param.level = MMAL_VIDEO_LEVEL_H264_11;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_2:
- + param.level = MMAL_VIDEO_LEVEL_H264_12;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_1_3:
- + param.level = MMAL_VIDEO_LEVEL_H264_13;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_0:
- + param.level = MMAL_VIDEO_LEVEL_H264_2;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_1:
- + param.level = MMAL_VIDEO_LEVEL_H264_21;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_2_2:
- + param.level = MMAL_VIDEO_LEVEL_H264_22;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_0:
- + param.level = MMAL_VIDEO_LEVEL_H264_3;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_1:
- + param.level = MMAL_VIDEO_LEVEL_H264_31;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_3_2:
- + param.level = MMAL_VIDEO_LEVEL_H264_32;
- + break;
- + case V4L2_MPEG_VIDEO_H264_LEVEL_4_0:
- + param.level = MMAL_VIDEO_LEVEL_H264_4;
- + break;
- + default:
- + /* Should never get here */
- + break;
- + }
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_VIDEO_ENCODE]->output[0],
- + mmal_ctrl->mmal_id,
- + ¶m, sizeof(param));
- + }
- + return ret;
- +}
- +
- +static int ctrl_set_scene_mode(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl *ctrl,
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl)
- +{
- + int ret = 0;
- + int shutter_speed;
- + struct vchiq_mmal_port *control;
- +
- + v4l2_dbg(0, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "scene mode selected %d, was %d\n", ctrl->val,
- + dev->scene_mode);
- + control = &dev->component[MMAL_COMPONENT_CAMERA]->control;
- +
- + if (ctrl->val == dev->scene_mode)
- + return 0;
- +
- + if (ctrl->val == V4L2_SCENE_MODE_NONE) {
- + /* Restore all user selections */
- + dev->scene_mode = V4L2_SCENE_MODE_NONE;
- +
- + if (dev->exposure_mode_user == MMAL_PARAM_EXPOSUREMODE_OFF)
- + shutter_speed = dev->manual_shutter_speed;
- + else
- + shutter_speed = 0;
- +
- + v4l2_dbg(0, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: scene mode none: shut_speed %d, exp_mode %d, metering %d\n",
- + __func__, shutter_speed, dev->exposure_mode_user,
- + dev->metering_mode);
- + ret = vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_SHUTTER_SPEED,
- + &shutter_speed,
- + sizeof(shutter_speed));
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_EXPOSURE_MODE,
- + &dev->exposure_mode_user,
- + sizeof(u32));
- + dev->exposure_mode_active = dev->exposure_mode_user;
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_EXP_METERING_MODE,
- + &dev->metering_mode,
- + sizeof(u32));
- + ret += set_framerate_params(dev);
- + } else {
- + /* Set up scene mode */
- + int i;
- + const struct v4l2_mmal_scene_config *scene = NULL;
- + int shutter_speed;
- + enum mmal_parameter_exposuremode exposure_mode;
- + enum mmal_parameter_exposuremeteringmode metering_mode;
- +
- + for (i = 0; i < ARRAY_SIZE(scene_configs); i++) {
- + if (scene_configs[i].v4l2_scene ==
- + ctrl->val) {
- + scene = &scene_configs[i];
- + break;
- + }
- + }
- + if (!scene)
- + return -EINVAL;
- + if (i >= ARRAY_SIZE(scene_configs))
- + return -EINVAL;
- +
- + /* Set all the values */
- + dev->scene_mode = ctrl->val;
- +
- + if (scene->exposure_mode == MMAL_PARAM_EXPOSUREMODE_OFF)
- + shutter_speed = dev->manual_shutter_speed;
- + else
- + shutter_speed = 0;
- + exposure_mode = scene->exposure_mode;
- + metering_mode = scene->metering_mode;
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: scene mode none: shut_speed %d, exp_mode %d, metering %d\n",
- + __func__, shutter_speed, exposure_mode, metering_mode);
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance, control,
- + MMAL_PARAMETER_SHUTTER_SPEED,
- + &shutter_speed,
- + sizeof(shutter_speed));
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + control,
- + MMAL_PARAMETER_EXPOSURE_MODE,
- + &exposure_mode,
- + sizeof(u32));
- + dev->exposure_mode_active = exposure_mode;
- + ret += vchiq_mmal_port_parameter_set(dev->instance, control,
- + MMAL_PARAMETER_EXPOSURE_MODE,
- + &exposure_mode,
- + sizeof(u32));
- + ret += vchiq_mmal_port_parameter_set(dev->instance, control,
- + MMAL_PARAMETER_EXP_METERING_MODE,
- + &metering_mode,
- + sizeof(u32));
- + ret += set_framerate_params(dev);
- + }
- + if (ret) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "%s: Setting scene to %d, ret=%d\n",
- + __func__, ctrl->val, ret);
- + ret = -EINVAL;
- + }
- + return 0;
- +}
- +
- +static int bm2835_mmal_s_ctrl(struct v4l2_ctrl *ctrl)
- +{
- + struct bm2835_mmal_dev *dev =
- + container_of(ctrl->handler, struct bm2835_mmal_dev,
- + ctrl_handler);
- + const struct bm2835_mmal_v4l2_ctrl *mmal_ctrl = ctrl->priv;
- + int ret;
- +
- + if ((mmal_ctrl == NULL) ||
- + (mmal_ctrl->id != ctrl->id) ||
- + (mmal_ctrl->setter == NULL)) {
- + pr_warn("mmal_ctrl:%p ctrl id:%d\n", mmal_ctrl, ctrl->id);
- + return -EINVAL;
- + }
- +
- + ret = mmal_ctrl->setter(dev, ctrl, mmal_ctrl);
- + if (ret)
- + pr_warn("ctrl id:%d/MMAL param %08X- returned ret %d\n",
- + ctrl->id, mmal_ctrl->mmal_id, ret);
- + if (mmal_ctrl->ignore_errors)
- + ret = 0;
- + return ret;
- +}
- +
- +static const struct v4l2_ctrl_ops bm2835_mmal_ctrl_ops = {
- + .s_ctrl = bm2835_mmal_s_ctrl,
- +};
- +
- +
- +
- +static const struct bm2835_mmal_v4l2_ctrl v4l2_ctrls[V4L2_CTRL_COUNT] = {
- + {
- + V4L2_CID_SATURATION, MMAL_CONTROL_TYPE_STD,
- + -100, 100, 0, 1, NULL,
- + MMAL_PARAMETER_SATURATION,
- + &ctrl_set_rational,
- + false
- + },
- + {
- + V4L2_CID_SHARPNESS, MMAL_CONTROL_TYPE_STD,
- + -100, 100, 0, 1, NULL,
- + MMAL_PARAMETER_SHARPNESS,
- + &ctrl_set_rational,
- + false
- + },
- + {
- + V4L2_CID_CONTRAST, MMAL_CONTROL_TYPE_STD,
- + -100, 100, 0, 1, NULL,
- + MMAL_PARAMETER_CONTRAST,
- + &ctrl_set_rational,
- + false
- + },
- + {
- + V4L2_CID_BRIGHTNESS, MMAL_CONTROL_TYPE_STD,
- + 0, 100, 50, 1, NULL,
- + MMAL_PARAMETER_BRIGHTNESS,
- + &ctrl_set_rational,
- + false
- + },
- + {
- + V4L2_CID_ISO_SENSITIVITY, MMAL_CONTROL_TYPE_INT_MENU,
- + 0, ARRAY_SIZE(iso_qmenu) - 1, 0, 1, iso_qmenu,
- + MMAL_PARAMETER_ISO,
- + &ctrl_set_value_menu,
- + false
- + },
- + {
- + V4L2_CID_IMAGE_STABILIZATION, MMAL_CONTROL_TYPE_STD,
- + 0, 1, 0, 1, NULL,
- + MMAL_PARAMETER_VIDEO_STABILISATION,
- + &ctrl_set_value,
- + false
- + },
- +/* {
- + 0, MMAL_CONTROL_TYPE_CLUSTER, 3, 1, 0, NULL, 0, NULL
- + }, */
- + {
- + V4L2_CID_EXPOSURE_AUTO, MMAL_CONTROL_TYPE_STD_MENU,
- + ~0x03, 3, V4L2_EXPOSURE_AUTO, 0, NULL,
- + MMAL_PARAMETER_EXPOSURE_MODE,
- + &ctrl_set_exposure,
- + false
- + },
- +/* todo this needs mixing in with set exposure
- + {
- + V4L2_CID_SCENE_MODE, MMAL_CONTROL_TYPE_STD_MENU,
- + },
- + */
- + {
- + V4L2_CID_EXPOSURE_ABSOLUTE, MMAL_CONTROL_TYPE_STD,
- + /* Units of 100usecs */
- + 1, 1*1000*10, 100*10, 1, NULL,
- + MMAL_PARAMETER_SHUTTER_SPEED,
- + &ctrl_set_exposure,
- + false
- + },
- + {
- + V4L2_CID_AUTO_EXPOSURE_BIAS, MMAL_CONTROL_TYPE_INT_MENU,
- + 0, ARRAY_SIZE(ev_bias_qmenu) - 1,
- + (ARRAY_SIZE(ev_bias_qmenu)+1)/2 - 1, 0, ev_bias_qmenu,
- + MMAL_PARAMETER_EXPOSURE_COMP,
- + &ctrl_set_value_ev,
- + false
- + },
- + {
- + V4L2_CID_EXPOSURE_AUTO_PRIORITY, MMAL_CONTROL_TYPE_STD,
- + 0, 1,
- + 0, 1, NULL,
- + 0, /* Dummy MMAL ID as it gets mapped into FPS range*/
- + &ctrl_set_exposure,
- + false
- + },
- + {
- + V4L2_CID_EXPOSURE_METERING,
- + MMAL_CONTROL_TYPE_STD_MENU,
- + ~0x7, 2, V4L2_EXPOSURE_METERING_AVERAGE, 0, NULL,
- + MMAL_PARAMETER_EXP_METERING_MODE,
- + &ctrl_set_metering_mode,
- + false
- + },
- + {
- + V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE,
- + MMAL_CONTROL_TYPE_STD_MENU,
- + ~0x3ff, 9, V4L2_WHITE_BALANCE_AUTO, 0, NULL,
- + MMAL_PARAMETER_AWB_MODE,
- + &ctrl_set_awb_mode,
- + false
- + },
- + {
- + V4L2_CID_RED_BALANCE, MMAL_CONTROL_TYPE_STD,
- + 1, 7999, 1000, 1, NULL,
- + MMAL_PARAMETER_CUSTOM_AWB_GAINS,
- + &ctrl_set_awb_gains,
- + false
- + },
- + {
- + V4L2_CID_BLUE_BALANCE, MMAL_CONTROL_TYPE_STD,
- + 1, 7999, 1000, 1, NULL,
- + MMAL_PARAMETER_CUSTOM_AWB_GAINS,
- + &ctrl_set_awb_gains,
- + false
- + },
- + {
- + V4L2_CID_COLORFX, MMAL_CONTROL_TYPE_STD_MENU,
- + 0, 15, V4L2_COLORFX_NONE, 0, NULL,
- + MMAL_PARAMETER_IMAGE_EFFECT,
- + &ctrl_set_image_effect,
- + false
- + },
- + {
- + V4L2_CID_COLORFX_CBCR, MMAL_CONTROL_TYPE_STD,
- + 0, 0xffff, 0x8080, 1, NULL,
- + MMAL_PARAMETER_COLOUR_EFFECT,
- + &ctrl_set_colfx,
- + false
- + },
- + {
- + V4L2_CID_ROTATE, MMAL_CONTROL_TYPE_STD,
- + 0, 360, 0, 90, NULL,
- + MMAL_PARAMETER_ROTATION,
- + &ctrl_set_rotate,
- + false
- + },
- + {
- + V4L2_CID_HFLIP, MMAL_CONTROL_TYPE_STD,
- + 0, 1, 0, 1, NULL,
- + MMAL_PARAMETER_MIRROR,
- + &ctrl_set_flip,
- + false
- + },
- + {
- + V4L2_CID_VFLIP, MMAL_CONTROL_TYPE_STD,
- + 0, 1, 0, 1, NULL,
- + MMAL_PARAMETER_MIRROR,
- + &ctrl_set_flip,
- + false
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_BITRATE_MODE, MMAL_CONTROL_TYPE_STD_MENU,
- + 0, ARRAY_SIZE(bitrate_mode_qmenu) - 1,
- + 0, 0, bitrate_mode_qmenu,
- + MMAL_PARAMETER_RATECONTROL,
- + &ctrl_set_bitrate_mode,
- + false
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_BITRATE, MMAL_CONTROL_TYPE_STD,
- + 25*1000, 25*1000*1000, 10*1000*1000, 25*1000, NULL,
- + MMAL_PARAMETER_VIDEO_BIT_RATE,
- + &ctrl_set_bitrate,
- + false
- + },
- + {
- + V4L2_CID_JPEG_COMPRESSION_QUALITY, MMAL_CONTROL_TYPE_STD,
- + 1, 100,
- + 30, 1, NULL,
- + MMAL_PARAMETER_JPEG_Q_FACTOR,
- + &ctrl_set_image_encode_output,
- + false
- + },
- + {
- + V4L2_CID_POWER_LINE_FREQUENCY, MMAL_CONTROL_TYPE_STD_MENU,
- + 0, ARRAY_SIZE(mains_freq_qmenu) - 1,
- + 1, 1, NULL,
- + MMAL_PARAMETER_FLICKER_AVOID,
- + &ctrl_set_flicker_avoidance,
- + false
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_REPEAT_SEQ_HEADER, MMAL_CONTROL_TYPE_STD,
- + 0, 1,
- + 0, 1, NULL,
- + MMAL_PARAMETER_VIDEO_ENCODE_INLINE_HEADER,
- + &ctrl_set_video_encode_param_output,
- + true /* Errors ignored as requires latest firmware to work */
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_H264_PROFILE,
- + MMAL_CONTROL_TYPE_STD_MENU,
- + ~((1<<V4L2_MPEG_VIDEO_H264_PROFILE_BASELINE) |
- + (1<<V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE) |
- + (1<<V4L2_MPEG_VIDEO_H264_PROFILE_MAIN) |
- + (1<<V4L2_MPEG_VIDEO_H264_PROFILE_HIGH)),
- + V4L2_MPEG_VIDEO_H264_PROFILE_HIGH,
- + V4L2_MPEG_VIDEO_H264_PROFILE_HIGH, 1, NULL,
- + MMAL_PARAMETER_PROFILE,
- + &ctrl_set_video_encode_profile_level,
- + false
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_H264_LEVEL, MMAL_CONTROL_TYPE_STD_MENU,
- + ~((1<<V4L2_MPEG_VIDEO_H264_LEVEL_1_0) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_1B) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_1_1) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_1_2) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_1_3) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_2_0) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_2_1) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_2_2) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_3_0) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_3_1) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_3_2) |
- + (1<<V4L2_MPEG_VIDEO_H264_LEVEL_4_0)),
- + V4L2_MPEG_VIDEO_H264_LEVEL_4_0,
- + V4L2_MPEG_VIDEO_H264_LEVEL_4_0, 1, NULL,
- + MMAL_PARAMETER_PROFILE,
- + &ctrl_set_video_encode_profile_level,
- + false
- + },
- + {
- + V4L2_CID_SCENE_MODE, MMAL_CONTROL_TYPE_STD_MENU,
- + -1, /* Min is computed at runtime */
- + V4L2_SCENE_MODE_TEXT,
- + V4L2_SCENE_MODE_NONE, 1, NULL,
- + MMAL_PARAMETER_PROFILE,
- + &ctrl_set_scene_mode,
- + false
- + },
- + {
- + V4L2_CID_MPEG_VIDEO_H264_I_PERIOD, MMAL_CONTROL_TYPE_STD,
- + 0, 0x7FFFFFFF, 60, 1, NULL,
- + MMAL_PARAMETER_INTRAPERIOD,
- + &ctrl_set_video_encode_param_output,
- + false
- + },
- +};
- +
- +int bm2835_mmal_set_all_camera_controls(struct bm2835_mmal_dev *dev)
- +{
- + int c;
- + int ret = 0;
- +
- + for (c = 0; c < V4L2_CTRL_COUNT; c++) {
- + if ((dev->ctrls[c]) && (v4l2_ctrls[c].setter)) {
- + ret = v4l2_ctrls[c].setter(dev, dev->ctrls[c],
- + &v4l2_ctrls[c]);
- + if (!v4l2_ctrls[c].ignore_errors && ret) {
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Failed when setting default values for ctrl %d\n",
- + c);
- + break;
- + }
- + }
- + }
- + return ret;
- +}
- +
- +int set_framerate_params(struct bm2835_mmal_dev *dev)
- +{
- + struct mmal_parameter_fps_range fps_range;
- + int ret;
- +
- + if ((dev->exposure_mode_active != MMAL_PARAM_EXPOSUREMODE_OFF) &&
- + (dev->exp_auto_priority)) {
- + /* Variable FPS. Define min FPS as 1fps.
- + * Max as max defined FPS.
- + */
- + fps_range.fps_low.num = 1;
- + fps_range.fps_low.den = 1;
- + fps_range.fps_high.num = dev->capture.timeperframe.denominator;
- + fps_range.fps_high.den = dev->capture.timeperframe.numerator;
- + } else {
- + /* Fixed FPS - set min and max to be the same */
- + fps_range.fps_low.num = fps_range.fps_high.num =
- + dev->capture.timeperframe.denominator;
- + fps_range.fps_low.den = fps_range.fps_high.den =
- + dev->capture.timeperframe.numerator;
- + }
- +
- + v4l2_dbg(1, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Set fps range to %d/%d to %d/%d\n",
- + fps_range.fps_low.num,
- + fps_range.fps_low.den,
- + fps_range.fps_high.num,
- + fps_range.fps_high.den
- + );
- +
- + ret = vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_PREVIEW],
- + MMAL_PARAMETER_FPS_RANGE,
- + &fps_range, sizeof(fps_range));
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_VIDEO],
- + MMAL_PARAMETER_FPS_RANGE,
- + &fps_range, sizeof(fps_range));
- + ret += vchiq_mmal_port_parameter_set(dev->instance,
- + &dev->component[MMAL_COMPONENT_CAMERA]->
- + output[MMAL_CAMERA_PORT_CAPTURE],
- + MMAL_PARAMETER_FPS_RANGE,
- + &fps_range, sizeof(fps_range));
- + if (ret)
- + v4l2_dbg(0, bcm2835_v4l2_debug, &dev->v4l2_dev,
- + "Failed to set fps ret %d\n",
- + ret);
- +
- + return ret;
- +
- +}
- +
- +int bm2835_mmal_init_controls(struct bm2835_mmal_dev *dev,
- + struct v4l2_ctrl_handler *hdl)
- +{
- + int c;
- + const struct bm2835_mmal_v4l2_ctrl *ctrl;
- +
- + v4l2_ctrl_handler_init(hdl, V4L2_CTRL_COUNT);
- +
- + for (c = 0; c < V4L2_CTRL_COUNT; c++) {
- + ctrl = &v4l2_ctrls[c];
- +
- + switch (ctrl->type) {
- + case MMAL_CONTROL_TYPE_STD:
- + dev->ctrls[c] = v4l2_ctrl_new_std(hdl,
- + &bm2835_mmal_ctrl_ops, ctrl->id,
- + ctrl->min, ctrl->max, ctrl->step, ctrl->def);
- + break;
- +
- + case MMAL_CONTROL_TYPE_STD_MENU:
- + {
- + int mask = ctrl->min;
- +
- + if (ctrl->id == V4L2_CID_SCENE_MODE) {
- + /* Special handling to work out the mask
- + * value based on the scene_configs array
- + * at runtime. Reduces the chance of
- + * mismatches.
- + */
- + int i;
- + mask = 1<<V4L2_SCENE_MODE_NONE;
- + for (i = 0;
- + i < ARRAY_SIZE(scene_configs);
- + i++) {
- + mask |= 1<<scene_configs[i].v4l2_scene;
- + }
- + mask = ~mask;
- + }
- +
- + dev->ctrls[c] = v4l2_ctrl_new_std_menu(hdl,
- + &bm2835_mmal_ctrl_ops, ctrl->id,
- + ctrl->max, mask, ctrl->def);
- + break;
- + }
- +
- + case MMAL_CONTROL_TYPE_INT_MENU:
- + dev->ctrls[c] = v4l2_ctrl_new_int_menu(hdl,
- + &bm2835_mmal_ctrl_ops, ctrl->id,
- + ctrl->max, ctrl->def, ctrl->imenu);
- + break;
- +
- + case MMAL_CONTROL_TYPE_CLUSTER:
- + /* skip this entry when constructing controls */
- + continue;
- + }
- +
- + if (hdl->error)
- + break;
- +
- + dev->ctrls[c]->priv = (void *)ctrl;
- + }
- +
- + if (hdl->error) {
- + pr_err("error adding control %d/%d id 0x%x\n", c,
- + V4L2_CTRL_COUNT, ctrl->id);
- + return hdl->error;
- + }
- +
- + for (c = 0; c < V4L2_CTRL_COUNT; c++) {
- + ctrl = &v4l2_ctrls[c];
- +
- + switch (ctrl->type) {
- + case MMAL_CONTROL_TYPE_CLUSTER:
- + v4l2_ctrl_auto_cluster(ctrl->min,
- + &dev->ctrls[c+1],
- + ctrl->max,
- + ctrl->def);
- + break;
- +
- + case MMAL_CONTROL_TYPE_STD:
- + case MMAL_CONTROL_TYPE_STD_MENU:
- + case MMAL_CONTROL_TYPE_INT_MENU:
- + break;
- + }
- +
- + }
- +
- + return 0;
- +}
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-common.h
- @@ -0,0 +1,53 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + *
- + * MMAL structures
- + *
- + */
- +
- +#define MMAL_FOURCC(a, b, c, d) ((a) | (b << 8) | (c << 16) | (d << 24))
- +#define MMAL_MAGIC MMAL_FOURCC('m', 'm', 'a', 'l')
- +
- +/** Special value signalling that time is not known */
- +#define MMAL_TIME_UNKNOWN (1LL<<63)
- +
- +/* mapping between v4l and mmal video modes */
- +struct mmal_fmt {
- + char *name;
- + u32 fourcc; /* v4l2 format id */
- + int flags; /* v4l2 flags field */
- + u32 mmal;
- + int depth;
- + u32 mmal_component; /* MMAL component index to be used to encode */
- + u32 ybbp; /* depth of first Y plane for planar formats */
- +};
- +
- +/* buffer for one video frame */
- +struct mmal_buffer {
- + /* v4l buffer data -- must be first */
- + struct vb2_v4l2_buffer vb;
- +
- + /* list of buffers available */
- + struct list_head list;
- +
- + void *buffer; /* buffer pointer */
- + unsigned long buffer_size; /* size of allocated buffer */
- +};
- +
- +/* */
- +struct mmal_colourfx {
- + s32 enable;
- + u32 u;
- + u32 v;
- +};
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-encodings.h
- @@ -0,0 +1,127 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +#ifndef MMAL_ENCODINGS_H
- +#define MMAL_ENCODINGS_H
- +
- +#define MMAL_ENCODING_H264 MMAL_FOURCC('H', '2', '6', '4')
- +#define MMAL_ENCODING_H263 MMAL_FOURCC('H', '2', '6', '3')
- +#define MMAL_ENCODING_MP4V MMAL_FOURCC('M', 'P', '4', 'V')
- +#define MMAL_ENCODING_MP2V MMAL_FOURCC('M', 'P', '2', 'V')
- +#define MMAL_ENCODING_MP1V MMAL_FOURCC('M', 'P', '1', 'V')
- +#define MMAL_ENCODING_WMV3 MMAL_FOURCC('W', 'M', 'V', '3')
- +#define MMAL_ENCODING_WMV2 MMAL_FOURCC('W', 'M', 'V', '2')
- +#define MMAL_ENCODING_WMV1 MMAL_FOURCC('W', 'M', 'V', '1')
- +#define MMAL_ENCODING_WVC1 MMAL_FOURCC('W', 'V', 'C', '1')
- +#define MMAL_ENCODING_VP8 MMAL_FOURCC('V', 'P', '8', ' ')
- +#define MMAL_ENCODING_VP7 MMAL_FOURCC('V', 'P', '7', ' ')
- +#define MMAL_ENCODING_VP6 MMAL_FOURCC('V', 'P', '6', ' ')
- +#define MMAL_ENCODING_THEORA MMAL_FOURCC('T', 'H', 'E', 'O')
- +#define MMAL_ENCODING_SPARK MMAL_FOURCC('S', 'P', 'R', 'K')
- +#define MMAL_ENCODING_MJPEG MMAL_FOURCC('M', 'J', 'P', 'G')
- +
- +#define MMAL_ENCODING_JPEG MMAL_FOURCC('J', 'P', 'E', 'G')
- +#define MMAL_ENCODING_GIF MMAL_FOURCC('G', 'I', 'F', ' ')
- +#define MMAL_ENCODING_PNG MMAL_FOURCC('P', 'N', 'G', ' ')
- +#define MMAL_ENCODING_PPM MMAL_FOURCC('P', 'P', 'M', ' ')
- +#define MMAL_ENCODING_TGA MMAL_FOURCC('T', 'G', 'A', ' ')
- +#define MMAL_ENCODING_BMP MMAL_FOURCC('B', 'M', 'P', ' ')
- +
- +#define MMAL_ENCODING_I420 MMAL_FOURCC('I', '4', '2', '0')
- +#define MMAL_ENCODING_I420_SLICE MMAL_FOURCC('S', '4', '2', '0')
- +#define MMAL_ENCODING_YV12 MMAL_FOURCC('Y', 'V', '1', '2')
- +#define MMAL_ENCODING_I422 MMAL_FOURCC('I', '4', '2', '2')
- +#define MMAL_ENCODING_I422_SLICE MMAL_FOURCC('S', '4', '2', '2')
- +#define MMAL_ENCODING_YUYV MMAL_FOURCC('Y', 'U', 'Y', 'V')
- +#define MMAL_ENCODING_YVYU MMAL_FOURCC('Y', 'V', 'Y', 'U')
- +#define MMAL_ENCODING_UYVY MMAL_FOURCC('U', 'Y', 'V', 'Y')
- +#define MMAL_ENCODING_VYUY MMAL_FOURCC('V', 'Y', 'U', 'Y')
- +#define MMAL_ENCODING_NV12 MMAL_FOURCC('N', 'V', '1', '2')
- +#define MMAL_ENCODING_NV21 MMAL_FOURCC('N', 'V', '2', '1')
- +#define MMAL_ENCODING_ARGB MMAL_FOURCC('A', 'R', 'G', 'B')
- +#define MMAL_ENCODING_RGBA MMAL_FOURCC('R', 'G', 'B', 'A')
- +#define MMAL_ENCODING_ABGR MMAL_FOURCC('A', 'B', 'G', 'R')
- +#define MMAL_ENCODING_BGRA MMAL_FOURCC('B', 'G', 'R', 'A')
- +#define MMAL_ENCODING_RGB16 MMAL_FOURCC('R', 'G', 'B', '2')
- +#define MMAL_ENCODING_RGB24 MMAL_FOURCC('R', 'G', 'B', '3')
- +#define MMAL_ENCODING_RGB32 MMAL_FOURCC('R', 'G', 'B', '4')
- +#define MMAL_ENCODING_BGR16 MMAL_FOURCC('B', 'G', 'R', '2')
- +#define MMAL_ENCODING_BGR24 MMAL_FOURCC('B', 'G', 'R', '3')
- +#define MMAL_ENCODING_BGR32 MMAL_FOURCC('B', 'G', 'R', '4')
- +
- +/** SAND Video (YUVUV128) format, native format understood by VideoCore.
- + * This format is *not* opaque - if requested you will receive full frames
- + * of YUV_UV video.
- + */
- +#define MMAL_ENCODING_YUVUV128 MMAL_FOURCC('S', 'A', 'N', 'D')
- +
- +/** VideoCore opaque image format, image handles are returned to
- + * the host but not the actual image data.
- + */
- +#define MMAL_ENCODING_OPAQUE MMAL_FOURCC('O', 'P', 'Q', 'V')
- +
- +/** An EGL image handle
- + */
- +#define MMAL_ENCODING_EGL_IMAGE MMAL_FOURCC('E', 'G', 'L', 'I')
- +
- +/* }@ */
- +
- +/** \name Pre-defined audio encodings */
- +/* @{ */
- +#define MMAL_ENCODING_PCM_UNSIGNED_BE MMAL_FOURCC('P', 'C', 'M', 'U')
- +#define MMAL_ENCODING_PCM_UNSIGNED_LE MMAL_FOURCC('p', 'c', 'm', 'u')
- +#define MMAL_ENCODING_PCM_SIGNED_BE MMAL_FOURCC('P', 'C', 'M', 'S')
- +#define MMAL_ENCODING_PCM_SIGNED_LE MMAL_FOURCC('p', 'c', 'm', 's')
- +#define MMAL_ENCODING_PCM_FLOAT_BE MMAL_FOURCC('P', 'C', 'M', 'F')
- +#define MMAL_ENCODING_PCM_FLOAT_LE MMAL_FOURCC('p', 'c', 'm', 'f')
- +
- +/* Pre-defined H264 encoding variants */
- +
- +/** ISO 14496-10 Annex B byte stream format */
- +#define MMAL_ENCODING_VARIANT_H264_DEFAULT 0
- +/** ISO 14496-15 AVC stream format */
- +#define MMAL_ENCODING_VARIANT_H264_AVC1 MMAL_FOURCC('A', 'V', 'C', '1')
- +/** Implicitly delineated NAL units without emulation prevention */
- +#define MMAL_ENCODING_VARIANT_H264_RAW MMAL_FOURCC('R', 'A', 'W', ' ')
- +
- +
- +/** \defgroup MmalColorSpace List of pre-defined video color spaces
- + * This defines a list of common color spaces. This list isn't exhaustive and
- + * is only provided as a convenience to avoid clients having to use FourCC
- + * codes directly. However components are allowed to define and use their own
- + * FourCC codes.
- + */
- +/* @{ */
- +
- +/** Unknown color space */
- +#define MMAL_COLOR_SPACE_UNKNOWN 0
- +/** ITU-R BT.601-5 [SDTV] */
- +#define MMAL_COLOR_SPACE_ITUR_BT601 MMAL_FOURCC('Y', '6', '0', '1')
- +/** ITU-R BT.709-3 [HDTV] */
- +#define MMAL_COLOR_SPACE_ITUR_BT709 MMAL_FOURCC('Y', '7', '0', '9')
- +/** JPEG JFIF */
- +#define MMAL_COLOR_SPACE_JPEG_JFIF MMAL_FOURCC('Y', 'J', 'F', 'I')
- +/** Title 47 Code of Federal Regulations (2003) 73.682 (a) (20) */
- +#define MMAL_COLOR_SPACE_FCC MMAL_FOURCC('Y', 'F', 'C', 'C')
- +/** Society of Motion Picture and Television Engineers 240M (1999) */
- +#define MMAL_COLOR_SPACE_SMPTE240M MMAL_FOURCC('Y', '2', '4', '0')
- +/** ITU-R BT.470-2 System M */
- +#define MMAL_COLOR_SPACE_BT470_2_M MMAL_FOURCC('Y', '_', '_', 'M')
- +/** ITU-R BT.470-2 System BG */
- +#define MMAL_COLOR_SPACE_BT470_2_BG MMAL_FOURCC('Y', '_', 'B', 'G')
- +/** JPEG JFIF, but with 16..255 luma */
- +#define MMAL_COLOR_SPACE_JFIF_Y16_255 MMAL_FOURCC('Y', 'Y', '1', '6')
- +/* @} MmalColorSpace List */
- +
- +#endif /* MMAL_ENCODINGS_H */
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-msg-common.h
- @@ -0,0 +1,50 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +#ifndef MMAL_MSG_COMMON_H
- +#define MMAL_MSG_COMMON_H
- +
- +enum mmal_msg_status {
- + MMAL_MSG_STATUS_SUCCESS = 0, /**< Success */
- + MMAL_MSG_STATUS_ENOMEM, /**< Out of memory */
- + MMAL_MSG_STATUS_ENOSPC, /**< Out of resources other than memory */
- + MMAL_MSG_STATUS_EINVAL, /**< Argument is invalid */
- + MMAL_MSG_STATUS_ENOSYS, /**< Function not implemented */
- + MMAL_MSG_STATUS_ENOENT, /**< No such file or directory */
- + MMAL_MSG_STATUS_ENXIO, /**< No such device or address */
- + MMAL_MSG_STATUS_EIO, /**< I/O error */
- + MMAL_MSG_STATUS_ESPIPE, /**< Illegal seek */
- + MMAL_MSG_STATUS_ECORRUPT, /**< Data is corrupt \attention */
- + MMAL_MSG_STATUS_ENOTREADY, /**< Component is not ready */
- + MMAL_MSG_STATUS_ECONFIG, /**< Component is not configured */
- + MMAL_MSG_STATUS_EISCONN, /**< Port is already connected */
- + MMAL_MSG_STATUS_ENOTCONN, /**< Port is disconnected */
- + MMAL_MSG_STATUS_EAGAIN, /**< Resource temporarily unavailable. */
- + MMAL_MSG_STATUS_EFAULT, /**< Bad address */
- +};
- +
- +struct mmal_rect {
- + s32 x; /**< x coordinate (from left) */
- + s32 y; /**< y coordinate (from top) */
- + s32 width; /**< width */
- + s32 height; /**< height */
- +};
- +
- +struct mmal_rational {
- + s32 num; /**< Numerator */
- + s32 den; /**< Denominator */
- +};
- +
- +#endif /* MMAL_MSG_COMMON_H */
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-msg-format.h
- @@ -0,0 +1,81 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +#ifndef MMAL_MSG_FORMAT_H
- +#define MMAL_MSG_FORMAT_H
- +
- +#include "mmal-msg-common.h"
- +
- +/* MMAL_ES_FORMAT_T */
- +
- +
- +struct mmal_audio_format {
- + u32 channels; /**< Number of audio channels */
- + u32 sample_rate; /**< Sample rate */
- +
- + u32 bits_per_sample; /**< Bits per sample */
- + u32 block_align; /**< Size of a block of data */
- +};
- +
- +struct mmal_video_format {
- + u32 width; /**< Width of frame in pixels */
- + u32 height; /**< Height of frame in rows of pixels */
- + struct mmal_rect crop; /**< Visible region of the frame */
- + struct mmal_rational frame_rate; /**< Frame rate */
- + struct mmal_rational par; /**< Pixel aspect ratio */
- +
- + /* FourCC specifying the color space of the video stream. See the
- + * \ref MmalColorSpace "pre-defined color spaces" for some examples.
- + */
- + u32 color_space;
- +};
- +
- +struct mmal_subpicture_format {
- + u32 x_offset;
- + u32 y_offset;
- +};
- +
- +union mmal_es_specific_format {
- + struct mmal_audio_format audio;
- + struct mmal_video_format video;
- + struct mmal_subpicture_format subpicture;
- +};
- +
- +/** Definition of an elementary stream format (MMAL_ES_FORMAT_T) */
- +struct mmal_es_format {
- + u32 type; /* enum mmal_es_type */
- +
- + u32 encoding; /* FourCC specifying encoding of the elementary stream.*/
- + u32 encoding_variant; /* FourCC specifying the specific
- + * encoding variant of the elementary
- + * stream.
- + */
- +
- + union mmal_es_specific_format *es; /* TODO: pointers in
- + * message serialisation?!?
- + */
- + /* Type specific
- + * information for the
- + * elementary stream
- + */
- +
- + u32 bitrate; /**< Bitrate in bits per second */
- + u32 flags; /**< Flags describing properties of the elementary stream. */
- +
- + u32 extradata_size; /**< Size of the codec specific data */
- + u8 *extradata; /**< Codec specific data */
- +};
- +
- +#endif /* MMAL_MSG_FORMAT_H */
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-msg-port.h
- @@ -0,0 +1,107 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +/* MMAL_PORT_TYPE_T */
- +enum mmal_port_type {
- + MMAL_PORT_TYPE_UNKNOWN = 0, /**< Unknown port type */
- + MMAL_PORT_TYPE_CONTROL, /**< Control port */
- + MMAL_PORT_TYPE_INPUT, /**< Input port */
- + MMAL_PORT_TYPE_OUTPUT, /**< Output port */
- + MMAL_PORT_TYPE_CLOCK, /**< Clock port */
- +};
- +
- +/** The port is pass-through and doesn't need buffer headers allocated */
- +#define MMAL_PORT_CAPABILITY_PASSTHROUGH 0x01
- +/** The port wants to allocate the buffer payloads.
- + * This signals a preference that payload allocation should be done
- + * on this port for efficiency reasons. */
- +#define MMAL_PORT_CAPABILITY_ALLOCATION 0x02
- +/** The port supports format change events.
- + * This applies to input ports and is used to let the client know
- + * whether the port supports being reconfigured via a format
- + * change event (i.e. without having to disable the port). */
- +#define MMAL_PORT_CAPABILITY_SUPPORTS_EVENT_FORMAT_CHANGE 0x04
- +
- +/* mmal port structure (MMAL_PORT_T)
- + *
- + * most elements are informational only, the pointer values for
- + * interogation messages are generally provided as additional
- + * strucures within the message. When used to set values only teh
- + * buffer_num, buffer_size and userdata parameters are writable.
- + */
- +struct mmal_port {
- + void *priv; /* Private member used by the framework */
- + const char *name; /* Port name. Used for debugging purposes (RO) */
- +
- + u32 type; /* Type of the port (RO) enum mmal_port_type */
- + u16 index; /* Index of the port in its type list (RO) */
- + u16 index_all; /* Index of the port in the list of all ports (RO) */
- +
- + u32 is_enabled; /* Indicates whether the port is enabled or not (RO) */
- + struct mmal_es_format *format; /* Format of the elementary stream */
- +
- + u32 buffer_num_min; /* Minimum number of buffers the port
- + * requires (RO). This is set by the
- + * component.
- + */
- +
- + u32 buffer_size_min; /* Minimum size of buffers the port
- + * requires (RO). This is set by the
- + * component.
- + */
- +
- + u32 buffer_alignment_min; /* Minimum alignment requirement for
- + * the buffers (RO). A value of
- + * zero means no special alignment
- + * requirements. This is set by the
- + * component.
- + */
- +
- + u32 buffer_num_recommended; /* Number of buffers the port
- + * recommends for optimal
- + * performance (RO). A value of
- + * zero means no special
- + * recommendation. This is set
- + * by the component.
- + */
- +
- + u32 buffer_size_recommended; /* Size of buffers the port
- + * recommends for optimal
- + * performance (RO). A value of
- + * zero means no special
- + * recommendation. This is set
- + * by the component.
- + */
- +
- + u32 buffer_num; /* Actual number of buffers the port will use.
- + * This is set by the client.
- + */
- +
- + u32 buffer_size; /* Actual maximum size of the buffers that
- + * will be sent to the port. This is set by
- + * the client.
- + */
- +
- + void *component; /* Component this port belongs to (Read Only) */
- +
- + void *userdata; /* Field reserved for use by the client */
- +
- + u32 capabilities; /* Flags describing the capabilities of a
- + * port (RO). Bitwise combination of \ref
- + * portcapabilities "Port capabilities"
- + * values.
- + */
- +
- +};
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-msg.h
- @@ -0,0 +1,404 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +/* all the data structures which serialise the MMAL protocol. note
- + * these are directly mapped onto the recived message data.
- + *
- + * BEWARE: They seem to *assume* pointers are u32 and that there is no
- + * structure padding!
- + *
- + * NOTE: this implementation uses kernel types to ensure sizes. Rather
- + * than assigning values to enums to force their size the
- + * implementation uses fixed size types and not the enums (though the
- + * comments have the actual enum type
- + */
- +
- +#define VC_MMAL_VER 15
- +#define VC_MMAL_MIN_VER 10
- +#define VC_MMAL_SERVER_NAME MAKE_FOURCC("mmal")
- +
- +/* max total message size is 512 bytes */
- +#define MMAL_MSG_MAX_SIZE 512
- +/* with six 32bit header elements max payload is therefore 488 bytes */
- +#define MMAL_MSG_MAX_PAYLOAD 488
- +
- +#include "mmal-msg-common.h"
- +#include "mmal-msg-format.h"
- +#include "mmal-msg-port.h"
- +
- +enum mmal_msg_type {
- + MMAL_MSG_TYPE_QUIT = 1,
- + MMAL_MSG_TYPE_SERVICE_CLOSED,
- + MMAL_MSG_TYPE_GET_VERSION,
- + MMAL_MSG_TYPE_COMPONENT_CREATE,
- + MMAL_MSG_TYPE_COMPONENT_DESTROY, /* 5 */
- + MMAL_MSG_TYPE_COMPONENT_ENABLE,
- + MMAL_MSG_TYPE_COMPONENT_DISABLE,
- + MMAL_MSG_TYPE_PORT_INFO_GET,
- + MMAL_MSG_TYPE_PORT_INFO_SET,
- + MMAL_MSG_TYPE_PORT_ACTION, /* 10 */
- + MMAL_MSG_TYPE_BUFFER_FROM_HOST,
- + MMAL_MSG_TYPE_BUFFER_TO_HOST,
- + MMAL_MSG_TYPE_GET_STATS,
- + MMAL_MSG_TYPE_PORT_PARAMETER_SET,
- + MMAL_MSG_TYPE_PORT_PARAMETER_GET, /* 15 */
- + MMAL_MSG_TYPE_EVENT_TO_HOST,
- + MMAL_MSG_TYPE_GET_CORE_STATS_FOR_PORT,
- + MMAL_MSG_TYPE_OPAQUE_ALLOCATOR,
- + MMAL_MSG_TYPE_CONSUME_MEM,
- + MMAL_MSG_TYPE_LMK, /* 20 */
- + MMAL_MSG_TYPE_OPAQUE_ALLOCATOR_DESC,
- + MMAL_MSG_TYPE_DRM_GET_LHS32,
- + MMAL_MSG_TYPE_DRM_GET_TIME,
- + MMAL_MSG_TYPE_BUFFER_FROM_HOST_ZEROLEN,
- + MMAL_MSG_TYPE_PORT_FLUSH, /* 25 */
- + MMAL_MSG_TYPE_HOST_LOG,
- + MMAL_MSG_TYPE_MSG_LAST
- +};
- +
- +/* port action request messages differ depending on the action type */
- +enum mmal_msg_port_action_type {
- + MMAL_MSG_PORT_ACTION_TYPE_UNKNOWN = 0, /* Unkown action */
- + MMAL_MSG_PORT_ACTION_TYPE_ENABLE, /* Enable a port */
- + MMAL_MSG_PORT_ACTION_TYPE_DISABLE, /* Disable a port */
- + MMAL_MSG_PORT_ACTION_TYPE_FLUSH, /* Flush a port */
- + MMAL_MSG_PORT_ACTION_TYPE_CONNECT, /* Connect ports */
- + MMAL_MSG_PORT_ACTION_TYPE_DISCONNECT, /* Disconnect ports */
- + MMAL_MSG_PORT_ACTION_TYPE_SET_REQUIREMENTS, /* Set buffer requirements*/
- +};
- +
- +struct mmal_msg_header {
- + u32 magic;
- + u32 type; /** enum mmal_msg_type */
- +
- + /* Opaque handle to the control service */
- + struct mmal_control_service *control_service;
- +
- + struct mmal_msg_context *context; /** a u32 per message context */
- + u32 status; /** The status of the vchiq operation */
- + u32 padding;
- +};
- +
- +/* Send from VC to host to report version */
- +struct mmal_msg_version {
- + u32 flags;
- + u32 major;
- + u32 minor;
- + u32 minimum;
- +};
- +
- +/* request to VC to create component */
- +struct mmal_msg_component_create {
- + void *client_component; /* component context */
- + char name[128];
- + u32 pid; /* For debug */
- +};
- +
- +/* reply from VC to component creation request */
- +struct mmal_msg_component_create_reply {
- + u32 status; /** enum mmal_msg_status - how does this differ to
- + * the one in the header?
- + */
- + u32 component_handle; /* VideoCore handle for component */
- + u32 input_num; /* Number of input ports */
- + u32 output_num; /* Number of output ports */
- + u32 clock_num; /* Number of clock ports */
- +};
- +
- +/* request to VC to destroy a component */
- +struct mmal_msg_component_destroy {
- + u32 component_handle;
- +};
- +
- +struct mmal_msg_component_destroy_reply {
- + u32 status; /** The component destruction status */
- +};
- +
- +
- +/* request and reply to VC to enable a component */
- +struct mmal_msg_component_enable {
- + u32 component_handle;
- +};
- +
- +struct mmal_msg_component_enable_reply {
- + u32 status; /** The component enable status */
- +};
- +
- +
- +/* request and reply to VC to disable a component */
- +struct mmal_msg_component_disable {
- + u32 component_handle;
- +};
- +
- +struct mmal_msg_component_disable_reply {
- + u32 status; /** The component disable status */
- +};
- +
- +/* request to VC to get port information */
- +struct mmal_msg_port_info_get {
- + u32 component_handle; /* component handle port is associated with */
- + u32 port_type; /* enum mmal_msg_port_type */
- + u32 index; /* port index to query */
- +};
- +
- +/* reply from VC to get port info request */
- +struct mmal_msg_port_info_get_reply {
- + u32 status; /** enum mmal_msg_status */
- + u32 component_handle; /* component handle port is associated with */
- + u32 port_type; /* enum mmal_msg_port_type */
- + u32 port_index; /* port indexed in query */
- + s32 found; /* unused */
- + u32 port_handle; /**< Handle to use for this port */
- + struct mmal_port port;
- + struct mmal_es_format format; /* elementry stream format */
- + union mmal_es_specific_format es; /* es type specific data */
- + u8 extradata[MMAL_FORMAT_EXTRADATA_MAX_SIZE]; /* es extra data */
- +};
- +
- +/* request to VC to set port information */
- +struct mmal_msg_port_info_set {
- + u32 component_handle;
- + u32 port_type; /* enum mmal_msg_port_type */
- + u32 port_index; /* port indexed in query */
- + struct mmal_port port;
- + struct mmal_es_format format;
- + union mmal_es_specific_format es;
- + u8 extradata[MMAL_FORMAT_EXTRADATA_MAX_SIZE];
- +};
- +
- +/* reply from VC to port info set request */
- +struct mmal_msg_port_info_set_reply {
- + u32 status;
- + u32 component_handle; /* component handle port is associated with */
- + u32 port_type; /* enum mmal_msg_port_type */
- + u32 index; /* port indexed in query */
- + s32 found; /* unused */
- + u32 port_handle; /**< Handle to use for this port */
- + struct mmal_port port;
- + struct mmal_es_format format;
- + union mmal_es_specific_format es;
- + u8 extradata[MMAL_FORMAT_EXTRADATA_MAX_SIZE];
- +};
- +
- +
- +/* port action requests that take a mmal_port as a parameter */
- +struct mmal_msg_port_action_port {
- + u32 component_handle;
- + u32 port_handle;
- + u32 action; /* enum mmal_msg_port_action_type */
- + struct mmal_port port;
- +};
- +
- +/* port action requests that take handles as a parameter */
- +struct mmal_msg_port_action_handle {
- + u32 component_handle;
- + u32 port_handle;
- + u32 action; /* enum mmal_msg_port_action_type */
- + u32 connect_component_handle;
- + u32 connect_port_handle;
- +};
- +
- +struct mmal_msg_port_action_reply {
- + u32 status; /** The port action operation status */
- +};
- +
- +
- +
- +
- +/* MMAL buffer transfer */
- +
- +/** Size of space reserved in a buffer message for short messages. */
- +#define MMAL_VC_SHORT_DATA 128
- +
- +/** Signals that the current payload is the end of the stream of data */
- +#define MMAL_BUFFER_HEADER_FLAG_EOS (1<<0)
- +/** Signals that the start of the current payload starts a frame */
- +#define MMAL_BUFFER_HEADER_FLAG_FRAME_START (1<<1)
- +/** Signals that the end of the current payload ends a frame */
- +#define MMAL_BUFFER_HEADER_FLAG_FRAME_END (1<<2)
- +/** Signals that the current payload contains only complete frames (>1) */
- +#define MMAL_BUFFER_HEADER_FLAG_FRAME \
- + (MMAL_BUFFER_HEADER_FLAG_FRAME_START|MMAL_BUFFER_HEADER_FLAG_FRAME_END)
- +/** Signals that the current payload is a keyframe (i.e. self decodable) */
- +#define MMAL_BUFFER_HEADER_FLAG_KEYFRAME (1<<3)
- +/** Signals a discontinuity in the stream of data (e.g. after a seek).
- + * Can be used for instance by a decoder to reset its state */
- +#define MMAL_BUFFER_HEADER_FLAG_DISCONTINUITY (1<<4)
- +/** Signals a buffer containing some kind of config data for the component
- + * (e.g. codec config data) */
- +#define MMAL_BUFFER_HEADER_FLAG_CONFIG (1<<5)
- +/** Signals an encrypted payload */
- +#define MMAL_BUFFER_HEADER_FLAG_ENCRYPTED (1<<6)
- +/** Signals a buffer containing side information */
- +#define MMAL_BUFFER_HEADER_FLAG_CODECSIDEINFO (1<<7)
- +/** Signals a buffer which is the snapshot/postview image from a stills
- + * capture
- + */
- +#define MMAL_BUFFER_HEADER_FLAGS_SNAPSHOT (1<<8)
- +/** Signals a buffer which contains data known to be corrupted */
- +#define MMAL_BUFFER_HEADER_FLAG_CORRUPTED (1<<9)
- +/** Signals that a buffer failed to be transmitted */
- +#define MMAL_BUFFER_HEADER_FLAG_TRANSMISSION_FAILED (1<<10)
- +
- +struct mmal_driver_buffer {
- + u32 magic;
- + u32 component_handle;
- + u32 port_handle;
- + void *client_context;
- +};
- +
- +/* buffer header */
- +struct mmal_buffer_header {
- + struct mmal_buffer_header *next; /* next header */
- + void *priv; /* framework private data */
- + u32 cmd;
- + void *data;
- + u32 alloc_size;
- + u32 length;
- + u32 offset;
- + u32 flags;
- + s64 pts;
- + s64 dts;
- + void *type;
- + void *user_data;
- +};
- +
- +struct mmal_buffer_header_type_specific {
- + union {
- + struct {
- + u32 planes;
- + u32 offset[4];
- + u32 pitch[4];
- + u32 flags;
- + } video;
- + } u;
- +};
- +
- +struct mmal_msg_buffer_from_host {
- + /* The front 32 bytes of the buffer header are copied
- + * back to us in the reply to allow for context. This
- + * area is used to store two mmal_driver_buffer structures to
- + * allow for multiple concurrent service users.
- + */
- + /* control data */
- + struct mmal_driver_buffer drvbuf;
- +
- + /* referenced control data for passthrough buffer management */
- + struct mmal_driver_buffer drvbuf_ref;
- + struct mmal_buffer_header buffer_header; /* buffer header itself */
- + struct mmal_buffer_header_type_specific buffer_header_type_specific;
- + s32 is_zero_copy;
- + s32 has_reference;
- +
- + /** allows short data to be xfered in control message */
- + u32 payload_in_message;
- + u8 short_data[MMAL_VC_SHORT_DATA];
- +};
- +
- +
- +/* port parameter setting */
- +
- +#define MMAL_WORKER_PORT_PARAMETER_SPACE 96
- +
- +struct mmal_msg_port_parameter_set {
- + u32 component_handle; /* component */
- + u32 port_handle; /* port */
- + u32 id; /* Parameter ID */
- + u32 size; /* Parameter size */
- + uint32_t value[MMAL_WORKER_PORT_PARAMETER_SPACE];
- +};
- +
- +struct mmal_msg_port_parameter_set_reply {
- + u32 status; /** enum mmal_msg_status todo: how does this
- + * differ to the one in the header?
- + */
- +};
- +
- +/* port parameter getting */
- +
- +struct mmal_msg_port_parameter_get {
- + u32 component_handle; /* component */
- + u32 port_handle; /* port */
- + u32 id; /* Parameter ID */
- + u32 size; /* Parameter size */
- +};
- +
- +struct mmal_msg_port_parameter_get_reply {
- + u32 status; /* Status of mmal_port_parameter_get call */
- + u32 id; /* Parameter ID */
- + u32 size; /* Parameter size */
- + uint32_t value[MMAL_WORKER_PORT_PARAMETER_SPACE];
- +};
- +
- +/* event messages */
- +#define MMAL_WORKER_EVENT_SPACE 256
- +
- +struct mmal_msg_event_to_host {
- + void *client_component; /* component context */
- +
- + u32 port_type;
- + u32 port_num;
- +
- + u32 cmd;
- + u32 length;
- + u8 data[MMAL_WORKER_EVENT_SPACE];
- + struct mmal_buffer_header *delayed_buffer;
- +};
- +
- +/* all mmal messages are serialised through this structure */
- +struct mmal_msg {
- + /* header */
- + struct mmal_msg_header h;
- + /* payload */
- + union {
- + struct mmal_msg_version version;
- +
- + struct mmal_msg_component_create component_create;
- + struct mmal_msg_component_create_reply component_create_reply;
- +
- + struct mmal_msg_component_destroy component_destroy;
- + struct mmal_msg_component_destroy_reply component_destroy_reply;
- +
- + struct mmal_msg_component_enable component_enable;
- + struct mmal_msg_component_enable_reply component_enable_reply;
- +
- + struct mmal_msg_component_disable component_disable;
- + struct mmal_msg_component_disable_reply component_disable_reply;
- +
- + struct mmal_msg_port_info_get port_info_get;
- + struct mmal_msg_port_info_get_reply port_info_get_reply;
- +
- + struct mmal_msg_port_info_set port_info_set;
- + struct mmal_msg_port_info_set_reply port_info_set_reply;
- +
- + struct mmal_msg_port_action_port port_action_port;
- + struct mmal_msg_port_action_handle port_action_handle;
- + struct mmal_msg_port_action_reply port_action_reply;
- +
- + struct mmal_msg_buffer_from_host buffer_from_host;
- +
- + struct mmal_msg_port_parameter_set port_parameter_set;
- + struct mmal_msg_port_parameter_set_reply
- + port_parameter_set_reply;
- + struct mmal_msg_port_parameter_get
- + port_parameter_get;
- + struct mmal_msg_port_parameter_get_reply
- + port_parameter_get_reply;
- +
- + struct mmal_msg_event_to_host event_to_host;
- +
- + u8 payload[MMAL_MSG_MAX_PAYLOAD];
- + } u;
- +};
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-parameters.h
- @@ -0,0 +1,656 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + */
- +
- +/* common parameters */
- +
- +/** @name Parameter groups
- + * Parameters are divided into groups, and then allocated sequentially within
- + * a group using an enum.
- + * @{
- + */
- +
- +/** Common parameter ID group, used with many types of component. */
- +#define MMAL_PARAMETER_GROUP_COMMON (0<<16)
- +/** Camera-specific parameter ID group. */
- +#define MMAL_PARAMETER_GROUP_CAMERA (1<<16)
- +/** Video-specific parameter ID group. */
- +#define MMAL_PARAMETER_GROUP_VIDEO (2<<16)
- +/** Audio-specific parameter ID group. */
- +#define MMAL_PARAMETER_GROUP_AUDIO (3<<16)
- +/** Clock-specific parameter ID group. */
- +#define MMAL_PARAMETER_GROUP_CLOCK (4<<16)
- +/** Miracast-specific parameter ID group. */
- +#define MMAL_PARAMETER_GROUP_MIRACAST (5<<16)
- +
- +/* Common parameters */
- +enum mmal_parameter_common_type {
- + MMAL_PARAMETER_UNUSED /**< Never a valid parameter ID */
- + = MMAL_PARAMETER_GROUP_COMMON,
- + MMAL_PARAMETER_SUPPORTED_ENCODINGS, /**< MMAL_PARAMETER_ENCODING_T */
- + MMAL_PARAMETER_URI, /**< MMAL_PARAMETER_URI_T */
- +
- + /** MMAL_PARAMETER_CHANGE_EVENT_REQUEST_T */
- + MMAL_PARAMETER_CHANGE_EVENT_REQUEST,
- +
- + /** MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_ZERO_COPY,
- +
- + /**< MMAL_PARAMETER_BUFFER_REQUIREMENTS_T */
- + MMAL_PARAMETER_BUFFER_REQUIREMENTS,
- +
- + MMAL_PARAMETER_STATISTICS, /**< MMAL_PARAMETER_STATISTICS_T */
- + MMAL_PARAMETER_CORE_STATISTICS, /**< MMAL_PARAMETER_CORE_STATISTICS_T */
- + MMAL_PARAMETER_MEM_USAGE, /**< MMAL_PARAMETER_MEM_USAGE_T */
- + MMAL_PARAMETER_BUFFER_FLAG_FILTER, /**< MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_SEEK, /**< MMAL_PARAMETER_SEEK_T */
- + MMAL_PARAMETER_POWERMON_ENABLE, /**< MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_LOGGING, /**< MMAL_PARAMETER_LOGGING_T */
- + MMAL_PARAMETER_SYSTEM_TIME, /**< MMAL_PARAMETER_UINT64_T */
- + MMAL_PARAMETER_NO_IMAGE_PADDING /**< MMAL_PARAMETER_BOOLEAN_T */
- +};
- +
- +/* camera parameters */
- +
- +enum mmal_parameter_camera_type {
- + /* 0 */
- + /** @ref MMAL_PARAMETER_THUMBNAIL_CONFIG_T */
- + MMAL_PARAMETER_THUMBNAIL_CONFIGURATION
- + = MMAL_PARAMETER_GROUP_CAMERA,
- + MMAL_PARAMETER_CAPTURE_QUALITY, /**< Unused? */
- + MMAL_PARAMETER_ROTATION, /**< @ref MMAL_PARAMETER_INT32_T */
- + MMAL_PARAMETER_EXIF_DISABLE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_EXIF, /**< @ref MMAL_PARAMETER_EXIF_T */
- + MMAL_PARAMETER_AWB_MODE, /**< @ref MMAL_PARAM_AWBMODE_T */
- + MMAL_PARAMETER_IMAGE_EFFECT, /**< @ref MMAL_PARAMETER_IMAGEFX_T */
- + MMAL_PARAMETER_COLOUR_EFFECT, /**< @ref MMAL_PARAMETER_COLOURFX_T */
- + MMAL_PARAMETER_FLICKER_AVOID, /**< @ref MMAL_PARAMETER_FLICKERAVOID_T */
- + MMAL_PARAMETER_FLASH, /**< @ref MMAL_PARAMETER_FLASH_T */
- + MMAL_PARAMETER_REDEYE, /**< @ref MMAL_PARAMETER_REDEYE_T */
- + MMAL_PARAMETER_FOCUS, /**< @ref MMAL_PARAMETER_FOCUS_T */
- + MMAL_PARAMETER_FOCAL_LENGTHS, /**< Unused? */
- + MMAL_PARAMETER_EXPOSURE_COMP, /**< @ref MMAL_PARAMETER_INT32_T */
- + MMAL_PARAMETER_ZOOM, /**< @ref MMAL_PARAMETER_SCALEFACTOR_T */
- + MMAL_PARAMETER_MIRROR, /**< @ref MMAL_PARAMETER_MIRROR_T */
- +
- + /* 0x10 */
- + MMAL_PARAMETER_CAMERA_NUM, /**< @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_CAPTURE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_EXPOSURE_MODE, /**< @ref MMAL_PARAMETER_EXPOSUREMODE_T */
- + MMAL_PARAMETER_EXP_METERING_MODE, /**< @ref MMAL_PARAMETER_EXPOSUREMETERINGMODE_T */
- + MMAL_PARAMETER_FOCUS_STATUS, /**< @ref MMAL_PARAMETER_FOCUS_STATUS_T */
- + MMAL_PARAMETER_CAMERA_CONFIG, /**< @ref MMAL_PARAMETER_CAMERA_CONFIG_T */
- + MMAL_PARAMETER_CAPTURE_STATUS, /**< @ref MMAL_PARAMETER_CAPTURE_STATUS_T */
- + MMAL_PARAMETER_FACE_TRACK, /**< @ref MMAL_PARAMETER_FACE_TRACK_T */
- + MMAL_PARAMETER_DRAW_BOX_FACES_AND_FOCUS, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_JPEG_Q_FACTOR, /**< @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_FRAME_RATE, /**< @ref MMAL_PARAMETER_FRAME_RATE_T */
- + MMAL_PARAMETER_USE_STC, /**< @ref MMAL_PARAMETER_CAMERA_STC_MODE_T */
- + MMAL_PARAMETER_CAMERA_INFO, /**< @ref MMAL_PARAMETER_CAMERA_INFO_T */
- + MMAL_PARAMETER_VIDEO_STABILISATION, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_FACE_TRACK_RESULTS, /**< @ref MMAL_PARAMETER_FACE_TRACK_RESULTS_T */
- + MMAL_PARAMETER_ENABLE_RAW_CAPTURE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- +
- + /* 0x20 */
- + MMAL_PARAMETER_DPF_FILE, /**< @ref MMAL_PARAMETER_URI_T */
- + MMAL_PARAMETER_ENABLE_DPF_FILE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_DPF_FAIL_IS_FATAL, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_CAPTURE_MODE, /**< @ref MMAL_PARAMETER_CAPTUREMODE_T */
- + MMAL_PARAMETER_FOCUS_REGIONS, /**< @ref MMAL_PARAMETER_FOCUS_REGIONS_T */
- + MMAL_PARAMETER_INPUT_CROP, /**< @ref MMAL_PARAMETER_INPUT_CROP_T */
- + MMAL_PARAMETER_SENSOR_INFORMATION, /**< @ref MMAL_PARAMETER_SENSOR_INFORMATION_T */
- + MMAL_PARAMETER_FLASH_SELECT, /**< @ref MMAL_PARAMETER_FLASH_SELECT_T */
- + MMAL_PARAMETER_FIELD_OF_VIEW, /**< @ref MMAL_PARAMETER_FIELD_OF_VIEW_T */
- + MMAL_PARAMETER_HIGH_DYNAMIC_RANGE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_DYNAMIC_RANGE_COMPRESSION, /**< @ref MMAL_PARAMETER_DRC_T */
- + MMAL_PARAMETER_ALGORITHM_CONTROL, /**< @ref MMAL_PARAMETER_ALGORITHM_CONTROL_T */
- + MMAL_PARAMETER_SHARPNESS, /**< @ref MMAL_PARAMETER_RATIONAL_T */
- + MMAL_PARAMETER_CONTRAST, /**< @ref MMAL_PARAMETER_RATIONAL_T */
- + MMAL_PARAMETER_BRIGHTNESS, /**< @ref MMAL_PARAMETER_RATIONAL_T */
- + MMAL_PARAMETER_SATURATION, /**< @ref MMAL_PARAMETER_RATIONAL_T */
- +
- + /* 0x30 */
- + MMAL_PARAMETER_ISO, /**< @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_ANTISHAKE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- +
- + /** @ref MMAL_PARAMETER_IMAGEFX_PARAMETERS_T */
- + MMAL_PARAMETER_IMAGE_EFFECT_PARAMETERS,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_CAMERA_BURST_CAPTURE,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_CAMERA_MIN_ISO,
- +
- + /** @ref MMAL_PARAMETER_CAMERA_USE_CASE_T */
- + MMAL_PARAMETER_CAMERA_USE_CASE,
- +
- + /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_CAPTURE_STATS_PASS,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_CAMERA_CUSTOM_SENSOR_CONFIG,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_ENABLE_REGISTER_FILE,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_REGISTER_FAIL_IS_FATAL,
- +
- + /** @ref MMAL_PARAMETER_CONFIGFILE_T */
- + MMAL_PARAMETER_CONFIGFILE_REGISTERS,
- +
- + /** @ref MMAL_PARAMETER_CONFIGFILE_CHUNK_T */
- + MMAL_PARAMETER_CONFIGFILE_CHUNK_REGISTERS,
- + MMAL_PARAMETER_JPEG_ATTACH_LOG, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_ZERO_SHUTTER_LAG, /**< @ref MMAL_PARAMETER_ZEROSHUTTERLAG_T */
- + MMAL_PARAMETER_FPS_RANGE, /**< @ref MMAL_PARAMETER_FPS_RANGE_T */
- + MMAL_PARAMETER_CAPTURE_EXPOSURE_COMP, /**< @ref MMAL_PARAMETER_INT32_T */
- +
- + /* 0x40 */
- + MMAL_PARAMETER_SW_SHARPEN_DISABLE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_FLASH_REQUIRED, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_SW_SATURATION_DISABLE, /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_SHUTTER_SPEED, /**< Takes a @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_CUSTOM_AWB_GAINS, /**< Takes a @ref MMAL_PARAMETER_AWB_GAINS_T */
- +};
- +
- +struct mmal_parameter_rational {
- + s32 num; /**< Numerator */
- + s32 den; /**< Denominator */
- +};
- +
- +enum mmal_parameter_camera_config_timestamp_mode {
- + MMAL_PARAM_TIMESTAMP_MODE_ZERO = 0, /* Always timestamp frames as 0 */
- + MMAL_PARAM_TIMESTAMP_MODE_RAW_STC, /* Use the raw STC value
- + * for the frame timestamp
- + */
- + MMAL_PARAM_TIMESTAMP_MODE_RESET_STC, /* Use the STC timestamp
- + * but subtract the
- + * timestamp of the first
- + * frame sent to give a
- + * zero based timestamp.
- + */
- +};
- +
- +struct mmal_parameter_fps_range {
- + /**< Low end of the permitted framerate range */
- + struct mmal_parameter_rational fps_low;
- + /**< High end of the permitted framerate range */
- + struct mmal_parameter_rational fps_high;
- +};
- +
- +
- +/* camera configuration parameter */
- +struct mmal_parameter_camera_config {
- + /* Parameters for setting up the image pools */
- + u32 max_stills_w; /* Max size of stills capture */
- + u32 max_stills_h;
- + u32 stills_yuv422; /* Allow YUV422 stills capture */
- + u32 one_shot_stills; /* Continuous or one shot stills captures. */
- +
- + u32 max_preview_video_w; /* Max size of the preview or video
- + * capture frames
- + */
- + u32 max_preview_video_h;
- + u32 num_preview_video_frames;
- +
- + /** Sets the height of the circular buffer for stills capture. */
- + u32 stills_capture_circular_buffer_height;
- +
- + /** Allows preview/encode to resume as fast as possible after the stills
- + * input frame has been received, and then processes the still frame in
- + * the background whilst preview/encode has resumed.
- + * Actual mode is controlled by MMAL_PARAMETER_CAPTURE_MODE.
- + */
- + u32 fast_preview_resume;
- +
- + /** Selects algorithm for timestamping frames if
- + * there is no clock component connected.
- + * enum mmal_parameter_camera_config_timestamp_mode
- + */
- + s32 use_stc_timestamp;
- +};
- +
- +
- +enum mmal_parameter_exposuremode {
- + MMAL_PARAM_EXPOSUREMODE_OFF,
- + MMAL_PARAM_EXPOSUREMODE_AUTO,
- + MMAL_PARAM_EXPOSUREMODE_NIGHT,
- + MMAL_PARAM_EXPOSUREMODE_NIGHTPREVIEW,
- + MMAL_PARAM_EXPOSUREMODE_BACKLIGHT,
- + MMAL_PARAM_EXPOSUREMODE_SPOTLIGHT,
- + MMAL_PARAM_EXPOSUREMODE_SPORTS,
- + MMAL_PARAM_EXPOSUREMODE_SNOW,
- + MMAL_PARAM_EXPOSUREMODE_BEACH,
- + MMAL_PARAM_EXPOSUREMODE_VERYLONG,
- + MMAL_PARAM_EXPOSUREMODE_FIXEDFPS,
- + MMAL_PARAM_EXPOSUREMODE_ANTISHAKE,
- + MMAL_PARAM_EXPOSUREMODE_FIREWORKS,
- +};
- +
- +enum mmal_parameter_exposuremeteringmode {
- + MMAL_PARAM_EXPOSUREMETERINGMODE_AVERAGE,
- + MMAL_PARAM_EXPOSUREMETERINGMODE_SPOT,
- + MMAL_PARAM_EXPOSUREMETERINGMODE_BACKLIT,
- + MMAL_PARAM_EXPOSUREMETERINGMODE_MATRIX,
- +};
- +
- +enum mmal_parameter_awbmode {
- + MMAL_PARAM_AWBMODE_OFF,
- + MMAL_PARAM_AWBMODE_AUTO,
- + MMAL_PARAM_AWBMODE_SUNLIGHT,
- + MMAL_PARAM_AWBMODE_CLOUDY,
- + MMAL_PARAM_AWBMODE_SHADE,
- + MMAL_PARAM_AWBMODE_TUNGSTEN,
- + MMAL_PARAM_AWBMODE_FLUORESCENT,
- + MMAL_PARAM_AWBMODE_INCANDESCENT,
- + MMAL_PARAM_AWBMODE_FLASH,
- + MMAL_PARAM_AWBMODE_HORIZON,
- +};
- +
- +enum mmal_parameter_imagefx {
- + MMAL_PARAM_IMAGEFX_NONE,
- + MMAL_PARAM_IMAGEFX_NEGATIVE,
- + MMAL_PARAM_IMAGEFX_SOLARIZE,
- + MMAL_PARAM_IMAGEFX_POSTERIZE,
- + MMAL_PARAM_IMAGEFX_WHITEBOARD,
- + MMAL_PARAM_IMAGEFX_BLACKBOARD,
- + MMAL_PARAM_IMAGEFX_SKETCH,
- + MMAL_PARAM_IMAGEFX_DENOISE,
- + MMAL_PARAM_IMAGEFX_EMBOSS,
- + MMAL_PARAM_IMAGEFX_OILPAINT,
- + MMAL_PARAM_IMAGEFX_HATCH,
- + MMAL_PARAM_IMAGEFX_GPEN,
- + MMAL_PARAM_IMAGEFX_PASTEL,
- + MMAL_PARAM_IMAGEFX_WATERCOLOUR,
- + MMAL_PARAM_IMAGEFX_FILM,
- + MMAL_PARAM_IMAGEFX_BLUR,
- + MMAL_PARAM_IMAGEFX_SATURATION,
- + MMAL_PARAM_IMAGEFX_COLOURSWAP,
- + MMAL_PARAM_IMAGEFX_WASHEDOUT,
- + MMAL_PARAM_IMAGEFX_POSTERISE,
- + MMAL_PARAM_IMAGEFX_COLOURPOINT,
- + MMAL_PARAM_IMAGEFX_COLOURBALANCE,
- + MMAL_PARAM_IMAGEFX_CARTOON,
- +};
- +
- +enum MMAL_PARAM_FLICKERAVOID_T {
- + MMAL_PARAM_FLICKERAVOID_OFF,
- + MMAL_PARAM_FLICKERAVOID_AUTO,
- + MMAL_PARAM_FLICKERAVOID_50HZ,
- + MMAL_PARAM_FLICKERAVOID_60HZ,
- + MMAL_PARAM_FLICKERAVOID_MAX = 0x7FFFFFFF
- +};
- +
- +struct mmal_parameter_awbgains {
- + struct mmal_parameter_rational r_gain; /**< Red gain */
- + struct mmal_parameter_rational b_gain; /**< Blue gain */
- +};
- +
- +/** Manner of video rate control */
- +enum mmal_parameter_rate_control_mode {
- + MMAL_VIDEO_RATECONTROL_DEFAULT,
- + MMAL_VIDEO_RATECONTROL_VARIABLE,
- + MMAL_VIDEO_RATECONTROL_CONSTANT,
- + MMAL_VIDEO_RATECONTROL_VARIABLE_SKIP_FRAMES,
- + MMAL_VIDEO_RATECONTROL_CONSTANT_SKIP_FRAMES
- +};
- +
- +enum mmal_video_profile {
- + MMAL_VIDEO_PROFILE_H263_BASELINE,
- + MMAL_VIDEO_PROFILE_H263_H320CODING,
- + MMAL_VIDEO_PROFILE_H263_BACKWARDCOMPATIBLE,
- + MMAL_VIDEO_PROFILE_H263_ISWV2,
- + MMAL_VIDEO_PROFILE_H263_ISWV3,
- + MMAL_VIDEO_PROFILE_H263_HIGHCOMPRESSION,
- + MMAL_VIDEO_PROFILE_H263_INTERNET,
- + MMAL_VIDEO_PROFILE_H263_INTERLACE,
- + MMAL_VIDEO_PROFILE_H263_HIGHLATENCY,
- + MMAL_VIDEO_PROFILE_MP4V_SIMPLE,
- + MMAL_VIDEO_PROFILE_MP4V_SIMPLESCALABLE,
- + MMAL_VIDEO_PROFILE_MP4V_CORE,
- + MMAL_VIDEO_PROFILE_MP4V_MAIN,
- + MMAL_VIDEO_PROFILE_MP4V_NBIT,
- + MMAL_VIDEO_PROFILE_MP4V_SCALABLETEXTURE,
- + MMAL_VIDEO_PROFILE_MP4V_SIMPLEFACE,
- + MMAL_VIDEO_PROFILE_MP4V_SIMPLEFBA,
- + MMAL_VIDEO_PROFILE_MP4V_BASICANIMATED,
- + MMAL_VIDEO_PROFILE_MP4V_HYBRID,
- + MMAL_VIDEO_PROFILE_MP4V_ADVANCEDREALTIME,
- + MMAL_VIDEO_PROFILE_MP4V_CORESCALABLE,
- + MMAL_VIDEO_PROFILE_MP4V_ADVANCEDCODING,
- + MMAL_VIDEO_PROFILE_MP4V_ADVANCEDCORE,
- + MMAL_VIDEO_PROFILE_MP4V_ADVANCEDSCALABLE,
- + MMAL_VIDEO_PROFILE_MP4V_ADVANCEDSIMPLE,
- + MMAL_VIDEO_PROFILE_H264_BASELINE,
- + MMAL_VIDEO_PROFILE_H264_MAIN,
- + MMAL_VIDEO_PROFILE_H264_EXTENDED,
- + MMAL_VIDEO_PROFILE_H264_HIGH,
- + MMAL_VIDEO_PROFILE_H264_HIGH10,
- + MMAL_VIDEO_PROFILE_H264_HIGH422,
- + MMAL_VIDEO_PROFILE_H264_HIGH444,
- + MMAL_VIDEO_PROFILE_H264_CONSTRAINED_BASELINE,
- + MMAL_VIDEO_PROFILE_DUMMY = 0x7FFFFFFF
- +};
- +
- +enum mmal_video_level {
- + MMAL_VIDEO_LEVEL_H263_10,
- + MMAL_VIDEO_LEVEL_H263_20,
- + MMAL_VIDEO_LEVEL_H263_30,
- + MMAL_VIDEO_LEVEL_H263_40,
- + MMAL_VIDEO_LEVEL_H263_45,
- + MMAL_VIDEO_LEVEL_H263_50,
- + MMAL_VIDEO_LEVEL_H263_60,
- + MMAL_VIDEO_LEVEL_H263_70,
- + MMAL_VIDEO_LEVEL_MP4V_0,
- + MMAL_VIDEO_LEVEL_MP4V_0b,
- + MMAL_VIDEO_LEVEL_MP4V_1,
- + MMAL_VIDEO_LEVEL_MP4V_2,
- + MMAL_VIDEO_LEVEL_MP4V_3,
- + MMAL_VIDEO_LEVEL_MP4V_4,
- + MMAL_VIDEO_LEVEL_MP4V_4a,
- + MMAL_VIDEO_LEVEL_MP4V_5,
- + MMAL_VIDEO_LEVEL_MP4V_6,
- + MMAL_VIDEO_LEVEL_H264_1,
- + MMAL_VIDEO_LEVEL_H264_1b,
- + MMAL_VIDEO_LEVEL_H264_11,
- + MMAL_VIDEO_LEVEL_H264_12,
- + MMAL_VIDEO_LEVEL_H264_13,
- + MMAL_VIDEO_LEVEL_H264_2,
- + MMAL_VIDEO_LEVEL_H264_21,
- + MMAL_VIDEO_LEVEL_H264_22,
- + MMAL_VIDEO_LEVEL_H264_3,
- + MMAL_VIDEO_LEVEL_H264_31,
- + MMAL_VIDEO_LEVEL_H264_32,
- + MMAL_VIDEO_LEVEL_H264_4,
- + MMAL_VIDEO_LEVEL_H264_41,
- + MMAL_VIDEO_LEVEL_H264_42,
- + MMAL_VIDEO_LEVEL_H264_5,
- + MMAL_VIDEO_LEVEL_H264_51,
- + MMAL_VIDEO_LEVEL_DUMMY = 0x7FFFFFFF
- +};
- +
- +struct mmal_parameter_video_profile {
- + enum mmal_video_profile profile;
- + enum mmal_video_level level;
- +};
- +
- +/* video parameters */
- +
- +enum mmal_parameter_video_type {
- + /** @ref MMAL_DISPLAYREGION_T */
- + MMAL_PARAMETER_DISPLAYREGION = MMAL_PARAMETER_GROUP_VIDEO,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_PROFILE_T */
- + MMAL_PARAMETER_SUPPORTED_PROFILES,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_PROFILE_T */
- + MMAL_PARAMETER_PROFILE,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_INTRAPERIOD,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_RATECONTROL_T */
- + MMAL_PARAMETER_RATECONTROL,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_NALUNITFORMAT_T */
- + MMAL_PARAMETER_NALUNITFORMAT,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_MINIMISE_FRAGMENTATION,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T.
- + * Setting the value to zero resets to the default (one slice per frame).
- + */
- + MMAL_PARAMETER_MB_ROWS_PER_SLICE,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_LEVEL_EXTENSION_T */
- + MMAL_PARAMETER_VIDEO_LEVEL_EXTENSION,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_EEDE_ENABLE_T */
- + MMAL_PARAMETER_VIDEO_EEDE_ENABLE,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_EEDE_LOSSRATE_T */
- + MMAL_PARAMETER_VIDEO_EEDE_LOSSRATE,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. Request an I-frame. */
- + MMAL_PARAMETER_VIDEO_REQUEST_I_FRAME,
- + /** @ref MMAL_PARAMETER_VIDEO_INTRA_REFRESH_T */
- + MMAL_PARAMETER_VIDEO_INTRA_REFRESH,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. */
- + MMAL_PARAMETER_VIDEO_IMMUTABLE_INPUT,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. Run-time bit rate control */
- + MMAL_PARAMETER_VIDEO_BIT_RATE,
- +
- + /** @ref MMAL_PARAMETER_FRAME_RATE_T */
- + MMAL_PARAMETER_VIDEO_FRAME_RATE,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_MIN_QUANT,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_MAX_QUANT,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_ENCODE_RC_MODEL_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_RC_MODEL,
- +
- + MMAL_PARAMETER_EXTRA_BUFFERS, /**< @ref MMAL_PARAMETER_UINT32_T. */
- + /** @ref MMAL_PARAMETER_UINT32_T.
- + * Changing this parameter from the default can reduce frame rate
- + * because image buffers need to be re-pitched.
- + */
- + MMAL_PARAMETER_VIDEO_ALIGN_HORIZ,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T.
- + * Changing this parameter from the default can reduce frame rate
- + * because image buffers need to be re-pitched.
- + */
- + MMAL_PARAMETER_VIDEO_ALIGN_VERT,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. */
- + MMAL_PARAMETER_VIDEO_DROPPABLE_PFRAMES,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_INITIAL_QUANT,
- +
- + /**< @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_QP_P,
- +
- + /**< @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_RC_SLICE_DQUANT,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_FRAME_LIMIT_BITS,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_PEAK_RATE,
- +
- + /* H264 specific parameters */
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_DISABLE_CABAC,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_LOW_LATENCY,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_AU_DELIMITERS,
- +
- + /** @ref MMAL_PARAMETER_UINT32_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_DEBLOCK_IDC,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_ENCODER_H264_MB_INTRA_MODES_T. */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_MB_INTRA_MODE,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_HEADER_ON_OPEN,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_PRECODE_FOR_QP,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_DRM_INIT_INFO_T. */
- + MMAL_PARAMETER_VIDEO_DRM_INIT_INFO,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_TIMESTAMP_FIFO,
- +
- + /** @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_DECODE_ERROR_CONCEALMENT,
- +
- + /** @ref MMAL_PARAMETER_VIDEO_DRM_PROTECT_BUFFER_T. */
- + MMAL_PARAMETER_VIDEO_DRM_PROTECT_BUFFER,
- +
- + /** @ref MMAL_PARAMETER_BYTES_T */
- + MMAL_PARAMETER_VIDEO_DECODE_CONFIG_VD3,
- +
- + /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_VCL_HRD_PARAMETERS,
- +
- + /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_H264_LOW_DELAY_HRD_FLAG,
- +
- + /**< @ref MMAL_PARAMETER_BOOLEAN_T */
- + MMAL_PARAMETER_VIDEO_ENCODE_INLINE_HEADER
- +};
- +
- +/** Valid mirror modes */
- +enum mmal_parameter_mirror {
- + MMAL_PARAM_MIRROR_NONE,
- + MMAL_PARAM_MIRROR_VERTICAL,
- + MMAL_PARAM_MIRROR_HORIZONTAL,
- + MMAL_PARAM_MIRROR_BOTH,
- +};
- +
- +enum mmal_parameter_displaytransform {
- + MMAL_DISPLAY_ROT0 = 0,
- + MMAL_DISPLAY_MIRROR_ROT0 = 1,
- + MMAL_DISPLAY_MIRROR_ROT180 = 2,
- + MMAL_DISPLAY_ROT180 = 3,
- + MMAL_DISPLAY_MIRROR_ROT90 = 4,
- + MMAL_DISPLAY_ROT270 = 5,
- + MMAL_DISPLAY_ROT90 = 6,
- + MMAL_DISPLAY_MIRROR_ROT270 = 7,
- +};
- +
- +enum mmal_parameter_displaymode {
- + MMAL_DISPLAY_MODE_FILL = 0,
- + MMAL_DISPLAY_MODE_LETTERBOX = 1,
- +};
- +
- +enum mmal_parameter_displayset {
- + MMAL_DISPLAY_SET_NONE = 0,
- + MMAL_DISPLAY_SET_NUM = 1,
- + MMAL_DISPLAY_SET_FULLSCREEN = 2,
- + MMAL_DISPLAY_SET_TRANSFORM = 4,
- + MMAL_DISPLAY_SET_DEST_RECT = 8,
- + MMAL_DISPLAY_SET_SRC_RECT = 0x10,
- + MMAL_DISPLAY_SET_MODE = 0x20,
- + MMAL_DISPLAY_SET_PIXEL = 0x40,
- + MMAL_DISPLAY_SET_NOASPECT = 0x80,
- + MMAL_DISPLAY_SET_LAYER = 0x100,
- + MMAL_DISPLAY_SET_COPYPROTECT = 0x200,
- + MMAL_DISPLAY_SET_ALPHA = 0x400,
- +};
- +
- +struct mmal_parameter_displayregion {
- + /** Bitfield that indicates which fields are set and should be
- + * used. All other fields will maintain their current value.
- + * \ref MMAL_DISPLAYSET_T defines the bits that can be
- + * combined.
- + */
- + u32 set;
- +
- + /** Describes the display output device, with 0 typically
- + * being a directly connected LCD display. The actual values
- + * will depend on the hardware. Code using hard-wired numbers
- + * (e.g. 2) is certain to fail.
- + */
- +
- + u32 display_num;
- + /** Indicates that we are using the full device screen area,
- + * rather than a window of the display. If zero, then
- + * dest_rect is used to specify a region of the display to
- + * use.
- + */
- +
- + s32 fullscreen;
- + /** Indicates any rotation or flipping used to map frames onto
- + * the natural display orientation.
- + */
- + u32 transform; /* enum mmal_parameter_displaytransform */
- +
- + /** Where to display the frame within the screen, if
- + * fullscreen is zero.
- + */
- + struct vchiq_mmal_rect dest_rect;
- +
- + /** Indicates which area of the frame to display. If all
- + * values are zero, the whole frame will be used.
- + */
- + struct vchiq_mmal_rect src_rect;
- +
- + /** If set to non-zero, indicates that any display scaling
- + * should disregard the aspect ratio of the frame region being
- + * displayed.
- + */
- + s32 noaspect;
- +
- + /** Indicates how the image should be scaled to fit the
- + * display. \code MMAL_DISPLAY_MODE_FILL \endcode indicates
- + * that the image should fill the screen by potentially
- + * cropping the frames. Setting \code mode \endcode to \code
- + * MMAL_DISPLAY_MODE_LETTERBOX \endcode indicates that all the
- + * source region should be displayed and black bars added if
- + * necessary.
- + */
- + u32 mode; /* enum mmal_parameter_displaymode */
- +
- + /** If non-zero, defines the width of a source pixel relative
- + * to \code pixel_y \endcode. If zero, then pixels default to
- + * being square.
- + */
- + u32 pixel_x;
- +
- + /** If non-zero, defines the height of a source pixel relative
- + * to \code pixel_x \endcode. If zero, then pixels default to
- + * being square.
- + */
- + u32 pixel_y;
- +
- + /** Sets the relative depth of the images, with greater values
- + * being in front of smaller values.
- + */
- + u32 layer;
- +
- + /** Set to non-zero to ensure copy protection is used on
- + * output.
- + */
- + s32 copyprotect_required;
- +
- + /** Level of opacity of the layer, where zero is fully
- + * transparent and 255 is fully opaque.
- + */
- + u32 alpha;
- +};
- +
- +#define MMAL_MAX_IMAGEFX_PARAMETERS 5
- +
- +struct mmal_parameter_imagefx_parameters {
- + enum mmal_parameter_imagefx effect;
- + u32 num_effect_params;
- + u32 effect_parameter[MMAL_MAX_IMAGEFX_PARAMETERS];
- +};
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-vchiq.c
- @@ -0,0 +1,1916 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + *
- + * V4L2 driver MMAL vchiq interface code
- + */
- +
- +#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- +
- +#include <linux/errno.h>
- +#include <linux/kernel.h>
- +#include <linux/mutex.h>
- +#include <linux/mm.h>
- +#include <linux/slab.h>
- +#include <linux/completion.h>
- +#include <linux/vmalloc.h>
- +#include <asm/cacheflush.h>
- +#include <media/videobuf2-vmalloc.h>
- +
- +#include "mmal-common.h"
- +#include "mmal-vchiq.h"
- +#include "mmal-msg.h"
- +
- +#define USE_VCHIQ_ARM
- +#include "interface/vchi/vchi.h"
- +
- +/* maximum number of components supported */
- +#define VCHIQ_MMAL_MAX_COMPONENTS 4
- +
- +/*#define FULL_MSG_DUMP 1*/
- +
- +#ifdef DEBUG
- +static const char *const msg_type_names[] = {
- + "UNKNOWN",
- + "QUIT",
- + "SERVICE_CLOSED",
- + "GET_VERSION",
- + "COMPONENT_CREATE",
- + "COMPONENT_DESTROY",
- + "COMPONENT_ENABLE",
- + "COMPONENT_DISABLE",
- + "PORT_INFO_GET",
- + "PORT_INFO_SET",
- + "PORT_ACTION",
- + "BUFFER_FROM_HOST",
- + "BUFFER_TO_HOST",
- + "GET_STATS",
- + "PORT_PARAMETER_SET",
- + "PORT_PARAMETER_GET",
- + "EVENT_TO_HOST",
- + "GET_CORE_STATS_FOR_PORT",
- + "OPAQUE_ALLOCATOR",
- + "CONSUME_MEM",
- + "LMK",
- + "OPAQUE_ALLOCATOR_DESC",
- + "DRM_GET_LHS32",
- + "DRM_GET_TIME",
- + "BUFFER_FROM_HOST_ZEROLEN",
- + "PORT_FLUSH",
- + "HOST_LOG",
- +};
- +#endif
- +
- +static const char *const port_action_type_names[] = {
- + "UNKNOWN",
- + "ENABLE",
- + "DISABLE",
- + "FLUSH",
- + "CONNECT",
- + "DISCONNECT",
- + "SET_REQUIREMENTS",
- +};
- +
- +#if defined(DEBUG)
- +#if defined(FULL_MSG_DUMP)
- +#define DBG_DUMP_MSG(MSG, MSG_LEN, TITLE) \
- + do { \
- + pr_debug(TITLE" type:%s(%d) length:%d\n", \
- + msg_type_names[(MSG)->h.type], \
- + (MSG)->h.type, (MSG_LEN)); \
- + print_hex_dump(KERN_DEBUG, "<<h: ", DUMP_PREFIX_OFFSET, \
- + 16, 4, (MSG), \
- + sizeof(struct mmal_msg_header), 1); \
- + print_hex_dump(KERN_DEBUG, "<<p: ", DUMP_PREFIX_OFFSET, \
- + 16, 4, \
- + ((u8 *)(MSG)) + sizeof(struct mmal_msg_header),\
- + (MSG_LEN) - sizeof(struct mmal_msg_header), 1); \
- + } while (0)
- +#else
- +#define DBG_DUMP_MSG(MSG, MSG_LEN, TITLE) \
- + { \
- + pr_debug(TITLE" type:%s(%d) length:%d\n", \
- + msg_type_names[(MSG)->h.type], \
- + (MSG)->h.type, (MSG_LEN)); \
- + }
- +#endif
- +#else
- +#define DBG_DUMP_MSG(MSG, MSG_LEN, TITLE)
- +#endif
- +
- +/* normal message context */
- +struct mmal_msg_context {
- + union {
- + struct {
- + /* work struct for defered callback - must come first */
- + struct work_struct work;
- + /* mmal instance */
- + struct vchiq_mmal_instance *instance;
- + /* mmal port */
- + struct vchiq_mmal_port *port;
- + /* actual buffer used to store bulk reply */
- + struct mmal_buffer *buffer;
- + /* amount of buffer used */
- + unsigned long buffer_used;
- + /* MMAL buffer flags */
- + u32 mmal_flags;
- + /* Presentation and Decode timestamps */
- + s64 pts;
- + s64 dts;
- +
- + int status; /* context status */
- +
- + } bulk; /* bulk data */
- +
- + struct {
- + /* message handle to release */
- + VCHI_HELD_MSG_T msg_handle;
- + /* pointer to received message */
- + struct mmal_msg *msg;
- + /* received message length */
- + u32 msg_len;
- + /* completion upon reply */
- + struct completion cmplt;
- + } sync; /* synchronous response */
- + } u;
- +
- +};
- +
- +struct vchiq_mmal_instance {
- + VCHI_SERVICE_HANDLE_T handle;
- +
- + /* ensure serialised access to service */
- + struct mutex vchiq_mutex;
- +
- + /* ensure serialised access to bulk operations */
- + struct mutex bulk_mutex;
- +
- + /* vmalloc page to receive scratch bulk xfers into */
- + void *bulk_scratch;
- +
- + /* component to use next */
- + int component_idx;
- + struct vchiq_mmal_component component[VCHIQ_MMAL_MAX_COMPONENTS];
- +};
- +
- +static struct mmal_msg_context *get_msg_context(struct vchiq_mmal_instance
- + *instance)
- +{
- + struct mmal_msg_context *msg_context;
- +
- + /* todo: should this be allocated from a pool to avoid kmalloc */
- + msg_context = kmalloc(sizeof(*msg_context), GFP_KERNEL);
- + memset(msg_context, 0, sizeof(*msg_context));
- +
- + return msg_context;
- +}
- +
- +static void release_msg_context(struct mmal_msg_context *msg_context)
- +{
- + kfree(msg_context);
- +}
- +
- +/* deals with receipt of event to host message */
- +static void event_to_host_cb(struct vchiq_mmal_instance *instance,
- + struct mmal_msg *msg, u32 msg_len)
- +{
- + pr_debug("unhandled event\n");
- + pr_debug("component:%p port type:%d num:%d cmd:0x%x length:%d\n",
- + msg->u.event_to_host.client_component,
- + msg->u.event_to_host.port_type,
- + msg->u.event_to_host.port_num,
- + msg->u.event_to_host.cmd, msg->u.event_to_host.length);
- +}
- +
- +/* workqueue scheduled callback
- + *
- + * we do this because it is important we do not call any other vchiq
- + * sync calls from witin the message delivery thread
- + */
- +static void buffer_work_cb(struct work_struct *work)
- +{
- + struct mmal_msg_context *msg_context = (struct mmal_msg_context *)work;
- +
- + msg_context->u.bulk.port->buffer_cb(msg_context->u.bulk.instance,
- + msg_context->u.bulk.port,
- + msg_context->u.bulk.status,
- + msg_context->u.bulk.buffer,
- + msg_context->u.bulk.buffer_used,
- + msg_context->u.bulk.mmal_flags,
- + msg_context->u.bulk.dts,
- + msg_context->u.bulk.pts);
- +
- + /* release message context */
- + release_msg_context(msg_context);
- +}
- +
- +/* enqueue a bulk receive for a given message context */
- +static int bulk_receive(struct vchiq_mmal_instance *instance,
- + struct mmal_msg *msg,
- + struct mmal_msg_context *msg_context)
- +{
- + unsigned long rd_len;
- + unsigned long flags = 0;
- + int ret;
- +
- + /* bulk mutex stops other bulk operations while we have a
- + * receive in progress - released in callback
- + */
- + ret = mutex_lock_interruptible(&instance->bulk_mutex);
- + if (ret != 0)
- + return ret;
- +
- + rd_len = msg->u.buffer_from_host.buffer_header.length;
- +
- + /* take buffer from queue */
- + spin_lock_irqsave(&msg_context->u.bulk.port->slock, flags);
- + if (list_empty(&msg_context->u.bulk.port->buffers)) {
- + spin_unlock_irqrestore(&msg_context->u.bulk.port->slock, flags);
- + pr_err("buffer list empty trying to submit bulk receive\n");
- +
- + /* todo: this is a serious error, we should never have
- + * commited a buffer_to_host operation to the mmal
- + * port without the buffer to back it up (underflow
- + * handling) and there is no obvious way to deal with
- + * this - how is the mmal servie going to react when
- + * we fail to do the xfer and reschedule a buffer when
- + * it arrives? perhaps a starved flag to indicate a
- + * waiting bulk receive?
- + */
- +
- + mutex_unlock(&instance->bulk_mutex);
- +
- + return -EINVAL;
- + }
- +
- + msg_context->u.bulk.buffer =
- + list_entry(msg_context->u.bulk.port->buffers.next,
- + struct mmal_buffer, list);
- + list_del(&msg_context->u.bulk.buffer->list);
- +
- + spin_unlock_irqrestore(&msg_context->u.bulk.port->slock, flags);
- +
- + /* ensure we do not overrun the available buffer */
- + if (rd_len > msg_context->u.bulk.buffer->buffer_size) {
- + rd_len = msg_context->u.bulk.buffer->buffer_size;
- + pr_warn("short read as not enough receive buffer space\n");
- + /* todo: is this the correct response, what happens to
- + * the rest of the message data?
- + */
- + }
- +
- + /* store length */
- + msg_context->u.bulk.buffer_used = rd_len;
- + msg_context->u.bulk.mmal_flags =
- + msg->u.buffer_from_host.buffer_header.flags;
- + msg_context->u.bulk.dts = msg->u.buffer_from_host.buffer_header.dts;
- + msg_context->u.bulk.pts = msg->u.buffer_from_host.buffer_header.pts;
- +
- + // only need to flush L1 cache here, as VCHIQ takes care of the L2
- + // cache.
- + __cpuc_flush_dcache_area(msg_context->u.bulk.buffer->buffer, rd_len);
- +
- + /* queue the bulk submission */
- + vchi_service_use(instance->handle);
- + ret = vchi_bulk_queue_receive(instance->handle,
- + msg_context->u.bulk.buffer->buffer,
- + /* Actual receive needs to be a multiple
- + * of 4 bytes
- + */
- + (rd_len + 3) & ~3,
- + VCHI_FLAGS_CALLBACK_WHEN_OP_COMPLETE |
- + VCHI_FLAGS_BLOCK_UNTIL_QUEUED,
- + msg_context);
- +
- + vchi_service_release(instance->handle);
- +
- + if (ret != 0) {
- + /* callback will not be clearing the mutex */
- + mutex_unlock(&instance->bulk_mutex);
- + }
- +
- + return ret;
- +}
- +
- +/* enque a dummy bulk receive for a given message context */
- +static int dummy_bulk_receive(struct vchiq_mmal_instance *instance,
- + struct mmal_msg_context *msg_context)
- +{
- + int ret;
- +
- + /* bulk mutex stops other bulk operations while we have a
- + * receive in progress - released in callback
- + */
- + ret = mutex_lock_interruptible(&instance->bulk_mutex);
- + if (ret != 0)
- + return ret;
- +
- + /* zero length indicates this was a dummy transfer */
- + msg_context->u.bulk.buffer_used = 0;
- +
- + /* queue the bulk submission */
- + vchi_service_use(instance->handle);
- +
- + ret = vchi_bulk_queue_receive(instance->handle,
- + instance->bulk_scratch,
- + 8,
- + VCHI_FLAGS_CALLBACK_WHEN_OP_COMPLETE |
- + VCHI_FLAGS_BLOCK_UNTIL_QUEUED,
- + msg_context);
- +
- + vchi_service_release(instance->handle);
- +
- + if (ret != 0) {
- + /* callback will not be clearing the mutex */
- + mutex_unlock(&instance->bulk_mutex);
- + }
- +
- + return ret;
- +}
- +
- +/* data in message, memcpy from packet into output buffer */
- +static int inline_receive(struct vchiq_mmal_instance *instance,
- + struct mmal_msg *msg,
- + struct mmal_msg_context *msg_context)
- +{
- + unsigned long flags = 0;
- +
- + /* take buffer from queue */
- + spin_lock_irqsave(&msg_context->u.bulk.port->slock, flags);
- + if (list_empty(&msg_context->u.bulk.port->buffers)) {
- + spin_unlock_irqrestore(&msg_context->u.bulk.port->slock, flags);
- + pr_err("buffer list empty trying to receive inline\n");
- +
- + /* todo: this is a serious error, we should never have
- + * commited a buffer_to_host operation to the mmal
- + * port without the buffer to back it up (with
- + * underflow handling) and there is no obvious way to
- + * deal with this. Less bad than the bulk case as we
- + * can just drop this on the floor but...unhelpful
- + */
- + return -EINVAL;
- + }
- +
- + msg_context->u.bulk.buffer =
- + list_entry(msg_context->u.bulk.port->buffers.next,
- + struct mmal_buffer, list);
- + list_del(&msg_context->u.bulk.buffer->list);
- +
- + spin_unlock_irqrestore(&msg_context->u.bulk.port->slock, flags);
- +
- + memcpy(msg_context->u.bulk.buffer->buffer,
- + msg->u.buffer_from_host.short_data,
- + msg->u.buffer_from_host.payload_in_message);
- +
- + msg_context->u.bulk.buffer_used =
- + msg->u.buffer_from_host.payload_in_message;
- +
- + return 0;
- +}
- +
- +/* queue the buffer availability with MMAL_MSG_TYPE_BUFFER_FROM_HOST */
- +static int
- +buffer_from_host(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port, struct mmal_buffer *buf)
- +{
- + struct mmal_msg_context *msg_context;
- + struct mmal_msg m;
- + int ret;
- +
- + pr_debug("instance:%p buffer:%p\n", instance->handle, buf);
- +
- + /* bulk mutex stops other bulk operations while we
- + * have a receive in progress
- + */
- + if (mutex_lock_interruptible(&instance->bulk_mutex))
- + return -EINTR;
- +
- + /* get context */
- + msg_context = get_msg_context(instance);
- + if (msg_context == NULL)
- + return -ENOMEM;
- +
- + /* store bulk message context for when data arrives */
- + msg_context->u.bulk.instance = instance;
- + msg_context->u.bulk.port = port;
- + msg_context->u.bulk.buffer = NULL; /* not valid until bulk xfer */
- + msg_context->u.bulk.buffer_used = 0;
- +
- + /* initialise work structure ready to schedule callback */
- + INIT_WORK(&msg_context->u.bulk.work, buffer_work_cb);
- +
- + /* prep the buffer from host message */
- + memset(&m, 0xbc, sizeof(m)); /* just to make debug clearer */
- +
- + m.h.type = MMAL_MSG_TYPE_BUFFER_FROM_HOST;
- + m.h.magic = MMAL_MAGIC;
- + m.h.context = msg_context;
- + m.h.status = 0;
- +
- + /* drvbuf is our private data passed back */
- + m.u.buffer_from_host.drvbuf.magic = MMAL_MAGIC;
- + m.u.buffer_from_host.drvbuf.component_handle = port->component->handle;
- + m.u.buffer_from_host.drvbuf.port_handle = port->handle;
- + m.u.buffer_from_host.drvbuf.client_context = msg_context;
- +
- + /* buffer header */
- + m.u.buffer_from_host.buffer_header.cmd = 0;
- + m.u.buffer_from_host.buffer_header.data = buf->buffer;
- + m.u.buffer_from_host.buffer_header.alloc_size = buf->buffer_size;
- + m.u.buffer_from_host.buffer_header.length = 0; /* nothing used yet */
- + m.u.buffer_from_host.buffer_header.offset = 0; /* no offset */
- + m.u.buffer_from_host.buffer_header.flags = 0; /* no flags */
- + m.u.buffer_from_host.buffer_header.pts = MMAL_TIME_UNKNOWN;
- + m.u.buffer_from_host.buffer_header.dts = MMAL_TIME_UNKNOWN;
- +
- + /* clear buffer type sepecific data */
- + memset(&m.u.buffer_from_host.buffer_header_type_specific, 0,
- + sizeof(m.u.buffer_from_host.buffer_header_type_specific));
- +
- + /* no payload in message */
- + m.u.buffer_from_host.payload_in_message = 0;
- +
- + vchi_service_use(instance->handle);
- +
- + ret = vchi_msg_queue(instance->handle, &m,
- + sizeof(struct mmal_msg_header) +
- + sizeof(m.u.buffer_from_host),
- + VCHI_FLAGS_BLOCK_UNTIL_QUEUED, NULL);
- +
- + if (ret != 0) {
- + release_msg_context(msg_context);
- + /* todo: is this correct error value? */
- + }
- +
- + vchi_service_release(instance->handle);
- +
- + mutex_unlock(&instance->bulk_mutex);
- +
- + return ret;
- +}
- +
- +/* submit a buffer to the mmal sevice
- + *
- + * the buffer_from_host uses size data from the ports next available
- + * mmal_buffer and deals with there being no buffer available by
- + * incrementing the underflow for later
- + */
- +static int port_buffer_from_host(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- + struct mmal_buffer *buf;
- + unsigned long flags = 0;
- +
- + if (!port->enabled)
- + return -EINVAL;
- +
- + /* peek buffer from queue */
- + spin_lock_irqsave(&port->slock, flags);
- + if (list_empty(&port->buffers)) {
- + port->buffer_underflow++;
- + spin_unlock_irqrestore(&port->slock, flags);
- + return -ENOSPC;
- + }
- +
- + buf = list_entry(port->buffers.next, struct mmal_buffer, list);
- +
- + spin_unlock_irqrestore(&port->slock, flags);
- +
- + /* issue buffer to mmal service */
- + ret = buffer_from_host(instance, port, buf);
- + if (ret) {
- + pr_err("adding buffer header failed\n");
- + /* todo: how should this be dealt with */
- + }
- +
- + return ret;
- +}
- +
- +/* deals with receipt of buffer to host message */
- +static void buffer_to_host_cb(struct vchiq_mmal_instance *instance,
- + struct mmal_msg *msg, u32 msg_len)
- +{
- + struct mmal_msg_context *msg_context;
- +
- + pr_debug("buffer_to_host_cb: instance:%p msg:%p msg_len:%d\n",
- + instance, msg, msg_len);
- +
- + if (msg->u.buffer_from_host.drvbuf.magic == MMAL_MAGIC) {
- + msg_context = msg->u.buffer_from_host.drvbuf.client_context;
- + } else {
- + pr_err("MMAL_MSG_TYPE_BUFFER_TO_HOST with bad magic\n");
- + return;
- + }
- +
- + if (msg->h.status != MMAL_MSG_STATUS_SUCCESS) {
- + /* message reception had an error */
- + pr_warn("error %d in reply\n", msg->h.status);
- +
- + msg_context->u.bulk.status = msg->h.status;
- +
- + } else if (msg->u.buffer_from_host.buffer_header.length == 0) {
- + /* empty buffer */
- + if (msg->u.buffer_from_host.buffer_header.flags &
- + MMAL_BUFFER_HEADER_FLAG_EOS) {
- + msg_context->u.bulk.status =
- + dummy_bulk_receive(instance, msg_context);
- + if (msg_context->u.bulk.status == 0)
- + return; /* successful bulk submission, bulk
- + * completion will trigger callback
- + */
- + } else {
- + /* do callback with empty buffer - not EOS though */
- + msg_context->u.bulk.status = 0;
- + msg_context->u.bulk.buffer_used = 0;
- + }
- + } else if (msg->u.buffer_from_host.payload_in_message == 0) {
- + /* data is not in message, queue a bulk receive */
- + msg_context->u.bulk.status =
- + bulk_receive(instance, msg, msg_context);
- + if (msg_context->u.bulk.status == 0)
- + return; /* successful bulk submission, bulk
- + * completion will trigger callback
- + */
- +
- + /* failed to submit buffer, this will end badly */
- + pr_err("error %d on bulk submission\n",
- + msg_context->u.bulk.status);
- +
- + } else if (msg->u.buffer_from_host.payload_in_message <=
- + MMAL_VC_SHORT_DATA) {
- + /* data payload within message */
- + msg_context->u.bulk.status = inline_receive(instance, msg,
- + msg_context);
- + } else {
- + pr_err("message with invalid short payload\n");
- +
- + /* signal error */
- + msg_context->u.bulk.status = -EINVAL;
- + msg_context->u.bulk.buffer_used =
- + msg->u.buffer_from_host.payload_in_message;
- + }
- +
- + /* replace the buffer header */
- + port_buffer_from_host(instance, msg_context->u.bulk.port);
- +
- + /* schedule the port callback */
- + schedule_work(&msg_context->u.bulk.work);
- +}
- +
- +static void bulk_receive_cb(struct vchiq_mmal_instance *instance,
- + struct mmal_msg_context *msg_context)
- +{
- + /* bulk receive operation complete */
- + mutex_unlock(&msg_context->u.bulk.instance->bulk_mutex);
- +
- + /* replace the buffer header */
- + port_buffer_from_host(msg_context->u.bulk.instance,
- + msg_context->u.bulk.port);
- +
- + msg_context->u.bulk.status = 0;
- +
- + /* schedule the port callback */
- + schedule_work(&msg_context->u.bulk.work);
- +}
- +
- +static void bulk_abort_cb(struct vchiq_mmal_instance *instance,
- + struct mmal_msg_context *msg_context)
- +{
- + pr_err("%s: bulk ABORTED msg_context:%p\n", __func__, msg_context);
- +
- + /* bulk receive operation complete */
- + mutex_unlock(&msg_context->u.bulk.instance->bulk_mutex);
- +
- + /* replace the buffer header */
- + port_buffer_from_host(msg_context->u.bulk.instance,
- + msg_context->u.bulk.port);
- +
- + msg_context->u.bulk.status = -EINTR;
- +
- + schedule_work(&msg_context->u.bulk.work);
- +}
- +
- +/* incoming event service callback */
- +static void service_callback(void *param,
- + const VCHI_CALLBACK_REASON_T reason,
- + void *bulk_ctx)
- +{
- + struct vchiq_mmal_instance *instance = param;
- + int status;
- + u32 msg_len;
- + struct mmal_msg *msg;
- + VCHI_HELD_MSG_T msg_handle;
- +
- + if (!instance) {
- + pr_err("Message callback passed NULL instance\n");
- + return;
- + }
- +
- + switch (reason) {
- + case VCHI_CALLBACK_MSG_AVAILABLE:
- + status = vchi_msg_hold(instance->handle, (void **)&msg,
- + &msg_len, VCHI_FLAGS_NONE, &msg_handle);
- + if (status) {
- + pr_err("Unable to dequeue a message (%d)\n", status);
- + break;
- + }
- +
- + DBG_DUMP_MSG(msg, msg_len, "<<< reply message");
- +
- + /* handling is different for buffer messages */
- + switch (msg->h.type) {
- +
- + case MMAL_MSG_TYPE_BUFFER_FROM_HOST:
- + vchi_held_msg_release(&msg_handle);
- + break;
- +
- + case MMAL_MSG_TYPE_EVENT_TO_HOST:
- + event_to_host_cb(instance, msg, msg_len);
- + vchi_held_msg_release(&msg_handle);
- +
- + break;
- +
- + case MMAL_MSG_TYPE_BUFFER_TO_HOST:
- + buffer_to_host_cb(instance, msg, msg_len);
- + vchi_held_msg_release(&msg_handle);
- + break;
- +
- + default:
- + /* messages dependant on header context to complete */
- +
- + /* todo: the msg.context really ought to be sanity
- + * checked before we just use it, afaict it comes back
- + * and is used raw from the videocore. Perhaps it
- + * should be verified the address lies in the kernel
- + * address space.
- + */
- + if (msg->h.context == NULL) {
- + pr_err("received message context was null!\n");
- + vchi_held_msg_release(&msg_handle);
- + break;
- + }
- +
- + /* fill in context values */
- + msg->h.context->u.sync.msg_handle = msg_handle;
- + msg->h.context->u.sync.msg = msg;
- + msg->h.context->u.sync.msg_len = msg_len;
- +
- + /* todo: should this check (completion_done()
- + * == 1) for no one waiting? or do we need a
- + * flag to tell us the completion has been
- + * interrupted so we can free the message and
- + * its context. This probably also solves the
- + * message arriving after interruption todo
- + * below
- + */
- +
- + /* complete message so caller knows it happened */
- + complete(&msg->h.context->u.sync.cmplt);
- + break;
- + }
- +
- + break;
- +
- + case VCHI_CALLBACK_BULK_RECEIVED:
- + bulk_receive_cb(instance, bulk_ctx);
- + break;
- +
- + case VCHI_CALLBACK_BULK_RECEIVE_ABORTED:
- + bulk_abort_cb(instance, bulk_ctx);
- + break;
- +
- + case VCHI_CALLBACK_SERVICE_CLOSED:
- + /* TODO: consider if this requires action if received when
- + * driver is not explicitly closing the service
- + */
- + break;
- +
- + default:
- + pr_err("Received unhandled message reason %d\n", reason);
- + break;
- + }
- +}
- +
- +static int send_synchronous_mmal_msg(struct vchiq_mmal_instance *instance,
- + struct mmal_msg *msg,
- + unsigned int payload_len,
- + struct mmal_msg **msg_out,
- + VCHI_HELD_MSG_T *msg_handle_out)
- +{
- + struct mmal_msg_context msg_context;
- + int ret;
- +
- + /* payload size must not cause message to exceed max size */
- + if (payload_len >
- + (MMAL_MSG_MAX_SIZE - sizeof(struct mmal_msg_header))) {
- + pr_err("payload length %d exceeds max:%d\n", payload_len,
- + (MMAL_MSG_MAX_SIZE - sizeof(struct mmal_msg_header)));
- + return -EINVAL;
- + }
- +
- + init_completion(&msg_context.u.sync.cmplt);
- +
- + msg->h.magic = MMAL_MAGIC;
- + msg->h.context = &msg_context;
- + msg->h.status = 0;
- +
- + DBG_DUMP_MSG(msg, (sizeof(struct mmal_msg_header) + payload_len),
- + ">>> sync message");
- +
- + vchi_service_use(instance->handle);
- +
- + ret = vchi_msg_queue(instance->handle,
- + msg,
- + sizeof(struct mmal_msg_header) + payload_len,
- + VCHI_FLAGS_BLOCK_UNTIL_QUEUED, NULL);
- +
- + vchi_service_release(instance->handle);
- +
- + if (ret) {
- + pr_err("error %d queuing message\n", ret);
- + return ret;
- + }
- +
- + ret = wait_for_completion_timeout(&msg_context.u.sync.cmplt, 3*HZ);
- + if (ret <= 0) {
- + pr_err("error %d waiting for sync completion\n", ret);
- + if (ret == 0)
- + ret = -ETIME;
- + /* todo: what happens if the message arrives after aborting */
- + return ret;
- + }
- +
- + *msg_out = msg_context.u.sync.msg;
- + *msg_handle_out = msg_context.u.sync.msg_handle;
- +
- + return 0;
- +}
- +
- +static void dump_port_info(struct vchiq_mmal_port *port)
- +{
- + pr_debug("port handle:0x%x enabled:%d\n", port->handle, port->enabled);
- +
- + pr_debug("buffer minimum num:%d size:%d align:%d\n",
- + port->minimum_buffer.num,
- + port->minimum_buffer.size, port->minimum_buffer.alignment);
- +
- + pr_debug("buffer recommended num:%d size:%d align:%d\n",
- + port->recommended_buffer.num,
- + port->recommended_buffer.size,
- + port->recommended_buffer.alignment);
- +
- + pr_debug("buffer current values num:%d size:%d align:%d\n",
- + port->current_buffer.num,
- + port->current_buffer.size, port->current_buffer.alignment);
- +
- + pr_debug("elementry stream: type:%d encoding:0x%x varient:0x%x\n",
- + port->format.type,
- + port->format.encoding, port->format.encoding_variant);
- +
- + pr_debug(" bitrate:%d flags:0x%x\n",
- + port->format.bitrate, port->format.flags);
- +
- + if (port->format.type == MMAL_ES_TYPE_VIDEO) {
- + pr_debug
- + ("es video format: width:%d height:%d colourspace:0x%x\n",
- + port->es.video.width, port->es.video.height,
- + port->es.video.color_space);
- +
- + pr_debug(" : crop xywh %d,%d,%d,%d\n",
- + port->es.video.crop.x,
- + port->es.video.crop.y,
- + port->es.video.crop.width, port->es.video.crop.height);
- + pr_debug(" : framerate %d/%d aspect %d/%d\n",
- + port->es.video.frame_rate.num,
- + port->es.video.frame_rate.den,
- + port->es.video.par.num, port->es.video.par.den);
- + }
- +}
- +
- +static void port_to_mmal_msg(struct vchiq_mmal_port *port, struct mmal_port *p)
- +{
- +
- + /* todo do readonly fields need setting at all? */
- + p->type = port->type;
- + p->index = port->index;
- + p->index_all = 0;
- + p->is_enabled = port->enabled;
- + p->buffer_num_min = port->minimum_buffer.num;
- + p->buffer_size_min = port->minimum_buffer.size;
- + p->buffer_alignment_min = port->minimum_buffer.alignment;
- + p->buffer_num_recommended = port->recommended_buffer.num;
- + p->buffer_size_recommended = port->recommended_buffer.size;
- +
- + /* only three writable fields in a port */
- + p->buffer_num = port->current_buffer.num;
- + p->buffer_size = port->current_buffer.size;
- + p->userdata = port;
- +}
- +
- +static int port_info_set(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + pr_debug("setting port info port %p\n", port);
- + if (!port)
- + return -1;
- + dump_port_info(port);
- +
- + m.h.type = MMAL_MSG_TYPE_PORT_INFO_SET;
- +
- + m.u.port_info_set.component_handle = port->component->handle;
- + m.u.port_info_set.port_type = port->type;
- + m.u.port_info_set.port_index = port->index;
- +
- + port_to_mmal_msg(port, &m.u.port_info_set.port);
- +
- + /* elementry stream format setup */
- + m.u.port_info_set.format.type = port->format.type;
- + m.u.port_info_set.format.encoding = port->format.encoding;
- + m.u.port_info_set.format.encoding_variant =
- + port->format.encoding_variant;
- + m.u.port_info_set.format.bitrate = port->format.bitrate;
- + m.u.port_info_set.format.flags = port->format.flags;
- +
- + memcpy(&m.u.port_info_set.es, &port->es,
- + sizeof(union mmal_es_specific_format));
- +
- + m.u.port_info_set.format.extradata_size = port->format.extradata_size;
- + memcpy(&m.u.port_info_set.extradata, port->format.extradata,
- + port->format.extradata_size);
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.port_info_set),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_INFO_SET) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + /* return operation status */
- + ret = -rmsg->u.port_info_get_reply.status;
- +
- + pr_debug("%s:result:%d component:0x%x port:%d\n", __func__, ret,
- + port->component->handle, port->handle);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +
- +}
- +
- +/* use port info get message to retrive port information */
- +static int port_info_get(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + /* port info time */
- + m.h.type = MMAL_MSG_TYPE_PORT_INFO_GET;
- + m.u.port_info_get.component_handle = port->component->handle;
- + m.u.port_info_get.port_type = port->type;
- + m.u.port_info_get.index = port->index;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.port_info_get),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_INFO_GET) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + /* return operation status */
- + ret = -rmsg->u.port_info_get_reply.status;
- + if (ret != MMAL_MSG_STATUS_SUCCESS)
- + goto release_msg;
- +
- + if (rmsg->u.port_info_get_reply.port.is_enabled == 0)
- + port->enabled = false;
- + else
- + port->enabled = true;
- +
- + /* copy the values out of the message */
- + port->handle = rmsg->u.port_info_get_reply.port_handle;
- +
- + /* port type and index cached to use on port info set becuase
- + * it does not use a port handle
- + */
- + port->type = rmsg->u.port_info_get_reply.port_type;
- + port->index = rmsg->u.port_info_get_reply.port_index;
- +
- + port->minimum_buffer.num =
- + rmsg->u.port_info_get_reply.port.buffer_num_min;
- + port->minimum_buffer.size =
- + rmsg->u.port_info_get_reply.port.buffer_size_min;
- + port->minimum_buffer.alignment =
- + rmsg->u.port_info_get_reply.port.buffer_alignment_min;
- +
- + port->recommended_buffer.alignment =
- + rmsg->u.port_info_get_reply.port.buffer_alignment_min;
- + port->recommended_buffer.num =
- + rmsg->u.port_info_get_reply.port.buffer_num_recommended;
- +
- + port->current_buffer.num = rmsg->u.port_info_get_reply.port.buffer_num;
- + port->current_buffer.size =
- + rmsg->u.port_info_get_reply.port.buffer_size;
- +
- + /* stream format */
- + port->format.type = rmsg->u.port_info_get_reply.format.type;
- + port->format.encoding = rmsg->u.port_info_get_reply.format.encoding;
- + port->format.encoding_variant =
- + rmsg->u.port_info_get_reply.format.encoding_variant;
- + port->format.bitrate = rmsg->u.port_info_get_reply.format.bitrate;
- + port->format.flags = rmsg->u.port_info_get_reply.format.flags;
- +
- + /* elementry stream format */
- + memcpy(&port->es,
- + &rmsg->u.port_info_get_reply.es,
- + sizeof(union mmal_es_specific_format));
- + port->format.es = &port->es;
- +
- + port->format.extradata_size =
- + rmsg->u.port_info_get_reply.format.extradata_size;
- + memcpy(port->format.extradata,
- + rmsg->u.port_info_get_reply.extradata,
- + port->format.extradata_size);
- +
- + pr_debug("received port info\n");
- + dump_port_info(port);
- +
- +release_msg:
- +
- + pr_debug("%s:result:%d component:0x%x port:%d\n",
- + __func__, ret, port->component->handle, port->handle);
- +
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* create comonent on vc */
- +static int create_component(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component,
- + const char *name)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + /* build component create message */
- + m.h.type = MMAL_MSG_TYPE_COMPONENT_CREATE;
- + m.u.component_create.client_component = component;
- + strncpy(m.u.component_create.name, name,
- + sizeof(m.u.component_create.name));
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.component_create),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != m.h.type) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.component_create_reply.status;
- + if (ret != MMAL_MSG_STATUS_SUCCESS)
- + goto release_msg;
- +
- + /* a valid component response received */
- + component->handle = rmsg->u.component_create_reply.component_handle;
- + component->inputs = rmsg->u.component_create_reply.input_num;
- + component->outputs = rmsg->u.component_create_reply.output_num;
- + component->clocks = rmsg->u.component_create_reply.clock_num;
- +
- + pr_debug("Component handle:0x%x in:%d out:%d clock:%d\n",
- + component->handle,
- + component->inputs, component->outputs, component->clocks);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* destroys a component on vc */
- +static int destroy_component(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_COMPONENT_DESTROY;
- + m.u.component_destroy.component_handle = component->handle;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.component_destroy),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != m.h.type) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.component_destroy_reply.status;
- +
- +release_msg:
- +
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* enable a component on vc */
- +static int enable_component(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_COMPONENT_ENABLE;
- + m.u.component_enable.component_handle = component->handle;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.component_enable),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != m.h.type) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.component_enable_reply.status;
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* disable a component on vc */
- +static int disable_component(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_COMPONENT_DISABLE;
- + m.u.component_disable.component_handle = component->handle;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.component_disable),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != m.h.type) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.component_disable_reply.status;
- +
- +release_msg:
- +
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* get version of mmal implementation */
- +static int get_version(struct vchiq_mmal_instance *instance,
- + u32 *major_out, u32 *minor_out)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_GET_VERSION;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.version),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != m.h.type) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + *major_out = rmsg->u.version.major;
- + *minor_out = rmsg->u.version.minor;
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* do a port action with a port as a parameter */
- +static int port_action_port(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + enum mmal_msg_port_action_type action_type)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_PORT_ACTION;
- + m.u.port_action_port.component_handle = port->component->handle;
- + m.u.port_action_port.port_handle = port->handle;
- + m.u.port_action_port.action = action_type;
- +
- + port_to_mmal_msg(port, &m.u.port_action_port.port);
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.port_action_port),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_ACTION) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.port_action_reply.status;
- +
- + pr_debug("%s:result:%d component:0x%x port:%d action:%s(%d)\n",
- + __func__,
- + ret, port->component->handle, port->handle,
- + port_action_type_names[action_type], action_type);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* do a port action with handles as parameters */
- +static int port_action_handle(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + enum mmal_msg_port_action_type action_type,
- + u32 connect_component_handle,
- + u32 connect_port_handle)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_PORT_ACTION;
- +
- + m.u.port_action_handle.component_handle = port->component->handle;
- + m.u.port_action_handle.port_handle = port->handle;
- + m.u.port_action_handle.action = action_type;
- +
- + m.u.port_action_handle.connect_component_handle =
- + connect_component_handle;
- + m.u.port_action_handle.connect_port_handle = connect_port_handle;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(m.u.port_action_handle),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_ACTION) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.port_action_reply.status;
- +
- + pr_debug("%s:result:%d component:0x%x port:%d action:%s(%d)" \
- + " connect component:0x%x connect port:%d\n",
- + __func__,
- + ret, port->component->handle, port->handle,
- + port_action_type_names[action_type],
- + action_type, connect_component_handle, connect_port_handle);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +static int port_parameter_set(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter_id, void *value, u32 value_size)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_PORT_PARAMETER_SET;
- +
- + m.u.port_parameter_set.component_handle = port->component->handle;
- + m.u.port_parameter_set.port_handle = port->handle;
- + m.u.port_parameter_set.id = parameter_id;
- + m.u.port_parameter_set.size = (2 * sizeof(u32)) + value_size;
- + memcpy(&m.u.port_parameter_set.value, value, value_size);
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + (4 * sizeof(u32)) + value_size,
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_PARAMETER_SET) {
- + /* got an unexpected message type in reply */
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.port_parameter_set_reply.status;
- +
- + pr_debug("%s:result:%d component:0x%x port:%d parameter:%d\n",
- + __func__,
- + ret, port->component->handle, port->handle, parameter_id);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +static int port_parameter_get(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter_id, void *value, u32 *value_size)
- +{
- + int ret;
- + struct mmal_msg m;
- + struct mmal_msg *rmsg;
- + VCHI_HELD_MSG_T rmsg_handle;
- +
- + m.h.type = MMAL_MSG_TYPE_PORT_PARAMETER_GET;
- +
- + m.u.port_parameter_get.component_handle = port->component->handle;
- + m.u.port_parameter_get.port_handle = port->handle;
- + m.u.port_parameter_get.id = parameter_id;
- + m.u.port_parameter_get.size = (2 * sizeof(u32)) + *value_size;
- +
- + ret = send_synchronous_mmal_msg(instance, &m,
- + sizeof(struct
- + mmal_msg_port_parameter_get),
- + &rmsg, &rmsg_handle);
- + if (ret)
- + return ret;
- +
- + if (rmsg->h.type != MMAL_MSG_TYPE_PORT_PARAMETER_GET) {
- + /* got an unexpected message type in reply */
- + pr_err("Incorrect reply type %d\n", rmsg->h.type);
- + ret = -EINVAL;
- + goto release_msg;
- + }
- +
- + ret = -rmsg->u.port_parameter_get_reply.status;
- + if (ret) {
- + /* Copy only as much as we have space for
- + * but report true size of parameter
- + */
- + memcpy(value, &rmsg->u.port_parameter_get_reply.value,
- + *value_size);
- + *value_size = rmsg->u.port_parameter_get_reply.size;
- + } else
- + memcpy(value, &rmsg->u.port_parameter_get_reply.value,
- + rmsg->u.port_parameter_get_reply.size);
- +
- + pr_debug("%s:result:%d component:0x%x port:%d parameter:%d\n", __func__,
- + ret, port->component->handle, port->handle, parameter_id);
- +
- +release_msg:
- + vchi_held_msg_release(&rmsg_handle);
- +
- + return ret;
- +}
- +
- +/* disables a port and drains buffers from it */
- +static int port_disable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- + struct list_head *q, *buf_head;
- + unsigned long flags = 0;
- +
- + if (!port->enabled)
- + return 0;
- +
- + port->enabled = false;
- +
- + ret = port_action_port(instance, port,
- + MMAL_MSG_PORT_ACTION_TYPE_DISABLE);
- + if (ret == 0) {
- +
- + /* drain all queued buffers on port */
- + spin_lock_irqsave(&port->slock, flags);
- +
- + list_for_each_safe(buf_head, q, &port->buffers) {
- + struct mmal_buffer *mmalbuf;
- + mmalbuf = list_entry(buf_head, struct mmal_buffer,
- + list);
- + list_del(buf_head);
- + if (port->buffer_cb)
- + port->buffer_cb(instance,
- + port, 0, mmalbuf, 0, 0,
- + MMAL_TIME_UNKNOWN,
- + MMAL_TIME_UNKNOWN);
- + }
- +
- + spin_unlock_irqrestore(&port->slock, flags);
- +
- + ret = port_info_get(instance, port);
- + }
- +
- + return ret;
- +}
- +
- +/* enable a port */
- +static int port_enable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + unsigned int hdr_count;
- + struct list_head *buf_head;
- + int ret;
- +
- + if (port->enabled)
- + return 0;
- +
- + /* ensure there are enough buffers queued to cover the buffer headers */
- + if (port->buffer_cb != NULL) {
- + hdr_count = 0;
- + list_for_each(buf_head, &port->buffers) {
- + hdr_count++;
- + }
- + if (hdr_count < port->current_buffer.num)
- + return -ENOSPC;
- + }
- +
- + ret = port_action_port(instance, port,
- + MMAL_MSG_PORT_ACTION_TYPE_ENABLE);
- + if (ret)
- + goto done;
- +
- + port->enabled = true;
- +
- + if (port->buffer_cb) {
- + /* send buffer headers to videocore */
- + hdr_count = 1;
- + list_for_each(buf_head, &port->buffers) {
- + struct mmal_buffer *mmalbuf;
- + mmalbuf = list_entry(buf_head, struct mmal_buffer,
- + list);
- + ret = buffer_from_host(instance, port, mmalbuf);
- + if (ret)
- + goto done;
- +
- + hdr_count++;
- + if (hdr_count > port->current_buffer.num)
- + break;
- + }
- + }
- +
- + ret = port_info_get(instance, port);
- +
- +done:
- + return ret;
- +}
- +
- +/* ------------------------------------------------------------------
- + * Exported API
- + *------------------------------------------------------------------*/
- +
- +int vchiq_mmal_port_set_format(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + ret = port_info_set(instance, port);
- + if (ret)
- + goto release_unlock;
- +
- + /* read what has actually been set */
- + ret = port_info_get(instance, port);
- +
- +release_unlock:
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +
- +}
- +
- +int vchiq_mmal_port_parameter_set(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter, void *value, u32 value_size)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + ret = port_parameter_set(instance, port, parameter, value, value_size);
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +int vchiq_mmal_port_parameter_get(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter, void *value, u32 *value_size)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + ret = port_parameter_get(instance, port, parameter, value, value_size);
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +/* enable a port
- + *
- + * enables a port and queues buffers for satisfying callbacks if we
- + * provide a callback handler
- + */
- +int vchiq_mmal_port_enable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + vchiq_mmal_buffer_cb buffer_cb)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + /* already enabled - noop */
- + if (port->enabled) {
- + ret = 0;
- + goto unlock;
- + }
- +
- + port->buffer_cb = buffer_cb;
- +
- + ret = port_enable(instance, port);
- +
- +unlock:
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +int vchiq_mmal_port_disable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + if (!port->enabled) {
- + mutex_unlock(&instance->vchiq_mutex);
- + return 0;
- + }
- +
- + ret = port_disable(instance, port);
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +/* ports will be connected in a tunneled manner so data buffers
- + * are not handled by client.
- + */
- +int vchiq_mmal_port_connect_tunnel(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *src,
- + struct vchiq_mmal_port *dst)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + /* disconnect ports if connected */
- + if (src->connected != NULL) {
- + ret = port_disable(instance, src);
- + if (ret) {
- + pr_err("failed disabling src port(%d)\n", ret);
- + goto release_unlock;
- + }
- +
- + /* do not need to disable the destination port as they
- + * are connected and it is done automatically
- + */
- +
- + ret = port_action_handle(instance, src,
- + MMAL_MSG_PORT_ACTION_TYPE_DISCONNECT,
- + src->connected->component->handle,
- + src->connected->handle);
- + if (ret < 0) {
- + pr_err("failed disconnecting src port\n");
- + goto release_unlock;
- + }
- + src->connected->enabled = false;
- + src->connected = NULL;
- + }
- +
- + if (dst == NULL) {
- + /* do not make new connection */
- + ret = 0;
- + pr_debug("not making new connection\n");
- + goto release_unlock;
- + }
- +
- + /* copy src port format to dst */
- + dst->format.encoding = src->format.encoding;
- + dst->es.video.width = src->es.video.width;
- + dst->es.video.height = src->es.video.height;
- + dst->es.video.crop.x = src->es.video.crop.x;
- + dst->es.video.crop.y = src->es.video.crop.y;
- + dst->es.video.crop.width = src->es.video.crop.width;
- + dst->es.video.crop.height = src->es.video.crop.height;
- + dst->es.video.frame_rate.num = src->es.video.frame_rate.num;
- + dst->es.video.frame_rate.den = src->es.video.frame_rate.den;
- +
- + /* set new format */
- + ret = port_info_set(instance, dst);
- + if (ret) {
- + pr_debug("setting port info failed\n");
- + goto release_unlock;
- + }
- +
- + /* read what has actually been set */
- + ret = port_info_get(instance, dst);
- + if (ret) {
- + pr_debug("read back port info failed\n");
- + goto release_unlock;
- + }
- +
- + /* connect two ports together */
- + ret = port_action_handle(instance, src,
- + MMAL_MSG_PORT_ACTION_TYPE_CONNECT,
- + dst->component->handle, dst->handle);
- + if (ret < 0) {
- + pr_debug("connecting port %d:%d to %d:%d failed\n",
- + src->component->handle, src->handle,
- + dst->component->handle, dst->handle);
- + goto release_unlock;
- + }
- + src->connected = dst;
- +
- +release_unlock:
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +int vchiq_mmal_submit_buffer(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + struct mmal_buffer *buffer)
- +{
- + unsigned long flags = 0;
- +
- + spin_lock_irqsave(&port->slock, flags);
- + list_add_tail(&buffer->list, &port->buffers);
- + spin_unlock_irqrestore(&port->slock, flags);
- +
- + /* the port previously underflowed because it was missing a
- + * mmal_buffer which has just been added, submit that buffer
- + * to the mmal service.
- + */
- + if (port->buffer_underflow) {
- + port_buffer_from_host(instance, port);
- + port->buffer_underflow--;
- + }
- +
- + return 0;
- +}
- +
- +/* Initialise a mmal component and its ports
- + *
- + */
- +int vchiq_mmal_component_init(struct vchiq_mmal_instance *instance,
- + const char *name,
- + struct vchiq_mmal_component **component_out)
- +{
- + int ret;
- + int idx; /* port index */
- + struct vchiq_mmal_component *component;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + if (instance->component_idx == VCHIQ_MMAL_MAX_COMPONENTS) {
- + ret = -EINVAL; /* todo is this correct error? */
- + goto unlock;
- + }
- +
- + component = &instance->component[instance->component_idx];
- +
- + ret = create_component(instance, component, name);
- + if (ret < 0)
- + goto unlock;
- +
- + /* ports info needs gathering */
- + component->control.type = MMAL_PORT_TYPE_CONTROL;
- + component->control.index = 0;
- + component->control.component = component;
- + spin_lock_init(&component->control.slock);
- + INIT_LIST_HEAD(&component->control.buffers);
- + ret = port_info_get(instance, &component->control);
- + if (ret < 0)
- + goto release_component;
- +
- + for (idx = 0; idx < component->inputs; idx++) {
- + component->input[idx].type = MMAL_PORT_TYPE_INPUT;
- + component->input[idx].index = idx;
- + component->input[idx].component = component;
- + spin_lock_init(&component->input[idx].slock);
- + INIT_LIST_HEAD(&component->input[idx].buffers);
- + ret = port_info_get(instance, &component->input[idx]);
- + if (ret < 0)
- + goto release_component;
- + }
- +
- + for (idx = 0; idx < component->outputs; idx++) {
- + component->output[idx].type = MMAL_PORT_TYPE_OUTPUT;
- + component->output[idx].index = idx;
- + component->output[idx].component = component;
- + spin_lock_init(&component->output[idx].slock);
- + INIT_LIST_HEAD(&component->output[idx].buffers);
- + ret = port_info_get(instance, &component->output[idx]);
- + if (ret < 0)
- + goto release_component;
- + }
- +
- + for (idx = 0; idx < component->clocks; idx++) {
- + component->clock[idx].type = MMAL_PORT_TYPE_CLOCK;
- + component->clock[idx].index = idx;
- + component->clock[idx].component = component;
- + spin_lock_init(&component->clock[idx].slock);
- + INIT_LIST_HEAD(&component->clock[idx].buffers);
- + ret = port_info_get(instance, &component->clock[idx]);
- + if (ret < 0)
- + goto release_component;
- + }
- +
- + instance->component_idx++;
- +
- + *component_out = component;
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return 0;
- +
- +release_component:
- + destroy_component(instance, component);
- +unlock:
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +/*
- + * cause a mmal component to be destroyed
- + */
- +int vchiq_mmal_component_finalise(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + if (component->enabled)
- + ret = disable_component(instance, component);
- +
- + ret = destroy_component(instance, component);
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +/*
- + * cause a mmal component to be enabled
- + */
- +int vchiq_mmal_component_enable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + if (component->enabled) {
- + mutex_unlock(&instance->vchiq_mutex);
- + return 0;
- + }
- +
- + ret = enable_component(instance, component);
- + if (ret == 0)
- + component->enabled = true;
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +/*
- + * cause a mmal component to be enabled
- + */
- +int vchiq_mmal_component_disable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + if (!component->enabled) {
- + mutex_unlock(&instance->vchiq_mutex);
- + return 0;
- + }
- +
- + ret = disable_component(instance, component);
- + if (ret == 0)
- + component->enabled = false;
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +int vchiq_mmal_version(struct vchiq_mmal_instance *instance,
- + u32 *major_out, u32 *minor_out)
- +{
- + int ret;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + ret = get_version(instance, major_out, minor_out);
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + return ret;
- +}
- +
- +int vchiq_mmal_finalise(struct vchiq_mmal_instance *instance)
- +{
- + int status = 0;
- +
- + if (instance == NULL)
- + return -EINVAL;
- +
- + if (mutex_lock_interruptible(&instance->vchiq_mutex))
- + return -EINTR;
- +
- + vchi_service_use(instance->handle);
- +
- + status = vchi_service_close(instance->handle);
- + if (status != 0)
- + pr_err("mmal-vchiq: VCHIQ close failed");
- +
- + mutex_unlock(&instance->vchiq_mutex);
- +
- + vfree(instance->bulk_scratch);
- +
- + kfree(instance);
- +
- + return status;
- +}
- +
- +int vchiq_mmal_init(struct vchiq_mmal_instance **out_instance)
- +{
- + int status;
- + struct vchiq_mmal_instance *instance;
- + static VCHI_CONNECTION_T *vchi_connection;
- + static VCHI_INSTANCE_T vchi_instance;
- + SERVICE_CREATION_T params = {
- + VCHI_VERSION_EX(VC_MMAL_VER, VC_MMAL_MIN_VER),
- + VC_MMAL_SERVER_NAME,
- + vchi_connection,
- + 0, /* rx fifo size (unused) */
- + 0, /* tx fifo size (unused) */
- + service_callback,
- + NULL, /* service callback parameter */
- + 1, /* unaligned bulk receives */
- + 1, /* unaligned bulk transmits */
- + 0 /* want crc check on bulk transfers */
- + };
- +
- + /* compile time checks to ensure structure size as they are
- + * directly (de)serialised from memory.
- + */
- +
- + /* ensure the header structure has packed to the correct size */
- + BUILD_BUG_ON(sizeof(struct mmal_msg_header) != 24);
- +
- + /* ensure message structure does not exceed maximum length */
- + BUILD_BUG_ON(sizeof(struct mmal_msg) > MMAL_MSG_MAX_SIZE);
- +
- + /* mmal port struct is correct size */
- + BUILD_BUG_ON(sizeof(struct mmal_port) != 64);
- +
- + /* create a vchi instance */
- + status = vchi_initialise(&vchi_instance);
- + if (status) {
- + pr_err("Failed to initialise VCHI instance (status=%d)\n",
- + status);
- + return -EIO;
- + }
- +
- + status = vchi_connect(NULL, 0, vchi_instance);
- + if (status) {
- + pr_err("Failed to connect VCHI instance (status=%d)\n", status);
- + return -EIO;
- + }
- +
- + instance = kmalloc(sizeof(*instance), GFP_KERNEL);
- + memset(instance, 0, sizeof(*instance));
- +
- + mutex_init(&instance->vchiq_mutex);
- + mutex_init(&instance->bulk_mutex);
- +
- + instance->bulk_scratch = vmalloc(PAGE_SIZE);
- +
- + params.callback_param = instance;
- +
- + status = vchi_service_open(vchi_instance, ¶ms, &instance->handle);
- + if (status) {
- + pr_err("Failed to open VCHI service connection (status=%d)\n",
- + status);
- + goto err_close_services;
- + }
- +
- + vchi_service_release(instance->handle);
- +
- + *out_instance = instance;
- +
- + return 0;
- +
- +err_close_services:
- +
- + vchi_service_close(instance->handle);
- + vfree(instance->bulk_scratch);
- + kfree(instance);
- + return -ENODEV;
- +}
- --- /dev/null
- +++ b/drivers/media/platform/bcm2835/mmal-vchiq.h
- @@ -0,0 +1,178 @@
- +/*
- + * Broadcom BM2835 V4L2 driver
- + *
- + * Copyright © 2013 Raspberry Pi (Trading) Ltd.
- + *
- + * This file is subject to the terms and conditions of the GNU General Public
- + * License. See the file COPYING in the main directory of this archive
- + * for more details.
- + *
- + * Authors: Vincent Sanders <vincent.sanders@collabora.co.uk>
- + * Dave Stevenson <dsteve@broadcom.com>
- + * Simon Mellor <simellor@broadcom.com>
- + * Luke Diamand <luked@broadcom.com>
- + *
- + * MMAL interface to VCHIQ message passing
- + */
- +
- +#ifndef MMAL_VCHIQ_H
- +#define MMAL_VCHIQ_H
- +
- +#include "mmal-msg-format.h"
- +
- +#define MAX_PORT_COUNT 4
- +
- +/* Maximum size of the format extradata. */
- +#define MMAL_FORMAT_EXTRADATA_MAX_SIZE 128
- +
- +struct vchiq_mmal_instance;
- +
- +enum vchiq_mmal_es_type {
- + MMAL_ES_TYPE_UNKNOWN, /**< Unknown elementary stream type */
- + MMAL_ES_TYPE_CONTROL, /**< Elementary stream of control commands */
- + MMAL_ES_TYPE_AUDIO, /**< Audio elementary stream */
- + MMAL_ES_TYPE_VIDEO, /**< Video elementary stream */
- + MMAL_ES_TYPE_SUBPICTURE /**< Sub-picture elementary stream */
- +};
- +
- +/* rectangle, used lots so it gets its own struct */
- +struct vchiq_mmal_rect {
- + s32 x;
- + s32 y;
- + s32 width;
- + s32 height;
- +};
- +
- +struct vchiq_mmal_port_buffer {
- + unsigned int num; /* number of buffers */
- + u32 size; /* size of buffers */
- + u32 alignment; /* alignment of buffers */
- +};
- +
- +struct vchiq_mmal_port;
- +
- +typedef void (*vchiq_mmal_buffer_cb)(
- + struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + int status, struct mmal_buffer *buffer,
- + unsigned long length, u32 mmal_flags, s64 dts, s64 pts);
- +
- +struct vchiq_mmal_port {
- + bool enabled;
- + u32 handle;
- + u32 type; /* port type, cached to use on port info set */
- + u32 index; /* port index, cached to use on port info set */
- +
- + /* component port belongs to, allows simple deref */
- + struct vchiq_mmal_component *component;
- +
- + struct vchiq_mmal_port *connected; /* port conencted to */
- +
- + /* buffer info */
- + struct vchiq_mmal_port_buffer minimum_buffer;
- + struct vchiq_mmal_port_buffer recommended_buffer;
- + struct vchiq_mmal_port_buffer current_buffer;
- +
- + /* stream format */
- + struct mmal_es_format format;
- + /* elementry stream format */
- + union mmal_es_specific_format es;
- +
- + /* data buffers to fill */
- + struct list_head buffers;
- + /* lock to serialise adding and removing buffers from list */
- + spinlock_t slock;
- + /* count of how many buffer header refils have failed because
- + * there was no buffer to satisfy them
- + */
- + int buffer_underflow;
- + /* callback on buffer completion */
- + vchiq_mmal_buffer_cb buffer_cb;
- + /* callback context */
- + void *cb_ctx;
- +};
- +
- +struct vchiq_mmal_component {
- + bool enabled;
- + u32 handle; /* VideoCore handle for component */
- + u32 inputs; /* Number of input ports */
- + u32 outputs; /* Number of output ports */
- + u32 clocks; /* Number of clock ports */
- + struct vchiq_mmal_port control; /* control port */
- + struct vchiq_mmal_port input[MAX_PORT_COUNT]; /* input ports */
- + struct vchiq_mmal_port output[MAX_PORT_COUNT]; /* output ports */
- + struct vchiq_mmal_port clock[MAX_PORT_COUNT]; /* clock ports */
- +};
- +
- +
- +int vchiq_mmal_init(struct vchiq_mmal_instance **out_instance);
- +int vchiq_mmal_finalise(struct vchiq_mmal_instance *instance);
- +
- +/* Initialise a mmal component and its ports
- +*
- +*/
- +int vchiq_mmal_component_init(
- + struct vchiq_mmal_instance *instance,
- + const char *name,
- + struct vchiq_mmal_component **component_out);
- +
- +int vchiq_mmal_component_finalise(
- + struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component);
- +
- +int vchiq_mmal_component_enable(
- + struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component);
- +
- +int vchiq_mmal_component_disable(
- + struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_component *component);
- +
- +
- +
- +/* enable a mmal port
- + *
- + * enables a port and if a buffer callback provided enque buffer
- + * headers as apropriate for the port.
- + */
- +int vchiq_mmal_port_enable(
- + struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + vchiq_mmal_buffer_cb buffer_cb);
- +
- +/* disable a port
- + *
- + * disable a port will dequeue any pending buffers
- + */
- +int vchiq_mmal_port_disable(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port);
- +
- +
- +int vchiq_mmal_port_parameter_set(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter,
- + void *value,
- + u32 value_size);
- +
- +int vchiq_mmal_port_parameter_get(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + u32 parameter,
- + void *value,
- + u32 *value_size);
- +
- +int vchiq_mmal_port_set_format(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port);
- +
- +int vchiq_mmal_port_connect_tunnel(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *src,
- + struct vchiq_mmal_port *dst);
- +
- +int vchiq_mmal_version(struct vchiq_mmal_instance *instance,
- + u32 *major_out,
- + u32 *minor_out);
- +
- +int vchiq_mmal_submit_buffer(struct vchiq_mmal_instance *instance,
- + struct vchiq_mmal_port *port,
- + struct mmal_buffer *buf);
- +
- +#endif /* MMAL_VCHIQ_H */
|