123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047 |
- /*
- * Gadget Driver for Android
- *
- * Copyright (C) 2008 Google, Inc.
- * Author: Mike Lockwood <lockwood@android.com>
- * Benoit Goby <benoit@android.com>
- *
- * This software is licensed under the terms of the GNU General Public
- * License version 2, as published by the Free Software Foundation, and
- * may be copied, distributed, and modified under those terms.
- *
- * 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.
- *
- */
- #include <linux/init.h>
- #include <linux/module.h>
- #include <linux/fs.h>
- #include <linux/delay.h>
- #include <linux/kernel.h>
- #include <linux/utsname.h>
- #include <linux/platform_device.h>
- #include <linux/pm_qos.h>
- #include <linux/of.h>
- #include <linux/usb/ch9.h>
- #include <linux/usb/composite.h>
- #include <linux/usb/gadget.h>
- #include <linux/usb/android.h>
- #include <mach/diag_dload.h>
- #include "gadget_chips.h"
- /*
- * Kbuild is not very cooperative with respect to linking separately
- * compiled library objects into one module. So for now we won't use
- * separate compilation ... ensuring init/exit sections work to shrink
- * the runtime footprint, and giving us at least some parts of what
- * a "gcc --combine ... part1.c part2.c part3.c ... " build would.
- */
- #include "usbstring.c"
- #include "config.c"
- #include "epautoconf.c"
- #include "composite.c"
- #include "f_diag.c"
- #include "f_qdss.c"
- #include "f_rmnet_smd.c"
- #include "f_rmnet_sdio.c"
- #include "f_rmnet_smd_sdio.c"
- #include "f_rmnet.c"
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- #include "f_gps.c"
- #endif
- #ifdef CONFIG_SND_PCM
- #include "f_audio_source.c"
- #endif
- #include "f_fs.c"
- #include "f_mass_storage.c"
- #include "u_serial.c"
- #include "u_sdio.c"
- #include "u_smd.c"
- #include "u_bam.c"
- #include "u_rmnet_ctrl_smd.c"
- #include "u_rmnet_ctrl_qti.c"
- #include "u_ctrl_hsic.c"
- #include "u_data_hsic.c"
- #include "u_ctrl_hsuart.c"
- #include "u_data_hsuart.c"
- #ifdef CONFIG_USB_DUN_SUPPORT
- #include "serial_acm.c"
- #endif
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- #include "f_serial.c"
- #endif
- #include "f_acm.c"
- #include "f_adb.c"
- #include "f_ccid.c"
- #ifdef CONFIG_SND_RAWMIDI
- #include "f_midi.c"
- #endif
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_MTP
- #include "f_mtp_samsung.c"
- #else
- #include "f_mtp.c"
- #endif
- #include "f_accessory.c"
- #define USB_ETH_RNDIS y
- #include "f_rndis.c"
- #include "rndis.c"
- #include "f_qc_ecm.c"
- #include "f_mbim.c"
- #include "u_bam_data.c"
- #include "f_ecm.c"
- #include "f_qc_rndis.c"
- #include "u_ether.c"
- #include "u_qc_ether.c"
- #include "f_hid.c"
- #ifdef CONFIG_TARGET_CORE
- #include "f_tcm.c"
- #endif
- #ifdef CONFIG_SND_PCM
- #include "u_uac1.c"
- #include "f_uac1.c"
- #endif
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- #include "f_ncm.c"
- #endif
- #ifdef CONFIG_USB_LOCK_SUPPORT_FOR_MDM
- #include <linux/power_supply.h>
- #endif
- MODULE_AUTHOR("Mike Lockwood");
- MODULE_DESCRIPTION("Android Composite USB Driver");
- MODULE_LICENSE("GPL");
- MODULE_VERSION("1.0");
- static const char longname[] = "Gadget Android";
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- static int composite_string_index;
- #endif
- /* Default vendor and product IDs, overridden by userspace */
- #define VENDOR_ID 0x18D1
- #define PRODUCT_ID 0x0001
- #define ANDROID_DEVICE_NODE_NAME_LENGTH 11
- /* f_midi configuration */
- #ifdef CONFIG_SND_RAWMIDI
- #define SNDRV_DEFAULT_IDX1 (-1)
- #define SNDRV_DEFAULT_STR1 NULL
- #define MIDI_INPUT_PORTS 1
- #define MIDI_OUTPUT_PORTS 1
- #define MIDI_BUFFER_SIZE 1024
- #define MIDI_QUEUE_LENGTH 32
- #endif
- struct android_usb_function {
- char *name;
- void *config;
- struct device *dev;
- char *dev_name;
- struct device_attribute **attributes;
- struct android_dev *android_dev;
- /* Optional: initialization during gadget bind */
- int (*init)(struct android_usb_function *, struct usb_composite_dev *);
- /* Optional: cleanup during gadget unbind */
- void (*cleanup)(struct android_usb_function *);
- /* Optional: called when the function is added the list of
- * enabled functions */
- void (*enable)(struct android_usb_function *);
- /* Optional: called when it is removed */
- void (*disable)(struct android_usb_function *);
- int (*bind_config)(struct android_usb_function *,
- struct usb_configuration *);
- /* Optional: called when the configuration is removed */
- void (*unbind_config)(struct android_usb_function *,
- struct usb_configuration *);
- /* Optional: handle ctrl requests before the device is configured */
- int (*ctrlrequest)(struct android_usb_function *,
- struct usb_composite_dev *,
- const struct usb_ctrlrequest *);
- };
- #if defined(CONFIG_SEC_H_PROJECT) || defined(CONFIG_SEC_F_PROJECT) || defined(CONFIG_SEC_K_PROJECT)
- u8 usb30en;
- extern int sec_set_speedlimit(struct usb_gadget *gadget,
- enum usb_device_speed speed);
- extern int sec_get_ss_host_available(struct usb_gadget *gadget);
- #endif
- struct android_usb_function_holder {
- struct android_usb_function *f;
- /* for android_conf.enabled_functions */
- struct list_head enabled_list;
- };
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_SIDESYNC
- #include "f_conn_gadget.c"
- #endif
- /**
- * struct android_dev - represents android USB gadget device
- * @name: device name.
- * @functions: an array of all the supported USB function
- * drivers that this gadget support but not necessarily
- * added to one of the gadget configurations.
- * @cdev: The internal composite device. Android gadget device
- * is a composite device, such that it can support configurations
- * with more than one function driver.
- * @dev: The kernel device that represents this android device.
- * @enabled: True if the android gadget is enabled, means all
- * the configurations were set and all function drivers were
- * bind and ready for USB enumeration.
- * @disable_depth: Number of times the device was disabled, after
- * symmetrical number of enables the device willl be enabled.
- * Used for controlling ADB userspace disable/enable requests.
- * @mutex: Internal mutex for protecting device member fields.
- * @pdata: Platform data fetched from the kernel device platfrom data.
- * @connected: True if got connect notification from the gadget UDC.
- * False if got disconnect notification from the gadget UDC.
- * @sw_connected: Equal to 'connected' only after the connect
- * notification was handled by the android gadget work function.
- * @suspended: True if got suspend notification from the gadget UDC.
- * False if got resume notification from the gadget UDC.
- * @sw_suspended: Equal to 'suspended' only after the susped
- * notification was handled by the android gadget work function.
- * @pm_qos: An attribute string that can be set by user space in order to
- * determine pm_qos policy. Set to 'high' for always demand pm_qos
- * when USB bus is connected and resumed. Set to 'low' for disable
- * any setting of pm_qos by this driver. Default = 'high'.
- * @work: workqueue used for handling notifications from the gadget UDC.
- * @configs: List of configurations currently configured into the device.
- * The android gadget supports more than one configuration. The host
- * may choose one configuration from the suggested.
- * @configs_num: Number of configurations currently configured and existing
- * in the configs list.
- * @list_item: This driver supports more than one android gadget device (for
- * example in order to support multiple USB cores), therefore this is
- * a item in a linked list of android devices.
- */
- struct android_dev {
- const char *name;
- struct android_usb_function **functions;
- struct usb_composite_dev *cdev;
- struct device *dev;
- void (*setup_complete)(struct usb_ep *ep,
- struct usb_request *req);
- bool enabled;
- int disable_depth;
- struct mutex mutex;
- struct android_usb_platform_data *pdata;
- bool connected;
- bool sw_connected;
- bool suspended;
- bool sw_suspended;
- char pm_qos[5];
- struct pm_qos_request pm_qos_req_dma;
- struct work_struct work;
- char ffs_aliases[256];
- #if defined(CONFIG_SEC_H_PROJECT)
- struct delayed_work usb_connection_work;
- int speaker_check_count;
- #endif
- #ifdef CONFIG_USB_LOCK_SUPPORT_FOR_MDM
- int usb_lock;
- #endif
- /* A list of struct android_configuration */
- struct list_head configs;
- int configs_num;
- /* A list node inside the android_dev_list */
- struct list_head list_item;
- };
- struct android_configuration {
- struct usb_configuration usb_config;
- /* A list of the functions supported by this config */
- struct list_head enabled_functions;
- /* A list node inside the struct android_dev.configs list */
- struct list_head list_item;
- };
- struct dload_struct __iomem *diag_dload;
- static struct class *android_class;
- static struct list_head android_dev_list;
- static int android_dev_count;
- static int android_bind_config(struct usb_configuration *c);
- static void android_unbind_config(struct usb_configuration *c);
- static struct android_dev *cdev_to_android_dev(struct usb_composite_dev *cdev);
- static struct android_configuration *alloc_android_config
- (struct android_dev *dev);
- static void free_android_config(struct android_dev *dev,
- struct android_configuration *conf);
- static int usb_diag_update_pid_and_serial_num(uint32_t pid, const char *snum);
- /* string IDs are assigned dynamically */
- #define STRING_MANUFACTURER_IDX 0
- #define STRING_PRODUCT_IDX 1
- #define STRING_SERIAL_IDX 2
- static char manufacturer_string[256];
- static char product_string[256];
- static char serial_string[256];
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- #include "u_ncm.c"
- #endif
- /* String Table */
- static struct usb_string strings_dev[] = {
- [STRING_MANUFACTURER_IDX].s = manufacturer_string,
- [STRING_PRODUCT_IDX].s = product_string,
- [STRING_SERIAL_IDX].s = serial_string,
- { } /* end of list */
- };
- static struct usb_gadget_strings stringtab_dev = {
- .language = 0x0409, /* en-us */
- .strings = strings_dev,
- };
- static struct usb_gadget_strings *dev_strings[] = {
- &stringtab_dev,
- NULL,
- };
- static struct usb_device_descriptor device_desc = {
- .bLength = sizeof(device_desc),
- .bDescriptorType = USB_DT_DEVICE,
- .bcdUSB = __constant_cpu_to_le16(0x0200),
- .bDeviceClass = USB_CLASS_PER_INTERFACE,
- .idVendor = __constant_cpu_to_le16(VENDOR_ID),
- .idProduct = __constant_cpu_to_le16(PRODUCT_ID),
- .bNumConfigurations = 1,
- };
- static struct usb_otg_descriptor otg_descriptor = {
- .bLength = sizeof otg_descriptor,
- .bDescriptorType = USB_DT_OTG,
- .bmAttributes = USB_OTG_SRP | USB_OTG_HNP,
- .bcdOTG = __constant_cpu_to_le16(0x0200),
- };
- static const struct usb_descriptor_header *otg_desc[] = {
- (struct usb_descriptor_header *) &otg_descriptor,
- NULL,
- };
- enum android_device_state {
- USB_DISCONNECTED,
- USB_CONNECTED,
- USB_CONFIGURED,
- USB_SUSPENDED,
- USB_RESUMED
- };
- static void android_pm_qos_update_latency(struct android_dev *dev, int vote)
- {
- struct android_usb_platform_data *pdata = dev->pdata;
- u32 swfi_latency = 0;
- static int last_vote = -1;
- if (!pdata || vote == last_vote
- || !pdata->swfi_latency)
- return;
- swfi_latency = pdata->swfi_latency + 1;
- if (vote)
- pm_qos_update_request(&dev->pm_qos_req_dma,
- swfi_latency);
- else
- pm_qos_update_request(&dev->pm_qos_req_dma,
- PM_QOS_DEFAULT_VALUE);
- last_vote = vote;
- }
- static void android_work(struct work_struct *data)
- {
- struct android_dev *dev = container_of(data, struct android_dev, work);
- struct usb_composite_dev *cdev = dev->cdev;
- char *disconnected[2] = { "USB_STATE=DISCONNECTED", NULL };
- char *connected[2] = { "USB_STATE=CONNECTED", NULL };
- char *configured[2] = { "USB_STATE=CONFIGURED", NULL };
- char *suspended[2] = { "USB_STATE=SUSPENDED", NULL };
- char *resumed[2] = { "USB_STATE=RESUMED", NULL };
- #if defined(CONFIG_SEC_H_PROJECT)
- char *cableconnect[2] = { "USB_CONNECTION=CONNECTED", NULL };
- #endif
- char **uevent_envp = NULL;
- static enum android_device_state last_uevent, next_state;
- unsigned long flags;
- int pm_qos_vote = -1;
- spin_lock_irqsave(&cdev->lock, flags);
- if (dev->suspended != dev->sw_suspended && cdev->config) {
- if (strncmp(dev->pm_qos, "low", 3))
- pm_qos_vote = dev->suspended ? 0 : 1;
- next_state = dev->suspended ? USB_SUSPENDED : USB_RESUMED;
- uevent_envp = dev->suspended ? suspended : resumed;
- } else if (cdev->config) {
- uevent_envp = configured;
- next_state = USB_CONFIGURED;
- } else if (dev->connected != dev->sw_connected) {
- uevent_envp = dev->connected ? connected : disconnected;
- next_state = dev->connected ? USB_CONNECTED : USB_DISCONNECTED;
- if (dev->connected && strncmp(dev->pm_qos, "low", 3))
- pm_qos_vote = 1;
- else if (!dev->connected || !strncmp(dev->pm_qos, "low", 3))
- pm_qos_vote = 0;
- }
- dev->sw_connected = dev->connected;
- dev->sw_suspended = dev->suspended;
- spin_unlock_irqrestore(&cdev->lock, flags);
- if (pm_qos_vote != -1)
- android_pm_qos_update_latency(dev, pm_qos_vote);
- if (uevent_envp) {
- /*
- * Some userspace modules, e.g. MTP, work correctly only if
- * CONFIGURED uevent is preceded by DISCONNECT uevent.
- * Check if we missed sending out a DISCONNECT uevent. This can
- * happen if host PC resets and configures device really quick.
- */
- if (((uevent_envp == connected) &&
- (last_uevent != USB_DISCONNECTED)) ||
- ((uevent_envp == configured) &&
- (last_uevent == USB_CONFIGURED))) {
- pr_info("%s: sent missed DISCONNECT event\n", __func__);
- kobject_uevent_env(&dev->dev->kobj, KOBJ_CHANGE,
- disconnected);
- // msleep(20);
- }
- /*
- * Before sending out CONFIGURED uevent give function drivers
- * a chance to wakeup userspace threads and notify disconnect
- */
- // if (uevent_envp == configured)
- // msleep(50);
- /* Do not notify on suspend / resume */
- if (next_state != USB_SUSPENDED && next_state != USB_RESUMED) {
- kobject_uevent_env(&dev->dev->kobj, KOBJ_CHANGE,
- uevent_envp);
- #if defined(CONFIG_SEC_H_PROJECT)
- if (uevent_envp == connected)
- kobject_uevent_env(&dev->dev->kobj, KOBJ_CHANGE, cableconnect);
- #endif
- last_uevent = next_state;
- }
- pr_info("%s: sent uevent %s\n", __func__, uevent_envp[0]);
- } else {
- pr_info("%s: did not send uevent (%d %d %pK)\n", __func__,
- dev->connected, dev->sw_connected, cdev->config);
- }
- }
- #if defined(CONFIG_SEC_H_PROJECT)
- extern int speaker_status;
- void usb_gadget_connect_work(struct work_struct *work)
- {
- struct android_dev *dev =
- container_of(work, struct android_dev, usb_connection_work.work);
- struct usb_composite_dev *cdev = dev->cdev;
- pr_info("%s: speaker_status: %d, speaker_check_count %d\n",
- __func__, speaker_status, dev->speaker_check_count);
- /* Max 3 sec */
- if (speaker_status > 0) {
- dev->speaker_check_count++;
- if (dev->speaker_check_count < 60) {
- schedule_delayed_work(&dev->usb_connection_work, msecs_to_jiffies(50));
- return;
- }
- }
- usb_gadget_connect(cdev->gadget);
- }
- void schedule_usb_gadget_connect_work(struct android_dev *dev)
- {
- char *ready[2] = { "USB_CONNECTION=READY", NULL };
- if (dev == NULL) return;
- pr_info("%s\n",__func__);
- if (work_busy(&dev->usb_connection_work.work)) {
- cancel_delayed_work(&dev->usb_connection_work);
- pr_info("%s canceling the work\n",__func__);
- }
- kobject_uevent_env(&dev->dev->kobj, KOBJ_CHANGE, ready);
- dev->speaker_check_count = 0;
- schedule_delayed_work(&dev->usb_connection_work, msecs_to_jiffies(1));
- }
- #endif
- static int android_enable(struct android_dev *dev)
- {
- struct usb_composite_dev *cdev = dev->cdev;
- struct android_configuration *conf;
- int err = 0;
- if (WARN_ON(!dev->disable_depth))
- return err;
- if (--dev->disable_depth == 0) {
- list_for_each_entry(conf, &dev->configs, list_item) {
- err = usb_add_config(cdev, &conf->usb_config,
- android_bind_config);
- if (err < 0) {
- pr_err("%s: usb_add_config failed : err: %d\n",
- __func__, err);
- dev->disable_depth++;
- return err;
- }
- }
- #if defined(CONFIG_SEC_H_PROJECT)
- schedule_usb_gadget_connect_work(dev);
- #else
- usb_gadget_connect(cdev->gadget);
- #endif
- }
- return err;
- }
- static void android_disable(struct android_dev *dev)
- {
- struct usb_composite_dev *cdev = dev->cdev;
- struct android_configuration *conf;
- if (dev->disable_depth++ == 0) {
- usb_gadget_disconnect(cdev->gadget);
- /* Cancel pending control requests */
- usb_ep_dequeue(cdev->gadget->ep0, cdev->req);
- list_for_each_entry(conf, &dev->configs, list_item)
- usb_remove_config(cdev, &conf->usb_config);
- }
- }
- /*-------------------------------------------------------------------------*/
- /* Supported functions initialization */
- struct functionfs_config {
- bool opened;
- bool enabled;
- struct ffs_data *data;
- struct android_dev *dev;
- };
- static int ffs_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- f->config = kzalloc(sizeof(struct functionfs_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return functionfs_init();
- }
- static void ffs_function_cleanup(struct android_usb_function *f)
- {
- functionfs_cleanup();
- kfree(f->config);
- }
- static void ffs_function_enable(struct android_usb_function *f)
- {
- struct android_dev *dev = f->android_dev;
- struct functionfs_config *config = f->config;
- config->enabled = true;
- /* Disable the gadget until the function is ready */
- if (!config->opened)
- android_disable(dev);
- }
- static void ffs_function_disable(struct android_usb_function *f)
- {
- struct android_dev *dev = f->android_dev;
- struct functionfs_config *config = f->config;
- config->enabled = false;
- /* Balance the disable that was called in closed_callback */
- if (!config->opened)
- android_enable(dev);
- }
- static int ffs_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct functionfs_config *config = f->config;
- return functionfs_bind_config(c->cdev, c, config->data);
- }
- static ssize_t
- ffs_aliases_show(struct device *pdev, struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev;
- int ret;
- dev = list_first_entry(&android_dev_list, struct android_dev,
- list_item);
- mutex_lock(&dev->mutex);
- ret = sprintf(buf, "%s\n", dev->ffs_aliases);
- mutex_unlock(&dev->mutex);
- return ret;
- }
- static ssize_t
- ffs_aliases_store(struct device *pdev, struct device_attribute *attr,
- const char *buf, size_t size)
- {
- struct android_dev *dev;
- char buff[256];
- dev = list_first_entry(&android_dev_list, struct android_dev,
- list_item);
- mutex_lock(&dev->mutex);
- if (dev->enabled) {
- mutex_unlock(&dev->mutex);
- return -EBUSY;
- }
- strlcpy(buff, buf, sizeof(buff));
- strlcpy(dev->ffs_aliases, strim(buff), sizeof(dev->ffs_aliases));
- mutex_unlock(&dev->mutex);
- return size;
- }
- static DEVICE_ATTR(aliases, S_IRUGO | S_IWUSR, ffs_aliases_show,
- ffs_aliases_store);
- static struct device_attribute *ffs_function_attributes[] = {
- &dev_attr_aliases,
- NULL
- };
- static struct android_usb_function ffs_function = {
- .name = "ffs",
- .init = ffs_function_init,
- .enable = ffs_function_enable,
- .disable = ffs_function_disable,
- .cleanup = ffs_function_cleanup,
- .bind_config = ffs_function_bind_config,
- .attributes = ffs_function_attributes,
- };
- static int functionfs_ready_callback(struct ffs_data *ffs)
- {
- struct android_dev *dev = ffs_function.android_dev;
- struct functionfs_config *config = ffs_function.config;
- int ret = 0;
- /* dev is null in case ADB is not in the composition */
- if (dev) {
- mutex_lock(&dev->mutex);
- ret = functionfs_bind(ffs, dev->cdev);
- if (ret) {
- mutex_unlock(&dev->mutex);
- return ret;
- }
- } else {
- /* android ffs_func requires daemon to start only after enable*/
- pr_debug("start adbd only in ADB composition\n");
- return -ENODEV;
- }
- config->data = ffs;
- config->opened = true;
- /* Save dev in case the adb function will get disabled */
- config->dev = dev;
- if (config->enabled)
- android_enable(dev);
- mutex_unlock(&dev->mutex);
- return 0;
- }
- static void functionfs_closed_callback(struct ffs_data *ffs)
- {
- struct android_dev *dev = ffs_function.android_dev;
- struct functionfs_config *config = ffs_function.config;
- /*
- * In case new composition is without ADB or ADB got disabled by the
- * time ffs_daemon was stopped then use saved one
- */
- if (!dev)
- dev = config->dev;
- /* fatal-error: It should never happen */
- if (!dev)
- pr_err("adb_closed_callback: config->dev is NULL");
- if (dev)
- mutex_lock(&dev->mutex);
- if (config->enabled && dev)
- android_disable(dev);
- config->dev = NULL;
- config->opened = false;
- config->data = NULL;
- functionfs_unbind(ffs);
- if (dev)
- mutex_unlock(&dev->mutex);
- }
- static void *functionfs_acquire_dev_callback(const char *dev_name)
- {
- return 0;
- }
- static void functionfs_release_dev_callback(struct ffs_data *ffs_data)
- {
- }
- struct adb_data {
- bool opened;
- bool enabled;
- struct android_dev *dev;
- };
- static int
- adb_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- f->config = kzalloc(sizeof(struct adb_data), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return adb_setup();
- }
- static void adb_function_cleanup(struct android_usb_function *f)
- {
- adb_cleanup();
- kfree(f->config);
- }
- static int
- adb_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return adb_bind_config(c);
- }
- static void adb_android_function_enable(struct android_usb_function *f)
- {
- struct android_dev *dev = f->android_dev;
- struct adb_data *data = f->config;
- data->enabled = true;
- /* Disable the gadget until adbd is ready */
- if (!data->opened)
- android_disable(dev);
- }
- static void adb_android_function_disable(struct android_usb_function *f)
- {
- struct android_dev *dev = f->android_dev;
- struct adb_data *data = f->config;
- data->enabled = false;
- /* Balance the disable that was called in closed_callback */
- if (!data->opened)
- android_enable(dev);
- }
- static struct android_usb_function adb_function = {
- .name = "adb",
- .enable = adb_android_function_enable,
- .disable = adb_android_function_disable,
- .init = adb_function_init,
- .cleanup = adb_function_cleanup,
- .bind_config = adb_function_bind_config,
- };
- static void adb_ready_callback(void)
- {
- struct android_dev *dev = adb_function.android_dev;
- struct adb_data *data = adb_function.config;
- /* dev is null in case ADB is not in the composition */
- if (dev)
- mutex_lock(&dev->mutex);
- /* Save dev in case the adb function will get disabled */
- data->dev = dev;
- data->opened = true;
- if (data->enabled && dev)
- android_enable(dev);
- if (dev)
- mutex_unlock(&dev->mutex);
- }
- static void adb_closed_callback(void)
- {
- struct adb_data *data = adb_function.config;
- struct android_dev *dev = adb_function.android_dev;
- /* In case new composition is without ADB, use saved one */
- if (!dev)
- dev = data->dev;
- if (!dev)
- pr_err("adb_closed_callback: data->dev is NULL");
- if (dev)
- mutex_lock(&dev->mutex);
- data->opened = false;
- if (data->enabled && dev)
- android_disable(dev);
- data->dev = NULL;
- if (dev)
- mutex_unlock(&dev->mutex);
- }
- #ifdef CONFIG_SND_RAWMIDI
- static int midi_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- struct midi_alsa_config *config;
- config = kzalloc(sizeof(struct midi_alsa_config), GFP_KERNEL);
- f->config = config;
- if (!config)
- return -ENOMEM;
- config->card = -1;
- config->device = -1;
- return 0;
- }
- static void midi_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- }
- static int midi_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct midi_alsa_config *config = f->config;
- return f_midi_bind_config(c, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1,
- MIDI_INPUT_PORTS, MIDI_OUTPUT_PORTS, MIDI_BUFFER_SIZE,
- MIDI_QUEUE_LENGTH, config);
- }
- static ssize_t midi_alsa_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct midi_alsa_config *config = f->config;
- /* print ALSA card and device numbers */
- return sprintf(buf, "%d %d\n", config->card, config->device);
- }
- static DEVICE_ATTR(alsa, S_IRUGO, midi_alsa_show, NULL);
- static struct device_attribute *midi_function_attributes[] = {
- &dev_attr_alsa,
- NULL
- };
- static struct android_usb_function midi_function = {
- .name = "midi",
- .init = midi_function_init,
- .cleanup = midi_function_cleanup,
- .bind_config = midi_function_bind_config,
- .attributes = midi_function_attributes,
- };
- #endif
- /*-------------------------------------------------------------------------*/
- /* Supported functions initialization */
- /* ACM */
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- #define MAX_ACM_INSTANCES 4
- struct acm_function_config {
- int instances;
- };
- static int acm_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- int ret;
- f->config = kzalloc(sizeof(struct acm_function_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- ret = gserial_setup(cdev->gadget, MAX_ACM_INSTANCES);
- return ret;
- }
- static void acm_function_cleanup(struct android_usb_function *f)
- {
- gserial_cleanup();
- kfree(f->config);
- f->config = NULL;
- }
- static int acm_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int i;
- int ret = 0;
- struct acm_function_config *config = f->config;
- for (i = 0; i < config->instances; i++) {
- ret = acm_bind_config(c, i);
- if (ret) {
- pr_err("Could not bind acm%u config\n", i);
- break;
- }
- }
- return ret;
- }
- static ssize_t acm_instances_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct acm_function_config *config = f->config;
- return sprintf(buf, "%d\n", config->instances);
- }
- static ssize_t acm_instances_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct acm_function_config *config = f->config;
- int value;
- sscanf(buf, "%d", &value);
- if (value > MAX_ACM_INSTANCES)
- value = MAX_ACM_INSTANCES;
- config->instances = value;
- return size;
- }
- static DEVICE_ATTR(instances, S_IRUGO | S_IWUSR,
- acm_instances_show, acm_instances_store);
- static struct device_attribute *acm_function_attributes[]
- = { &dev_attr_instances, NULL };
- static struct android_usb_function acm_function = {
- .name = "acm",
- .init = acm_function_init,
- .cleanup = acm_function_cleanup,
- .bind_config = acm_function_bind_config,
- .attributes = acm_function_attributes,
- };
- #else
- static char acm_transports[32]; /*enabled ACM ports - "tty[,sdio]"*/
- static ssize_t acm_transports_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(acm_transports, buff, sizeof(acm_transports));
- return size;
- }
- static DEVICE_ATTR(acm_transports, S_IWUSR, NULL, acm_transports_store);
- static struct device_attribute *acm_function_attributes[] = {
- &dev_attr_acm_transports,
- NULL
- };
- static void acm_function_cleanup(struct android_usb_function *f)
- {
- gserial_cleanup();
- }
- static int
- acm_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- char *name;
- char buf[32], *b;
- int err = -1, i;
- static int acm_initialized, ports;
- if (acm_initialized)
- goto bind_config;
- acm_initialized = 1;
- strlcpy(buf, acm_transports, sizeof(buf));
- b = strim(buf);
- while (b) {
- name = strsep(&b, ",");
- if (name) {
- err = acm_init_port(ports, name);
- if (err) {
- pr_err("acm: Cannot open port '%s'", name);
- goto out;
- }
- ports++;
- }
- }
- err = acm_port_setup(c);
- if (err) {
- pr_err("acm: Cannot setup transports");
- goto out;
- }
- bind_config:
- for (i = 0; i < ports; i++) {
- err = acm_bind_config(c, i);
- if (err) {
- pr_err("acm: bind_config failed for port %d", i);
- goto out;
- }
- }
- out:
- return err;
- }
- static struct android_usb_function acm_function = {
- .name = "acm",
- .cleanup = acm_function_cleanup,
- .bind_config = acm_function_bind_config,
- .attributes = acm_function_attributes,
- };
- #endif
- /* RMNET_SMD */
- static int rmnet_smd_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return rmnet_smd_bind_config(c);
- }
- static struct android_usb_function rmnet_smd_function = {
- .name = "rmnet_smd",
- .bind_config = rmnet_smd_function_bind_config,
- };
- /* RMNET_SDIO */
- static int rmnet_sdio_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return rmnet_sdio_function_add(c);
- }
- static struct android_usb_function rmnet_sdio_function = {
- .name = "rmnet_sdio",
- .bind_config = rmnet_sdio_function_bind_config,
- };
- /* RMNET_SMD_SDIO */
- static int rmnet_smd_sdio_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return rmnet_smd_sdio_init();
- }
- static void rmnet_smd_sdio_function_cleanup(struct android_usb_function *f)
- {
- rmnet_smd_sdio_cleanup();
- }
- static int rmnet_smd_sdio_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return rmnet_smd_sdio_function_add(c);
- }
- static struct device_attribute *rmnet_smd_sdio_attributes[] = {
- &dev_attr_transport, NULL };
- static struct android_usb_function rmnet_smd_sdio_function = {
- .name = "rmnet_smd_sdio",
- .init = rmnet_smd_sdio_function_init,
- .cleanup = rmnet_smd_sdio_function_cleanup,
- .bind_config = rmnet_smd_sdio_bind_config,
- .attributes = rmnet_smd_sdio_attributes,
- };
- /*rmnet transport string format(per port):"ctrl0,data0,ctrl1,data1..." */
- #define MAX_XPORT_STR_LEN 50
- static char rmnet_transports[MAX_XPORT_STR_LEN];
- /*rmnet transport name string - "rmnet_hsic[,rmnet_hsusb]" */
- static char rmnet_xport_names[MAX_XPORT_STR_LEN];
- static void rmnet_function_cleanup(struct android_usb_function *f)
- {
- frmnet_cleanup();
- }
- static int rmnet_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int i;
- int err = 0;
- char *ctrl_name;
- char *data_name;
- char *tname = NULL;
- char buf[MAX_XPORT_STR_LEN], *b;
- char xport_name_buf[MAX_XPORT_STR_LEN], *tb;
- static int rmnet_initialized, ports;
- if (!rmnet_initialized) {
- rmnet_initialized = 1;
- strlcpy(buf, rmnet_transports, sizeof(buf));
- b = strim(buf);
- strlcpy(xport_name_buf, rmnet_xport_names,
- sizeof(xport_name_buf));
- tb = strim(xport_name_buf);
- while (b) {
- ctrl_name = strsep(&b, ",");
- data_name = strsep(&b, ",");
- if (ctrl_name && data_name) {
- if (tb)
- tname = strsep(&tb, ",");
- err = frmnet_init_port(ctrl_name, data_name,
- tname);
- if (err) {
- pr_err("rmnet: Cannot open ctrl port:"
- "'%s' data port:'%s'\n",
- ctrl_name, data_name);
- goto out;
- }
- ports++;
- }
- }
- err = rmnet_gport_setup();
- if (err) {
- pr_err("rmnet: Cannot setup transports");
- goto out;
- }
- }
- for (i = 0; i < ports; i++) {
- err = frmnet_bind_config(c, i);
- if (err) {
- pr_err("Could not bind rmnet%u config\n", i);
- break;
- }
- }
- out:
- return err;
- }
- static ssize_t rmnet_transports_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- return snprintf(buf, PAGE_SIZE, "%s\n", rmnet_transports);
- }
- static ssize_t rmnet_transports_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(rmnet_transports, buff, sizeof(rmnet_transports));
- return size;
- }
- static ssize_t rmnet_xport_names_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- return snprintf(buf, PAGE_SIZE, "%s\n", rmnet_xport_names);
- }
- static ssize_t rmnet_xport_names_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(rmnet_xport_names, buff, sizeof(rmnet_xport_names));
- return size;
- }
- static struct device_attribute dev_attr_rmnet_transports =
- __ATTR(transports, S_IRUGO | S_IWUSR,
- rmnet_transports_show,
- rmnet_transports_store);
- static struct device_attribute dev_attr_rmnet_xport_names =
- __ATTR(transport_names, S_IRUGO | S_IWUSR,
- rmnet_xport_names_show,
- rmnet_xport_names_store);
- static struct device_attribute *rmnet_function_attributes[] = {
- &dev_attr_rmnet_transports,
- &dev_attr_rmnet_xport_names,
- NULL };
- static struct android_usb_function rmnet_function = {
- .name = "rmnet",
- .cleanup = rmnet_function_cleanup,
- .bind_config = rmnet_function_bind_config,
- .attributes = rmnet_function_attributes,
- };
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- static void gps_function_cleanup(struct android_usb_function *f)
- {
- gps_cleanup();
- }
- static int gps_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int err;
- static int gps_initialized;
- if (!gps_initialized) {
- gps_initialized = 1;
- err = gps_init_port();
- if (err) {
- pr_err("gps: Cannot init gps port");
- return err;
- }
- }
- err = gps_gport_setup();
- if (err) {
- pr_err("gps: Cannot setup transports");
- return err;
- }
- err = gps_bind_config(c);
- if (err) {
- pr_err("Could not bind gps config\n");
- return err;
- }
- return 0;
- }
- static struct android_usb_function gps_function = {
- .name = "gps",
- .cleanup = gps_function_cleanup,
- .bind_config = gps_function_bind_config,
- };
- #endif
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- /* ncm */
- struct ncm_function_config {
- u8 ethaddr[ETH_ALEN];
- };
- static int
- ncm_function_init(struct android_usb_function *f, struct usb_composite_dev *c)
- {
- f->config = kzalloc(sizeof(struct ncm_function_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return 0;
- }
- static void ncm_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- f->config = NULL;
- }
- static int
- ncm_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct ncm_function_config *ncm = f->config;
- int ret;
- if (!ncm) {
- pr_err("%s: ncm config is null\n", __func__);
- return -EINVAL;
- }
- pr_info("%s MAC: %02X:%02X:%02X:%02X:%02X:%02X\n", __func__,
- ncm->ethaddr[0], ncm->ethaddr[1], ncm->ethaddr[2],
- ncm->ethaddr[3], ncm->ethaddr[4], ncm->ethaddr[5]);
- ret = gether_setup_name(c->cdev->gadget, ncm->ethaddr, "ncm");
- if (ret) {
- pr_err("%s: gether setup failed err:%d\n", __func__, ret);
- return ret;
- }
- ret = ncm_bind_config(c, ncm->ethaddr);
- if (ret) {
- pr_err("%s: ncm bind config failed err:%d", __func__, ret);
- gether_cleanup();
- return ret;
- }
- return ret;
- }
- static void ncm_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- gether_cleanup();
- }
- static ssize_t ncm_ethaddr_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct ncm_function_config *ncm = f->config;
- return snprintf(buf, PAGE_SIZE, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- ncm->ethaddr[0], ncm->ethaddr[1], ncm->ethaddr[2],
- ncm->ethaddr[3], ncm->ethaddr[4], ncm->ethaddr[5]);
- }
- static ssize_t ncm_ethaddr_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct ncm_function_config *ncm = f->config;
- if (sscanf(buf, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- (int *)&ncm->ethaddr[0], (int *)&ncm->ethaddr[1],
- (int *)&ncm->ethaddr[2], (int *)&ncm->ethaddr[3],
- (int *)&ncm->ethaddr[4], (int *)&ncm->ethaddr[5]) == 6)
- return size;
- return -EINVAL;
- }
- static DEVICE_ATTR(ncm_ethaddr, S_IRUGO | S_IWUSR, ncm_ethaddr_show,
- ncm_ethaddr_store);
- static struct device_attribute *ncm_function_attributes[] = {
- &dev_attr_ncm_ethaddr,
- NULL
- };
- static struct android_usb_function ncm_function = {
- .name = "ncm",
- .init = ncm_function_init,
- .cleanup = ncm_function_cleanup,
- .bind_config = ncm_function_bind_config,
- .unbind_config = ncm_function_unbind_config,
- .attributes = ncm_function_attributes,
- };
- #endif
- /* ecm transport string */
- static char ecm_transports[MAX_XPORT_STR_LEN];
- struct ecm_function_config {
- u8 ethaddr[ETH_ALEN];
- };
- static int ecm_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- f->config = kzalloc(sizeof(struct ecm_function_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return 0;
- }
- static void ecm_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- f->config = NULL;
- }
- static int ecm_qc_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int ret;
- char *trans;
- struct ecm_function_config *ecm = f->config;
- if (!ecm) {
- pr_err("%s: ecm_pdata\n", __func__);
- return -EINVAL;
- }
- pr_debug("%s: ecm_transport is %s", __func__, ecm_transports);
- trans = strim(ecm_transports);
- if (strcmp("BAM2BAM_IPA", trans)) {
- ret = gether_qc_setup_name(c->cdev->gadget,
- ecm->ethaddr, "ecm");
- if (ret) {
- pr_err("%s: gether_setup failed\n", __func__);
- return ret;
- }
- }
- return ecm_qc_bind_config(c, ecm->ethaddr, trans);
- }
- static void ecm_qc_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- char *trans = strim(ecm_transports);
- if (strcmp("BAM2BAM_IPA", trans))
- gether_qc_cleanup_name("ecm0");
- }
- static ssize_t ecm_ethaddr_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct ecm_function_config *ecm = f->config;
- return snprintf(buf, PAGE_SIZE, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- ecm->ethaddr[0], ecm->ethaddr[1], ecm->ethaddr[2],
- ecm->ethaddr[3], ecm->ethaddr[4], ecm->ethaddr[5]);
- }
- static ssize_t ecm_ethaddr_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct ecm_function_config *ecm = f->config;
- if (sscanf(buf, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- (int *)&ecm->ethaddr[0], (int *)&ecm->ethaddr[1],
- (int *)&ecm->ethaddr[2], (int *)&ecm->ethaddr[3],
- (int *)&ecm->ethaddr[4], (int *)&ecm->ethaddr[5]) == 6)
- return size;
- return -EINVAL;
- }
- static DEVICE_ATTR(ecm_ethaddr, S_IRUGO | S_IWUSR, ecm_ethaddr_show,
- ecm_ethaddr_store);
- static ssize_t ecm_transports_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- return snprintf(buf, PAGE_SIZE, "%s\n", ecm_transports);
- }
- static ssize_t ecm_transports_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- strlcpy(ecm_transports, buf, sizeof(ecm_transports));
- return size;
- }
- static DEVICE_ATTR(ecm_transports, S_IRUGO | S_IWUSR, ecm_transports_show,
- ecm_transports_store);
- static struct device_attribute *ecm_function_attributes[] = {
- &dev_attr_ecm_transports,
- &dev_attr_ecm_ethaddr,
- NULL
- };
- static struct android_usb_function ecm_qc_function = {
- .name = "ecm_qc",
- .init = ecm_function_init,
- .cleanup = ecm_function_cleanup,
- .bind_config = ecm_qc_function_bind_config,
- .unbind_config = ecm_qc_function_unbind_config,
- .attributes = ecm_function_attributes,
- };
- /* MBIM - used with BAM */
- #define MAX_MBIM_INSTANCES 1
- static int mbim_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return mbim_init(MAX_MBIM_INSTANCES);
- }
- static void mbim_function_cleanup(struct android_usb_function *f)
- {
- fmbim_cleanup();
- }
- /* mbim transport string */
- static char mbim_transports[MAX_XPORT_STR_LEN];
- static int mbim_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- char *trans;
- pr_debug("%s: mbim transport is %s", __func__, mbim_transports);
- trans = strim(mbim_transports);
- return mbim_bind_config(c, 0, trans);
- }
- static int mbim_function_ctrlrequest(struct android_usb_function *f,
- struct usb_composite_dev *cdev,
- const struct usb_ctrlrequest *c)
- {
- return mbim_ctrlrequest(cdev, c);
- }
- static ssize_t mbim_transports_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- return snprintf(buf, PAGE_SIZE, "%s\n", mbim_transports);
- }
- static ssize_t mbim_transports_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- strlcpy(mbim_transports, buf, sizeof(mbim_transports));
- return size;
- }
- static DEVICE_ATTR(mbim_transports, S_IRUGO | S_IWUSR, mbim_transports_show,
- mbim_transports_store);
- static struct device_attribute *mbim_function_attributes[] = {
- &dev_attr_mbim_transports,
- NULL
- };
- static struct android_usb_function mbim_function = {
- .name = "usb_mbim",
- .cleanup = mbim_function_cleanup,
- .bind_config = mbim_function_bind_config,
- .init = mbim_function_init,
- .ctrlrequest = mbim_function_ctrlrequest,
- .attributes = mbim_function_attributes,
- };
- #ifdef CONFIG_SND_PCM
- /* PERIPHERAL AUDIO */
- static int audio_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return audio_bind_config(c);
- }
- static struct android_usb_function audio_function = {
- .name = "audio",
- .bind_config = audio_function_bind_config,
- };
- #endif
- static int hid_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return ghid_setup(cdev->gadget, 2);
- }
- static void hid_function_cleanup(struct android_usb_function *f)
- {
- ghid_cleanup();
- }
- static int hid_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return hidg_bind_config(c, NULL, 0);
- }
- static struct android_usb_function hid_function = {
- .name = "hid",
- .init = hid_function_init,
- .cleanup = hid_function_cleanup,
- .bind_config = hid_function_bind_config,
- };
- /* DIAG */
- static char diag_clients[32]; /*enabled DIAG clients- "diag[,diag_mdm]" */
- static ssize_t clients_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(diag_clients, buff, sizeof(diag_clients));
- return size;
- }
- static DEVICE_ATTR(clients, S_IWUSR, NULL, clients_store);
- static struct device_attribute *diag_function_attributes[] =
- { &dev_attr_clients, NULL };
- static int diag_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return diag_setup();
- }
- static void diag_function_cleanup(struct android_usb_function *f)
- {
- diag_cleanup();
- }
- static int diag_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- char *name;
- char buf[32], *b;
- int once = 0, err = -1;
- int (*notify)(uint32_t, const char *);
- struct android_dev *dev = cdev_to_android_dev(c->cdev);
- strlcpy(buf, diag_clients, sizeof(buf));
- b = strim(buf);
- while (b) {
- notify = NULL;
- name = strsep(&b, ",");
- /* Allow only first diag channel to update pid and serial no */
- if (!once++) {
- if (dev->pdata && dev->pdata->update_pid_and_serial_num)
- notify = dev->pdata->update_pid_and_serial_num;
- else
- notify = usb_diag_update_pid_and_serial_num;
- }
- if (name) {
- err = diag_function_add(c, name, notify);
- if (err)
- pr_err("diag: Cannot open channel '%s'", name);
- }
- }
- return err;
- }
- static struct android_usb_function diag_function = {
- .name = "diag",
- .init = diag_function_init,
- .cleanup = diag_function_cleanup,
- .bind_config = diag_function_bind_config,
- .attributes = diag_function_attributes,
- };
- /* DEBUG */
- static int qdss_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return qdss_setup();
- }
- static void qdss_function_cleanup(struct android_usb_function *f)
- {
- qdss_cleanup();
- }
- static int qdss_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int err = -1;
- err = qdss_bind_config(c, "qdss");
- if (err)
- pr_err("qdss: Cannot open channel qdss");
- return err;
- }
- static struct android_usb_function qdss_function = {
- .name = "qdss",
- .init = qdss_function_init,
- .cleanup = qdss_function_cleanup,
- .bind_config = qdss_function_bind_config,
- };
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- /* SERIAL */
- static char serial_transports[32]; /*enabled FSERIAL ports - "tty[,sdio]"*/
- static ssize_t serial_transports_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(serial_transports, buff, sizeof(serial_transports));
- return size;
- }
- /*enabled FSERIAL transport names - "serial_hsic[,serial_hsusb]"*/
- static char serial_xport_names[32];
- static ssize_t serial_xport_names_store(
- struct device *device, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- strlcpy(serial_xport_names, buff, sizeof(serial_xport_names));
- return size;
- }
- static ssize_t serial_xport_names_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- return snprintf(buf, PAGE_SIZE, "%s\n", serial_xport_names);
- }
- static DEVICE_ATTR(transports, S_IWUSR, NULL, serial_transports_store);
- static struct device_attribute dev_attr_serial_xport_names =
- __ATTR(transport_names, S_IRUGO | S_IWUSR,
- serial_xport_names_show,
- serial_xport_names_store);
- static struct device_attribute *serial_function_attributes[] = {
- &dev_attr_transports,
- &dev_attr_serial_xport_names,
- NULL };
- static void serial_function_cleanup(struct android_usb_function *f)
- {
- gserial_cleanup();
- }
- static int serial_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- char *name, *xport_name = NULL;
- char buf[32], *b, xport_name_buf[32], *tb;
- int err = -1, i;
- static int serial_initialized = 0, ports = 0;
- if (serial_initialized)
- goto bind_config;
- serial_initialized = 1;
- strlcpy(buf, serial_transports, sizeof(buf));
- b = strim(buf);
- strlcpy(xport_name_buf, serial_xport_names, sizeof(xport_name_buf));
- tb = strim(xport_name_buf);
- while (b) {
- name = strsep(&b, ",");
- if (name) {
- if (tb)
- xport_name = strsep(&tb, ",");
- err = gserial_init_port(ports, name, xport_name);
- if (err) {
- pr_err("serial: Cannot open port '%s'", name);
- goto out;
- }
- ports++;
- }
- }
- err = gport_setup(c);
- if (err) {
- pr_err("serial: Cannot setup transports");
- goto out;
- }
- bind_config:
- for (i = 0; i < ports; i++) {
- err = gser_bind_config(c, i);
- if (err) {
- pr_err("serial: bind_config failed for port %d", i);
- goto out;
- }
- }
- out:
- return err;
- }
- static struct android_usb_function serial_function = {
- .name = "serial",
- .cleanup = serial_function_cleanup,
- .bind_config = serial_function_bind_config,
- .attributes = serial_function_attributes,
- };
- #endif
- /* CCID */
- static int ccid_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return ccid_setup();
- }
- static void ccid_function_cleanup(struct android_usb_function *f)
- {
- ccid_cleanup();
- }
- static int ccid_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return ccid_bind_config(c);
- }
- static struct android_usb_function ccid_function = {
- .name = "ccid",
- .init = ccid_function_init,
- .cleanup = ccid_function_cleanup,
- .bind_config = ccid_function_bind_config,
- };
- static int
- mtp_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return mtp_setup();
- }
- static void mtp_function_cleanup(struct android_usb_function *f)
- {
- mtp_cleanup();
- }
- static int
- mtp_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return mtp_bind_config(c, false);
- }
- static int
- ptp_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- /* nothing to do - initialization is handled by mtp_function_init */
- return 0;
- }
- static void ptp_function_cleanup(struct android_usb_function *f)
- {
- /* nothing to do - cleanup is handled by mtp_function_cleanup */
- }
- static int
- ptp_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return mtp_bind_config(c, true);
- }
- static int mtp_function_ctrlrequest(struct android_usb_function *f,
- struct usb_composite_dev *cdev,
- const struct usb_ctrlrequest *c)
- {
- return mtp_ctrlrequest(cdev, c);
- }
- static struct android_usb_function mtp_function = {
- .name = "mtp",
- .init = mtp_function_init,
- .cleanup = mtp_function_cleanup,
- .bind_config = mtp_function_bind_config,
- .ctrlrequest = mtp_function_ctrlrequest,
- };
- /* PTP function is same as MTP with slightly different interface descriptor */
- static struct android_usb_function ptp_function = {
- .name = "ptp",
- .init = ptp_function_init,
- .cleanup = ptp_function_cleanup,
- .bind_config = ptp_function_bind_config,
- .ctrlrequest = mtp_function_ctrlrequest,
- };
- struct rndis_function_config {
- u8 ethaddr[ETH_ALEN];
- u32 vendorID;
- u8 max_pkt_per_xfer;
- char manufacturer[256];
- /* "Wireless" RNDIS; auto-detected by Windows */
- bool wceis;
- };
- static int
- rndis_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- f->config = kzalloc(sizeof(struct rndis_function_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return 0;
- }
- static void rndis_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- f->config = NULL;
- }
- static int rndis_qc_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- f->config = kzalloc(sizeof(struct rndis_function_config), GFP_KERNEL);
- if (!f->config)
- return -ENOMEM;
- return rndis_qc_init();
- }
- static void rndis_qc_function_cleanup(struct android_usb_function *f)
- {
- rndis_qc_cleanup();
- kfree(f->config);
- }
- static int
- rndis_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int ret;
- struct rndis_function_config *rndis = f->config;
- if (!rndis) {
- pr_err("%s: rndis_pdata\n", __func__);
- return -1;
- }
- pr_info("%s MAC: %02X:%02X:%02X:%02X:%02X:%02X\n", __func__,
- rndis->ethaddr[0], rndis->ethaddr[1], rndis->ethaddr[2],
- rndis->ethaddr[3], rndis->ethaddr[4], rndis->ethaddr[5]);
- ret = gether_setup_name(c->cdev->gadget, rndis->ethaddr, "rndis");
- if (ret) {
- pr_err("%s: gether_setup failed\n", __func__);
- return ret;
- }
- if (rndis->wceis) {
- /* "Wireless" RNDIS; auto-detected by Windows */
- rndis_iad_descriptor.bFunctionClass =
- USB_CLASS_WIRELESS_CONTROLLER;
- rndis_iad_descriptor.bFunctionSubClass = 0x01;
- rndis_iad_descriptor.bFunctionProtocol = 0x03;
- rndis_control_intf.bInterfaceClass =
- USB_CLASS_WIRELESS_CONTROLLER;
- rndis_control_intf.bInterfaceSubClass = 0x01;
- rndis_control_intf.bInterfaceProtocol = 0x03;
- }
- return rndis_bind_config_vendor(c, rndis->ethaddr, rndis->vendorID,
- rndis->manufacturer);
- }
- static int rndis_qc_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int ret;
- struct rndis_function_config *rndis = f->config;
- if (!rndis) {
- pr_err("%s: rndis_pdata\n", __func__);
- return -EINVAL;
- }
- pr_info("%s MAC: %02X:%02X:%02X:%02X:%02X:%02X\n", __func__,
- rndis->ethaddr[0], rndis->ethaddr[1], rndis->ethaddr[2],
- rndis->ethaddr[3], rndis->ethaddr[4], rndis->ethaddr[5]);
- ret = gether_qc_setup_name(c->cdev->gadget, rndis->ethaddr, "rndis");
- if (ret) {
- pr_err("%s: gether_setup failed\n", __func__);
- return ret;
- }
- if (rndis->wceis) {
- /* "Wireless" RNDIS; auto-detected by Windows */
- rndis_qc_iad_descriptor.bFunctionClass =
- USB_CLASS_WIRELESS_CONTROLLER;
- rndis_qc_iad_descriptor.bFunctionSubClass = 0x01;
- rndis_qc_iad_descriptor.bFunctionProtocol = 0x03;
- rndis_qc_control_intf.bInterfaceClass =
- USB_CLASS_WIRELESS_CONTROLLER;
- rndis_qc_control_intf.bInterfaceSubClass = 0x01;
- rndis_qc_control_intf.bInterfaceProtocol = 0x03;
- }
- return rndis_qc_bind_config_vendor(c, rndis->ethaddr, rndis->vendorID,
- rndis->manufacturer,
- rndis->max_pkt_per_xfer);
- }
- static void rndis_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- gether_cleanup();
- }
- static void rndis_qc_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- gether_qc_cleanup_name("rndis0");
- }
- static ssize_t rndis_manufacturer_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- return snprintf(buf, PAGE_SIZE, "%s\n", config->manufacturer);
- }
- static ssize_t rndis_manufacturer_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- if (size >= sizeof(config->manufacturer))
- return -EINVAL;
- if (sscanf(buf, "%255s", config->manufacturer) == 1)
- return size;
- return -1;
- }
- static DEVICE_ATTR(manufacturer, S_IRUGO | S_IWUSR, rndis_manufacturer_show,
- rndis_manufacturer_store);
- static ssize_t rndis_wceis_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- return snprintf(buf, PAGE_SIZE, "%d\n", config->wceis);
- }
- static ssize_t rndis_wceis_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- int value;
- if (sscanf(buf, "%d", &value) == 1) {
- config->wceis = value;
- return size;
- }
- return -EINVAL;
- }
- static DEVICE_ATTR(wceis, S_IRUGO | S_IWUSR, rndis_wceis_show,
- rndis_wceis_store);
- static ssize_t rndis_ethaddr_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *rndis = f->config;
- return snprintf(buf, PAGE_SIZE, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- rndis->ethaddr[0], rndis->ethaddr[1], rndis->ethaddr[2],
- rndis->ethaddr[3], rndis->ethaddr[4], rndis->ethaddr[5]);
- }
- static ssize_t rndis_ethaddr_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *rndis = f->config;
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- int i;
- char *src;
- for (i = 0; i < ETH_ALEN; i++)
- rndis->ethaddr[i] = 0;
- /* create a fake MAC address from our serial number.
- * first byte is 0x02 to signify locally administered.
- */
- rndis->ethaddr[0] = 0x02;
- src = serial_string;
- for (i = 0; (i < 256) && *src; i++) {
- /* XOR the USB serial across the remaining bytes */
- rndis->ethaddr[i % (ETH_ALEN - 1) + 1] ^= *src++;
- }
- return size;
- #else
- if (sscanf(buf, "%02x:%02x:%02x:%02x:%02x:%02x\n",
- (int *)&rndis->ethaddr[0], (int *)&rndis->ethaddr[1],
- (int *)&rndis->ethaddr[2], (int *)&rndis->ethaddr[3],
- (int *)&rndis->ethaddr[4], (int *)&rndis->ethaddr[5]) == 6)
- return size;
- return -EINVAL;
- #endif
- }
- static DEVICE_ATTR(ethaddr, S_IRUGO | S_IWUSR, rndis_ethaddr_show,
- rndis_ethaddr_store);
- static ssize_t rndis_vendorID_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- return snprintf(buf, PAGE_SIZE, "%04x\n", config->vendorID);
- }
- static ssize_t rndis_vendorID_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- int value;
- if (sscanf(buf, "%04x", &value) == 1) {
- config->vendorID = value;
- return size;
- }
- return -EINVAL;
- }
- static DEVICE_ATTR(vendorID, S_IRUGO | S_IWUSR, rndis_vendorID_show,
- rndis_vendorID_store);
- static ssize_t rndis_max_pkt_per_xfer_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- return snprintf(buf, PAGE_SIZE, "%d\n", config->max_pkt_per_xfer);
- }
- static ssize_t rndis_max_pkt_per_xfer_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct rndis_function_config *config = f->config;
- int value;
- if (sscanf(buf, "%d", &value) == 1) {
- config->max_pkt_per_xfer = value;
- return size;
- }
- return -EINVAL;
- }
- static DEVICE_ATTR(max_pkt_per_xfer, S_IRUGO | S_IWUSR,
- rndis_max_pkt_per_xfer_show,
- rndis_max_pkt_per_xfer_store);
- static struct device_attribute *rndis_function_attributes[] = {
- &dev_attr_manufacturer,
- &dev_attr_wceis,
- &dev_attr_ethaddr,
- &dev_attr_vendorID,
- &dev_attr_max_pkt_per_xfer,
- NULL
- };
- static struct android_usb_function rndis_function = {
- .name = "rndis",
- .init = rndis_function_init,
- .cleanup = rndis_function_cleanup,
- .bind_config = rndis_function_bind_config,
- .unbind_config = rndis_function_unbind_config,
- .attributes = rndis_function_attributes,
- };
- static struct android_usb_function rndis_qc_function = {
- .name = "rndis_qc",
- .init = rndis_qc_function_init,
- .cleanup = rndis_qc_function_cleanup,
- .bind_config = rndis_qc_function_bind_config,
- .unbind_config = rndis_qc_function_unbind_config,
- .attributes = rndis_function_attributes,
- };
- static int ecm_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- int ret;
- struct ecm_function_config *ecm = f->config;
- if (!ecm) {
- pr_err("%s: ecm_pdata\n", __func__);
- return -EINVAL;
- }
- ret = gether_setup_name(c->cdev->gadget, ecm->ethaddr, "ecm");
- if (ret) {
- pr_err("%s: gether_setup failed\n", __func__);
- return ret;
- }
- ret = ecm_bind_config(c, ecm->ethaddr);
- if (ret) {
- pr_err("%s: ecm_bind_config failed\n", __func__);
- gether_cleanup();
- }
- return ret;
- }
- static void ecm_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- gether_cleanup();
- }
- static struct android_usb_function ecm_function = {
- .name = "ecm",
- .init = ecm_function_init,
- .cleanup = ecm_function_cleanup,
- .bind_config = ecm_function_bind_config,
- .unbind_config = ecm_function_unbind_config,
- .attributes = ecm_function_attributes,
- };
- struct mass_storage_function_config {
- struct fsg_config fsg;
- struct fsg_common *common;
- };
- static int mass_storage_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- struct android_dev *dev = cdev_to_android_dev(cdev);
- #endif
- struct mass_storage_function_config *config;
- struct fsg_common *common;
- int err;
- int i;
- const char *name[3];
- config = kzalloc(sizeof(struct mass_storage_function_config),
- GFP_KERNEL);
- if (!config)
- return -ENOMEM;
- config->fsg.nluns = 1;
- name[0] = "lun";
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- config->fsg.luns[0].cdrom = 1;
- config->fsg.luns[0].ro = 0;
- config->fsg.luns[0].removable = 1;
- #else
- if (dev->pdata && dev->pdata->cdrom) {
- config->fsg.luns[config->fsg.nluns].cdrom = 1;
- config->fsg.luns[config->fsg.nluns].ro = 1;
- config->fsg.luns[config->fsg.nluns].removable = 0;
- name[config->fsg.nluns] = "lun0";
- config->fsg.nluns++;
- }
- if (dev->pdata && dev->pdata->internal_ums) {
- config->fsg.luns[config->fsg.nluns].cdrom = 0;
- config->fsg.luns[config->fsg.nluns].ro = 0;
- config->fsg.luns[config->fsg.nluns].removable = 1;
- name[config->fsg.nluns] = "lun1";
- config->fsg.nluns++;
- }
- config->fsg.luns[0].removable = 1;
- #endif
- common = fsg_common_init(NULL, cdev, &config->fsg);
- if (IS_ERR(common)) {
- kfree(config);
- return PTR_ERR(common);
- }
- for (i = 0; i < config->fsg.nluns; i++) {
- err = sysfs_create_link(&f->dev->kobj,
- &common->luns[i].dev.kobj,
- name[i]);
- if (err)
- goto error;
- }
- config->common = common;
- f->config = config;
- return 0;
- error:
- for (; i > 0 ; i--)
- sysfs_remove_link(&f->dev->kobj, name[i-1]);
- fsg_common_release(&common->ref);
- kfree(config);
- return err;
- }
- static void mass_storage_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- f->config = NULL;
- }
- static int mass_storage_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct mass_storage_function_config *config = f->config;
- return fsg_bind_config(c->cdev, c, config->common);
- }
- static ssize_t mass_storage_inquiry_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- return snprintf(buf, PAGE_SIZE, "%s\n", config->common->inquiry_string);
- }
- static ssize_t mass_storage_inquiry_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- if (size >= sizeof(config->common->inquiry_string))
- return -EINVAL;
- if (sscanf(buf, "%28s", config->common->inquiry_string) != 1)
- return -EINVAL;
- return size;
- }
- static DEVICE_ATTR(inquiry_string, S_IRUGO | S_IWUSR,
- mass_storage_inquiry_show,
- mass_storage_inquiry_store);
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- static ssize_t mass_storage_vendor_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- return sprintf(buf, "%s\n", config->common->vendor_string);
- }
- static ssize_t mass_storage_vendor_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- if (size >= sizeof(config->common->vendor_string))
- return -EINVAL;
- if (sscanf(buf, "%s", config->common->vendor_string) != 1)
- return -EINVAL;
- printk(KERN_DEBUG "%s: vendor %s", __func__,
- config->common->vendor_string);
- return size;
- }
- static DEVICE_ATTR(vendor_string, S_IRUGO | S_IWUSR,
- mass_storage_vendor_show,
- mass_storage_vendor_store);
- static ssize_t mass_storage_product_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- return sprintf(buf, "%s\n", config->common->product_string);
- }
- static ssize_t mass_storage_product_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- if (size >= sizeof(config->common->product_string))
- return -EINVAL;
- if (sscanf(buf, "%s", config->common->product_string) != 1)
- return -EINVAL;
- printk(KERN_DEBUG "%s: product %s", __func__,
- config->common->product_string);
- return size;
- }
- static DEVICE_ATTR(product_string, S_IRUGO | S_IWUSR,
- mass_storage_product_show,
- mass_storage_product_store);
- static ssize_t sua_version_info_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- int ret;
- ret = sprintf(buf, "%s\r\n",config->common-> version_string);
- printk(KERN_DEBUG "usb: %s version %s\n", __func__, buf);
- return ret;
- }
- /*
- /sys/class/android_usb/android0/f_mass_storage/sua_version_info
- */
- static ssize_t sua_version_info_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t size)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct mass_storage_function_config *config = f->config;
- int len=0;
- if (size < sizeof(config->common-> version_string))
- memcpy(config->common-> version_string,buf,size);
- else
- {
- len=sizeof(config->common-> version_string);
- memcpy(config->common-> version_string,buf,len-1);
- }
- printk(KERN_DEBUG "usb: %s buf=%s[%d], %s\n", __func__, buf,sizeof(buf),config->common-> version_string);
- return size;
- }
- static DEVICE_ATTR(sua_version_info, S_IRUGO | S_IWUSR,
- sua_version_info_show, sua_version_info_store);
- #endif
- static struct device_attribute *mass_storage_function_attributes[] = {
- &dev_attr_inquiry_string,
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- &dev_attr_vendor_string,
- &dev_attr_product_string,
- &dev_attr_sua_version_info,
- #endif
- NULL
- };
- static struct android_usb_function mass_storage_function = {
- .name = "mass_storage",
- .init = mass_storage_function_init,
- .cleanup = mass_storage_function_cleanup,
- .bind_config = mass_storage_function_bind_config,
- .attributes = mass_storage_function_attributes,
- };
- static int accessory_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return acc_setup();
- }
- static void accessory_function_cleanup(struct android_usb_function *f)
- {
- acc_cleanup();
- }
- static int accessory_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return acc_bind_config(c);
- }
- static int accessory_function_ctrlrequest(struct android_usb_function *f,
- struct usb_composite_dev *cdev,
- const struct usb_ctrlrequest *c)
- {
- return acc_ctrlrequest(cdev, c);
- }
- static struct android_usb_function accessory_function = {
- .name = "accessory",
- .init = accessory_function_init,
- .cleanup = accessory_function_cleanup,
- .bind_config = accessory_function_bind_config,
- .ctrlrequest = accessory_function_ctrlrequest,
- };
- #ifdef CONFIG_SND_PCM
- static int audio_source_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- struct audio_source_config *config;
- config = kzalloc(sizeof(struct audio_source_config), GFP_KERNEL);
- if (!config)
- return -ENOMEM;
- config->card = -1;
- config->device = -1;
- f->config = config;
- return 0;
- }
- static void audio_source_function_cleanup(struct android_usb_function *f)
- {
- kfree(f->config);
- }
- static int audio_source_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct audio_source_config *config = f->config;
- return audio_source_bind_config(c, config);
- }
- static void audio_source_function_unbind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- struct audio_source_config *config = f->config;
- config->card = -1;
- config->device = -1;
- }
- static ssize_t audio_source_pcm_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct android_usb_function *f = dev_get_drvdata(dev);
- struct audio_source_config *config = f->config;
- /* print PCM card and device numbers */
- return snprintf(buf, PAGE_SIZE,
- "%d %d\n", config->card, config->device);
- }
- static DEVICE_ATTR(pcm, S_IRUGO | S_IWUSR, audio_source_pcm_show, NULL);
- static struct device_attribute *audio_source_function_attributes[] = {
- &dev_attr_pcm,
- NULL
- };
- static struct android_usb_function audio_source_function = {
- .name = "audio_source",
- .init = audio_source_function_init,
- .cleanup = audio_source_function_cleanup,
- .bind_config = audio_source_function_bind_config,
- .unbind_config = audio_source_function_unbind_config,
- .attributes = audio_source_function_attributes,
- };
- #endif
- static int android_uasp_connect_cb(bool connect)
- {
- /*
- * TODO
- * We may have to disable gadget till UASP configfs nodes
- * are configured which includes mapping LUN with the
- * backing file. It is a fundamental difference between
- * f_mass_storage and f_tcp. That means UASP can not be
- * in default composition.
- *
- * For now, assume that UASP configfs nodes are configured
- * before enabling android gadget. Or cable should be
- * reconnected after mapping the LUN.
- *
- * Also consider making UASP to respond to Host requests when
- * Lun is not mapped.
- */
- pr_debug("UASP %s\n", connect ? "connect" : "disconnect");
- return 0;
- }
- static int uasp_function_init(struct android_usb_function *f,
- struct usb_composite_dev *cdev)
- {
- return f_tcm_init(&android_uasp_connect_cb);
- }
- static void uasp_function_cleanup(struct android_usb_function *f)
- {
- f_tcm_exit();
- }
- static int uasp_function_bind_config(struct android_usb_function *f,
- struct usb_configuration *c)
- {
- return tcm_bind_config(c);
- }
- static struct android_usb_function uasp_function = {
- .name = "uasp",
- .init = uasp_function_init,
- .cleanup = uasp_function_cleanup,
- .bind_config = uasp_function_bind_config,
- };
- static struct android_usb_function *supported_functions[] = {
- &ffs_function,
- &mbim_function,
- &ecm_qc_function,
- #ifdef CONFIG_SND_PCM
- &audio_function,
- #endif
- &rmnet_smd_function,
- &rmnet_sdio_function,
- &rmnet_smd_sdio_function,
- &rmnet_function,
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- &gps_function,
- #endif
- &diag_function,
- &qdss_function,
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- &serial_function,
- #endif
- &adb_function,
- #ifdef CONFIG_SND_RAWMIDI
- &midi_function,
- #endif
- &ccid_function,
- &acm_function,
- &mtp_function,
- &ptp_function,
- &rndis_function,
- &rndis_qc_function,
- &ecm_function,
- &ncm_function,
- &mass_storage_function,
- &accessory_function,
- &hid_function,
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_SIDESYNC
- &conn_gadget_function,
- #endif
- #ifdef CONFIG_SND_PCM
- &audio_source_function,
- #endif
- &uasp_function,
- NULL
- };
- static void android_cleanup_functions(struct android_usb_function **functions)
- {
- struct android_usb_function *f;
- struct device_attribute **attrs;
- struct device_attribute *attr;
- while (*functions) {
- f = *functions++;
- if (f->dev) {
- device_destroy(android_class, f->dev->devt);
- kfree(f->dev_name);
- } else
- continue;
- if (f->cleanup)
- f->cleanup(f);
- attrs = f->attributes;
- if (attrs) {
- while ((attr = *attrs++))
- device_remove_file(f->dev, attr);
- }
- }
- }
- static int android_init_functions(struct android_usb_function **functions,
- struct usb_composite_dev *cdev)
- {
- struct android_dev *dev = cdev_to_android_dev(cdev);
- struct android_usb_function *f;
- struct device_attribute **attrs;
- struct device_attribute *attr;
- int err = 0;
- int index = 1; /* index 0 is for android0 device */
- for (; (f = *functions++); index++) {
- f->dev_name = kasprintf(GFP_KERNEL, "f_%s", f->name);
- f->android_dev = NULL;
- if (!f->dev_name) {
- err = -ENOMEM;
- goto err_out;
- }
- f->dev = device_create(android_class, dev->dev,
- MKDEV(0, index), f, f->dev_name);
- if (IS_ERR(f->dev)) {
- pr_err("%s: Failed to create dev %s", __func__,
- f->dev_name);
- err = PTR_ERR(f->dev);
- f->dev = NULL;
- goto err_create;
- }
- if (f->init) {
- err = f->init(f, cdev);
- if (err) {
- pr_err("%s: Failed to init %s", __func__,
- f->name);
- goto err_init;
- }
- }
- attrs = f->attributes;
- if (attrs) {
- while ((attr = *attrs++) && !err)
- err = device_create_file(f->dev, attr);
- }
- if (err) {
- pr_err("%s: Failed to create function %s attributes",
- __func__, f->name);
- goto err_attrs;
- }
- }
- return 0;
- err_attrs:
- for (attr = *(attrs -= 2); attrs != f->attributes; attr = *(attrs--))
- device_remove_file(f->dev, attr);
- if (f->cleanup)
- f->cleanup(f);
- err_init:
- device_destroy(android_class, f->dev->devt);
- err_create:
- f->dev = NULL;
- kfree(f->dev_name);
- err_out:
- android_cleanup_functions(dev->functions);
- return err;
- }
- static int
- android_bind_enabled_functions(struct android_dev *dev,
- struct usb_configuration *c)
- {
- struct android_usb_function_holder *f_holder;
- struct android_configuration *conf =
- container_of(c, struct android_configuration, usb_config);
- int ret;
- list_for_each_entry(f_holder, &conf->enabled_functions, enabled_list) {
- ret = f_holder->f->bind_config(f_holder->f, c);
- if (ret) {
- pr_err("%s: %s failed\n", __func__, f_holder->f->name);
- while (!list_empty(&c->functions)) {
- struct usb_function *f;
- f = list_first_entry(&c->functions,
- struct usb_function, list);
- list_del(&f->list);
- if (f->unbind)
- f->unbind(c, f);
- }
- if (c->unbind)
- c->unbind(c);
- return ret;
- }
- }
- return 0;
- }
- static void
- android_unbind_enabled_functions(struct android_dev *dev,
- struct usb_configuration *c)
- {
- struct android_usb_function_holder *f_holder;
- struct android_configuration *conf =
- container_of(c, struct android_configuration, usb_config);
- list_for_each_entry(f_holder, &conf->enabled_functions, enabled_list) {
- if (f_holder->f->unbind_config)
- f_holder->f->unbind_config(f_holder->f, c);
- }
- }
- static inline void check_streaming_func(struct usb_gadget *gadget,
- struct android_usb_platform_data *pdata,
- char *name)
- {
- int i;
- for (i = 0; i < pdata->streaming_func_count; i++) {
- if (!strcmp(name,
- pdata->streaming_func[i])) {
- pr_debug("set streaming_enabled to true\n");
- gadget->streaming_enabled = true;
- break;
- }
- }
- }
- static int android_enable_function(struct android_dev *dev,
- struct android_configuration *conf,
- char *name)
- {
- struct android_usb_function **functions = dev->functions;
- struct android_usb_function *f;
- struct android_usb_function_holder *f_holder;
- struct android_usb_platform_data *pdata = dev->pdata;
- struct usb_gadget *gadget = dev->cdev->gadget;
- while ((f = *functions++)) {
- if (!strcmp(name, f->name)) {
- if (f->android_dev && f->android_dev != dev)
- pr_err("%s is enabled in other device\n",
- f->name);
- else {
- f_holder = kzalloc(sizeof(*f_holder),
- GFP_KERNEL);
- if (!f_holder) {
- pr_err("Failed to alloc f_holder\n");
- return -ENOMEM;
- }
- f->android_dev = dev;
- f_holder->f = f;
- list_add_tail(&f_holder->enabled_list,
- &conf->enabled_functions);
- pr_debug("func:%s is enabled.\n", f->name);
- /*
- * compare enable function with streaming func
- * list and based on the same request streaming.
- */
- check_streaming_func(gadget, pdata, f->name);
- return 0;
- }
- }
- }
- return -EINVAL;
- }
- /*-------------------------------------------------------------------------*/
- /* /sys/class/android_usb/android%d/ interface */
- static ssize_t remote_wakeup_show(struct device *pdev,
- struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct android_configuration *conf;
- /*
- * Show the wakeup attribute of the first configuration,
- * since all configurations have the same wakeup attribute
- */
- if (dev->configs_num == 0)
- return 0;
- conf = list_entry(dev->configs.next,
- struct android_configuration,
- list_item);
- return snprintf(buf, PAGE_SIZE, "%d\n",
- !!(conf->usb_config.bmAttributes &
- USB_CONFIG_ATT_WAKEUP));
- }
- static ssize_t remote_wakeup_store(struct device *pdev,
- struct device_attribute *attr, const char *buff, size_t size)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct android_configuration *conf;
- int enable = 0;
- sscanf(buff, "%d", &enable);
- pr_debug("android_usb: %s remote wakeup\n",
- enable ? "enabling" : "disabling");
- list_for_each_entry(conf, &dev->configs, list_item)
- if (enable)
- conf->usb_config.bmAttributes |=
- USB_CONFIG_ATT_WAKEUP;
- else
- conf->usb_config.bmAttributes &=
- ~USB_CONFIG_ATT_WAKEUP;
- return size;
- }
- static ssize_t
- functions_show(struct device *pdev, struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct android_configuration *conf;
- struct android_usb_function_holder *f_holder;
- char *buff = buf;
- mutex_lock(&dev->mutex);
- list_for_each_entry(conf, &dev->configs, list_item) {
- if (buff != buf)
- *(buff-1) = ':';
- list_for_each_entry(f_holder, &conf->enabled_functions,
- enabled_list)
- buff += snprintf(buff, PAGE_SIZE, "%s,",
- f_holder->f->name);
- }
- mutex_unlock(&dev->mutex);
- if (buff != buf)
- *(buff-1) = '\n';
- return buff - buf;
- }
- static ssize_t
- functions_store(struct device *pdev, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct list_head *curr_conf = &dev->configs;
- struct android_configuration *conf;
- char *conf_str;
- struct android_usb_function_holder *f_holder;
- char *name;
- char buf[256], *b;
- char aliases[256], *a;
- int err;
- int is_ffs;
- int ffs_enabled = 0;
- mutex_lock(&dev->mutex);
- if (dev->enabled) {
- mutex_unlock(&dev->mutex);
- return -EBUSY;
- }
- /* Clear previous enabled list */
- list_for_each_entry(conf, &dev->configs, list_item) {
- while (conf->enabled_functions.next !=
- &conf->enabled_functions) {
- f_holder = list_entry(conf->enabled_functions.next,
- typeof(*f_holder),
- enabled_list);
- f_holder->f->android_dev = NULL;
- list_del(&f_holder->enabled_list);
- kfree(f_holder);
- }
- INIT_LIST_HEAD(&conf->enabled_functions);
- }
- strlcpy(buf, buff, sizeof(buf));
- b = strim(buf);
- dev->cdev->gadget->streaming_enabled = false;
- while (b) {
- conf_str = strsep(&b, ":");
- if (conf_str) {
- /* If the next not equal to the head, take it */
- if (curr_conf->next != &dev->configs)
- conf = list_entry(curr_conf->next,
- struct android_configuration,
- list_item);
- else
- conf = alloc_android_config(dev);
- curr_conf = curr_conf->next;
- }
- while (conf_str) {
- name = strsep(&conf_str, ",");
- is_ffs = 0;
- strlcpy(aliases, dev->ffs_aliases, sizeof(aliases));
- a = aliases;
- while (a) {
- char *alias = strsep(&a, ",");
- if (alias && !strcmp(name, alias)) {
- is_ffs = 1;
- break;
- }
- }
- if (is_ffs) {
- if (ffs_enabled)
- continue;
- err = android_enable_function(dev, conf, "ffs");
- if (err)
- pr_err("android_usb: Cannot enable ffs (%d)",
- err);
- else
- ffs_enabled = 1;
- continue;
- }
- err = android_enable_function(dev, conf, name);
- if (err)
- pr_err("android_usb: Cannot enable '%s' (%d)",
- name, err);
- }
- }
- /* Free uneeded configurations if exists */
- while (curr_conf->next != &dev->configs) {
- conf = list_entry(curr_conf->next,
- struct android_configuration, list_item);
- free_android_config(dev, conf);
- }
- mutex_unlock(&dev->mutex);
- return size;
- }
- static ssize_t enable_show(struct device *pdev, struct device_attribute *attr,
- char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- printk(KERN_DEBUG "usb: %s dev->enabled=%d\n", __func__, dev->enabled);
- return snprintf(buf, PAGE_SIZE, "%d\n", dev->enabled);
- }
- static ssize_t enable_store(struct device *pdev, struct device_attribute *attr,
- const char *buff, size_t size)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct usb_composite_dev *cdev = dev->cdev;
- struct android_usb_function_holder *f_holder;
- struct android_configuration *conf;
- int enabled = 0;
- bool audio_enabled = false;
- static DEFINE_RATELIMIT_STATE(rl, 10*HZ, 1);
- int err = 0;
- if (!cdev)
- return -ENODEV;
- mutex_lock(&dev->mutex);
- sscanf(buff, "%d", &enabled);
- printk(KERN_INFO "usb: enabled: %d, dev->enabled: %d\n",
- enabled, dev->enabled);
- if (enabled && !dev->enabled) {
- /*
- * Update values in composite driver's copy of
- * device descriptor.
- */
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- cdev->next_string_id = composite_string_index;
- strncpy(manufacturer_string, "SAMSUNG",
- sizeof(manufacturer_string) - 1);
- strncpy(product_string, "SAMSUNG_Android",
- sizeof(product_string) - 1);
- #else
- cdev->next_string_id = 0;
- #endif
- cdev->desc.idVendor = device_desc.idVendor;
- cdev->desc.idProduct = device_desc.idProduct;
- if (device_desc.bcdDevice)
- cdev->desc.bcdDevice = device_desc.bcdDevice;
- cdev->desc.bDeviceClass = device_desc.bDeviceClass;
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- list_for_each_entry(conf, &dev->configs, list_item)
- list_for_each_entry(f_holder, &conf->enabled_functions,
- enabled_list) {
- printk(KERN_DEBUG "usb: %s f_holder->f:%s\n",
- __func__, f_holder->f->name);
- if (!strcmp(f_holder->f->name, "acm")) {
- printk(KERN_DEBUG "usb: acm is enabled. (bcdDevice=0x400)\n");
- /* Samsung KIES needs fixed bcdDevice number */
- cdev->desc.bcdDevice = cpu_to_le16(0x0400);
- }
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_SIDESYNC
- if (!strcmp(f_holder->f->name, "conn_gadget")) {
- if(cdev->desc.bcdDevice == cpu_to_le16(0x0400)) {
- printk(KERN_DEBUG "usb: conn_gadget + kies (bcdDevice=0xC00)\n");
- cdev->desc.bcdDevice = cpu_to_le16(0x0C00);
- } else {
- printk(KERN_DEBUG "usb: conn_gadget only (bcdDevice=0x800)\n");
- cdev->desc.bcdDevice = cpu_to_le16(0x0800);
- }
- }
- #endif
- }
- strncpy(manufacturer_string, "SAMSUNG", sizeof(manufacturer_string) - 1);
- strncpy(product_string, "SAMSUNG_Android", sizeof(product_string) - 1);
- #endif
- cdev->desc.bDeviceSubClass = device_desc.bDeviceSubClass;
- cdev->desc.bDeviceProtocol = device_desc.bDeviceProtocol;
- printk(KERN_DEBUG "usb: %s vendor=%x,product=%x,bcdDevice=%x",
- __func__, cdev->desc.idVendor,
- cdev->desc.idProduct, cdev->desc.bcdDevice);
- printk(KERN_DEBUG ",Class=%x,SubClass=%x,Protocol=%x\n",
- cdev->desc.bDeviceClass,
- cdev->desc.bDeviceSubClass,
- cdev->desc.bDeviceProtocol);
- printk(KERN_DEBUG "usb: %s next cmd : usb_add_config\n",
- __func__);
- /* Audio dock accessory is unable to enumerate device if
- * pull-up is enabled immediately. The enumeration is
- * reliable with 100 msec delay.
- */
- list_for_each_entry(conf, &dev->configs, list_item)
- list_for_each_entry(f_holder, &conf->enabled_functions,
- enabled_list) {
- if (f_holder->f->enable)
- f_holder->f->enable(f_holder->f);
- if (!strncmp(f_holder->f->name,
- "audio_source", 12))
- audio_enabled = true;
- }
- #ifndef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- if (audio_enabled)
- msleep(100);
- #endif
- err = android_enable(dev);
- if (err < 0) {
- pr_err("%s: android_enable failed\n", __func__);
- dev->connected = 0;
- dev->enabled = false;
- mutex_unlock(&dev->mutex);
- return size;
- }
- dev->enabled = true;
- } else if (!enabled && dev->enabled) {
- android_disable(dev);
- list_for_each_entry(conf, &dev->configs, list_item)
- list_for_each_entry(f_holder, &conf->enabled_functions,
- enabled_list) {
- if (f_holder->f->disable)
- f_holder->f->disable(f_holder->f);
- }
- dev->enabled = false;
- } else if (!enabled) {
- usb_gadget_disconnect(cdev->gadget);
- dev->enabled = false;
- } else if (__ratelimit(&rl)) {
- pr_err("android_usb: already %s\n",
- dev->enabled ? "enabled" : "disabled");
- }
- mutex_unlock(&dev->mutex);
- return size;
- }
- static ssize_t pm_qos_show(struct device *pdev,
- struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- return snprintf(buf, PAGE_SIZE, "%s\n", dev->pm_qos);
- }
- static ssize_t pm_qos_store(struct device *pdev,
- struct device_attribute *attr,
- const char *buff, size_t size)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- strlcpy(dev->pm_qos, buff, sizeof(dev->pm_qos));
- return size;
- }
- static ssize_t state_show(struct device *pdev, struct device_attribute *attr,
- char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct usb_composite_dev *cdev = dev->cdev;
- char *state = "DISCONNECTED";
- unsigned long flags;
- if (!cdev)
- goto out;
- spin_lock_irqsave(&cdev->lock, flags);
- if (cdev->config)
- state = "CONFIGURED";
- else if (dev->connected)
- state = "CONNECTED";
- spin_unlock_irqrestore(&cdev->lock, flags);
- out:
- printk(KERN_DEBUG "usb: %s buf=%s\n", __func__, state);
- return snprintf(buf, PAGE_SIZE, "%s\n", state);
- }
- #define DESCRIPTOR_ATTR(field, format_string) \
- static ssize_t \
- field ## _show(struct device *dev, struct device_attribute *attr, \
- char *buf) \
- { \
- return snprintf(buf, PAGE_SIZE, \
- format_string, device_desc.field); \
- } \
- static ssize_t \
- field ## _store(struct device *dev, struct device_attribute *attr, \
- const char *buf, size_t size) \
- { \
- int value; \
- if (sscanf(buf, format_string, &value) == 1) { \
- device_desc.field = value; \
- return size; \
- } \
- return -1; \
- } \
- static DEVICE_ATTR(field, S_IRUGO | S_IWUSR, field ## _show, field ## _store);
- #define DESCRIPTOR_STRING_ATTR(field, buffer) \
- static ssize_t \
- field ## _show(struct device *dev, struct device_attribute *attr, \
- char *buf) \
- { \
- return snprintf(buf, PAGE_SIZE, "%s", buffer); \
- } \
- static ssize_t \
- field ## _store(struct device *dev, struct device_attribute *attr, \
- const char *buf, size_t size) \
- { \
- if (size >= sizeof(buffer)) \
- return -EINVAL; \
- strlcpy(buffer, buf, sizeof(buffer)); \
- strim(buffer); \
- return size; \
- } \
- static DEVICE_ATTR(field, S_IRUGO | S_IWUSR, field ## _show, field ## _store);
- DESCRIPTOR_ATTR(idVendor, "%04x\n")
- DESCRIPTOR_ATTR(idProduct, "%04x\n")
- DESCRIPTOR_ATTR(bcdDevice, "%04x\n")
- DESCRIPTOR_ATTR(bDeviceClass, "%d\n")
- DESCRIPTOR_ATTR(bDeviceSubClass, "%d\n")
- DESCRIPTOR_ATTR(bDeviceProtocol, "%d\n")
- DESCRIPTOR_STRING_ATTR(iManufacturer, manufacturer_string)
- DESCRIPTOR_STRING_ATTR(iProduct, product_string)
- DESCRIPTOR_STRING_ATTR(iSerial, serial_string)
- static ssize_t
- bcdUSB_show(struct device *pdev, struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- if(dev->cdev) {
- return sprintf(buf, "%04x\n", dev->cdev->desc.bcdUSB);
- } else {
- return sprintf(buf, "%04x\n", device_desc.bcdUSB);
- }
- }
- #if defined(CONFIG_SEC_H_PROJECT) || defined(CONFIG_SEC_F_PROJECT) || defined(CONFIG_SEC_K_PROJECT)
- static ssize_t
- usb30en_show(struct device *pdev, struct device_attribute *attr, char *buf)
- {
- return sprintf(buf, "%01x\n", usb30en);
- }
- extern void set_redriver_power(int on);
- static ssize_t usb30en_store (struct device *pdev,
- struct device_attribute *attr,
- const char *buf, size_t size)
- {
- int value;
- struct android_dev *dev = dev_get_drvdata(pdev);
- if(get_host_os_type() == 0) {
- usb30en = 0;
- printk(KERN_ERR "usb: %s MAC host is always disabled \n", __func__);
- return size;
- }
- if (sscanf(buf, "%d", &value) == 1) {
- if (dev->cdev ) {
- if(usb30en == value){
- printk(KERN_INFO "usb: %s ignore!! usb30en(%d), value(%d)\n",
- __func__, usb30en, value);
- return size;
- }
- usb30en = value;
- usb_gadget_disconnect(dev->cdev->gadget);
- sec_set_speedlimit(dev->cdev->gadget,
- (usb30en ? USB_SPEED_SUPER : USB_SPEED_HIGH));
- printk(KERN_DEBUG "usb: %s B4 disconectng gadget\n", __func__);
- msleep(200);
- #if defined(CONFIG_SEC_H_PROJECT)
- if (!usb30en)
- schedule_usb_gadget_connect_work(dev);
- else
- #endif
- usb_gadget_connect(dev->cdev->gadget);
- printk(KERN_DEBUG "usb: %s after usb_gadget_connect\n", __func__);
- return size;
- }
- }
- printk(KERN_ERR "usb: %s Failed to set the usb30_en value\n", __func__);
- return -1;
- }
- static ssize_t ss_host_available_show(struct device *pdev,
- struct device_attribute *attr, char *buf)
- {
- int value = 0;
- int ss_host_available;
- int windowsos;
- struct android_dev *dev = dev_get_drvdata(pdev);
- if (dev->cdev ) {
- ss_host_available = sec_get_ss_host_available(dev->cdev->gadget);
- windowsos = get_host_os_type();
- if((ss_host_available == 1) && ( windowsos == 1)) {
- printk(KERN_ERR "usb: %s superspeed available \n", __func__);
- value = 1;
- }
- printk(KERN_ERR "usb: %s ss_host_available(%d), windowsos(%d)\n",
- __func__, ss_host_available, windowsos);
- } else {
- printk(KERN_ERR "usb: %s gadget not available \n", __func__);
- value = -1;
- }
- return sprintf(buf, "%d\n", value);
- }
- static ssize_t macos_show(struct device *pdev,
- struct device_attribute *attr, char *buf)
- {
- int value = 1;
- int ss_host_available;
- int windowsos;
- struct android_dev *dev = dev_get_drvdata(pdev);
- if (dev->cdev ) {
- ss_host_available = sec_get_ss_host_available(dev->cdev->gadget);
- windowsos = get_host_os_type();
- if((ss_host_available == 1) && ( windowsos == 1)) {
- printk(KERN_ERR "usb: %s superspeed available \n", __func__);
- value = 0;
- }
- printk(KERN_ERR "usb: %s ss_host_available(%d), windowsos(%d)\n",
- __func__, ss_host_available, windowsos);
- } else {
- printk(KERN_ERR "usb: %s gadget not available \n", __func__);
- value = -1;
- }
- return sprintf(buf, "%d\n", value);
- }
- #endif
- #ifdef CONFIG_USB_LOCK_SUPPORT_FOR_MDM
- static ssize_t show_usb_device_lock_state(struct device *pdev,
- struct device_attribute *attr, char *buf)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- if (!dev->usb_lock)
- return snprintf(buf, PAGE_SIZE, "USB_UNLOCK\n");
- else
- return snprintf(buf, PAGE_SIZE, "USB_LOCK\n");
- }
- static ssize_t store_usb_device_lock_state(struct device *pdev,
- struct device_attribute *attr, const char *buff, size_t count)
- {
- struct android_dev *dev = dev_get_drvdata(pdev);
- struct power_supply *psy;
- psy = power_supply_get_by_name("dwc-usb");
- if (!psy) {
- pr_info("%s: couldn't get usb power supply\n", __func__);
- return -EINVAL;
- }
- if (!strncmp(buff, "0", 1)){
- mutex_lock(&dev->mutex);
- dev->usb_lock = 0;
- android_enable(dev);
- mutex_unlock(&dev->mutex);
- } else if (!strncmp(buff, "1", 1)){
- mutex_lock(&dev->mutex);
- dev->usb_lock = 1;
- android_disable(dev);
- mutex_unlock(&dev->mutex);
- } else {
- pr_warn("%s: Wrong command\n", __func__);
- return count;
- }
- if(dev->usb_lock) {
- power_supply_set_present(psy, 0);
- pr_info("[%s][%d] : usb disconnect for support MDM\n",
- __func__,__LINE__);
- }
- return count;
- }
- #endif
- static DEVICE_ATTR(bcdUSB, S_IRUGO | S_IWUSR, bcdUSB_show, NULL);
- #if defined(CONFIG_SEC_H_PROJECT) || defined(CONFIG_SEC_F_PROJECT) || defined(CONFIG_SEC_K_PROJECT)
- static DEVICE_ATTR(usb30en,S_IRUGO | S_IWUSR, usb30en_show, usb30en_store);
- static DEVICE_ATTR(ss_host_available,S_IRUGO | S_IWUSR, ss_host_available_show, NULL);
- static DEVICE_ATTR(macos,S_IRUGO | S_IWUSR, macos_show, NULL);
- #endif
- static DEVICE_ATTR(functions, S_IRUGO | S_IWUSR, functions_show,
- functions_store);
- static DEVICE_ATTR(enable, S_IRUGO | S_IWUSR, enable_show, enable_store);
- static DEVICE_ATTR(pm_qos, S_IRUGO | S_IWUSR,
- pm_qos_show, pm_qos_store);
- static DEVICE_ATTR(state, S_IRUGO, state_show, NULL);
- static DEVICE_ATTR(remote_wakeup, S_IRUGO | S_IWUSR,
- remote_wakeup_show, remote_wakeup_store);
- #ifdef CONFIG_USB_LOCK_SUPPORT_FOR_MDM
- static DEVICE_ATTR(usb_lock, S_IRUGO | S_IWUSR,
- show_usb_device_lock_state, store_usb_device_lock_state);
- #endif
- static struct device_attribute *android_usb_attributes[] = {
- &dev_attr_idVendor,
- &dev_attr_idProduct,
- &dev_attr_bcdDevice,
- &dev_attr_bDeviceClass,
- &dev_attr_bDeviceSubClass,
- &dev_attr_bDeviceProtocol,
- &dev_attr_iManufacturer,
- &dev_attr_iProduct,
- &dev_attr_iSerial,
- &dev_attr_functions,
- &dev_attr_enable,
- &dev_attr_pm_qos,
- &dev_attr_state,
- &dev_attr_bcdUSB,
- #if defined(CONFIG_SEC_H_PROJECT) || defined(CONFIG_SEC_F_PROJECT) || defined(CONFIG_SEC_K_PROJECT)
- &dev_attr_usb30en,
- &dev_attr_ss_host_available,
- &dev_attr_macos,
- #endif
- &dev_attr_remote_wakeup,
- #ifdef CONFIG_USB_LOCK_SUPPORT_FOR_MDM
- &dev_attr_usb_lock,
- #endif
- NULL
- };
- /*-------------------------------------------------------------------------*/
- /* Composite driver */
- static int android_bind_config(struct usb_configuration *c)
- {
- struct android_dev *dev = cdev_to_android_dev(c->cdev);
- int ret = 0;
- ret = android_bind_enabled_functions(dev, c);
- if (ret)
- return ret;
- return 0;
- }
- static void android_unbind_config(struct usb_configuration *c)
- {
- struct android_dev *dev = cdev_to_android_dev(c->cdev);
- android_unbind_enabled_functions(dev, c);
- }
- static int android_bind(struct usb_composite_dev *cdev)
- {
- struct android_dev *dev;
- struct usb_gadget *gadget = cdev->gadget;
- struct android_configuration *conf;
- int gcnum, id, ret;
- /* Bind to the last android_dev that was probed */
- dev = list_entry(android_dev_list.prev, struct android_dev, list_item);
- dev->cdev = cdev;
- /* Save the default handler */
- dev->setup_complete = cdev->req->complete;
- printk(KERN_DEBUG "usb: %s disconnect\n", __func__);
- /*
- * Start disconnected. Userspace will connect the gadget once
- * it is done configuring the functions.
- */
- usb_gadget_disconnect(gadget);
- /* Init the supported functions only once, on the first android_dev */
- if (android_dev_count == 1) {
- ret = android_init_functions(dev->functions, cdev);
- if (ret)
- return ret;
- }
- /* Allocate string descriptor numbers ... note that string
- * contents can be overridden by the composite_dev glue.
- */
- id = usb_string_id(cdev);
- if (id < 0)
- return id;
- strings_dev[STRING_MANUFACTURER_IDX].id = id;
- device_desc.iManufacturer = id;
- id = usb_string_id(cdev);
- if (id < 0)
- return id;
- strings_dev[STRING_PRODUCT_IDX].id = id;
- device_desc.iProduct = id;
- /* Default strings - should be updated by userspace */
- strlcpy(manufacturer_string, "Android",
- sizeof(manufacturer_string) - 1);
- strlcpy(product_string, "Android", sizeof(product_string) - 1);
- strlcpy(serial_string, "0123456789ABCDEF", sizeof(serial_string) - 1);
- id = usb_string_id(cdev);
- if (id < 0)
- return id;
- strings_dev[STRING_SERIAL_IDX].id = id;
- device_desc.iSerialNumber = id;
- if (gadget_is_otg(cdev->gadget))
- list_for_each_entry(conf, &dev->configs, list_item)
- conf->usb_config.descriptors = otg_desc;
- gcnum = usb_gadget_controller_number(gadget);
- if (gcnum >= 0)
- device_desc.bcdDevice = cpu_to_le16(0x0200 + gcnum);
- else {
- pr_warning("%s: controller '%s' not recognized\n",
- longname, gadget->name);
- device_desc.bcdDevice = __constant_cpu_to_le16(0x9999);
- }
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- composite_string_index = 4;
- #endif
- return 0;
- }
- static int android_usb_unbind(struct usb_composite_dev *cdev)
- {
- struct android_dev *dev = cdev_to_android_dev(cdev);
- printk(KERN_DEBUG "usb: %s\n", __func__);
- manufacturer_string[0] = '\0';
- product_string[0] = '\0';
- serial_string[0] = '0';
- cancel_work_sync(&dev->work);
- android_cleanup_functions(dev->functions);
- return 0;
- }
- static struct usb_composite_driver android_usb_driver = {
- .name = "android_usb",
- .dev = &device_desc,
- .strings = dev_strings,
- .unbind = android_usb_unbind,
- #if defined(CONFIG_SEC_LT03_PROJECT) || defined(CONFIG_SEC_MONDRIAN_PROJECT)\
- || defined(CONFIG_SEC_KS01_PROJECT) || defined(CONFIG_SEC_PICASSO_PROJECT)\
- || defined(CONFIG_SEC_KACTIVE_PROJECT) || defined(CONFIG_SEC_FRESCO_PROJECT)\
- || defined(CONFIG_SEC_KSPORTS_PROJECT) || defined(CONFIG_SEC_JACTIVE_PROJECT)\
- || defined(CONFIG_SEC_S_PROJECT) || defined(CONFIG_SEC_PATEK_PROJECT)\
- || defined(CONFIG_SEC_CHAGALL_PROJECT) || defined(CONFIG_SEC_KLIMT_PROJECT)\
- || defined(CONFIG_MACH_JS01LTEDCM) ||defined(CONFIG_MACH_JSGLTE_CHN_CMCC)
- .max_speed = USB_SPEED_HIGH
- #else
- .max_speed = USB_SPEED_SUPER
- #endif
- };
- static int
- android_setup(struct usb_gadget *gadget, const struct usb_ctrlrequest *c)
- {
- struct usb_composite_dev *cdev = get_gadget_data(gadget);
- struct android_dev *dev = cdev_to_android_dev(cdev);
- struct usb_request *req = cdev->req;
- struct android_usb_function *f;
- struct android_usb_function_holder *f_holder;
- struct android_configuration *conf;
- int value = -EOPNOTSUPP;
- unsigned long flags;
- bool do_work = false;
- bool prev_configured = false;
- req->zero = 0;
- req->complete = composite_setup_complete;
- req->length = 0;
- req->complete = dev->setup_complete;
- gadget->ep0->driver_data = cdev;
- list_for_each_entry(conf, &dev->configs, list_item)
- list_for_each_entry(f_holder,
- &conf->enabled_functions,
- enabled_list) {
- f = f_holder->f;
- if (f->ctrlrequest) {
- value = f->ctrlrequest(f, cdev, c);
- if (value >= 0)
- break;
- }
- }
- /*
- * skip the work when 2nd set config arrives
- * with same value from the host.
- */
- if (cdev->config)
- prev_configured = true;
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- if (value < 0)
- value = terminal_ctrl_request(cdev, c);
- #endif
- /* Special case the accessory function.
- * It needs to handle control requests before it is enabled.
- */
- if (value < 0)
- value = acc_ctrlrequest(cdev, c);
- if (value < 0)
- value = composite_setup(gadget, c);
- spin_lock_irqsave(&cdev->lock, flags);
- if (!dev->connected) {
- dev->connected = 1;
- do_work = true;
- } else if (c->bRequest == USB_REQ_SET_CONFIGURATION &&
- cdev->config) {
- if (!prev_configured)
- do_work = true;
- }
- spin_unlock_irqrestore(&cdev->lock, flags);
- if (do_work)
- schedule_work(&dev->work);
- return value;
- }
- static void android_disconnect(struct usb_gadget *gadget)
- {
- struct usb_composite_dev *cdev = get_gadget_data(gadget);
- struct android_dev *dev = cdev_to_android_dev(cdev);
- unsigned long flags;
- composite_disconnect(gadget);
- /* accessory HID support can be active while the
- accessory function is not actually enabled,
- so we need to inform it when we are disconnected.
- */
- acc_disconnect();
- spin_lock_irqsave(&cdev->lock, flags);
- dev->connected = 0;
- schedule_work(&dev->work);
- spin_unlock_irqrestore(&cdev->lock, flags);
- }
- static void android_suspend(struct usb_gadget *gadget)
- {
- struct usb_composite_dev *cdev = get_gadget_data(gadget);
- struct android_dev *dev = cdev_to_android_dev(cdev);
- unsigned long flags;
- spin_lock_irqsave(&cdev->lock, flags);
- if (!dev->suspended) {
- dev->suspended = 1;
- schedule_work(&dev->work);
- }
- spin_unlock_irqrestore(&cdev->lock, flags);
- composite_suspend(gadget);
- }
- static void android_resume(struct usb_gadget *gadget)
- {
- struct usb_composite_dev *cdev = get_gadget_data(gadget);
- struct android_dev *dev = cdev_to_android_dev(cdev);
- unsigned long flags;
- spin_lock_irqsave(&cdev->lock, flags);
- if (dev->suspended) {
- dev->suspended = 0;
- schedule_work(&dev->work);
- }
- spin_unlock_irqrestore(&cdev->lock, flags);
- composite_resume(gadget);
- }
- static int android_create_device(struct android_dev *dev, u8 usb_core_id)
- {
- struct device_attribute **attrs = android_usb_attributes;
- struct device_attribute *attr;
- char device_node_name[ANDROID_DEVICE_NODE_NAME_LENGTH];
- int err;
- /*
- * The primary usb core should always have usb_core_id=0, since
- * Android user space is currently interested in android0 events.
- */
- snprintf(device_node_name, ANDROID_DEVICE_NODE_NAME_LENGTH,
- "android%d", usb_core_id);
- dev->dev = device_create(android_class, NULL,
- MKDEV(0, 0), NULL, device_node_name);
- if (IS_ERR(dev->dev))
- return PTR_ERR(dev->dev);
- dev_set_drvdata(dev->dev, dev);
- while ((attr = *attrs++)) {
- err = device_create_file(dev->dev, attr);
- if (err) {
- device_destroy(android_class, dev->dev->devt);
- return err;
- }
- }
- return 0;
- }
- static void android_destroy_device(struct android_dev *dev)
- {
- struct device_attribute **attrs = android_usb_attributes;
- struct device_attribute *attr;
- while ((attr = *attrs++))
- device_remove_file(dev->dev, attr);
- device_destroy(android_class, dev->dev->devt);
- }
- static struct android_dev *cdev_to_android_dev(struct usb_composite_dev *cdev)
- {
- struct android_dev *dev = NULL;
- /* Find the android dev from the list */
- list_for_each_entry(dev, &android_dev_list, list_item) {
- if (dev->cdev == cdev)
- break;
- }
- return dev;
- }
- static struct android_configuration *alloc_android_config
- (struct android_dev *dev)
- {
- struct android_configuration *conf;
- conf = kzalloc(sizeof(*conf), GFP_KERNEL);
- if (!conf) {
- pr_err("%s(): Failed to alloc memory for android conf\n",
- __func__);
- return ERR_PTR(-ENOMEM);
- }
- dev->configs_num++;
- conf->usb_config.label = dev->name;
- conf->usb_config.unbind = android_unbind_config;
- conf->usb_config.bConfigurationValue = dev->configs_num;
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- conf->usb_config.bmAttributes
- = USB_CONFIG_ATT_ONE | USB_CONFIG_ATT_SELFPOWER;
- conf->usb_config.bMaxPower = 0x30; /* 96ma */
- #endif
- INIT_LIST_HEAD(&conf->enabled_functions);
- list_add_tail(&conf->list_item, &dev->configs);
- return conf;
- }
- static void free_android_config(struct android_dev *dev,
- struct android_configuration *conf)
- {
- list_del(&conf->list_item);
- dev->configs_num--;
- kfree(conf);
- }
- static int usb_diag_update_pid_and_serial_num(u32 pid, const char *snum)
- {
- struct dload_struct local_diag_dload = { 0 };
- int *src, *dst, i;
- if (!diag_dload) {
- pr_debug("%s: unable to update PID and serial_no\n", __func__);
- return -ENODEV;
- }
- pr_debug("%s: dload:%pK pid:%x serial_num:%s\n",
- __func__, diag_dload, pid, snum);
- /* update pid */
- local_diag_dload.magic_struct.pid = PID_MAGIC_ID;
- local_diag_dload.pid = pid;
- /* update serial number */
- if (!snum) {
- local_diag_dload.magic_struct.serial_num = 0;
- memset(&local_diag_dload.serial_number, 0,
- SERIAL_NUMBER_LENGTH);
- } else {
- local_diag_dload.magic_struct.serial_num = SERIAL_NUM_MAGIC_ID;
- strlcpy((char *)&local_diag_dload.serial_number, snum,
- SERIAL_NUMBER_LENGTH);
- }
- /* Copy to shared struct (accesses need to be 32 bit aligned) */
- src = (int *)&local_diag_dload;
- dst = (int *)diag_dload;
- for (i = 0; i < sizeof(*diag_dload) / 4; i++)
- *dst++ = *src++;
- return 0;
- }
- static int __devinit android_probe(struct platform_device *pdev)
- {
- struct android_usb_platform_data *pdata;
- struct android_dev *android_dev;
- struct resource *res;
- int ret = 0, i, len = 0;
- if (pdev->dev.of_node) {
- dev_dbg(&pdev->dev, "device tree enabled\n");
- pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
- if (!pdata) {
- pr_err("unable to allocate platform data\n");
- return -ENOMEM;
- }
- of_property_read_u32(pdev->dev.of_node,
- "qcom,android-usb-swfi-latency",
- &pdata->swfi_latency);
- pdata->cdrom = of_property_read_bool(pdev->dev.of_node,
- "qcom,android-usb-cdrom");
- pdata->internal_ums = of_property_read_bool(pdev->dev.of_node,
- "qcom,android-usb-internal-ums");
- len = of_property_count_strings(pdev->dev.of_node,
- "qcom,streaming-func");
- if (len > MAX_STREAMING_FUNCS) {
- pr_err("Invalid number of functions used.\n");
- return -EINVAL;
- }
- for (i = 0; i < len; i++) {
- const char *name = NULL;
- of_property_read_string_index(pdev->dev.of_node,
- "qcom,streaming-func", i, &name);
- if (!name)
- continue;
- if (sizeof(name) > FUNC_NAME_LEN) {
- pr_err("Function name is bigger than allowed.\n");
- continue;
- }
- strlcpy(pdata->streaming_func[i], name,
- sizeof(pdata->streaming_func[i]));
- pr_debug("name of streaming function:%s\n",
- pdata->streaming_func[i]);
- }
- pdata->streaming_func_count = len;
- } else {
- pdata = pdev->dev.platform_data;
- }
- if (!android_class) {
- android_class = class_create(THIS_MODULE, "android_usb");
- if (IS_ERR(android_class))
- return PTR_ERR(android_class);
- }
- android_dev = kzalloc(sizeof(*android_dev), GFP_KERNEL);
- if (!android_dev) {
- pr_err("%s(): Failed to alloc memory for android_dev\n",
- __func__);
- ret = -ENOMEM;
- goto err_alloc;
- }
- android_dev->name = pdev->name;
- android_dev->disable_depth = 1;
- android_dev->functions = supported_functions;
- android_dev->configs_num = 0;
- INIT_LIST_HEAD(&android_dev->configs);
- INIT_WORK(&android_dev->work, android_work);
- #if defined(CONFIG_SEC_H_PROJECT)
- INIT_DELAYED_WORK(&android_dev->usb_connection_work, usb_gadget_connect_work);
- #endif
- mutex_init(&android_dev->mutex);
- android_dev->pdata = pdata;
- list_add_tail(&android_dev->list_item, &android_dev_list);
- android_dev_count++;
- if (pdata)
- composite_driver.usb_core_id = pdata->usb_core_id;
- else
- composite_driver.usb_core_id = 0; /*To backward compatibility*/
- res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
- if (res) {
- diag_dload = devm_ioremap(&pdev->dev, res->start,
- resource_size(res));
- if (!diag_dload) {
- dev_err(&pdev->dev, "ioremap failed\n");
- ret = -ENOMEM;
- goto err_dev;
- }
- } else {
- dev_dbg(&pdev->dev, "failed to get mem resource\n");
- }
- ret = android_create_device(android_dev, composite_driver.usb_core_id);
- if (ret) {
- pr_err("%s(): android_create_device failed\n", __func__);
- goto err_dev;
- }
- #ifdef CONFIG_USB_ANDROID_SAMSUNG_COMPOSITE
- /* Create below sysfs
- * /sys/class/android_usb/android0/terminal_version
- */
- ret = create_terminal_attribute(&android_dev->dev);
- if (ret) {
- printk(KERN_ERR "usb: %s To create terminal_atttrr is failed\n",
- __func__);
- return ret;
- }
- #endif
- ret = usb_composite_probe(&android_usb_driver, android_bind);
- if (ret) {
- pr_err("%s(): Failed to register android "
- "composite driver\n", __func__);
- goto err_probe;
- }
- /* pm qos request to prevent apps idle power collapse */
- if (pdata && pdata->swfi_latency)
- pm_qos_add_request(&android_dev->pm_qos_req_dma,
- PM_QOS_CPU_DMA_LATENCY, PM_QOS_DEFAULT_VALUE);
- strlcpy(android_dev->pm_qos, "high", sizeof(android_dev->pm_qos));
- #ifdef CONFIG_USB_DUN_SUPPORT
- ret = modem_misc_register();
- if (ret) {
- printk(KERN_ERR "usb: %s modem misc register is failed\n",
- __func__);
- goto err_probe;
- }
- #endif
- return ret;
- err_probe:
- android_destroy_device(android_dev);
- err_dev:
- list_del(&android_dev->list_item);
- android_dev_count--;
- kfree(android_dev);
- err_alloc:
- if (list_empty(&android_dev_list)) {
- class_destroy(android_class);
- android_class = NULL;
- }
- return ret;
- }
- static int android_remove(struct platform_device *pdev)
- {
- struct android_dev *dev = NULL;
- struct android_usb_platform_data *pdata = pdev->dev.platform_data;
- int usb_core_id = 0;
- if (pdata)
- usb_core_id = pdata->usb_core_id;
- /* Find the android dev from the list */
- list_for_each_entry(dev, &android_dev_list, list_item) {
- if (!dev->pdata)
- break; /*To backward compatibility*/
- if (dev->pdata->usb_core_id == usb_core_id)
- break;
- }
- if (dev) {
- android_destroy_device(dev);
- if (pdata && pdata->swfi_latency)
- pm_qos_remove_request(&dev->pm_qos_req_dma);
- list_del(&dev->list_item);
- android_dev_count--;
- kfree(dev);
- }
- if (list_empty(&android_dev_list)) {
- class_destroy(android_class);
- android_class = NULL;
- usb_composite_unregister(&android_usb_driver);
- }
- return 0;
- }
- static const struct platform_device_id android_id_table[] __devinitconst = {
- {
- .name = "android_usb",
- },
- {
- .name = "android_usb_hsic",
- },
- };
- static struct of_device_id usb_android_dt_match[] = {
- { .compatible = "qcom,android-usb",
- },
- {}
- };
- static struct platform_driver android_platform_driver = {
- .driver = {
- .name = "android_usb",
- .of_match_table = usb_android_dt_match,
- },
- .probe = android_probe,
- .remove = android_remove,
- .id_table = android_id_table,
- };
- static int __init init(void)
- {
- int ret;
- /* Override composite driver functions */
- composite_driver.setup = android_setup;
- composite_driver.disconnect = android_disconnect;
- composite_driver.suspend = android_suspend;
- composite_driver.resume = android_resume;
- INIT_LIST_HEAD(&android_dev_list);
- android_dev_count = 0;
- ret = platform_driver_register(&android_platform_driver);
- if (ret) {
- pr_err("%s(): Failed to register android"
- "platform driver\n", __func__);
- }
- return ret;
- }
- module_init(init);
- static void __exit cleanup(void)
- {
- platform_driver_unregister(&android_platform_driver);
- }
- module_exit(cleanup);
|