cifssmb.c 192 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562
  1. /*
  2. * fs/cifs/cifssmb.c
  3. *
  4. * Copyright (C) International Business Machines Corp., 2002,2010
  5. * Author(s): Steve French ([email protected])
  6. *
  7. * Contains the routines for constructing the SMB PDUs themselves
  8. *
  9. * This library is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU Lesser General Public License as published
  11. * by the Free Software Foundation; either version 2.1 of the License, or
  12. * (at your option) any later version.
  13. *
  14. * This library is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
  17. * the GNU Lesser General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU Lesser General Public License
  20. * along with this library; if not, write to the Free Software
  21. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  22. */
  23. /* SMB/CIFS PDU handling routines here - except for leftovers in connect.c */
  24. /* These are mostly routines that operate on a pathname, or on a tree id */
  25. /* (mounted volume), but there are eight handle based routines which must be */
  26. /* treated slightly differently for reconnection purposes since we never */
  27. /* want to reuse a stale file handle and only the caller knows the file info */
  28. #include <linux/fs.h>
  29. #include <linux/kernel.h>
  30. #include <linux/vfs.h>
  31. #include <linux/slab.h>
  32. #include <linux/posix_acl_xattr.h>
  33. #include <linux/pagemap.h>
  34. #include <linux/swap.h>
  35. #include <linux/task_io_accounting_ops.h>
  36. #include <asm/uaccess.h>
  37. #include "cifspdu.h"
  38. #include "cifsglob.h"
  39. #include "cifsacl.h"
  40. #include "cifsproto.h"
  41. #include "cifs_unicode.h"
  42. #include "cifs_debug.h"
  43. #include "fscache.h"
  44. #ifdef CONFIG_CIFS_POSIX
  45. static struct {
  46. int index;
  47. char *name;
  48. } protocols[] = {
  49. #ifdef CONFIG_CIFS_WEAK_PW_HASH
  50. {LANMAN_PROT, "\2LM1.2X002"},
  51. {LANMAN2_PROT, "\2LANMAN2.1"},
  52. #endif /* weak password hashing for legacy clients */
  53. {CIFS_PROT, "\2NT LM 0.12"},
  54. {POSIX_PROT, "\2POSIX 2"},
  55. {BAD_PROT, "\2"}
  56. };
  57. #else
  58. static struct {
  59. int index;
  60. char *name;
  61. } protocols[] = {
  62. #ifdef CONFIG_CIFS_WEAK_PW_HASH
  63. {LANMAN_PROT, "\2LM1.2X002"},
  64. {LANMAN2_PROT, "\2LANMAN2.1"},
  65. #endif /* weak password hashing for legacy clients */
  66. {CIFS_PROT, "\2NT LM 0.12"},
  67. {BAD_PROT, "\2"}
  68. };
  69. #endif
  70. /* define the number of elements in the cifs dialect array */
  71. #ifdef CONFIG_CIFS_POSIX
  72. #ifdef CONFIG_CIFS_WEAK_PW_HASH
  73. #define CIFS_NUM_PROT 4
  74. #else
  75. #define CIFS_NUM_PROT 2
  76. #endif /* CIFS_WEAK_PW_HASH */
  77. #else /* not posix */
  78. #ifdef CONFIG_CIFS_WEAK_PW_HASH
  79. #define CIFS_NUM_PROT 3
  80. #else
  81. #define CIFS_NUM_PROT 1
  82. #endif /* CONFIG_CIFS_WEAK_PW_HASH */
  83. #endif /* CIFS_POSIX */
  84. /*
  85. * Mark as invalid, all open files on tree connections since they
  86. * were closed when session to server was lost.
  87. */
  88. void
  89. cifs_mark_open_files_invalid(struct cifs_tcon *tcon)
  90. {
  91. struct cifsFileInfo *open_file = NULL;
  92. struct list_head *tmp;
  93. struct list_head *tmp1;
  94. /* list all files open on tree connection and mark them invalid */
  95. spin_lock(&tcon->open_file_lock);
  96. list_for_each_safe(tmp, tmp1, &tcon->openFileList) {
  97. open_file = list_entry(tmp, struct cifsFileInfo, tlist);
  98. open_file->invalidHandle = true;
  99. open_file->oplock_break_cancelled = true;
  100. }
  101. spin_unlock(&tcon->open_file_lock);
  102. /*
  103. * BB Add call to invalidate_inodes(sb) for all superblocks mounted
  104. * to this tcon.
  105. */
  106. }
  107. /* reconnect the socket, tcon, and smb session if needed */
  108. static int
  109. cifs_reconnect_tcon(struct cifs_tcon *tcon, int smb_command)
  110. {
  111. int rc;
  112. struct cifs_ses *ses;
  113. struct TCP_Server_Info *server;
  114. struct nls_table *nls_codepage;
  115. /*
  116. * SMBs NegProt, SessSetup, uLogoff do not have tcon yet so check for
  117. * tcp and smb session status done differently for those three - in the
  118. * calling routine
  119. */
  120. if (!tcon)
  121. return 0;
  122. ses = tcon->ses;
  123. server = ses->server;
  124. /*
  125. * only tree disconnect, open, and write, (and ulogoff which does not
  126. * have tcon) are allowed as we start force umount
  127. */
  128. if (tcon->tidStatus == CifsExiting) {
  129. if (smb_command != SMB_COM_WRITE_ANDX &&
  130. smb_command != SMB_COM_OPEN_ANDX &&
  131. smb_command != SMB_COM_TREE_DISCONNECT) {
  132. cifs_dbg(FYI, "can not send cmd %d while umounting\n",
  133. smb_command);
  134. return -ENODEV;
  135. }
  136. }
  137. /*
  138. * Give demultiplex thread up to 10 seconds to reconnect, should be
  139. * greater than cifs socket timeout which is 7 seconds
  140. */
  141. while (server->tcpStatus == CifsNeedReconnect) {
  142. rc = wait_event_interruptible_timeout(server->response_q,
  143. (server->tcpStatus != CifsNeedReconnect),
  144. 10 * HZ);
  145. if (rc < 0) {
  146. cifs_dbg(FYI, "%s: aborting reconnect due to a received"
  147. " signal by the process\n", __func__);
  148. return -ERESTARTSYS;
  149. }
  150. /* are we still trying to reconnect? */
  151. if (server->tcpStatus != CifsNeedReconnect)
  152. break;
  153. /*
  154. * on "soft" mounts we wait once. Hard mounts keep
  155. * retrying until process is killed or server comes
  156. * back on-line
  157. */
  158. if (!tcon->retry) {
  159. cifs_dbg(FYI, "gave up waiting on reconnect in smb_init\n");
  160. return -EHOSTDOWN;
  161. }
  162. }
  163. if (!ses->need_reconnect && !tcon->need_reconnect)
  164. return 0;
  165. nls_codepage = load_nls_default();
  166. /*
  167. * need to prevent multiple threads trying to simultaneously
  168. * reconnect the same SMB session
  169. */
  170. mutex_lock(&ses->session_mutex);
  171. rc = cifs_negotiate_protocol(0, ses);
  172. if (rc == 0 && ses->need_reconnect)
  173. rc = cifs_setup_session(0, ses, nls_codepage);
  174. /* do we need to reconnect tcon? */
  175. if (rc || !tcon->need_reconnect) {
  176. mutex_unlock(&ses->session_mutex);
  177. goto out;
  178. }
  179. cifs_mark_open_files_invalid(tcon);
  180. rc = CIFSTCon(0, ses, tcon->treeName, tcon, nls_codepage);
  181. mutex_unlock(&ses->session_mutex);
  182. cifs_dbg(FYI, "reconnect tcon rc = %d\n", rc);
  183. if (rc)
  184. goto out;
  185. atomic_inc(&tconInfoReconnectCount);
  186. /* tell server Unix caps we support */
  187. if (ses->capabilities & CAP_UNIX)
  188. reset_cifs_unix_caps(0, tcon, NULL, NULL);
  189. /*
  190. * Removed call to reopen open files here. It is safer (and faster) to
  191. * reopen files one at a time as needed in read and write.
  192. *
  193. * FIXME: what about file locks? don't we need to reclaim them ASAP?
  194. */
  195. out:
  196. /*
  197. * Check if handle based operation so we know whether we can continue
  198. * or not without returning to caller to reset file handle
  199. */
  200. switch (smb_command) {
  201. case SMB_COM_READ_ANDX:
  202. case SMB_COM_WRITE_ANDX:
  203. case SMB_COM_CLOSE:
  204. case SMB_COM_FIND_CLOSE2:
  205. case SMB_COM_LOCKING_ANDX:
  206. rc = -EAGAIN;
  207. }
  208. unload_nls(nls_codepage);
  209. return rc;
  210. }
  211. /* Allocate and return pointer to an SMB request buffer, and set basic
  212. SMB information in the SMB header. If the return code is zero, this
  213. function must have filled in request_buf pointer */
  214. static int
  215. small_smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
  216. void **request_buf)
  217. {
  218. int rc;
  219. rc = cifs_reconnect_tcon(tcon, smb_command);
  220. if (rc)
  221. return rc;
  222. *request_buf = cifs_small_buf_get();
  223. if (*request_buf == NULL) {
  224. /* BB should we add a retry in here if not a writepage? */
  225. return -ENOMEM;
  226. }
  227. header_assemble((struct smb_hdr *) *request_buf, smb_command,
  228. tcon, wct);
  229. if (tcon != NULL)
  230. cifs_stats_inc(&tcon->num_smbs_sent);
  231. return 0;
  232. }
  233. int
  234. small_smb_init_no_tc(const int smb_command, const int wct,
  235. struct cifs_ses *ses, void **request_buf)
  236. {
  237. int rc;
  238. struct smb_hdr *buffer;
  239. rc = small_smb_init(smb_command, wct, NULL, request_buf);
  240. if (rc)
  241. return rc;
  242. buffer = (struct smb_hdr *)*request_buf;
  243. buffer->Mid = get_next_mid(ses->server);
  244. if (ses->capabilities & CAP_UNICODE)
  245. buffer->Flags2 |= SMBFLG2_UNICODE;
  246. if (ses->capabilities & CAP_STATUS32)
  247. buffer->Flags2 |= SMBFLG2_ERR_STATUS;
  248. /* uid, tid can stay at zero as set in header assemble */
  249. /* BB add support for turning on the signing when
  250. this function is used after 1st of session setup requests */
  251. return rc;
  252. }
  253. /* If the return code is zero, this function must fill in request_buf pointer */
  254. static int
  255. __smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
  256. void **request_buf, void **response_buf)
  257. {
  258. *request_buf = cifs_buf_get();
  259. if (*request_buf == NULL) {
  260. /* BB should we add a retry in here if not a writepage? */
  261. return -ENOMEM;
  262. }
  263. /* Although the original thought was we needed the response buf for */
  264. /* potential retries of smb operations it turns out we can determine */
  265. /* from the mid flags when the request buffer can be resent without */
  266. /* having to use a second distinct buffer for the response */
  267. if (response_buf)
  268. *response_buf = *request_buf;
  269. header_assemble((struct smb_hdr *) *request_buf, smb_command, tcon,
  270. wct);
  271. if (tcon != NULL)
  272. cifs_stats_inc(&tcon->num_smbs_sent);
  273. return 0;
  274. }
  275. /* If the return code is zero, this function must fill in request_buf pointer */
  276. static int
  277. smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
  278. void **request_buf, void **response_buf)
  279. {
  280. int rc;
  281. rc = cifs_reconnect_tcon(tcon, smb_command);
  282. if (rc)
  283. return rc;
  284. return __smb_init(smb_command, wct, tcon, request_buf, response_buf);
  285. }
  286. static int
  287. smb_init_no_reconnect(int smb_command, int wct, struct cifs_tcon *tcon,
  288. void **request_buf, void **response_buf)
  289. {
  290. if (tcon->ses->need_reconnect || tcon->need_reconnect)
  291. return -EHOSTDOWN;
  292. return __smb_init(smb_command, wct, tcon, request_buf, response_buf);
  293. }
  294. static int validate_t2(struct smb_t2_rsp *pSMB)
  295. {
  296. unsigned int total_size;
  297. /* check for plausible wct */
  298. if (pSMB->hdr.WordCount < 10)
  299. goto vt2_err;
  300. /* check for parm and data offset going beyond end of smb */
  301. if (get_unaligned_le16(&pSMB->t2_rsp.ParameterOffset) > 1024 ||
  302. get_unaligned_le16(&pSMB->t2_rsp.DataOffset) > 1024)
  303. goto vt2_err;
  304. total_size = get_unaligned_le16(&pSMB->t2_rsp.ParameterCount);
  305. if (total_size >= 512)
  306. goto vt2_err;
  307. /* check that bcc is at least as big as parms + data, and that it is
  308. * less than negotiated smb buffer
  309. */
  310. total_size += get_unaligned_le16(&pSMB->t2_rsp.DataCount);
  311. if (total_size > get_bcc(&pSMB->hdr) ||
  312. total_size >= CIFSMaxBufSize + MAX_CIFS_HDR_SIZE)
  313. goto vt2_err;
  314. return 0;
  315. vt2_err:
  316. cifs_dump_mem("Invalid transact2 SMB: ", (char *)pSMB,
  317. sizeof(struct smb_t2_rsp) + 16);
  318. return -EINVAL;
  319. }
  320. static int
  321. decode_ext_sec_blob(struct cifs_ses *ses, NEGOTIATE_RSP *pSMBr)
  322. {
  323. int rc = 0;
  324. u16 count;
  325. char *guid = pSMBr->u.extended_response.GUID;
  326. struct TCP_Server_Info *server = ses->server;
  327. count = get_bcc(&pSMBr->hdr);
  328. if (count < SMB1_CLIENT_GUID_SIZE)
  329. return -EIO;
  330. spin_lock(&cifs_tcp_ses_lock);
  331. if (server->srv_count > 1) {
  332. spin_unlock(&cifs_tcp_ses_lock);
  333. if (memcmp(server->server_GUID, guid, SMB1_CLIENT_GUID_SIZE) != 0) {
  334. cifs_dbg(FYI, "server UID changed\n");
  335. memcpy(server->server_GUID, guid, SMB1_CLIENT_GUID_SIZE);
  336. }
  337. } else {
  338. spin_unlock(&cifs_tcp_ses_lock);
  339. memcpy(server->server_GUID, guid, SMB1_CLIENT_GUID_SIZE);
  340. }
  341. if (count == SMB1_CLIENT_GUID_SIZE) {
  342. server->sec_ntlmssp = true;
  343. } else {
  344. count -= SMB1_CLIENT_GUID_SIZE;
  345. rc = decode_negTokenInit(
  346. pSMBr->u.extended_response.SecurityBlob, count, server);
  347. if (rc != 1)
  348. return -EINVAL;
  349. }
  350. return 0;
  351. }
  352. int
  353. cifs_enable_signing(struct TCP_Server_Info *server, bool mnt_sign_required)
  354. {
  355. bool srv_sign_required = server->sec_mode & server->vals->signing_required;
  356. bool srv_sign_enabled = server->sec_mode & server->vals->signing_enabled;
  357. bool mnt_sign_enabled = global_secflags & CIFSSEC_MAY_SIGN;
  358. /*
  359. * Is signing required by mnt options? If not then check
  360. * global_secflags to see if it is there.
  361. */
  362. if (!mnt_sign_required)
  363. mnt_sign_required = ((global_secflags & CIFSSEC_MUST_SIGN) ==
  364. CIFSSEC_MUST_SIGN);
  365. /*
  366. * If signing is required then it's automatically enabled too,
  367. * otherwise, check to see if the secflags allow it.
  368. */
  369. mnt_sign_enabled = mnt_sign_required ? mnt_sign_required :
  370. (global_secflags & CIFSSEC_MAY_SIGN);
  371. /* If server requires signing, does client allow it? */
  372. if (srv_sign_required) {
  373. if (!mnt_sign_enabled) {
  374. cifs_dbg(VFS, "Server requires signing, but it's disabled in SecurityFlags!");
  375. return -ENOTSUPP;
  376. }
  377. server->sign = true;
  378. }
  379. /* If client requires signing, does server allow it? */
  380. if (mnt_sign_required) {
  381. if (!srv_sign_enabled) {
  382. cifs_dbg(VFS, "Server does not support signing!");
  383. return -ENOTSUPP;
  384. }
  385. server->sign = true;
  386. }
  387. return 0;
  388. }
  389. #ifdef CONFIG_CIFS_WEAK_PW_HASH
  390. static int
  391. decode_lanman_negprot_rsp(struct TCP_Server_Info *server, NEGOTIATE_RSP *pSMBr)
  392. {
  393. __s16 tmp;
  394. struct lanman_neg_rsp *rsp = (struct lanman_neg_rsp *)pSMBr;
  395. if (server->dialect != LANMAN_PROT && server->dialect != LANMAN2_PROT)
  396. return -EOPNOTSUPP;
  397. server->sec_mode = le16_to_cpu(rsp->SecurityMode);
  398. server->maxReq = min_t(unsigned int,
  399. le16_to_cpu(rsp->MaxMpxCount),
  400. cifs_max_pending);
  401. set_credits(server, server->maxReq);
  402. server->maxBuf = le16_to_cpu(rsp->MaxBufSize);
  403. /* even though we do not use raw we might as well set this
  404. accurately, in case we ever find a need for it */
  405. if ((le16_to_cpu(rsp->RawMode) & RAW_ENABLE) == RAW_ENABLE) {
  406. server->max_rw = 0xFF00;
  407. server->capabilities = CAP_MPX_MODE | CAP_RAW_MODE;
  408. } else {
  409. server->max_rw = 0;/* do not need to use raw anyway */
  410. server->capabilities = CAP_MPX_MODE;
  411. }
  412. tmp = (__s16)le16_to_cpu(rsp->ServerTimeZone);
  413. if (tmp == -1) {
  414. /* OS/2 often does not set timezone therefore
  415. * we must use server time to calc time zone.
  416. * Could deviate slightly from the right zone.
  417. * Smallest defined timezone difference is 15 minutes
  418. * (i.e. Nepal). Rounding up/down is done to match
  419. * this requirement.
  420. */
  421. int val, seconds, remain, result;
  422. struct timespec ts, utc;
  423. utc = CURRENT_TIME;
  424. ts = cnvrtDosUnixTm(rsp->SrvTime.Date,
  425. rsp->SrvTime.Time, 0);
  426. cifs_dbg(FYI, "SrvTime %d sec since 1970 (utc: %d) diff: %d\n",
  427. (int)ts.tv_sec, (int)utc.tv_sec,
  428. (int)(utc.tv_sec - ts.tv_sec));
  429. val = (int)(utc.tv_sec - ts.tv_sec);
  430. seconds = abs(val);
  431. result = (seconds / MIN_TZ_ADJ) * MIN_TZ_ADJ;
  432. remain = seconds % MIN_TZ_ADJ;
  433. if (remain >= (MIN_TZ_ADJ / 2))
  434. result += MIN_TZ_ADJ;
  435. if (val < 0)
  436. result = -result;
  437. server->timeAdj = result;
  438. } else {
  439. server->timeAdj = (int)tmp;
  440. server->timeAdj *= 60; /* also in seconds */
  441. }
  442. cifs_dbg(FYI, "server->timeAdj: %d seconds\n", server->timeAdj);
  443. /* BB get server time for time conversions and add
  444. code to use it and timezone since this is not UTC */
  445. if (rsp->EncryptionKeyLength ==
  446. cpu_to_le16(CIFS_CRYPTO_KEY_SIZE)) {
  447. memcpy(server->cryptkey, rsp->EncryptionKey,
  448. CIFS_CRYPTO_KEY_SIZE);
  449. } else if (server->sec_mode & SECMODE_PW_ENCRYPT) {
  450. return -EIO; /* need cryptkey unless plain text */
  451. }
  452. cifs_dbg(FYI, "LANMAN negotiated\n");
  453. return 0;
  454. }
  455. #else
  456. static inline int
  457. decode_lanman_negprot_rsp(struct TCP_Server_Info *server, NEGOTIATE_RSP *pSMBr)
  458. {
  459. cifs_dbg(VFS, "mount failed, cifs module not built with CIFS_WEAK_PW_HASH support\n");
  460. return -EOPNOTSUPP;
  461. }
  462. #endif
  463. static bool
  464. should_set_ext_sec_flag(enum securityEnum sectype)
  465. {
  466. switch (sectype) {
  467. case RawNTLMSSP:
  468. case Kerberos:
  469. return true;
  470. case Unspecified:
  471. if (global_secflags &
  472. (CIFSSEC_MAY_KRB5 | CIFSSEC_MAY_NTLMSSP))
  473. return true;
  474. /* Fallthrough */
  475. default:
  476. return false;
  477. }
  478. }
  479. int
  480. CIFSSMBNegotiate(const unsigned int xid, struct cifs_ses *ses)
  481. {
  482. NEGOTIATE_REQ *pSMB;
  483. NEGOTIATE_RSP *pSMBr;
  484. int rc = 0;
  485. int bytes_returned;
  486. int i;
  487. struct TCP_Server_Info *server = ses->server;
  488. u16 count;
  489. if (!server) {
  490. WARN(1, "%s: server is NULL!\n", __func__);
  491. return -EIO;
  492. }
  493. rc = smb_init(SMB_COM_NEGOTIATE, 0, NULL /* no tcon yet */ ,
  494. (void **) &pSMB, (void **) &pSMBr);
  495. if (rc)
  496. return rc;
  497. pSMB->hdr.Mid = get_next_mid(server);
  498. pSMB->hdr.Flags2 |= (SMBFLG2_UNICODE | SMBFLG2_ERR_STATUS);
  499. if (should_set_ext_sec_flag(ses->sectype)) {
  500. cifs_dbg(FYI, "Requesting extended security.");
  501. pSMB->hdr.Flags2 |= SMBFLG2_EXT_SEC;
  502. }
  503. count = 0;
  504. /*
  505. * We know that all the name entries in the protocols array
  506. * are short (< 16 bytes anyway) and are NUL terminated.
  507. */
  508. for (i = 0; i < CIFS_NUM_PROT; i++) {
  509. size_t len = strlen(protocols[i].name) + 1;
  510. memcpy(pSMB->DialectsArray+count, protocols[i].name, len);
  511. count += len;
  512. }
  513. inc_rfc1001_len(pSMB, count);
  514. pSMB->ByteCount = cpu_to_le16(count);
  515. rc = SendReceive(xid, ses, (struct smb_hdr *) pSMB,
  516. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  517. if (rc != 0)
  518. goto neg_err_exit;
  519. server->dialect = le16_to_cpu(pSMBr->DialectIndex);
  520. cifs_dbg(FYI, "Dialect: %d\n", server->dialect);
  521. /* Check wct = 1 error case */
  522. if ((pSMBr->hdr.WordCount < 13) || (server->dialect == BAD_PROT)) {
  523. /* core returns wct = 1, but we do not ask for core - otherwise
  524. small wct just comes when dialect index is -1 indicating we
  525. could not negotiate a common dialect */
  526. rc = -EOPNOTSUPP;
  527. goto neg_err_exit;
  528. } else if (pSMBr->hdr.WordCount == 13) {
  529. server->negflavor = CIFS_NEGFLAVOR_LANMAN;
  530. rc = decode_lanman_negprot_rsp(server, pSMBr);
  531. goto signing_check;
  532. } else if (pSMBr->hdr.WordCount != 17) {
  533. /* unknown wct */
  534. rc = -EOPNOTSUPP;
  535. goto neg_err_exit;
  536. }
  537. /* else wct == 17, NTLM or better */
  538. server->sec_mode = pSMBr->SecurityMode;
  539. if ((server->sec_mode & SECMODE_USER) == 0)
  540. cifs_dbg(FYI, "share mode security\n");
  541. /* one byte, so no need to convert this or EncryptionKeyLen from
  542. little endian */
  543. server->maxReq = min_t(unsigned int, le16_to_cpu(pSMBr->MaxMpxCount),
  544. cifs_max_pending);
  545. set_credits(server, server->maxReq);
  546. /* probably no need to store and check maxvcs */
  547. server->maxBuf = le32_to_cpu(pSMBr->MaxBufferSize);
  548. server->max_rw = le32_to_cpu(pSMBr->MaxRawSize);
  549. cifs_dbg(NOISY, "Max buf = %d\n", ses->server->maxBuf);
  550. server->capabilities = le32_to_cpu(pSMBr->Capabilities);
  551. server->timeAdj = (int)(__s16)le16_to_cpu(pSMBr->ServerTimeZone);
  552. server->timeAdj *= 60;
  553. if (pSMBr->EncryptionKeyLength == CIFS_CRYPTO_KEY_SIZE) {
  554. server->negflavor = CIFS_NEGFLAVOR_UNENCAP;
  555. memcpy(ses->server->cryptkey, pSMBr->u.EncryptionKey,
  556. CIFS_CRYPTO_KEY_SIZE);
  557. } else if (pSMBr->hdr.Flags2 & SMBFLG2_EXT_SEC ||
  558. server->capabilities & CAP_EXTENDED_SECURITY) {
  559. server->negflavor = CIFS_NEGFLAVOR_EXTENDED;
  560. rc = decode_ext_sec_blob(ses, pSMBr);
  561. } else if (server->sec_mode & SECMODE_PW_ENCRYPT) {
  562. rc = -EIO; /* no crypt key only if plain text pwd */
  563. } else {
  564. server->negflavor = CIFS_NEGFLAVOR_UNENCAP;
  565. server->capabilities &= ~CAP_EXTENDED_SECURITY;
  566. }
  567. signing_check:
  568. if (!rc)
  569. rc = cifs_enable_signing(server, ses->sign);
  570. neg_err_exit:
  571. cifs_buf_release(pSMB);
  572. cifs_dbg(FYI, "negprot rc %d\n", rc);
  573. return rc;
  574. }
  575. int
  576. CIFSSMBTDis(const unsigned int xid, struct cifs_tcon *tcon)
  577. {
  578. struct smb_hdr *smb_buffer;
  579. int rc = 0;
  580. cifs_dbg(FYI, "In tree disconnect\n");
  581. /* BB: do we need to check this? These should never be NULL. */
  582. if ((tcon->ses == NULL) || (tcon->ses->server == NULL))
  583. return -EIO;
  584. /*
  585. * No need to return error on this operation if tid invalidated and
  586. * closed on server already e.g. due to tcp session crashing. Also,
  587. * the tcon is no longer on the list, so no need to take lock before
  588. * checking this.
  589. */
  590. if ((tcon->need_reconnect) || (tcon->ses->need_reconnect))
  591. return 0;
  592. rc = small_smb_init(SMB_COM_TREE_DISCONNECT, 0, tcon,
  593. (void **)&smb_buffer);
  594. if (rc)
  595. return rc;
  596. rc = SendReceiveNoRsp(xid, tcon->ses, (char *)smb_buffer, 0);
  597. if (rc)
  598. cifs_dbg(FYI, "Tree disconnect failed %d\n", rc);
  599. /* No need to return error on this operation if tid invalidated and
  600. closed on server already e.g. due to tcp session crashing */
  601. if (rc == -EAGAIN)
  602. rc = 0;
  603. return rc;
  604. }
  605. /*
  606. * This is a no-op for now. We're not really interested in the reply, but
  607. * rather in the fact that the server sent one and that server->lstrp
  608. * gets updated.
  609. *
  610. * FIXME: maybe we should consider checking that the reply matches request?
  611. */
  612. static void
  613. cifs_echo_callback(struct mid_q_entry *mid)
  614. {
  615. struct TCP_Server_Info *server = mid->callback_data;
  616. mutex_lock(&server->srv_mutex);
  617. DeleteMidQEntry(mid);
  618. mutex_unlock(&server->srv_mutex);
  619. add_credits(server, 1, CIFS_ECHO_OP);
  620. }
  621. int
  622. CIFSSMBEcho(struct TCP_Server_Info *server)
  623. {
  624. ECHO_REQ *smb;
  625. int rc = 0;
  626. struct kvec iov;
  627. struct smb_rqst rqst = { .rq_iov = &iov,
  628. .rq_nvec = 1 };
  629. cifs_dbg(FYI, "In echo request\n");
  630. rc = small_smb_init(SMB_COM_ECHO, 0, NULL, (void **)&smb);
  631. if (rc)
  632. return rc;
  633. if (server->capabilities & CAP_UNICODE)
  634. smb->hdr.Flags2 |= SMBFLG2_UNICODE;
  635. /* set up echo request */
  636. smb->hdr.Tid = 0xffff;
  637. smb->hdr.WordCount = 1;
  638. put_unaligned_le16(1, &smb->EchoCount);
  639. put_bcc(1, &smb->hdr);
  640. smb->Data[0] = 'a';
  641. inc_rfc1001_len(smb, 3);
  642. iov.iov_base = smb;
  643. iov.iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4;
  644. rc = cifs_call_async(server, &rqst, NULL, cifs_echo_callback,
  645. server, CIFS_ASYNC_OP | CIFS_ECHO_OP);
  646. if (rc)
  647. cifs_dbg(FYI, "Echo request failed: %d\n", rc);
  648. cifs_small_buf_release(smb);
  649. return rc;
  650. }
  651. int
  652. CIFSSMBLogoff(const unsigned int xid, struct cifs_ses *ses)
  653. {
  654. LOGOFF_ANDX_REQ *pSMB;
  655. int rc = 0;
  656. cifs_dbg(FYI, "In SMBLogoff for session disconnect\n");
  657. /*
  658. * BB: do we need to check validity of ses and server? They should
  659. * always be valid since we have an active reference. If not, that
  660. * should probably be a BUG()
  661. */
  662. if (!ses || !ses->server)
  663. return -EIO;
  664. mutex_lock(&ses->session_mutex);
  665. if (ses->need_reconnect)
  666. goto session_already_dead; /* no need to send SMBlogoff if uid
  667. already closed due to reconnect */
  668. rc = small_smb_init(SMB_COM_LOGOFF_ANDX, 2, NULL, (void **)&pSMB);
  669. if (rc) {
  670. mutex_unlock(&ses->session_mutex);
  671. return rc;
  672. }
  673. pSMB->hdr.Mid = get_next_mid(ses->server);
  674. if (ses->server->sign)
  675. pSMB->hdr.Flags2 |= SMBFLG2_SECURITY_SIGNATURE;
  676. pSMB->hdr.Uid = ses->Suid;
  677. pSMB->AndXCommand = 0xFF;
  678. rc = SendReceiveNoRsp(xid, ses, (char *) pSMB, 0);
  679. session_already_dead:
  680. mutex_unlock(&ses->session_mutex);
  681. /* if session dead then we do not need to do ulogoff,
  682. since server closed smb session, no sense reporting
  683. error */
  684. if (rc == -EAGAIN)
  685. rc = 0;
  686. return rc;
  687. }
  688. int
  689. CIFSPOSIXDelFile(const unsigned int xid, struct cifs_tcon *tcon,
  690. const char *fileName, __u16 type,
  691. const struct nls_table *nls_codepage, int remap)
  692. {
  693. TRANSACTION2_SPI_REQ *pSMB = NULL;
  694. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  695. struct unlink_psx_rq *pRqD;
  696. int name_len;
  697. int rc = 0;
  698. int bytes_returned = 0;
  699. __u16 params, param_offset, offset, byte_count;
  700. cifs_dbg(FYI, "In POSIX delete\n");
  701. PsxDelete:
  702. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  703. (void **) &pSMBr);
  704. if (rc)
  705. return rc;
  706. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  707. name_len =
  708. cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
  709. PATH_MAX, nls_codepage, remap);
  710. name_len++; /* trailing null */
  711. name_len *= 2;
  712. } else { /* BB add path length overrun check */
  713. name_len = strnlen(fileName, PATH_MAX);
  714. name_len++; /* trailing null */
  715. strncpy(pSMB->FileName, fileName, name_len);
  716. }
  717. params = 6 + name_len;
  718. pSMB->MaxParameterCount = cpu_to_le16(2);
  719. pSMB->MaxDataCount = 0; /* BB double check this with jra */
  720. pSMB->MaxSetupCount = 0;
  721. pSMB->Reserved = 0;
  722. pSMB->Flags = 0;
  723. pSMB->Timeout = 0;
  724. pSMB->Reserved2 = 0;
  725. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  726. InformationLevel) - 4;
  727. offset = param_offset + params;
  728. /* Setup pointer to Request Data (inode type) */
  729. pRqD = (struct unlink_psx_rq *)(((char *)&pSMB->hdr.Protocol) + offset);
  730. pRqD->type = cpu_to_le16(type);
  731. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  732. pSMB->DataOffset = cpu_to_le16(offset);
  733. pSMB->SetupCount = 1;
  734. pSMB->Reserved3 = 0;
  735. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  736. byte_count = 3 /* pad */ + params + sizeof(struct unlink_psx_rq);
  737. pSMB->DataCount = cpu_to_le16(sizeof(struct unlink_psx_rq));
  738. pSMB->TotalDataCount = cpu_to_le16(sizeof(struct unlink_psx_rq));
  739. pSMB->ParameterCount = cpu_to_le16(params);
  740. pSMB->TotalParameterCount = pSMB->ParameterCount;
  741. pSMB->InformationLevel = cpu_to_le16(SMB_POSIX_UNLINK);
  742. pSMB->Reserved4 = 0;
  743. inc_rfc1001_len(pSMB, byte_count);
  744. pSMB->ByteCount = cpu_to_le16(byte_count);
  745. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  746. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  747. if (rc)
  748. cifs_dbg(FYI, "Posix delete returned %d\n", rc);
  749. cifs_buf_release(pSMB);
  750. cifs_stats_inc(&tcon->stats.cifs_stats.num_deletes);
  751. if (rc == -EAGAIN)
  752. goto PsxDelete;
  753. return rc;
  754. }
  755. int
  756. CIFSSMBDelFile(const unsigned int xid, struct cifs_tcon *tcon, const char *name,
  757. struct cifs_sb_info *cifs_sb)
  758. {
  759. DELETE_FILE_REQ *pSMB = NULL;
  760. DELETE_FILE_RSP *pSMBr = NULL;
  761. int rc = 0;
  762. int bytes_returned;
  763. int name_len;
  764. int remap = cifs_remap(cifs_sb);
  765. DelFileRetry:
  766. rc = smb_init(SMB_COM_DELETE, 1, tcon, (void **) &pSMB,
  767. (void **) &pSMBr);
  768. if (rc)
  769. return rc;
  770. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  771. name_len = cifsConvertToUTF16((__le16 *) pSMB->fileName, name,
  772. PATH_MAX, cifs_sb->local_nls,
  773. remap);
  774. name_len++; /* trailing null */
  775. name_len *= 2;
  776. } else { /* BB improve check for buffer overruns BB */
  777. name_len = strnlen(name, PATH_MAX);
  778. name_len++; /* trailing null */
  779. strncpy(pSMB->fileName, name, name_len);
  780. }
  781. pSMB->SearchAttributes =
  782. cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM);
  783. pSMB->BufferFormat = 0x04;
  784. inc_rfc1001_len(pSMB, name_len + 1);
  785. pSMB->ByteCount = cpu_to_le16(name_len + 1);
  786. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  787. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  788. cifs_stats_inc(&tcon->stats.cifs_stats.num_deletes);
  789. if (rc)
  790. cifs_dbg(FYI, "Error in RMFile = %d\n", rc);
  791. cifs_buf_release(pSMB);
  792. if (rc == -EAGAIN)
  793. goto DelFileRetry;
  794. return rc;
  795. }
  796. int
  797. CIFSSMBRmDir(const unsigned int xid, struct cifs_tcon *tcon, const char *name,
  798. struct cifs_sb_info *cifs_sb)
  799. {
  800. DELETE_DIRECTORY_REQ *pSMB = NULL;
  801. DELETE_DIRECTORY_RSP *pSMBr = NULL;
  802. int rc = 0;
  803. int bytes_returned;
  804. int name_len;
  805. int remap = cifs_remap(cifs_sb);
  806. cifs_dbg(FYI, "In CIFSSMBRmDir\n");
  807. RmDirRetry:
  808. rc = smb_init(SMB_COM_DELETE_DIRECTORY, 0, tcon, (void **) &pSMB,
  809. (void **) &pSMBr);
  810. if (rc)
  811. return rc;
  812. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  813. name_len = cifsConvertToUTF16((__le16 *) pSMB->DirName, name,
  814. PATH_MAX, cifs_sb->local_nls,
  815. remap);
  816. name_len++; /* trailing null */
  817. name_len *= 2;
  818. } else { /* BB improve check for buffer overruns BB */
  819. name_len = strnlen(name, PATH_MAX);
  820. name_len++; /* trailing null */
  821. strncpy(pSMB->DirName, name, name_len);
  822. }
  823. pSMB->BufferFormat = 0x04;
  824. inc_rfc1001_len(pSMB, name_len + 1);
  825. pSMB->ByteCount = cpu_to_le16(name_len + 1);
  826. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  827. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  828. cifs_stats_inc(&tcon->stats.cifs_stats.num_rmdirs);
  829. if (rc)
  830. cifs_dbg(FYI, "Error in RMDir = %d\n", rc);
  831. cifs_buf_release(pSMB);
  832. if (rc == -EAGAIN)
  833. goto RmDirRetry;
  834. return rc;
  835. }
  836. int
  837. CIFSSMBMkDir(const unsigned int xid, struct cifs_tcon *tcon, const char *name,
  838. struct cifs_sb_info *cifs_sb)
  839. {
  840. int rc = 0;
  841. CREATE_DIRECTORY_REQ *pSMB = NULL;
  842. CREATE_DIRECTORY_RSP *pSMBr = NULL;
  843. int bytes_returned;
  844. int name_len;
  845. int remap = cifs_remap(cifs_sb);
  846. cifs_dbg(FYI, "In CIFSSMBMkDir\n");
  847. MkDirRetry:
  848. rc = smb_init(SMB_COM_CREATE_DIRECTORY, 0, tcon, (void **) &pSMB,
  849. (void **) &pSMBr);
  850. if (rc)
  851. return rc;
  852. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  853. name_len = cifsConvertToUTF16((__le16 *) pSMB->DirName, name,
  854. PATH_MAX, cifs_sb->local_nls,
  855. remap);
  856. name_len++; /* trailing null */
  857. name_len *= 2;
  858. } else { /* BB improve check for buffer overruns BB */
  859. name_len = strnlen(name, PATH_MAX);
  860. name_len++; /* trailing null */
  861. strncpy(pSMB->DirName, name, name_len);
  862. }
  863. pSMB->BufferFormat = 0x04;
  864. inc_rfc1001_len(pSMB, name_len + 1);
  865. pSMB->ByteCount = cpu_to_le16(name_len + 1);
  866. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  867. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  868. cifs_stats_inc(&tcon->stats.cifs_stats.num_mkdirs);
  869. if (rc)
  870. cifs_dbg(FYI, "Error in Mkdir = %d\n", rc);
  871. cifs_buf_release(pSMB);
  872. if (rc == -EAGAIN)
  873. goto MkDirRetry;
  874. return rc;
  875. }
  876. int
  877. CIFSPOSIXCreate(const unsigned int xid, struct cifs_tcon *tcon,
  878. __u32 posix_flags, __u64 mode, __u16 *netfid,
  879. FILE_UNIX_BASIC_INFO *pRetData, __u32 *pOplock,
  880. const char *name, const struct nls_table *nls_codepage,
  881. int remap)
  882. {
  883. TRANSACTION2_SPI_REQ *pSMB = NULL;
  884. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  885. int name_len;
  886. int rc = 0;
  887. int bytes_returned = 0;
  888. __u16 params, param_offset, offset, byte_count, count;
  889. OPEN_PSX_REQ *pdata;
  890. OPEN_PSX_RSP *psx_rsp;
  891. cifs_dbg(FYI, "In POSIX Create\n");
  892. PsxCreat:
  893. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  894. (void **) &pSMBr);
  895. if (rc)
  896. return rc;
  897. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  898. name_len =
  899. cifsConvertToUTF16((__le16 *) pSMB->FileName, name,
  900. PATH_MAX, nls_codepage, remap);
  901. name_len++; /* trailing null */
  902. name_len *= 2;
  903. } else { /* BB improve the check for buffer overruns BB */
  904. name_len = strnlen(name, PATH_MAX);
  905. name_len++; /* trailing null */
  906. strncpy(pSMB->FileName, name, name_len);
  907. }
  908. params = 6 + name_len;
  909. count = sizeof(OPEN_PSX_REQ);
  910. pSMB->MaxParameterCount = cpu_to_le16(2);
  911. pSMB->MaxDataCount = cpu_to_le16(1000); /* large enough */
  912. pSMB->MaxSetupCount = 0;
  913. pSMB->Reserved = 0;
  914. pSMB->Flags = 0;
  915. pSMB->Timeout = 0;
  916. pSMB->Reserved2 = 0;
  917. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  918. InformationLevel) - 4;
  919. offset = param_offset + params;
  920. pdata = (OPEN_PSX_REQ *)(((char *)&pSMB->hdr.Protocol) + offset);
  921. pdata->Level = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
  922. pdata->Permissions = cpu_to_le64(mode);
  923. pdata->PosixOpenFlags = cpu_to_le32(posix_flags);
  924. pdata->OpenFlags = cpu_to_le32(*pOplock);
  925. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  926. pSMB->DataOffset = cpu_to_le16(offset);
  927. pSMB->SetupCount = 1;
  928. pSMB->Reserved3 = 0;
  929. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  930. byte_count = 3 /* pad */ + params + count;
  931. pSMB->DataCount = cpu_to_le16(count);
  932. pSMB->ParameterCount = cpu_to_le16(params);
  933. pSMB->TotalDataCount = pSMB->DataCount;
  934. pSMB->TotalParameterCount = pSMB->ParameterCount;
  935. pSMB->InformationLevel = cpu_to_le16(SMB_POSIX_OPEN);
  936. pSMB->Reserved4 = 0;
  937. inc_rfc1001_len(pSMB, byte_count);
  938. pSMB->ByteCount = cpu_to_le16(byte_count);
  939. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  940. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  941. if (rc) {
  942. cifs_dbg(FYI, "Posix create returned %d\n", rc);
  943. goto psx_create_err;
  944. }
  945. cifs_dbg(FYI, "copying inode info\n");
  946. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  947. if (rc || get_bcc(&pSMBr->hdr) < sizeof(OPEN_PSX_RSP)) {
  948. rc = -EIO; /* bad smb */
  949. goto psx_create_err;
  950. }
  951. /* copy return information to pRetData */
  952. psx_rsp = (OPEN_PSX_RSP *)((char *) &pSMBr->hdr.Protocol
  953. + le16_to_cpu(pSMBr->t2.DataOffset));
  954. *pOplock = le16_to_cpu(psx_rsp->OplockFlags);
  955. if (netfid)
  956. *netfid = psx_rsp->Fid; /* cifs fid stays in le */
  957. /* Let caller know file was created so we can set the mode. */
  958. /* Do we care about the CreateAction in any other cases? */
  959. if (cpu_to_le32(FILE_CREATE) == psx_rsp->CreateAction)
  960. *pOplock |= CIFS_CREATE_ACTION;
  961. /* check to make sure response data is there */
  962. if (psx_rsp->ReturnedLevel != cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC)) {
  963. pRetData->Type = cpu_to_le32(-1); /* unknown */
  964. cifs_dbg(NOISY, "unknown type\n");
  965. } else {
  966. if (get_bcc(&pSMBr->hdr) < sizeof(OPEN_PSX_RSP)
  967. + sizeof(FILE_UNIX_BASIC_INFO)) {
  968. cifs_dbg(VFS, "Open response data too small\n");
  969. pRetData->Type = cpu_to_le32(-1);
  970. goto psx_create_err;
  971. }
  972. memcpy((char *) pRetData,
  973. (char *)psx_rsp + sizeof(OPEN_PSX_RSP),
  974. sizeof(FILE_UNIX_BASIC_INFO));
  975. }
  976. psx_create_err:
  977. cifs_buf_release(pSMB);
  978. if (posix_flags & SMB_O_DIRECTORY)
  979. cifs_stats_inc(&tcon->stats.cifs_stats.num_posixmkdirs);
  980. else
  981. cifs_stats_inc(&tcon->stats.cifs_stats.num_posixopens);
  982. if (rc == -EAGAIN)
  983. goto PsxCreat;
  984. return rc;
  985. }
  986. static __u16 convert_disposition(int disposition)
  987. {
  988. __u16 ofun = 0;
  989. switch (disposition) {
  990. case FILE_SUPERSEDE:
  991. ofun = SMBOPEN_OCREATE | SMBOPEN_OTRUNC;
  992. break;
  993. case FILE_OPEN:
  994. ofun = SMBOPEN_OAPPEND;
  995. break;
  996. case FILE_CREATE:
  997. ofun = SMBOPEN_OCREATE;
  998. break;
  999. case FILE_OPEN_IF:
  1000. ofun = SMBOPEN_OCREATE | SMBOPEN_OAPPEND;
  1001. break;
  1002. case FILE_OVERWRITE:
  1003. ofun = SMBOPEN_OTRUNC;
  1004. break;
  1005. case FILE_OVERWRITE_IF:
  1006. ofun = SMBOPEN_OCREATE | SMBOPEN_OTRUNC;
  1007. break;
  1008. default:
  1009. cifs_dbg(FYI, "unknown disposition %d\n", disposition);
  1010. ofun = SMBOPEN_OAPPEND; /* regular open */
  1011. }
  1012. return ofun;
  1013. }
  1014. static int
  1015. access_flags_to_smbopen_mode(const int access_flags)
  1016. {
  1017. int masked_flags = access_flags & (GENERIC_READ | GENERIC_WRITE);
  1018. if (masked_flags == GENERIC_READ)
  1019. return SMBOPEN_READ;
  1020. else if (masked_flags == GENERIC_WRITE)
  1021. return SMBOPEN_WRITE;
  1022. /* just go for read/write */
  1023. return SMBOPEN_READWRITE;
  1024. }
  1025. int
  1026. SMBLegacyOpen(const unsigned int xid, struct cifs_tcon *tcon,
  1027. const char *fileName, const int openDisposition,
  1028. const int access_flags, const int create_options, __u16 *netfid,
  1029. int *pOplock, FILE_ALL_INFO *pfile_info,
  1030. const struct nls_table *nls_codepage, int remap)
  1031. {
  1032. int rc = -EACCES;
  1033. OPENX_REQ *pSMB = NULL;
  1034. OPENX_RSP *pSMBr = NULL;
  1035. int bytes_returned;
  1036. int name_len;
  1037. __u16 count;
  1038. OldOpenRetry:
  1039. rc = smb_init(SMB_COM_OPEN_ANDX, 15, tcon, (void **) &pSMB,
  1040. (void **) &pSMBr);
  1041. if (rc)
  1042. return rc;
  1043. pSMB->AndXCommand = 0xFF; /* none */
  1044. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  1045. count = 1; /* account for one byte pad to word boundary */
  1046. name_len =
  1047. cifsConvertToUTF16((__le16 *) (pSMB->fileName + 1),
  1048. fileName, PATH_MAX, nls_codepage, remap);
  1049. name_len++; /* trailing null */
  1050. name_len *= 2;
  1051. } else { /* BB improve check for buffer overruns BB */
  1052. count = 0; /* no pad */
  1053. name_len = strnlen(fileName, PATH_MAX);
  1054. name_len++; /* trailing null */
  1055. strncpy(pSMB->fileName, fileName, name_len);
  1056. }
  1057. if (*pOplock & REQ_OPLOCK)
  1058. pSMB->OpenFlags = cpu_to_le16(REQ_OPLOCK);
  1059. else if (*pOplock & REQ_BATCHOPLOCK)
  1060. pSMB->OpenFlags = cpu_to_le16(REQ_BATCHOPLOCK);
  1061. pSMB->OpenFlags |= cpu_to_le16(REQ_MORE_INFO);
  1062. pSMB->Mode = cpu_to_le16(access_flags_to_smbopen_mode(access_flags));
  1063. pSMB->Mode |= cpu_to_le16(0x40); /* deny none */
  1064. /* set file as system file if special file such
  1065. as fifo and server expecting SFU style and
  1066. no Unix extensions */
  1067. if (create_options & CREATE_OPTION_SPECIAL)
  1068. pSMB->FileAttributes = cpu_to_le16(ATTR_SYSTEM);
  1069. else /* BB FIXME BB */
  1070. pSMB->FileAttributes = cpu_to_le16(0/*ATTR_NORMAL*/);
  1071. if (create_options & CREATE_OPTION_READONLY)
  1072. pSMB->FileAttributes |= cpu_to_le16(ATTR_READONLY);
  1073. /* BB FIXME BB */
  1074. /* pSMB->CreateOptions = cpu_to_le32(create_options &
  1075. CREATE_OPTIONS_MASK); */
  1076. /* BB FIXME END BB */
  1077. pSMB->Sattr = cpu_to_le16(ATTR_HIDDEN | ATTR_SYSTEM | ATTR_DIRECTORY);
  1078. pSMB->OpenFunction = cpu_to_le16(convert_disposition(openDisposition));
  1079. count += name_len;
  1080. inc_rfc1001_len(pSMB, count);
  1081. pSMB->ByteCount = cpu_to_le16(count);
  1082. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  1083. (struct smb_hdr *)pSMBr, &bytes_returned, 0);
  1084. cifs_stats_inc(&tcon->stats.cifs_stats.num_opens);
  1085. if (rc) {
  1086. cifs_dbg(FYI, "Error in Open = %d\n", rc);
  1087. } else {
  1088. /* BB verify if wct == 15 */
  1089. /* *pOplock = pSMBr->OplockLevel; */ /* BB take from action field*/
  1090. *netfid = pSMBr->Fid; /* cifs fid stays in le */
  1091. /* Let caller know file was created so we can set the mode. */
  1092. /* Do we care about the CreateAction in any other cases? */
  1093. /* BB FIXME BB */
  1094. /* if (cpu_to_le32(FILE_CREATE) == pSMBr->CreateAction)
  1095. *pOplock |= CIFS_CREATE_ACTION; */
  1096. /* BB FIXME END */
  1097. if (pfile_info) {
  1098. pfile_info->CreationTime = 0; /* BB convert CreateTime*/
  1099. pfile_info->LastAccessTime = 0; /* BB fixme */
  1100. pfile_info->LastWriteTime = 0; /* BB fixme */
  1101. pfile_info->ChangeTime = 0; /* BB fixme */
  1102. pfile_info->Attributes =
  1103. cpu_to_le32(le16_to_cpu(pSMBr->FileAttributes));
  1104. /* the file_info buf is endian converted by caller */
  1105. pfile_info->AllocationSize =
  1106. cpu_to_le64(le32_to_cpu(pSMBr->EndOfFile));
  1107. pfile_info->EndOfFile = pfile_info->AllocationSize;
  1108. pfile_info->NumberOfLinks = cpu_to_le32(1);
  1109. pfile_info->DeletePending = 0;
  1110. }
  1111. }
  1112. cifs_buf_release(pSMB);
  1113. if (rc == -EAGAIN)
  1114. goto OldOpenRetry;
  1115. return rc;
  1116. }
  1117. int
  1118. CIFS_open(const unsigned int xid, struct cifs_open_parms *oparms, int *oplock,
  1119. FILE_ALL_INFO *buf)
  1120. {
  1121. int rc = -EACCES;
  1122. OPEN_REQ *req = NULL;
  1123. OPEN_RSP *rsp = NULL;
  1124. int bytes_returned;
  1125. int name_len;
  1126. __u16 count;
  1127. struct cifs_sb_info *cifs_sb = oparms->cifs_sb;
  1128. struct cifs_tcon *tcon = oparms->tcon;
  1129. int remap = cifs_remap(cifs_sb);
  1130. const struct nls_table *nls = cifs_sb->local_nls;
  1131. int create_options = oparms->create_options;
  1132. int desired_access = oparms->desired_access;
  1133. int disposition = oparms->disposition;
  1134. const char *path = oparms->path;
  1135. openRetry:
  1136. rc = smb_init(SMB_COM_NT_CREATE_ANDX, 24, tcon, (void **)&req,
  1137. (void **)&rsp);
  1138. if (rc)
  1139. return rc;
  1140. /* no commands go after this */
  1141. req->AndXCommand = 0xFF;
  1142. if (req->hdr.Flags2 & SMBFLG2_UNICODE) {
  1143. /* account for one byte pad to word boundary */
  1144. count = 1;
  1145. name_len = cifsConvertToUTF16((__le16 *)(req->fileName + 1),
  1146. path, PATH_MAX, nls, remap);
  1147. /* trailing null */
  1148. name_len++;
  1149. name_len *= 2;
  1150. req->NameLength = cpu_to_le16(name_len);
  1151. } else {
  1152. /* BB improve check for buffer overruns BB */
  1153. /* no pad */
  1154. count = 0;
  1155. name_len = strnlen(path, PATH_MAX);
  1156. /* trailing null */
  1157. name_len++;
  1158. req->NameLength = cpu_to_le16(name_len);
  1159. strncpy(req->fileName, path, name_len);
  1160. }
  1161. if (*oplock & REQ_OPLOCK)
  1162. req->OpenFlags = cpu_to_le32(REQ_OPLOCK);
  1163. else if (*oplock & REQ_BATCHOPLOCK)
  1164. req->OpenFlags = cpu_to_le32(REQ_BATCHOPLOCK);
  1165. req->DesiredAccess = cpu_to_le32(desired_access);
  1166. req->AllocationSize = 0;
  1167. /*
  1168. * Set file as system file if special file such as fifo and server
  1169. * expecting SFU style and no Unix extensions.
  1170. */
  1171. if (create_options & CREATE_OPTION_SPECIAL)
  1172. req->FileAttributes = cpu_to_le32(ATTR_SYSTEM);
  1173. else
  1174. req->FileAttributes = cpu_to_le32(ATTR_NORMAL);
  1175. /*
  1176. * XP does not handle ATTR_POSIX_SEMANTICS but it helps speed up case
  1177. * sensitive checks for other servers such as Samba.
  1178. */
  1179. if (tcon->ses->capabilities & CAP_UNIX)
  1180. req->FileAttributes |= cpu_to_le32(ATTR_POSIX_SEMANTICS);
  1181. if (create_options & CREATE_OPTION_READONLY)
  1182. req->FileAttributes |= cpu_to_le32(ATTR_READONLY);
  1183. req->ShareAccess = cpu_to_le32(FILE_SHARE_ALL);
  1184. req->CreateDisposition = cpu_to_le32(disposition);
  1185. req->CreateOptions = cpu_to_le32(create_options & CREATE_OPTIONS_MASK);
  1186. /* BB Expirement with various impersonation levels and verify */
  1187. req->ImpersonationLevel = cpu_to_le32(SECURITY_IMPERSONATION);
  1188. req->SecurityFlags = SECURITY_CONTEXT_TRACKING|SECURITY_EFFECTIVE_ONLY;
  1189. count += name_len;
  1190. inc_rfc1001_len(req, count);
  1191. req->ByteCount = cpu_to_le16(count);
  1192. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *)req,
  1193. (struct smb_hdr *)rsp, &bytes_returned, 0);
  1194. cifs_stats_inc(&tcon->stats.cifs_stats.num_opens);
  1195. if (rc) {
  1196. cifs_dbg(FYI, "Error in Open = %d\n", rc);
  1197. cifs_buf_release(req);
  1198. if (rc == -EAGAIN)
  1199. goto openRetry;
  1200. return rc;
  1201. }
  1202. /* 1 byte no need to le_to_cpu */
  1203. *oplock = rsp->OplockLevel;
  1204. /* cifs fid stays in le */
  1205. oparms->fid->netfid = rsp->Fid;
  1206. /* Let caller know file was created so we can set the mode. */
  1207. /* Do we care about the CreateAction in any other cases? */
  1208. if (cpu_to_le32(FILE_CREATE) == rsp->CreateAction)
  1209. *oplock |= CIFS_CREATE_ACTION;
  1210. if (buf) {
  1211. /* copy from CreationTime to Attributes */
  1212. memcpy((char *)buf, (char *)&rsp->CreationTime, 36);
  1213. /* the file_info buf is endian converted by caller */
  1214. buf->AllocationSize = rsp->AllocationSize;
  1215. buf->EndOfFile = rsp->EndOfFile;
  1216. buf->NumberOfLinks = cpu_to_le32(1);
  1217. buf->DeletePending = 0;
  1218. }
  1219. cifs_buf_release(req);
  1220. return rc;
  1221. }
  1222. /*
  1223. * Discard any remaining data in the current SMB. To do this, we borrow the
  1224. * current bigbuf.
  1225. */
  1226. static int
  1227. discard_remaining_data(struct TCP_Server_Info *server)
  1228. {
  1229. unsigned int rfclen = get_rfc1002_length(server->smallbuf);
  1230. int remaining = rfclen + 4 - server->total_read;
  1231. while (remaining > 0) {
  1232. int length;
  1233. length = cifs_read_from_socket(server, server->bigbuf,
  1234. min_t(unsigned int, remaining,
  1235. CIFSMaxBufSize + MAX_HEADER_SIZE(server)));
  1236. if (length < 0)
  1237. return length;
  1238. server->total_read += length;
  1239. remaining -= length;
  1240. }
  1241. return 0;
  1242. }
  1243. static int
  1244. cifs_readv_discard(struct TCP_Server_Info *server, struct mid_q_entry *mid)
  1245. {
  1246. int length;
  1247. struct cifs_readdata *rdata = mid->callback_data;
  1248. length = discard_remaining_data(server);
  1249. dequeue_mid(mid, rdata->result);
  1250. mid->resp_buf = server->smallbuf;
  1251. server->smallbuf = NULL;
  1252. return length;
  1253. }
  1254. int
  1255. cifs_readv_receive(struct TCP_Server_Info *server, struct mid_q_entry *mid)
  1256. {
  1257. int length, len;
  1258. unsigned int data_offset, data_len;
  1259. struct cifs_readdata *rdata = mid->callback_data;
  1260. char *buf = server->smallbuf;
  1261. unsigned int buflen = get_rfc1002_length(buf) + 4;
  1262. cifs_dbg(FYI, "%s: mid=%llu offset=%llu bytes=%u\n",
  1263. __func__, mid->mid, rdata->offset, rdata->bytes);
  1264. /*
  1265. * read the rest of READ_RSP header (sans Data array), or whatever we
  1266. * can if there's not enough data. At this point, we've read down to
  1267. * the Mid.
  1268. */
  1269. len = min_t(unsigned int, buflen, server->vals->read_rsp_size) -
  1270. HEADER_SIZE(server) + 1;
  1271. length = cifs_read_from_socket(server,
  1272. buf + HEADER_SIZE(server) - 1, len);
  1273. if (length < 0)
  1274. return length;
  1275. server->total_read += length;
  1276. if (server->ops->is_session_expired &&
  1277. server->ops->is_session_expired(buf)) {
  1278. cifs_reconnect(server);
  1279. wake_up(&server->response_q);
  1280. return -1;
  1281. }
  1282. if (server->ops->is_status_pending &&
  1283. server->ops->is_status_pending(buf, server, 0)) {
  1284. discard_remaining_data(server);
  1285. return -1;
  1286. }
  1287. /* Was the SMB read successful? */
  1288. rdata->result = server->ops->map_error(buf, false);
  1289. if (rdata->result != 0) {
  1290. cifs_dbg(FYI, "%s: server returned error %d\n",
  1291. __func__, rdata->result);
  1292. return cifs_readv_discard(server, mid);
  1293. }
  1294. /* Is there enough to get to the rest of the READ_RSP header? */
  1295. if (server->total_read < server->vals->read_rsp_size) {
  1296. cifs_dbg(FYI, "%s: server returned short header. got=%u expected=%zu\n",
  1297. __func__, server->total_read,
  1298. server->vals->read_rsp_size);
  1299. rdata->result = -EIO;
  1300. return cifs_readv_discard(server, mid);
  1301. }
  1302. data_offset = server->ops->read_data_offset(buf) + 4;
  1303. if (data_offset < server->total_read) {
  1304. /*
  1305. * win2k8 sometimes sends an offset of 0 when the read
  1306. * is beyond the EOF. Treat it as if the data starts just after
  1307. * the header.
  1308. */
  1309. cifs_dbg(FYI, "%s: data offset (%u) inside read response header\n",
  1310. __func__, data_offset);
  1311. data_offset = server->total_read;
  1312. } else if (data_offset > MAX_CIFS_SMALL_BUFFER_SIZE) {
  1313. /* data_offset is beyond the end of smallbuf */
  1314. cifs_dbg(FYI, "%s: data offset (%u) beyond end of smallbuf\n",
  1315. __func__, data_offset);
  1316. rdata->result = -EIO;
  1317. return cifs_readv_discard(server, mid);
  1318. }
  1319. cifs_dbg(FYI, "%s: total_read=%u data_offset=%u\n",
  1320. __func__, server->total_read, data_offset);
  1321. len = data_offset - server->total_read;
  1322. if (len > 0) {
  1323. /* read any junk before data into the rest of smallbuf */
  1324. length = cifs_read_from_socket(server,
  1325. buf + server->total_read, len);
  1326. if (length < 0)
  1327. return length;
  1328. server->total_read += length;
  1329. }
  1330. /* set up first iov for signature check */
  1331. rdata->iov.iov_base = buf;
  1332. rdata->iov.iov_len = server->total_read;
  1333. cifs_dbg(FYI, "0: iov_base=%p iov_len=%zu\n",
  1334. rdata->iov.iov_base, rdata->iov.iov_len);
  1335. /* how much data is in the response? */
  1336. data_len = server->ops->read_data_length(buf);
  1337. if (data_offset + data_len > buflen) {
  1338. /* data_len is corrupt -- discard frame */
  1339. rdata->result = -EIO;
  1340. return cifs_readv_discard(server, mid);
  1341. }
  1342. length = rdata->read_into_pages(server, rdata, data_len);
  1343. if (length < 0)
  1344. return length;
  1345. server->total_read += length;
  1346. cifs_dbg(FYI, "total_read=%u buflen=%u remaining=%u\n",
  1347. server->total_read, buflen, data_len);
  1348. /* discard anything left over */
  1349. if (server->total_read < buflen)
  1350. return cifs_readv_discard(server, mid);
  1351. dequeue_mid(mid, false);
  1352. mid->resp_buf = server->smallbuf;
  1353. server->smallbuf = NULL;
  1354. return length;
  1355. }
  1356. static void
  1357. cifs_readv_callback(struct mid_q_entry *mid)
  1358. {
  1359. struct cifs_readdata *rdata = mid->callback_data;
  1360. struct cifs_tcon *tcon = tlink_tcon(rdata->cfile->tlink);
  1361. struct TCP_Server_Info *server = tcon->ses->server;
  1362. struct smb_rqst rqst = { .rq_iov = &rdata->iov,
  1363. .rq_nvec = 1,
  1364. .rq_pages = rdata->pages,
  1365. .rq_npages = rdata->nr_pages,
  1366. .rq_pagesz = rdata->pagesz,
  1367. .rq_tailsz = rdata->tailsz };
  1368. cifs_dbg(FYI, "%s: mid=%llu state=%d result=%d bytes=%u\n",
  1369. __func__, mid->mid, mid->mid_state, rdata->result,
  1370. rdata->bytes);
  1371. switch (mid->mid_state) {
  1372. case MID_RESPONSE_RECEIVED:
  1373. /* result already set, check signature */
  1374. if (server->sign) {
  1375. int rc = 0;
  1376. rc = cifs_verify_signature(&rqst, server,
  1377. mid->sequence_number);
  1378. if (rc)
  1379. cifs_dbg(VFS, "SMB signature verification returned error = %d\n",
  1380. rc);
  1381. }
  1382. /* FIXME: should this be counted toward the initiating task? */
  1383. task_io_account_read(rdata->got_bytes);
  1384. cifs_stats_bytes_read(tcon, rdata->got_bytes);
  1385. break;
  1386. case MID_REQUEST_SUBMITTED:
  1387. case MID_RETRY_NEEDED:
  1388. rdata->result = -EAGAIN;
  1389. if (server->sign && rdata->got_bytes)
  1390. /* reset bytes number since we can not check a sign */
  1391. rdata->got_bytes = 0;
  1392. /* FIXME: should this be counted toward the initiating task? */
  1393. task_io_account_read(rdata->got_bytes);
  1394. cifs_stats_bytes_read(tcon, rdata->got_bytes);
  1395. break;
  1396. default:
  1397. rdata->result = -EIO;
  1398. }
  1399. queue_work(cifsiod_wq, &rdata->work);
  1400. mutex_lock(&server->srv_mutex);
  1401. DeleteMidQEntry(mid);
  1402. mutex_unlock(&server->srv_mutex);
  1403. add_credits(server, 1, 0);
  1404. }
  1405. /* cifs_async_readv - send an async write, and set up mid to handle result */
  1406. int
  1407. cifs_async_readv(struct cifs_readdata *rdata)
  1408. {
  1409. int rc;
  1410. READ_REQ *smb = NULL;
  1411. int wct;
  1412. struct cifs_tcon *tcon = tlink_tcon(rdata->cfile->tlink);
  1413. struct smb_rqst rqst = { .rq_iov = &rdata->iov,
  1414. .rq_nvec = 1 };
  1415. cifs_dbg(FYI, "%s: offset=%llu bytes=%u\n",
  1416. __func__, rdata->offset, rdata->bytes);
  1417. if (tcon->ses->capabilities & CAP_LARGE_FILES)
  1418. wct = 12;
  1419. else {
  1420. wct = 10; /* old style read */
  1421. if ((rdata->offset >> 32) > 0) {
  1422. /* can not handle this big offset for old */
  1423. return -EIO;
  1424. }
  1425. }
  1426. rc = small_smb_init(SMB_COM_READ_ANDX, wct, tcon, (void **)&smb);
  1427. if (rc)
  1428. return rc;
  1429. smb->hdr.Pid = cpu_to_le16((__u16)rdata->pid);
  1430. smb->hdr.PidHigh = cpu_to_le16((__u16)(rdata->pid >> 16));
  1431. smb->AndXCommand = 0xFF; /* none */
  1432. smb->Fid = rdata->cfile->fid.netfid;
  1433. smb->OffsetLow = cpu_to_le32(rdata->offset & 0xFFFFFFFF);
  1434. if (wct == 12)
  1435. smb->OffsetHigh = cpu_to_le32(rdata->offset >> 32);
  1436. smb->Remaining = 0;
  1437. smb->MaxCount = cpu_to_le16(rdata->bytes & 0xFFFF);
  1438. smb->MaxCountHigh = cpu_to_le32(rdata->bytes >> 16);
  1439. if (wct == 12)
  1440. smb->ByteCount = 0;
  1441. else {
  1442. /* old style read */
  1443. struct smb_com_readx_req *smbr =
  1444. (struct smb_com_readx_req *)smb;
  1445. smbr->ByteCount = 0;
  1446. }
  1447. /* 4 for RFC1001 length + 1 for BCC */
  1448. rdata->iov.iov_base = smb;
  1449. rdata->iov.iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4;
  1450. kref_get(&rdata->refcount);
  1451. rc = cifs_call_async(tcon->ses->server, &rqst, cifs_readv_receive,
  1452. cifs_readv_callback, rdata, 0);
  1453. if (rc == 0)
  1454. cifs_stats_inc(&tcon->stats.cifs_stats.num_reads);
  1455. else
  1456. kref_put(&rdata->refcount, cifs_readdata_release);
  1457. cifs_small_buf_release(smb);
  1458. return rc;
  1459. }
  1460. int
  1461. CIFSSMBRead(const unsigned int xid, struct cifs_io_parms *io_parms,
  1462. unsigned int *nbytes, char **buf, int *pbuf_type)
  1463. {
  1464. int rc = -EACCES;
  1465. READ_REQ *pSMB = NULL;
  1466. READ_RSP *pSMBr = NULL;
  1467. char *pReadData = NULL;
  1468. int wct;
  1469. int resp_buf_type = 0;
  1470. struct kvec iov[1];
  1471. __u32 pid = io_parms->pid;
  1472. __u16 netfid = io_parms->netfid;
  1473. __u64 offset = io_parms->offset;
  1474. struct cifs_tcon *tcon = io_parms->tcon;
  1475. unsigned int count = io_parms->length;
  1476. cifs_dbg(FYI, "Reading %d bytes on fid %d\n", count, netfid);
  1477. if (tcon->ses->capabilities & CAP_LARGE_FILES)
  1478. wct = 12;
  1479. else {
  1480. wct = 10; /* old style read */
  1481. if ((offset >> 32) > 0) {
  1482. /* can not handle this big offset for old */
  1483. return -EIO;
  1484. }
  1485. }
  1486. *nbytes = 0;
  1487. rc = small_smb_init(SMB_COM_READ_ANDX, wct, tcon, (void **) &pSMB);
  1488. if (rc)
  1489. return rc;
  1490. pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
  1491. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
  1492. /* tcon and ses pointer are checked in smb_init */
  1493. if (tcon->ses->server == NULL)
  1494. return -ECONNABORTED;
  1495. pSMB->AndXCommand = 0xFF; /* none */
  1496. pSMB->Fid = netfid;
  1497. pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
  1498. if (wct == 12)
  1499. pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
  1500. pSMB->Remaining = 0;
  1501. pSMB->MaxCount = cpu_to_le16(count & 0xFFFF);
  1502. pSMB->MaxCountHigh = cpu_to_le32(count >> 16);
  1503. if (wct == 12)
  1504. pSMB->ByteCount = 0; /* no need to do le conversion since 0 */
  1505. else {
  1506. /* old style read */
  1507. struct smb_com_readx_req *pSMBW =
  1508. (struct smb_com_readx_req *)pSMB;
  1509. pSMBW->ByteCount = 0;
  1510. }
  1511. iov[0].iov_base = (char *)pSMB;
  1512. iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
  1513. rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovecs */,
  1514. &resp_buf_type, CIFS_LOG_ERROR);
  1515. cifs_stats_inc(&tcon->stats.cifs_stats.num_reads);
  1516. pSMBr = (READ_RSP *)iov[0].iov_base;
  1517. if (rc) {
  1518. cifs_dbg(VFS, "Send error in read = %d\n", rc);
  1519. } else {
  1520. int data_length = le16_to_cpu(pSMBr->DataLengthHigh);
  1521. data_length = data_length << 16;
  1522. data_length += le16_to_cpu(pSMBr->DataLength);
  1523. *nbytes = data_length;
  1524. /*check that DataLength would not go beyond end of SMB */
  1525. if ((data_length > CIFSMaxBufSize)
  1526. || (data_length > count)) {
  1527. cifs_dbg(FYI, "bad length %d for count %d\n",
  1528. data_length, count);
  1529. rc = -EIO;
  1530. *nbytes = 0;
  1531. } else {
  1532. pReadData = (char *) (&pSMBr->hdr.Protocol) +
  1533. le16_to_cpu(pSMBr->DataOffset);
  1534. /* if (rc = copy_to_user(buf, pReadData, data_length)) {
  1535. cifs_dbg(VFS, "Faulting on read rc = %d\n",rc);
  1536. rc = -EFAULT;
  1537. }*/ /* can not use copy_to_user when using page cache*/
  1538. if (*buf)
  1539. memcpy(*buf, pReadData, data_length);
  1540. }
  1541. }
  1542. /* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
  1543. if (*buf) {
  1544. free_rsp_buf(resp_buf_type, iov[0].iov_base);
  1545. } else if (resp_buf_type != CIFS_NO_BUFFER) {
  1546. /* return buffer to caller to free */
  1547. *buf = iov[0].iov_base;
  1548. if (resp_buf_type == CIFS_SMALL_BUFFER)
  1549. *pbuf_type = CIFS_SMALL_BUFFER;
  1550. else if (resp_buf_type == CIFS_LARGE_BUFFER)
  1551. *pbuf_type = CIFS_LARGE_BUFFER;
  1552. } /* else no valid buffer on return - leave as null */
  1553. /* Note: On -EAGAIN error only caller can retry on handle based calls
  1554. since file handle passed in no longer valid */
  1555. return rc;
  1556. }
  1557. int
  1558. CIFSSMBWrite(const unsigned int xid, struct cifs_io_parms *io_parms,
  1559. unsigned int *nbytes, const char *buf)
  1560. {
  1561. int rc = -EACCES;
  1562. WRITE_REQ *pSMB = NULL;
  1563. WRITE_RSP *pSMBr = NULL;
  1564. int bytes_returned, wct;
  1565. __u32 bytes_sent;
  1566. __u16 byte_count;
  1567. __u32 pid = io_parms->pid;
  1568. __u16 netfid = io_parms->netfid;
  1569. __u64 offset = io_parms->offset;
  1570. struct cifs_tcon *tcon = io_parms->tcon;
  1571. unsigned int count = io_parms->length;
  1572. *nbytes = 0;
  1573. /* cifs_dbg(FYI, "write at %lld %d bytes\n", offset, count);*/
  1574. if (tcon->ses == NULL)
  1575. return -ECONNABORTED;
  1576. if (tcon->ses->capabilities & CAP_LARGE_FILES)
  1577. wct = 14;
  1578. else {
  1579. wct = 12;
  1580. if ((offset >> 32) > 0) {
  1581. /* can not handle big offset for old srv */
  1582. return -EIO;
  1583. }
  1584. }
  1585. rc = smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **) &pSMB,
  1586. (void **) &pSMBr);
  1587. if (rc)
  1588. return rc;
  1589. pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
  1590. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
  1591. /* tcon and ses pointer are checked in smb_init */
  1592. if (tcon->ses->server == NULL)
  1593. return -ECONNABORTED;
  1594. pSMB->AndXCommand = 0xFF; /* none */
  1595. pSMB->Fid = netfid;
  1596. pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
  1597. if (wct == 14)
  1598. pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
  1599. pSMB->Reserved = 0xFFFFFFFF;
  1600. pSMB->WriteMode = 0;
  1601. pSMB->Remaining = 0;
  1602. /* Can increase buffer size if buffer is big enough in some cases ie we
  1603. can send more if LARGE_WRITE_X capability returned by the server and if
  1604. our buffer is big enough or if we convert to iovecs on socket writes
  1605. and eliminate the copy to the CIFS buffer */
  1606. if (tcon->ses->capabilities & CAP_LARGE_WRITE_X) {
  1607. bytes_sent = min_t(const unsigned int, CIFSMaxBufSize, count);
  1608. } else {
  1609. bytes_sent = (tcon->ses->server->maxBuf - MAX_CIFS_HDR_SIZE)
  1610. & ~0xFF;
  1611. }
  1612. if (bytes_sent > count)
  1613. bytes_sent = count;
  1614. pSMB->DataOffset =
  1615. cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
  1616. if (buf)
  1617. memcpy(pSMB->Data, buf, bytes_sent);
  1618. else if (count != 0) {
  1619. /* No buffer */
  1620. cifs_buf_release(pSMB);
  1621. return -EINVAL;
  1622. } /* else setting file size with write of zero bytes */
  1623. if (wct == 14)
  1624. byte_count = bytes_sent + 1; /* pad */
  1625. else /* wct == 12 */
  1626. byte_count = bytes_sent + 5; /* bigger pad, smaller smb hdr */
  1627. pSMB->DataLengthLow = cpu_to_le16(bytes_sent & 0xFFFF);
  1628. pSMB->DataLengthHigh = cpu_to_le16(bytes_sent >> 16);
  1629. inc_rfc1001_len(pSMB, byte_count);
  1630. if (wct == 14)
  1631. pSMB->ByteCount = cpu_to_le16(byte_count);
  1632. else { /* old style write has byte count 4 bytes earlier
  1633. so 4 bytes pad */
  1634. struct smb_com_writex_req *pSMBW =
  1635. (struct smb_com_writex_req *)pSMB;
  1636. pSMBW->ByteCount = cpu_to_le16(byte_count);
  1637. }
  1638. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  1639. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  1640. cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
  1641. if (rc) {
  1642. cifs_dbg(FYI, "Send error in write = %d\n", rc);
  1643. } else {
  1644. *nbytes = le16_to_cpu(pSMBr->CountHigh);
  1645. *nbytes = (*nbytes) << 16;
  1646. *nbytes += le16_to_cpu(pSMBr->Count);
  1647. /*
  1648. * Mask off high 16 bits when bytes written as returned by the
  1649. * server is greater than bytes requested by the client. Some
  1650. * OS/2 servers are known to set incorrect CountHigh values.
  1651. */
  1652. if (*nbytes > count)
  1653. *nbytes &= 0xFFFF;
  1654. }
  1655. cifs_buf_release(pSMB);
  1656. /* Note: On -EAGAIN error only caller can retry on handle based calls
  1657. since file handle passed in no longer valid */
  1658. return rc;
  1659. }
  1660. void
  1661. cifs_writedata_release(struct kref *refcount)
  1662. {
  1663. struct cifs_writedata *wdata = container_of(refcount,
  1664. struct cifs_writedata, refcount);
  1665. if (wdata->cfile)
  1666. cifsFileInfo_put(wdata->cfile);
  1667. kfree(wdata);
  1668. }
  1669. /*
  1670. * Write failed with a retryable error. Resend the write request. It's also
  1671. * possible that the page was redirtied so re-clean the page.
  1672. */
  1673. static void
  1674. cifs_writev_requeue(struct cifs_writedata *wdata)
  1675. {
  1676. int i, rc = 0;
  1677. struct inode *inode = d_inode(wdata->cfile->dentry);
  1678. struct TCP_Server_Info *server;
  1679. unsigned int rest_len;
  1680. server = tlink_tcon(wdata->cfile->tlink)->ses->server;
  1681. i = 0;
  1682. rest_len = wdata->bytes;
  1683. do {
  1684. struct cifs_writedata *wdata2;
  1685. unsigned int j, nr_pages, wsize, tailsz, cur_len;
  1686. wsize = server->ops->wp_retry_size(inode);
  1687. if (wsize < rest_len) {
  1688. nr_pages = wsize / PAGE_SIZE;
  1689. if (!nr_pages) {
  1690. rc = -ENOTSUPP;
  1691. break;
  1692. }
  1693. cur_len = nr_pages * PAGE_SIZE;
  1694. tailsz = PAGE_SIZE;
  1695. } else {
  1696. nr_pages = DIV_ROUND_UP(rest_len, PAGE_SIZE);
  1697. cur_len = rest_len;
  1698. tailsz = rest_len - (nr_pages - 1) * PAGE_SIZE;
  1699. }
  1700. wdata2 = cifs_writedata_alloc(nr_pages, cifs_writev_complete);
  1701. if (!wdata2) {
  1702. rc = -ENOMEM;
  1703. break;
  1704. }
  1705. for (j = 0; j < nr_pages; j++) {
  1706. wdata2->pages[j] = wdata->pages[i + j];
  1707. lock_page(wdata2->pages[j]);
  1708. clear_page_dirty_for_io(wdata2->pages[j]);
  1709. }
  1710. wdata2->sync_mode = wdata->sync_mode;
  1711. wdata2->nr_pages = nr_pages;
  1712. wdata2->offset = page_offset(wdata2->pages[0]);
  1713. wdata2->pagesz = PAGE_SIZE;
  1714. wdata2->tailsz = tailsz;
  1715. wdata2->bytes = cur_len;
  1716. wdata2->cfile = find_writable_file(CIFS_I(inode), false);
  1717. if (!wdata2->cfile) {
  1718. cifs_dbg(VFS, "No writable handles for inode\n");
  1719. rc = -EBADF;
  1720. break;
  1721. }
  1722. wdata2->pid = wdata2->cfile->pid;
  1723. rc = server->ops->async_writev(wdata2, cifs_writedata_release);
  1724. for (j = 0; j < nr_pages; j++) {
  1725. unlock_page(wdata2->pages[j]);
  1726. if (rc != 0 && rc != -EAGAIN) {
  1727. SetPageError(wdata2->pages[j]);
  1728. end_page_writeback(wdata2->pages[j]);
  1729. put_page(wdata2->pages[j]);
  1730. }
  1731. }
  1732. if (rc) {
  1733. kref_put(&wdata2->refcount, cifs_writedata_release);
  1734. if (rc == -EAGAIN)
  1735. continue;
  1736. break;
  1737. }
  1738. rest_len -= cur_len;
  1739. i += nr_pages;
  1740. } while (i < wdata->nr_pages);
  1741. mapping_set_error(inode->i_mapping, rc);
  1742. kref_put(&wdata->refcount, cifs_writedata_release);
  1743. }
  1744. void
  1745. cifs_writev_complete(struct work_struct *work)
  1746. {
  1747. struct cifs_writedata *wdata = container_of(work,
  1748. struct cifs_writedata, work);
  1749. struct inode *inode = d_inode(wdata->cfile->dentry);
  1750. int i = 0;
  1751. if (wdata->result == 0) {
  1752. spin_lock(&inode->i_lock);
  1753. cifs_update_eof(CIFS_I(inode), wdata->offset, wdata->bytes);
  1754. spin_unlock(&inode->i_lock);
  1755. cifs_stats_bytes_written(tlink_tcon(wdata->cfile->tlink),
  1756. wdata->bytes);
  1757. } else if (wdata->sync_mode == WB_SYNC_ALL && wdata->result == -EAGAIN)
  1758. return cifs_writev_requeue(wdata);
  1759. for (i = 0; i < wdata->nr_pages; i++) {
  1760. struct page *page = wdata->pages[i];
  1761. if (wdata->result == -EAGAIN)
  1762. __set_page_dirty_nobuffers(page);
  1763. else if (wdata->result < 0)
  1764. SetPageError(page);
  1765. end_page_writeback(page);
  1766. put_page(page);
  1767. }
  1768. if (wdata->result != -EAGAIN)
  1769. mapping_set_error(inode->i_mapping, wdata->result);
  1770. kref_put(&wdata->refcount, cifs_writedata_release);
  1771. }
  1772. struct cifs_writedata *
  1773. cifs_writedata_alloc(unsigned int nr_pages, work_func_t complete)
  1774. {
  1775. struct cifs_writedata *wdata;
  1776. /* writedata + number of page pointers */
  1777. wdata = kzalloc(sizeof(*wdata) +
  1778. sizeof(struct page *) * nr_pages, GFP_NOFS);
  1779. if (wdata != NULL) {
  1780. kref_init(&wdata->refcount);
  1781. INIT_LIST_HEAD(&wdata->list);
  1782. init_completion(&wdata->done);
  1783. INIT_WORK(&wdata->work, complete);
  1784. }
  1785. return wdata;
  1786. }
  1787. /*
  1788. * Check the mid_state and signature on received buffer (if any), and queue the
  1789. * workqueue completion task.
  1790. */
  1791. static void
  1792. cifs_writev_callback(struct mid_q_entry *mid)
  1793. {
  1794. struct cifs_writedata *wdata = mid->callback_data;
  1795. struct cifs_tcon *tcon = tlink_tcon(wdata->cfile->tlink);
  1796. struct TCP_Server_Info *server = tcon->ses->server;
  1797. unsigned int written;
  1798. WRITE_RSP *smb = (WRITE_RSP *)mid->resp_buf;
  1799. switch (mid->mid_state) {
  1800. case MID_RESPONSE_RECEIVED:
  1801. wdata->result = cifs_check_receive(mid, tcon->ses->server, 0);
  1802. if (wdata->result != 0)
  1803. break;
  1804. written = le16_to_cpu(smb->CountHigh);
  1805. written <<= 16;
  1806. written += le16_to_cpu(smb->Count);
  1807. /*
  1808. * Mask off high 16 bits when bytes written as returned
  1809. * by the server is greater than bytes requested by the
  1810. * client. OS/2 servers are known to set incorrect
  1811. * CountHigh values.
  1812. */
  1813. if (written > wdata->bytes)
  1814. written &= 0xFFFF;
  1815. if (written < wdata->bytes)
  1816. wdata->result = -ENOSPC;
  1817. else
  1818. wdata->bytes = written;
  1819. break;
  1820. case MID_REQUEST_SUBMITTED:
  1821. case MID_RETRY_NEEDED:
  1822. wdata->result = -EAGAIN;
  1823. break;
  1824. default:
  1825. wdata->result = -EIO;
  1826. break;
  1827. }
  1828. queue_work(cifsiod_wq, &wdata->work);
  1829. mutex_lock(&server->srv_mutex);
  1830. DeleteMidQEntry(mid);
  1831. mutex_unlock(&server->srv_mutex);
  1832. add_credits(tcon->ses->server, 1, 0);
  1833. }
  1834. /* cifs_async_writev - send an async write, and set up mid to handle result */
  1835. int
  1836. cifs_async_writev(struct cifs_writedata *wdata,
  1837. void (*release)(struct kref *kref))
  1838. {
  1839. int rc = -EACCES;
  1840. WRITE_REQ *smb = NULL;
  1841. int wct;
  1842. struct cifs_tcon *tcon = tlink_tcon(wdata->cfile->tlink);
  1843. struct kvec iov;
  1844. struct smb_rqst rqst = { };
  1845. if (tcon->ses->capabilities & CAP_LARGE_FILES) {
  1846. wct = 14;
  1847. } else {
  1848. wct = 12;
  1849. if (wdata->offset >> 32 > 0) {
  1850. /* can not handle big offset for old srv */
  1851. return -EIO;
  1852. }
  1853. }
  1854. rc = small_smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **)&smb);
  1855. if (rc)
  1856. goto async_writev_out;
  1857. smb->hdr.Pid = cpu_to_le16((__u16)wdata->pid);
  1858. smb->hdr.PidHigh = cpu_to_le16((__u16)(wdata->pid >> 16));
  1859. smb->AndXCommand = 0xFF; /* none */
  1860. smb->Fid = wdata->cfile->fid.netfid;
  1861. smb->OffsetLow = cpu_to_le32(wdata->offset & 0xFFFFFFFF);
  1862. if (wct == 14)
  1863. smb->OffsetHigh = cpu_to_le32(wdata->offset >> 32);
  1864. smb->Reserved = 0xFFFFFFFF;
  1865. smb->WriteMode = 0;
  1866. smb->Remaining = 0;
  1867. smb->DataOffset =
  1868. cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
  1869. /* 4 for RFC1001 length + 1 for BCC */
  1870. iov.iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4 + 1;
  1871. iov.iov_base = smb;
  1872. rqst.rq_iov = &iov;
  1873. rqst.rq_nvec = 1;
  1874. rqst.rq_pages = wdata->pages;
  1875. rqst.rq_npages = wdata->nr_pages;
  1876. rqst.rq_pagesz = wdata->pagesz;
  1877. rqst.rq_tailsz = wdata->tailsz;
  1878. cifs_dbg(FYI, "async write at %llu %u bytes\n",
  1879. wdata->offset, wdata->bytes);
  1880. smb->DataLengthLow = cpu_to_le16(wdata->bytes & 0xFFFF);
  1881. smb->DataLengthHigh = cpu_to_le16(wdata->bytes >> 16);
  1882. if (wct == 14) {
  1883. inc_rfc1001_len(&smb->hdr, wdata->bytes + 1);
  1884. put_bcc(wdata->bytes + 1, &smb->hdr);
  1885. } else {
  1886. /* wct == 12 */
  1887. struct smb_com_writex_req *smbw =
  1888. (struct smb_com_writex_req *)smb;
  1889. inc_rfc1001_len(&smbw->hdr, wdata->bytes + 5);
  1890. put_bcc(wdata->bytes + 5, &smbw->hdr);
  1891. iov.iov_len += 4; /* pad bigger by four bytes */
  1892. }
  1893. kref_get(&wdata->refcount);
  1894. rc = cifs_call_async(tcon->ses->server, &rqst, NULL,
  1895. cifs_writev_callback, wdata, 0);
  1896. if (rc == 0)
  1897. cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
  1898. else
  1899. kref_put(&wdata->refcount, release);
  1900. async_writev_out:
  1901. cifs_small_buf_release(smb);
  1902. return rc;
  1903. }
  1904. int
  1905. CIFSSMBWrite2(const unsigned int xid, struct cifs_io_parms *io_parms,
  1906. unsigned int *nbytes, struct kvec *iov, int n_vec)
  1907. {
  1908. int rc = -EACCES;
  1909. WRITE_REQ *pSMB = NULL;
  1910. int wct;
  1911. int smb_hdr_len;
  1912. int resp_buf_type = 0;
  1913. __u32 pid = io_parms->pid;
  1914. __u16 netfid = io_parms->netfid;
  1915. __u64 offset = io_parms->offset;
  1916. struct cifs_tcon *tcon = io_parms->tcon;
  1917. unsigned int count = io_parms->length;
  1918. *nbytes = 0;
  1919. cifs_dbg(FYI, "write2 at %lld %d bytes\n", (long long)offset, count);
  1920. if (tcon->ses->capabilities & CAP_LARGE_FILES) {
  1921. wct = 14;
  1922. } else {
  1923. wct = 12;
  1924. if ((offset >> 32) > 0) {
  1925. /* can not handle big offset for old srv */
  1926. return -EIO;
  1927. }
  1928. }
  1929. rc = small_smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **) &pSMB);
  1930. if (rc)
  1931. return rc;
  1932. pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
  1933. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
  1934. /* tcon and ses pointer are checked in smb_init */
  1935. if (tcon->ses->server == NULL)
  1936. return -ECONNABORTED;
  1937. pSMB->AndXCommand = 0xFF; /* none */
  1938. pSMB->Fid = netfid;
  1939. pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
  1940. if (wct == 14)
  1941. pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
  1942. pSMB->Reserved = 0xFFFFFFFF;
  1943. pSMB->WriteMode = 0;
  1944. pSMB->Remaining = 0;
  1945. pSMB->DataOffset =
  1946. cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
  1947. pSMB->DataLengthLow = cpu_to_le16(count & 0xFFFF);
  1948. pSMB->DataLengthHigh = cpu_to_le16(count >> 16);
  1949. /* header + 1 byte pad */
  1950. smb_hdr_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 1;
  1951. if (wct == 14)
  1952. inc_rfc1001_len(pSMB, count + 1);
  1953. else /* wct == 12 */
  1954. inc_rfc1001_len(pSMB, count + 5); /* smb data starts later */
  1955. if (wct == 14)
  1956. pSMB->ByteCount = cpu_to_le16(count + 1);
  1957. else /* wct == 12 */ /* bigger pad, smaller smb hdr, keep offset ok */ {
  1958. struct smb_com_writex_req *pSMBW =
  1959. (struct smb_com_writex_req *)pSMB;
  1960. pSMBW->ByteCount = cpu_to_le16(count + 5);
  1961. }
  1962. iov[0].iov_base = pSMB;
  1963. if (wct == 14)
  1964. iov[0].iov_len = smb_hdr_len + 4;
  1965. else /* wct == 12 pad bigger by four bytes */
  1966. iov[0].iov_len = smb_hdr_len + 8;
  1967. rc = SendReceive2(xid, tcon->ses, iov, n_vec + 1, &resp_buf_type, 0);
  1968. cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
  1969. if (rc) {
  1970. cifs_dbg(FYI, "Send error Write2 = %d\n", rc);
  1971. } else if (resp_buf_type == 0) {
  1972. /* presumably this can not happen, but best to be safe */
  1973. rc = -EIO;
  1974. } else {
  1975. WRITE_RSP *pSMBr = (WRITE_RSP *)iov[0].iov_base;
  1976. *nbytes = le16_to_cpu(pSMBr->CountHigh);
  1977. *nbytes = (*nbytes) << 16;
  1978. *nbytes += le16_to_cpu(pSMBr->Count);
  1979. /*
  1980. * Mask off high 16 bits when bytes written as returned by the
  1981. * server is greater than bytes requested by the client. OS/2
  1982. * servers are known to set incorrect CountHigh values.
  1983. */
  1984. if (*nbytes > count)
  1985. *nbytes &= 0xFFFF;
  1986. }
  1987. /* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
  1988. free_rsp_buf(resp_buf_type, iov[0].iov_base);
  1989. /* Note: On -EAGAIN error only caller can retry on handle based calls
  1990. since file handle passed in no longer valid */
  1991. return rc;
  1992. }
  1993. int cifs_lockv(const unsigned int xid, struct cifs_tcon *tcon,
  1994. const __u16 netfid, const __u8 lock_type, const __u32 num_unlock,
  1995. const __u32 num_lock, LOCKING_ANDX_RANGE *buf)
  1996. {
  1997. int rc = 0;
  1998. LOCK_REQ *pSMB = NULL;
  1999. struct kvec iov[2];
  2000. int resp_buf_type;
  2001. __u16 count;
  2002. cifs_dbg(FYI, "cifs_lockv num lock %d num unlock %d\n",
  2003. num_lock, num_unlock);
  2004. rc = small_smb_init(SMB_COM_LOCKING_ANDX, 8, tcon, (void **) &pSMB);
  2005. if (rc)
  2006. return rc;
  2007. pSMB->Timeout = 0;
  2008. pSMB->NumberOfLocks = cpu_to_le16(num_lock);
  2009. pSMB->NumberOfUnlocks = cpu_to_le16(num_unlock);
  2010. pSMB->LockType = lock_type;
  2011. pSMB->AndXCommand = 0xFF; /* none */
  2012. pSMB->Fid = netfid; /* netfid stays le */
  2013. count = (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
  2014. inc_rfc1001_len(pSMB, count);
  2015. pSMB->ByteCount = cpu_to_le16(count);
  2016. iov[0].iov_base = (char *)pSMB;
  2017. iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4 -
  2018. (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
  2019. iov[1].iov_base = (char *)buf;
  2020. iov[1].iov_len = (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
  2021. cifs_stats_inc(&tcon->stats.cifs_stats.num_locks);
  2022. rc = SendReceive2(xid, tcon->ses, iov, 2, &resp_buf_type, CIFS_NO_RESP);
  2023. if (rc)
  2024. cifs_dbg(FYI, "Send error in cifs_lockv = %d\n", rc);
  2025. return rc;
  2026. }
  2027. int
  2028. CIFSSMBLock(const unsigned int xid, struct cifs_tcon *tcon,
  2029. const __u16 smb_file_id, const __u32 netpid, const __u64 len,
  2030. const __u64 offset, const __u32 numUnlock,
  2031. const __u32 numLock, const __u8 lockType,
  2032. const bool waitFlag, const __u8 oplock_level)
  2033. {
  2034. int rc = 0;
  2035. LOCK_REQ *pSMB = NULL;
  2036. /* LOCK_RSP *pSMBr = NULL; */ /* No response data other than rc to parse */
  2037. int bytes_returned;
  2038. int flags = 0;
  2039. __u16 count;
  2040. cifs_dbg(FYI, "CIFSSMBLock timeout %d numLock %d\n",
  2041. (int)waitFlag, numLock);
  2042. rc = small_smb_init(SMB_COM_LOCKING_ANDX, 8, tcon, (void **) &pSMB);
  2043. if (rc)
  2044. return rc;
  2045. if (lockType == LOCKING_ANDX_OPLOCK_RELEASE) {
  2046. /* no response expected */
  2047. flags = CIFS_ASYNC_OP | CIFS_OBREAK_OP;
  2048. pSMB->Timeout = 0;
  2049. } else if (waitFlag) {
  2050. flags = CIFS_BLOCKING_OP; /* blocking operation, no timeout */
  2051. pSMB->Timeout = cpu_to_le32(-1);/* blocking - do not time out */
  2052. } else {
  2053. pSMB->Timeout = 0;
  2054. }
  2055. pSMB->NumberOfLocks = cpu_to_le16(numLock);
  2056. pSMB->NumberOfUnlocks = cpu_to_le16(numUnlock);
  2057. pSMB->LockType = lockType;
  2058. pSMB->OplockLevel = oplock_level;
  2059. pSMB->AndXCommand = 0xFF; /* none */
  2060. pSMB->Fid = smb_file_id; /* netfid stays le */
  2061. if ((numLock != 0) || (numUnlock != 0)) {
  2062. pSMB->Locks[0].Pid = cpu_to_le16(netpid);
  2063. /* BB where to store pid high? */
  2064. pSMB->Locks[0].LengthLow = cpu_to_le32((u32)len);
  2065. pSMB->Locks[0].LengthHigh = cpu_to_le32((u32)(len>>32));
  2066. pSMB->Locks[0].OffsetLow = cpu_to_le32((u32)offset);
  2067. pSMB->Locks[0].OffsetHigh = cpu_to_le32((u32)(offset>>32));
  2068. count = sizeof(LOCKING_ANDX_RANGE);
  2069. } else {
  2070. /* oplock break */
  2071. count = 0;
  2072. }
  2073. inc_rfc1001_len(pSMB, count);
  2074. pSMB->ByteCount = cpu_to_le16(count);
  2075. if (waitFlag) {
  2076. rc = SendReceiveBlockingLock(xid, tcon, (struct smb_hdr *) pSMB,
  2077. (struct smb_hdr *) pSMB, &bytes_returned);
  2078. cifs_small_buf_release(pSMB);
  2079. } else {
  2080. rc = SendReceiveNoRsp(xid, tcon->ses, (char *)pSMB, flags);
  2081. /* SMB buffer freed by function above */
  2082. }
  2083. cifs_stats_inc(&tcon->stats.cifs_stats.num_locks);
  2084. if (rc)
  2085. cifs_dbg(FYI, "Send error in Lock = %d\n", rc);
  2086. /* Note: On -EAGAIN error only caller can retry on handle based calls
  2087. since file handle passed in no longer valid */
  2088. return rc;
  2089. }
  2090. int
  2091. CIFSSMBPosixLock(const unsigned int xid, struct cifs_tcon *tcon,
  2092. const __u16 smb_file_id, const __u32 netpid,
  2093. const loff_t start_offset, const __u64 len,
  2094. struct file_lock *pLockData, const __u16 lock_type,
  2095. const bool waitFlag)
  2096. {
  2097. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  2098. struct smb_com_transaction2_sfi_rsp *pSMBr = NULL;
  2099. struct cifs_posix_lock *parm_data;
  2100. int rc = 0;
  2101. int timeout = 0;
  2102. int bytes_returned = 0;
  2103. int resp_buf_type = 0;
  2104. __u16 params, param_offset, offset, byte_count, count;
  2105. struct kvec iov[1];
  2106. cifs_dbg(FYI, "Posix Lock\n");
  2107. rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
  2108. if (rc)
  2109. return rc;
  2110. pSMBr = (struct smb_com_transaction2_sfi_rsp *)pSMB;
  2111. params = 6;
  2112. pSMB->MaxSetupCount = 0;
  2113. pSMB->Reserved = 0;
  2114. pSMB->Flags = 0;
  2115. pSMB->Reserved2 = 0;
  2116. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  2117. offset = param_offset + params;
  2118. count = sizeof(struct cifs_posix_lock);
  2119. pSMB->MaxParameterCount = cpu_to_le16(2);
  2120. pSMB->MaxDataCount = cpu_to_le16(1000); /* BB find max SMB from sess */
  2121. pSMB->SetupCount = 1;
  2122. pSMB->Reserved3 = 0;
  2123. if (pLockData)
  2124. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
  2125. else
  2126. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  2127. byte_count = 3 /* pad */ + params + count;
  2128. pSMB->DataCount = cpu_to_le16(count);
  2129. pSMB->ParameterCount = cpu_to_le16(params);
  2130. pSMB->TotalDataCount = pSMB->DataCount;
  2131. pSMB->TotalParameterCount = pSMB->ParameterCount;
  2132. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  2133. parm_data = (struct cifs_posix_lock *)
  2134. (((char *) &pSMB->hdr.Protocol) + offset);
  2135. parm_data->lock_type = cpu_to_le16(lock_type);
  2136. if (waitFlag) {
  2137. timeout = CIFS_BLOCKING_OP; /* blocking operation, no timeout */
  2138. parm_data->lock_flags = cpu_to_le16(1);
  2139. pSMB->Timeout = cpu_to_le32(-1);
  2140. } else
  2141. pSMB->Timeout = 0;
  2142. parm_data->pid = cpu_to_le32(netpid);
  2143. parm_data->start = cpu_to_le64(start_offset);
  2144. parm_data->length = cpu_to_le64(len); /* normalize negative numbers */
  2145. pSMB->DataOffset = cpu_to_le16(offset);
  2146. pSMB->Fid = smb_file_id;
  2147. pSMB->InformationLevel = cpu_to_le16(SMB_SET_POSIX_LOCK);
  2148. pSMB->Reserved4 = 0;
  2149. inc_rfc1001_len(pSMB, byte_count);
  2150. pSMB->ByteCount = cpu_to_le16(byte_count);
  2151. if (waitFlag) {
  2152. rc = SendReceiveBlockingLock(xid, tcon, (struct smb_hdr *) pSMB,
  2153. (struct smb_hdr *) pSMBr, &bytes_returned);
  2154. } else {
  2155. iov[0].iov_base = (char *)pSMB;
  2156. iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
  2157. rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovecs */,
  2158. &resp_buf_type, timeout);
  2159. pSMB = NULL; /* request buf already freed by SendReceive2. Do
  2160. not try to free it twice below on exit */
  2161. pSMBr = (struct smb_com_transaction2_sfi_rsp *)iov[0].iov_base;
  2162. }
  2163. if (rc) {
  2164. cifs_dbg(FYI, "Send error in Posix Lock = %d\n", rc);
  2165. } else if (pLockData) {
  2166. /* lock structure can be returned on get */
  2167. __u16 data_offset;
  2168. __u16 data_count;
  2169. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  2170. if (rc || get_bcc(&pSMBr->hdr) < sizeof(*parm_data)) {
  2171. rc = -EIO; /* bad smb */
  2172. goto plk_err_exit;
  2173. }
  2174. data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  2175. data_count = le16_to_cpu(pSMBr->t2.DataCount);
  2176. if (data_count < sizeof(struct cifs_posix_lock)) {
  2177. rc = -EIO;
  2178. goto plk_err_exit;
  2179. }
  2180. parm_data = (struct cifs_posix_lock *)
  2181. ((char *)&pSMBr->hdr.Protocol + data_offset);
  2182. if (parm_data->lock_type == cpu_to_le16(CIFS_UNLCK))
  2183. pLockData->fl_type = F_UNLCK;
  2184. else {
  2185. if (parm_data->lock_type ==
  2186. cpu_to_le16(CIFS_RDLCK))
  2187. pLockData->fl_type = F_RDLCK;
  2188. else if (parm_data->lock_type ==
  2189. cpu_to_le16(CIFS_WRLCK))
  2190. pLockData->fl_type = F_WRLCK;
  2191. pLockData->fl_start = le64_to_cpu(parm_data->start);
  2192. pLockData->fl_end = pLockData->fl_start +
  2193. le64_to_cpu(parm_data->length) - 1;
  2194. pLockData->fl_pid = le32_to_cpu(parm_data->pid);
  2195. }
  2196. }
  2197. plk_err_exit:
  2198. if (pSMB)
  2199. cifs_small_buf_release(pSMB);
  2200. free_rsp_buf(resp_buf_type, iov[0].iov_base);
  2201. /* Note: On -EAGAIN error only caller can retry on handle based calls
  2202. since file handle passed in no longer valid */
  2203. return rc;
  2204. }
  2205. int
  2206. CIFSSMBClose(const unsigned int xid, struct cifs_tcon *tcon, int smb_file_id)
  2207. {
  2208. int rc = 0;
  2209. CLOSE_REQ *pSMB = NULL;
  2210. cifs_dbg(FYI, "In CIFSSMBClose\n");
  2211. /* do not retry on dead session on close */
  2212. rc = small_smb_init(SMB_COM_CLOSE, 3, tcon, (void **) &pSMB);
  2213. if (rc == -EAGAIN)
  2214. return 0;
  2215. if (rc)
  2216. return rc;
  2217. pSMB->FileID = (__u16) smb_file_id;
  2218. pSMB->LastWriteTime = 0xFFFFFFFF;
  2219. pSMB->ByteCount = 0;
  2220. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  2221. cifs_stats_inc(&tcon->stats.cifs_stats.num_closes);
  2222. if (rc) {
  2223. if (rc != -EINTR) {
  2224. /* EINTR is expected when user ctl-c to kill app */
  2225. cifs_dbg(VFS, "Send error in Close = %d\n", rc);
  2226. }
  2227. }
  2228. /* Since session is dead, file will be closed on server already */
  2229. if (rc == -EAGAIN)
  2230. rc = 0;
  2231. return rc;
  2232. }
  2233. int
  2234. CIFSSMBFlush(const unsigned int xid, struct cifs_tcon *tcon, int smb_file_id)
  2235. {
  2236. int rc = 0;
  2237. FLUSH_REQ *pSMB = NULL;
  2238. cifs_dbg(FYI, "In CIFSSMBFlush\n");
  2239. rc = small_smb_init(SMB_COM_FLUSH, 1, tcon, (void **) &pSMB);
  2240. if (rc)
  2241. return rc;
  2242. pSMB->FileID = (__u16) smb_file_id;
  2243. pSMB->ByteCount = 0;
  2244. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  2245. cifs_stats_inc(&tcon->stats.cifs_stats.num_flushes);
  2246. if (rc)
  2247. cifs_dbg(VFS, "Send error in Flush = %d\n", rc);
  2248. return rc;
  2249. }
  2250. int
  2251. CIFSSMBRename(const unsigned int xid, struct cifs_tcon *tcon,
  2252. const char *from_name, const char *to_name,
  2253. struct cifs_sb_info *cifs_sb)
  2254. {
  2255. int rc = 0;
  2256. RENAME_REQ *pSMB = NULL;
  2257. RENAME_RSP *pSMBr = NULL;
  2258. int bytes_returned;
  2259. int name_len, name_len2;
  2260. __u16 count;
  2261. int remap = cifs_remap(cifs_sb);
  2262. cifs_dbg(FYI, "In CIFSSMBRename\n");
  2263. renameRetry:
  2264. rc = smb_init(SMB_COM_RENAME, 1, tcon, (void **) &pSMB,
  2265. (void **) &pSMBr);
  2266. if (rc)
  2267. return rc;
  2268. pSMB->BufferFormat = 0x04;
  2269. pSMB->SearchAttributes =
  2270. cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
  2271. ATTR_DIRECTORY);
  2272. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2273. name_len = cifsConvertToUTF16((__le16 *) pSMB->OldFileName,
  2274. from_name, PATH_MAX,
  2275. cifs_sb->local_nls, remap);
  2276. name_len++; /* trailing null */
  2277. name_len *= 2;
  2278. pSMB->OldFileName[name_len] = 0x04; /* pad */
  2279. /* protocol requires ASCII signature byte on Unicode string */
  2280. pSMB->OldFileName[name_len + 1] = 0x00;
  2281. name_len2 =
  2282. cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
  2283. to_name, PATH_MAX, cifs_sb->local_nls,
  2284. remap);
  2285. name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
  2286. name_len2 *= 2; /* convert to bytes */
  2287. } else { /* BB improve the check for buffer overruns BB */
  2288. name_len = strnlen(from_name, PATH_MAX);
  2289. name_len++; /* trailing null */
  2290. strncpy(pSMB->OldFileName, from_name, name_len);
  2291. name_len2 = strnlen(to_name, PATH_MAX);
  2292. name_len2++; /* trailing null */
  2293. pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
  2294. strncpy(&pSMB->OldFileName[name_len + 1], to_name, name_len2);
  2295. name_len2++; /* trailing null */
  2296. name_len2++; /* signature byte */
  2297. }
  2298. count = 1 /* 1st signature byte */ + name_len + name_len2;
  2299. inc_rfc1001_len(pSMB, count);
  2300. pSMB->ByteCount = cpu_to_le16(count);
  2301. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2302. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2303. cifs_stats_inc(&tcon->stats.cifs_stats.num_renames);
  2304. if (rc)
  2305. cifs_dbg(FYI, "Send error in rename = %d\n", rc);
  2306. cifs_buf_release(pSMB);
  2307. if (rc == -EAGAIN)
  2308. goto renameRetry;
  2309. return rc;
  2310. }
  2311. int CIFSSMBRenameOpenFile(const unsigned int xid, struct cifs_tcon *pTcon,
  2312. int netfid, const char *target_name,
  2313. const struct nls_table *nls_codepage, int remap)
  2314. {
  2315. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  2316. struct smb_com_transaction2_sfi_rsp *pSMBr = NULL;
  2317. struct set_file_rename *rename_info;
  2318. char *data_offset;
  2319. char dummy_string[30];
  2320. int rc = 0;
  2321. int bytes_returned = 0;
  2322. int len_of_str;
  2323. __u16 params, param_offset, offset, count, byte_count;
  2324. cifs_dbg(FYI, "Rename to File by handle\n");
  2325. rc = smb_init(SMB_COM_TRANSACTION2, 15, pTcon, (void **) &pSMB,
  2326. (void **) &pSMBr);
  2327. if (rc)
  2328. return rc;
  2329. params = 6;
  2330. pSMB->MaxSetupCount = 0;
  2331. pSMB->Reserved = 0;
  2332. pSMB->Flags = 0;
  2333. pSMB->Timeout = 0;
  2334. pSMB->Reserved2 = 0;
  2335. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  2336. offset = param_offset + params;
  2337. data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
  2338. rename_info = (struct set_file_rename *) data_offset;
  2339. pSMB->MaxParameterCount = cpu_to_le16(2);
  2340. pSMB->MaxDataCount = cpu_to_le16(1000); /* BB find max SMB from sess */
  2341. pSMB->SetupCount = 1;
  2342. pSMB->Reserved3 = 0;
  2343. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  2344. byte_count = 3 /* pad */ + params;
  2345. pSMB->ParameterCount = cpu_to_le16(params);
  2346. pSMB->TotalParameterCount = pSMB->ParameterCount;
  2347. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  2348. pSMB->DataOffset = cpu_to_le16(offset);
  2349. /* construct random name ".cifs_tmp<inodenum><mid>" */
  2350. rename_info->overwrite = cpu_to_le32(1);
  2351. rename_info->root_fid = 0;
  2352. /* unicode only call */
  2353. if (target_name == NULL) {
  2354. sprintf(dummy_string, "cifs%x", pSMB->hdr.Mid);
  2355. len_of_str =
  2356. cifsConvertToUTF16((__le16 *)rename_info->target_name,
  2357. dummy_string, 24, nls_codepage, remap);
  2358. } else {
  2359. len_of_str =
  2360. cifsConvertToUTF16((__le16 *)rename_info->target_name,
  2361. target_name, PATH_MAX, nls_codepage,
  2362. remap);
  2363. }
  2364. rename_info->target_name_len = cpu_to_le32(2 * len_of_str);
  2365. count = 12 /* sizeof(struct set_file_rename) */ + (2 * len_of_str);
  2366. byte_count += count;
  2367. pSMB->DataCount = cpu_to_le16(count);
  2368. pSMB->TotalDataCount = pSMB->DataCount;
  2369. pSMB->Fid = netfid;
  2370. pSMB->InformationLevel =
  2371. cpu_to_le16(SMB_SET_FILE_RENAME_INFORMATION);
  2372. pSMB->Reserved4 = 0;
  2373. inc_rfc1001_len(pSMB, byte_count);
  2374. pSMB->ByteCount = cpu_to_le16(byte_count);
  2375. rc = SendReceive(xid, pTcon->ses, (struct smb_hdr *) pSMB,
  2376. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2377. cifs_stats_inc(&pTcon->stats.cifs_stats.num_t2renames);
  2378. if (rc)
  2379. cifs_dbg(FYI, "Send error in Rename (by file handle) = %d\n",
  2380. rc);
  2381. cifs_buf_release(pSMB);
  2382. /* Note: On -EAGAIN error only caller can retry on handle based calls
  2383. since file handle passed in no longer valid */
  2384. return rc;
  2385. }
  2386. int
  2387. CIFSSMBCopy(const unsigned int xid, struct cifs_tcon *tcon,
  2388. const char *fromName, const __u16 target_tid, const char *toName,
  2389. const int flags, const struct nls_table *nls_codepage, int remap)
  2390. {
  2391. int rc = 0;
  2392. COPY_REQ *pSMB = NULL;
  2393. COPY_RSP *pSMBr = NULL;
  2394. int bytes_returned;
  2395. int name_len, name_len2;
  2396. __u16 count;
  2397. cifs_dbg(FYI, "In CIFSSMBCopy\n");
  2398. copyRetry:
  2399. rc = smb_init(SMB_COM_COPY, 1, tcon, (void **) &pSMB,
  2400. (void **) &pSMBr);
  2401. if (rc)
  2402. return rc;
  2403. pSMB->BufferFormat = 0x04;
  2404. pSMB->Tid2 = target_tid;
  2405. pSMB->Flags = cpu_to_le16(flags & COPY_TREE);
  2406. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2407. name_len = cifsConvertToUTF16((__le16 *) pSMB->OldFileName,
  2408. fromName, PATH_MAX, nls_codepage,
  2409. remap);
  2410. name_len++; /* trailing null */
  2411. name_len *= 2;
  2412. pSMB->OldFileName[name_len] = 0x04; /* pad */
  2413. /* protocol requires ASCII signature byte on Unicode string */
  2414. pSMB->OldFileName[name_len + 1] = 0x00;
  2415. name_len2 =
  2416. cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
  2417. toName, PATH_MAX, nls_codepage, remap);
  2418. name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
  2419. name_len2 *= 2; /* convert to bytes */
  2420. } else { /* BB improve the check for buffer overruns BB */
  2421. name_len = strnlen(fromName, PATH_MAX);
  2422. name_len++; /* trailing null */
  2423. strncpy(pSMB->OldFileName, fromName, name_len);
  2424. name_len2 = strnlen(toName, PATH_MAX);
  2425. name_len2++; /* trailing null */
  2426. pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
  2427. strncpy(&pSMB->OldFileName[name_len + 1], toName, name_len2);
  2428. name_len2++; /* trailing null */
  2429. name_len2++; /* signature byte */
  2430. }
  2431. count = 1 /* 1st signature byte */ + name_len + name_len2;
  2432. inc_rfc1001_len(pSMB, count);
  2433. pSMB->ByteCount = cpu_to_le16(count);
  2434. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2435. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2436. if (rc) {
  2437. cifs_dbg(FYI, "Send error in copy = %d with %d files copied\n",
  2438. rc, le16_to_cpu(pSMBr->CopyCount));
  2439. }
  2440. cifs_buf_release(pSMB);
  2441. if (rc == -EAGAIN)
  2442. goto copyRetry;
  2443. return rc;
  2444. }
  2445. int
  2446. CIFSUnixCreateSymLink(const unsigned int xid, struct cifs_tcon *tcon,
  2447. const char *fromName, const char *toName,
  2448. const struct nls_table *nls_codepage, int remap)
  2449. {
  2450. TRANSACTION2_SPI_REQ *pSMB = NULL;
  2451. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  2452. char *data_offset;
  2453. int name_len;
  2454. int name_len_target;
  2455. int rc = 0;
  2456. int bytes_returned = 0;
  2457. __u16 params, param_offset, offset, byte_count;
  2458. cifs_dbg(FYI, "In Symlink Unix style\n");
  2459. createSymLinkRetry:
  2460. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  2461. (void **) &pSMBr);
  2462. if (rc)
  2463. return rc;
  2464. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2465. name_len =
  2466. cifsConvertToUTF16((__le16 *) pSMB->FileName, fromName,
  2467. /* find define for this maxpathcomponent */
  2468. PATH_MAX, nls_codepage, remap);
  2469. name_len++; /* trailing null */
  2470. name_len *= 2;
  2471. } else { /* BB improve the check for buffer overruns BB */
  2472. name_len = strnlen(fromName, PATH_MAX);
  2473. name_len++; /* trailing null */
  2474. strncpy(pSMB->FileName, fromName, name_len);
  2475. }
  2476. params = 6 + name_len;
  2477. pSMB->MaxSetupCount = 0;
  2478. pSMB->Reserved = 0;
  2479. pSMB->Flags = 0;
  2480. pSMB->Timeout = 0;
  2481. pSMB->Reserved2 = 0;
  2482. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  2483. InformationLevel) - 4;
  2484. offset = param_offset + params;
  2485. data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
  2486. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2487. name_len_target =
  2488. cifsConvertToUTF16((__le16 *) data_offset, toName,
  2489. /* find define for this maxpathcomponent */
  2490. PATH_MAX, nls_codepage, remap);
  2491. name_len_target++; /* trailing null */
  2492. name_len_target *= 2;
  2493. } else { /* BB improve the check for buffer overruns BB */
  2494. name_len_target = strnlen(toName, PATH_MAX);
  2495. name_len_target++; /* trailing null */
  2496. strncpy(data_offset, toName, name_len_target);
  2497. }
  2498. pSMB->MaxParameterCount = cpu_to_le16(2);
  2499. /* BB find exact max on data count below from sess */
  2500. pSMB->MaxDataCount = cpu_to_le16(1000);
  2501. pSMB->SetupCount = 1;
  2502. pSMB->Reserved3 = 0;
  2503. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  2504. byte_count = 3 /* pad */ + params + name_len_target;
  2505. pSMB->DataCount = cpu_to_le16(name_len_target);
  2506. pSMB->ParameterCount = cpu_to_le16(params);
  2507. pSMB->TotalDataCount = pSMB->DataCount;
  2508. pSMB->TotalParameterCount = pSMB->ParameterCount;
  2509. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  2510. pSMB->DataOffset = cpu_to_le16(offset);
  2511. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_LINK);
  2512. pSMB->Reserved4 = 0;
  2513. inc_rfc1001_len(pSMB, byte_count);
  2514. pSMB->ByteCount = cpu_to_le16(byte_count);
  2515. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2516. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2517. cifs_stats_inc(&tcon->stats.cifs_stats.num_symlinks);
  2518. if (rc)
  2519. cifs_dbg(FYI, "Send error in SetPathInfo create symlink = %d\n",
  2520. rc);
  2521. cifs_buf_release(pSMB);
  2522. if (rc == -EAGAIN)
  2523. goto createSymLinkRetry;
  2524. return rc;
  2525. }
  2526. int
  2527. CIFSUnixCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
  2528. const char *fromName, const char *toName,
  2529. const struct nls_table *nls_codepage, int remap)
  2530. {
  2531. TRANSACTION2_SPI_REQ *pSMB = NULL;
  2532. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  2533. char *data_offset;
  2534. int name_len;
  2535. int name_len_target;
  2536. int rc = 0;
  2537. int bytes_returned = 0;
  2538. __u16 params, param_offset, offset, byte_count;
  2539. cifs_dbg(FYI, "In Create Hard link Unix style\n");
  2540. createHardLinkRetry:
  2541. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  2542. (void **) &pSMBr);
  2543. if (rc)
  2544. return rc;
  2545. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2546. name_len = cifsConvertToUTF16((__le16 *) pSMB->FileName, toName,
  2547. PATH_MAX, nls_codepage, remap);
  2548. name_len++; /* trailing null */
  2549. name_len *= 2;
  2550. } else { /* BB improve the check for buffer overruns BB */
  2551. name_len = strnlen(toName, PATH_MAX);
  2552. name_len++; /* trailing null */
  2553. strncpy(pSMB->FileName, toName, name_len);
  2554. }
  2555. params = 6 + name_len;
  2556. pSMB->MaxSetupCount = 0;
  2557. pSMB->Reserved = 0;
  2558. pSMB->Flags = 0;
  2559. pSMB->Timeout = 0;
  2560. pSMB->Reserved2 = 0;
  2561. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  2562. InformationLevel) - 4;
  2563. offset = param_offset + params;
  2564. data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
  2565. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2566. name_len_target =
  2567. cifsConvertToUTF16((__le16 *) data_offset, fromName,
  2568. PATH_MAX, nls_codepage, remap);
  2569. name_len_target++; /* trailing null */
  2570. name_len_target *= 2;
  2571. } else { /* BB improve the check for buffer overruns BB */
  2572. name_len_target = strnlen(fromName, PATH_MAX);
  2573. name_len_target++; /* trailing null */
  2574. strncpy(data_offset, fromName, name_len_target);
  2575. }
  2576. pSMB->MaxParameterCount = cpu_to_le16(2);
  2577. /* BB find exact max on data count below from sess*/
  2578. pSMB->MaxDataCount = cpu_to_le16(1000);
  2579. pSMB->SetupCount = 1;
  2580. pSMB->Reserved3 = 0;
  2581. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  2582. byte_count = 3 /* pad */ + params + name_len_target;
  2583. pSMB->ParameterCount = cpu_to_le16(params);
  2584. pSMB->TotalParameterCount = pSMB->ParameterCount;
  2585. pSMB->DataCount = cpu_to_le16(name_len_target);
  2586. pSMB->TotalDataCount = pSMB->DataCount;
  2587. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  2588. pSMB->DataOffset = cpu_to_le16(offset);
  2589. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_HLINK);
  2590. pSMB->Reserved4 = 0;
  2591. inc_rfc1001_len(pSMB, byte_count);
  2592. pSMB->ByteCount = cpu_to_le16(byte_count);
  2593. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2594. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2595. cifs_stats_inc(&tcon->stats.cifs_stats.num_hardlinks);
  2596. if (rc)
  2597. cifs_dbg(FYI, "Send error in SetPathInfo (hard link) = %d\n",
  2598. rc);
  2599. cifs_buf_release(pSMB);
  2600. if (rc == -EAGAIN)
  2601. goto createHardLinkRetry;
  2602. return rc;
  2603. }
  2604. int
  2605. CIFSCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
  2606. const char *from_name, const char *to_name,
  2607. struct cifs_sb_info *cifs_sb)
  2608. {
  2609. int rc = 0;
  2610. NT_RENAME_REQ *pSMB = NULL;
  2611. RENAME_RSP *pSMBr = NULL;
  2612. int bytes_returned;
  2613. int name_len, name_len2;
  2614. __u16 count;
  2615. int remap = cifs_remap(cifs_sb);
  2616. cifs_dbg(FYI, "In CIFSCreateHardLink\n");
  2617. winCreateHardLinkRetry:
  2618. rc = smb_init(SMB_COM_NT_RENAME, 4, tcon, (void **) &pSMB,
  2619. (void **) &pSMBr);
  2620. if (rc)
  2621. return rc;
  2622. pSMB->SearchAttributes =
  2623. cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
  2624. ATTR_DIRECTORY);
  2625. pSMB->Flags = cpu_to_le16(CREATE_HARD_LINK);
  2626. pSMB->ClusterCount = 0;
  2627. pSMB->BufferFormat = 0x04;
  2628. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2629. name_len =
  2630. cifsConvertToUTF16((__le16 *) pSMB->OldFileName, from_name,
  2631. PATH_MAX, cifs_sb->local_nls, remap);
  2632. name_len++; /* trailing null */
  2633. name_len *= 2;
  2634. /* protocol specifies ASCII buffer format (0x04) for unicode */
  2635. pSMB->OldFileName[name_len] = 0x04;
  2636. pSMB->OldFileName[name_len + 1] = 0x00; /* pad */
  2637. name_len2 =
  2638. cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
  2639. to_name, PATH_MAX, cifs_sb->local_nls,
  2640. remap);
  2641. name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
  2642. name_len2 *= 2; /* convert to bytes */
  2643. } else { /* BB improve the check for buffer overruns BB */
  2644. name_len = strnlen(from_name, PATH_MAX);
  2645. name_len++; /* trailing null */
  2646. strncpy(pSMB->OldFileName, from_name, name_len);
  2647. name_len2 = strnlen(to_name, PATH_MAX);
  2648. name_len2++; /* trailing null */
  2649. pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
  2650. strncpy(&pSMB->OldFileName[name_len + 1], to_name, name_len2);
  2651. name_len2++; /* trailing null */
  2652. name_len2++; /* signature byte */
  2653. }
  2654. count = 1 /* string type byte */ + name_len + name_len2;
  2655. inc_rfc1001_len(pSMB, count);
  2656. pSMB->ByteCount = cpu_to_le16(count);
  2657. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2658. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2659. cifs_stats_inc(&tcon->stats.cifs_stats.num_hardlinks);
  2660. if (rc)
  2661. cifs_dbg(FYI, "Send error in hard link (NT rename) = %d\n", rc);
  2662. cifs_buf_release(pSMB);
  2663. if (rc == -EAGAIN)
  2664. goto winCreateHardLinkRetry;
  2665. return rc;
  2666. }
  2667. int
  2668. CIFSSMBUnixQuerySymLink(const unsigned int xid, struct cifs_tcon *tcon,
  2669. const unsigned char *searchName, char **symlinkinfo,
  2670. const struct nls_table *nls_codepage, int remap)
  2671. {
  2672. /* SMB_QUERY_FILE_UNIX_LINK */
  2673. TRANSACTION2_QPI_REQ *pSMB = NULL;
  2674. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  2675. int rc = 0;
  2676. int bytes_returned;
  2677. int name_len;
  2678. __u16 params, byte_count;
  2679. char *data_start;
  2680. cifs_dbg(FYI, "In QPathSymLinkInfo (Unix) for path %s\n", searchName);
  2681. querySymLinkRetry:
  2682. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  2683. (void **) &pSMBr);
  2684. if (rc)
  2685. return rc;
  2686. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  2687. name_len =
  2688. cifsConvertToUTF16((__le16 *) pSMB->FileName,
  2689. searchName, PATH_MAX, nls_codepage,
  2690. remap);
  2691. name_len++; /* trailing null */
  2692. name_len *= 2;
  2693. } else { /* BB improve the check for buffer overruns BB */
  2694. name_len = strnlen(searchName, PATH_MAX);
  2695. name_len++; /* trailing null */
  2696. strncpy(pSMB->FileName, searchName, name_len);
  2697. }
  2698. params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
  2699. pSMB->TotalDataCount = 0;
  2700. pSMB->MaxParameterCount = cpu_to_le16(2);
  2701. pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
  2702. pSMB->MaxSetupCount = 0;
  2703. pSMB->Reserved = 0;
  2704. pSMB->Flags = 0;
  2705. pSMB->Timeout = 0;
  2706. pSMB->Reserved2 = 0;
  2707. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  2708. struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
  2709. pSMB->DataCount = 0;
  2710. pSMB->DataOffset = 0;
  2711. pSMB->SetupCount = 1;
  2712. pSMB->Reserved3 = 0;
  2713. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  2714. byte_count = params + 1 /* pad */ ;
  2715. pSMB->TotalParameterCount = cpu_to_le16(params);
  2716. pSMB->ParameterCount = pSMB->TotalParameterCount;
  2717. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_LINK);
  2718. pSMB->Reserved4 = 0;
  2719. inc_rfc1001_len(pSMB, byte_count);
  2720. pSMB->ByteCount = cpu_to_le16(byte_count);
  2721. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2722. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2723. if (rc) {
  2724. cifs_dbg(FYI, "Send error in QuerySymLinkInfo = %d\n", rc);
  2725. } else {
  2726. /* decode response */
  2727. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  2728. /* BB also check enough total bytes returned */
  2729. if (rc || get_bcc(&pSMBr->hdr) < 2)
  2730. rc = -EIO;
  2731. else {
  2732. bool is_unicode;
  2733. u16 count = le16_to_cpu(pSMBr->t2.DataCount);
  2734. data_start = ((char *) &pSMBr->hdr.Protocol) +
  2735. le16_to_cpu(pSMBr->t2.DataOffset);
  2736. if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
  2737. is_unicode = true;
  2738. else
  2739. is_unicode = false;
  2740. /* BB FIXME investigate remapping reserved chars here */
  2741. *symlinkinfo = cifs_strndup_from_utf16(data_start,
  2742. count, is_unicode, nls_codepage);
  2743. if (!*symlinkinfo)
  2744. rc = -ENOMEM;
  2745. }
  2746. }
  2747. cifs_buf_release(pSMB);
  2748. if (rc == -EAGAIN)
  2749. goto querySymLinkRetry;
  2750. return rc;
  2751. }
  2752. /*
  2753. * Recent Windows versions now create symlinks more frequently
  2754. * and they use the "reparse point" mechanism below. We can of course
  2755. * do symlinks nicely to Samba and other servers which support the
  2756. * CIFS Unix Extensions and we can also do SFU symlinks and "client only"
  2757. * "MF" symlinks optionally, but for recent Windows we really need to
  2758. * reenable the code below and fix the cifs_symlink callers to handle this.
  2759. * In the interim this code has been moved to its own config option so
  2760. * it is not compiled in by default until callers fixed up and more tested.
  2761. */
  2762. int
  2763. CIFSSMBQuerySymLink(const unsigned int xid, struct cifs_tcon *tcon,
  2764. __u16 fid, char **symlinkinfo,
  2765. const struct nls_table *nls_codepage)
  2766. {
  2767. int rc = 0;
  2768. int bytes_returned;
  2769. struct smb_com_transaction_ioctl_req *pSMB;
  2770. struct smb_com_transaction_ioctl_rsp *pSMBr;
  2771. bool is_unicode;
  2772. unsigned int sub_len;
  2773. char *sub_start;
  2774. struct reparse_symlink_data *reparse_buf;
  2775. struct reparse_posix_data *posix_buf;
  2776. __u32 data_offset, data_count;
  2777. char *end_of_smb;
  2778. cifs_dbg(FYI, "In Windows reparse style QueryLink for fid %u\n", fid);
  2779. rc = smb_init(SMB_COM_NT_TRANSACT, 23, tcon, (void **) &pSMB,
  2780. (void **) &pSMBr);
  2781. if (rc)
  2782. return rc;
  2783. pSMB->TotalParameterCount = 0 ;
  2784. pSMB->TotalDataCount = 0;
  2785. pSMB->MaxParameterCount = cpu_to_le32(2);
  2786. /* BB find exact data count max from sess structure BB */
  2787. pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
  2788. pSMB->MaxSetupCount = 4;
  2789. pSMB->Reserved = 0;
  2790. pSMB->ParameterOffset = 0;
  2791. pSMB->DataCount = 0;
  2792. pSMB->DataOffset = 0;
  2793. pSMB->SetupCount = 4;
  2794. pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_IOCTL);
  2795. pSMB->ParameterCount = pSMB->TotalParameterCount;
  2796. pSMB->FunctionCode = cpu_to_le32(FSCTL_GET_REPARSE_POINT);
  2797. pSMB->IsFsctl = 1; /* FSCTL */
  2798. pSMB->IsRootFlag = 0;
  2799. pSMB->Fid = fid; /* file handle always le */
  2800. pSMB->ByteCount = 0;
  2801. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2802. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2803. if (rc) {
  2804. cifs_dbg(FYI, "Send error in QueryReparseLinkInfo = %d\n", rc);
  2805. goto qreparse_out;
  2806. }
  2807. data_offset = le32_to_cpu(pSMBr->DataOffset);
  2808. data_count = le32_to_cpu(pSMBr->DataCount);
  2809. if (get_bcc(&pSMBr->hdr) < 2 || data_offset > 512) {
  2810. /* BB also check enough total bytes returned */
  2811. rc = -EIO; /* bad smb */
  2812. goto qreparse_out;
  2813. }
  2814. if (!data_count || (data_count > 2048)) {
  2815. rc = -EIO;
  2816. cifs_dbg(FYI, "Invalid return data count on get reparse info ioctl\n");
  2817. goto qreparse_out;
  2818. }
  2819. end_of_smb = 2 + get_bcc(&pSMBr->hdr) + (char *)&pSMBr->ByteCount;
  2820. reparse_buf = (struct reparse_symlink_data *)
  2821. ((char *)&pSMBr->hdr.Protocol + data_offset);
  2822. if ((char *)reparse_buf >= end_of_smb) {
  2823. rc = -EIO;
  2824. goto qreparse_out;
  2825. }
  2826. if (reparse_buf->ReparseTag == cpu_to_le32(IO_REPARSE_TAG_NFS)) {
  2827. cifs_dbg(FYI, "NFS style reparse tag\n");
  2828. posix_buf = (struct reparse_posix_data *)reparse_buf;
  2829. if (posix_buf->InodeType != cpu_to_le64(NFS_SPECFILE_LNK)) {
  2830. cifs_dbg(FYI, "unsupported file type 0x%llx\n",
  2831. le64_to_cpu(posix_buf->InodeType));
  2832. rc = -EOPNOTSUPP;
  2833. goto qreparse_out;
  2834. }
  2835. is_unicode = true;
  2836. sub_len = le16_to_cpu(reparse_buf->ReparseDataLength);
  2837. if (posix_buf->PathBuffer + sub_len > end_of_smb) {
  2838. cifs_dbg(FYI, "reparse buf beyond SMB\n");
  2839. rc = -EIO;
  2840. goto qreparse_out;
  2841. }
  2842. *symlinkinfo = cifs_strndup_from_utf16(posix_buf->PathBuffer,
  2843. sub_len, is_unicode, nls_codepage);
  2844. goto qreparse_out;
  2845. } else if (reparse_buf->ReparseTag !=
  2846. cpu_to_le32(IO_REPARSE_TAG_SYMLINK)) {
  2847. rc = -EOPNOTSUPP;
  2848. goto qreparse_out;
  2849. }
  2850. /* Reparse tag is NTFS symlink */
  2851. sub_start = le16_to_cpu(reparse_buf->SubstituteNameOffset) +
  2852. reparse_buf->PathBuffer;
  2853. sub_len = le16_to_cpu(reparse_buf->SubstituteNameLength);
  2854. if (sub_start + sub_len > end_of_smb) {
  2855. cifs_dbg(FYI, "reparse buf beyond SMB\n");
  2856. rc = -EIO;
  2857. goto qreparse_out;
  2858. }
  2859. if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
  2860. is_unicode = true;
  2861. else
  2862. is_unicode = false;
  2863. /* BB FIXME investigate remapping reserved chars here */
  2864. *symlinkinfo = cifs_strndup_from_utf16(sub_start, sub_len, is_unicode,
  2865. nls_codepage);
  2866. if (!*symlinkinfo)
  2867. rc = -ENOMEM;
  2868. qreparse_out:
  2869. cifs_buf_release(pSMB);
  2870. /*
  2871. * Note: On -EAGAIN error only caller can retry on handle based calls
  2872. * since file handle passed in no longer valid.
  2873. */
  2874. return rc;
  2875. }
  2876. int
  2877. CIFSSMB_set_compression(const unsigned int xid, struct cifs_tcon *tcon,
  2878. __u16 fid)
  2879. {
  2880. int rc = 0;
  2881. int bytes_returned;
  2882. struct smb_com_transaction_compr_ioctl_req *pSMB;
  2883. struct smb_com_transaction_ioctl_rsp *pSMBr;
  2884. cifs_dbg(FYI, "Set compression for %u\n", fid);
  2885. rc = smb_init(SMB_COM_NT_TRANSACT, 23, tcon, (void **) &pSMB,
  2886. (void **) &pSMBr);
  2887. if (rc)
  2888. return rc;
  2889. pSMB->compression_state = cpu_to_le16(COMPRESSION_FORMAT_DEFAULT);
  2890. pSMB->TotalParameterCount = 0;
  2891. pSMB->TotalDataCount = cpu_to_le32(2);
  2892. pSMB->MaxParameterCount = 0;
  2893. pSMB->MaxDataCount = 0;
  2894. pSMB->MaxSetupCount = 4;
  2895. pSMB->Reserved = 0;
  2896. pSMB->ParameterOffset = 0;
  2897. pSMB->DataCount = cpu_to_le32(2);
  2898. pSMB->DataOffset =
  2899. cpu_to_le32(offsetof(struct smb_com_transaction_compr_ioctl_req,
  2900. compression_state) - 4); /* 84 */
  2901. pSMB->SetupCount = 4;
  2902. pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_IOCTL);
  2903. pSMB->ParameterCount = 0;
  2904. pSMB->FunctionCode = cpu_to_le32(FSCTL_SET_COMPRESSION);
  2905. pSMB->IsFsctl = 1; /* FSCTL */
  2906. pSMB->IsRootFlag = 0;
  2907. pSMB->Fid = fid; /* file handle always le */
  2908. /* 3 byte pad, followed by 2 byte compress state */
  2909. pSMB->ByteCount = cpu_to_le16(5);
  2910. inc_rfc1001_len(pSMB, 5);
  2911. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  2912. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  2913. if (rc)
  2914. cifs_dbg(FYI, "Send error in SetCompression = %d\n", rc);
  2915. cifs_buf_release(pSMB);
  2916. /*
  2917. * Note: On -EAGAIN error only caller can retry on handle based calls
  2918. * since file handle passed in no longer valid.
  2919. */
  2920. return rc;
  2921. }
  2922. #ifdef CONFIG_CIFS_POSIX
  2923. /*Convert an Access Control Entry from wire format to local POSIX xattr format*/
  2924. static void cifs_convert_ace(struct posix_acl_xattr_entry *ace,
  2925. struct cifs_posix_ace *cifs_ace)
  2926. {
  2927. /* u8 cifs fields do not need le conversion */
  2928. ace->e_perm = cpu_to_le16(cifs_ace->cifs_e_perm);
  2929. ace->e_tag = cpu_to_le16(cifs_ace->cifs_e_tag);
  2930. ace->e_id = cpu_to_le32(le64_to_cpu(cifs_ace->cifs_uid));
  2931. /*
  2932. cifs_dbg(FYI, "perm %d tag %d id %d\n",
  2933. ace->e_perm, ace->e_tag, ace->e_id);
  2934. */
  2935. return;
  2936. }
  2937. /* Convert ACL from CIFS POSIX wire format to local Linux POSIX ACL xattr */
  2938. static int cifs_copy_posix_acl(char *trgt, char *src, const int buflen,
  2939. const int acl_type, const int size_of_data_area)
  2940. {
  2941. int size = 0;
  2942. int i;
  2943. __u16 count;
  2944. struct cifs_posix_ace *pACE;
  2945. struct cifs_posix_acl *cifs_acl = (struct cifs_posix_acl *)src;
  2946. struct posix_acl_xattr_header *local_acl = (void *)trgt;
  2947. if (le16_to_cpu(cifs_acl->version) != CIFS_ACL_VERSION)
  2948. return -EOPNOTSUPP;
  2949. if (acl_type == ACL_TYPE_ACCESS) {
  2950. count = le16_to_cpu(cifs_acl->access_entry_count);
  2951. pACE = &cifs_acl->ace_array[0];
  2952. size = sizeof(struct cifs_posix_acl);
  2953. size += sizeof(struct cifs_posix_ace) * count;
  2954. /* check if we would go beyond end of SMB */
  2955. if (size_of_data_area < size) {
  2956. cifs_dbg(FYI, "bad CIFS POSIX ACL size %d vs. %d\n",
  2957. size_of_data_area, size);
  2958. return -EINVAL;
  2959. }
  2960. } else if (acl_type == ACL_TYPE_DEFAULT) {
  2961. count = le16_to_cpu(cifs_acl->access_entry_count);
  2962. size = sizeof(struct cifs_posix_acl);
  2963. size += sizeof(struct cifs_posix_ace) * count;
  2964. /* skip past access ACEs to get to default ACEs */
  2965. pACE = &cifs_acl->ace_array[count];
  2966. count = le16_to_cpu(cifs_acl->default_entry_count);
  2967. size += sizeof(struct cifs_posix_ace) * count;
  2968. /* check if we would go beyond end of SMB */
  2969. if (size_of_data_area < size)
  2970. return -EINVAL;
  2971. } else {
  2972. /* illegal type */
  2973. return -EINVAL;
  2974. }
  2975. size = posix_acl_xattr_size(count);
  2976. if ((buflen == 0) || (local_acl == NULL)) {
  2977. /* used to query ACL EA size */
  2978. } else if (size > buflen) {
  2979. return -ERANGE;
  2980. } else /* buffer big enough */ {
  2981. struct posix_acl_xattr_entry *ace = (void *)(local_acl + 1);
  2982. local_acl->a_version = cpu_to_le32(POSIX_ACL_XATTR_VERSION);
  2983. for (i = 0; i < count ; i++) {
  2984. cifs_convert_ace(&ace[i], pACE);
  2985. pACE++;
  2986. }
  2987. }
  2988. return size;
  2989. }
  2990. static __u16 convert_ace_to_cifs_ace(struct cifs_posix_ace *cifs_ace,
  2991. const struct posix_acl_xattr_entry *local_ace)
  2992. {
  2993. __u16 rc = 0; /* 0 = ACL converted ok */
  2994. cifs_ace->cifs_e_perm = le16_to_cpu(local_ace->e_perm);
  2995. cifs_ace->cifs_e_tag = le16_to_cpu(local_ace->e_tag);
  2996. /* BB is there a better way to handle the large uid? */
  2997. if (local_ace->e_id == cpu_to_le32(-1)) {
  2998. /* Probably no need to le convert -1 on any arch but can not hurt */
  2999. cifs_ace->cifs_uid = cpu_to_le64(-1);
  3000. } else
  3001. cifs_ace->cifs_uid = cpu_to_le64(le32_to_cpu(local_ace->e_id));
  3002. /*
  3003. cifs_dbg(FYI, "perm %d tag %d id %d\n",
  3004. ace->e_perm, ace->e_tag, ace->e_id);
  3005. */
  3006. return rc;
  3007. }
  3008. /* Convert ACL from local Linux POSIX xattr to CIFS POSIX ACL wire format */
  3009. static __u16 ACL_to_cifs_posix(char *parm_data, const char *pACL,
  3010. const int buflen, const int acl_type)
  3011. {
  3012. __u16 rc = 0;
  3013. struct cifs_posix_acl *cifs_acl = (struct cifs_posix_acl *)parm_data;
  3014. struct posix_acl_xattr_header *local_acl = (void *)pACL;
  3015. struct posix_acl_xattr_entry *ace = (void *)(local_acl + 1);
  3016. int count;
  3017. int i;
  3018. if ((buflen == 0) || (pACL == NULL) || (cifs_acl == NULL))
  3019. return 0;
  3020. count = posix_acl_xattr_count((size_t)buflen);
  3021. cifs_dbg(FYI, "setting acl with %d entries from buf of length %d and version of %d\n",
  3022. count, buflen, le32_to_cpu(local_acl->a_version));
  3023. if (le32_to_cpu(local_acl->a_version) != 2) {
  3024. cifs_dbg(FYI, "unknown POSIX ACL version %d\n",
  3025. le32_to_cpu(local_acl->a_version));
  3026. return 0;
  3027. }
  3028. cifs_acl->version = cpu_to_le16(1);
  3029. if (acl_type == ACL_TYPE_ACCESS) {
  3030. cifs_acl->access_entry_count = cpu_to_le16(count);
  3031. cifs_acl->default_entry_count = cpu_to_le16(0xFFFF);
  3032. } else if (acl_type == ACL_TYPE_DEFAULT) {
  3033. cifs_acl->default_entry_count = cpu_to_le16(count);
  3034. cifs_acl->access_entry_count = cpu_to_le16(0xFFFF);
  3035. } else {
  3036. cifs_dbg(FYI, "unknown ACL type %d\n", acl_type);
  3037. return 0;
  3038. }
  3039. for (i = 0; i < count; i++) {
  3040. rc = convert_ace_to_cifs_ace(&cifs_acl->ace_array[i], &ace[i]);
  3041. if (rc != 0) {
  3042. /* ACE not converted */
  3043. break;
  3044. }
  3045. }
  3046. if (rc == 0) {
  3047. rc = (__u16)(count * sizeof(struct cifs_posix_ace));
  3048. rc += sizeof(struct cifs_posix_acl);
  3049. /* BB add check to make sure ACL does not overflow SMB */
  3050. }
  3051. return rc;
  3052. }
  3053. int
  3054. CIFSSMBGetPosixACL(const unsigned int xid, struct cifs_tcon *tcon,
  3055. const unsigned char *searchName,
  3056. char *acl_inf, const int buflen, const int acl_type,
  3057. const struct nls_table *nls_codepage, int remap)
  3058. {
  3059. /* SMB_QUERY_POSIX_ACL */
  3060. TRANSACTION2_QPI_REQ *pSMB = NULL;
  3061. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  3062. int rc = 0;
  3063. int bytes_returned;
  3064. int name_len;
  3065. __u16 params, byte_count;
  3066. cifs_dbg(FYI, "In GetPosixACL (Unix) for path %s\n", searchName);
  3067. queryAclRetry:
  3068. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3069. (void **) &pSMBr);
  3070. if (rc)
  3071. return rc;
  3072. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3073. name_len =
  3074. cifsConvertToUTF16((__le16 *) pSMB->FileName,
  3075. searchName, PATH_MAX, nls_codepage,
  3076. remap);
  3077. name_len++; /* trailing null */
  3078. name_len *= 2;
  3079. pSMB->FileName[name_len] = 0;
  3080. pSMB->FileName[name_len+1] = 0;
  3081. } else { /* BB improve the check for buffer overruns BB */
  3082. name_len = strnlen(searchName, PATH_MAX);
  3083. name_len++; /* trailing null */
  3084. strncpy(pSMB->FileName, searchName, name_len);
  3085. }
  3086. params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
  3087. pSMB->TotalDataCount = 0;
  3088. pSMB->MaxParameterCount = cpu_to_le16(2);
  3089. /* BB find exact max data count below from sess structure BB */
  3090. pSMB->MaxDataCount = cpu_to_le16(4000);
  3091. pSMB->MaxSetupCount = 0;
  3092. pSMB->Reserved = 0;
  3093. pSMB->Flags = 0;
  3094. pSMB->Timeout = 0;
  3095. pSMB->Reserved2 = 0;
  3096. pSMB->ParameterOffset = cpu_to_le16(
  3097. offsetof(struct smb_com_transaction2_qpi_req,
  3098. InformationLevel) - 4);
  3099. pSMB->DataCount = 0;
  3100. pSMB->DataOffset = 0;
  3101. pSMB->SetupCount = 1;
  3102. pSMB->Reserved3 = 0;
  3103. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  3104. byte_count = params + 1 /* pad */ ;
  3105. pSMB->TotalParameterCount = cpu_to_le16(params);
  3106. pSMB->ParameterCount = pSMB->TotalParameterCount;
  3107. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_POSIX_ACL);
  3108. pSMB->Reserved4 = 0;
  3109. inc_rfc1001_len(pSMB, byte_count);
  3110. pSMB->ByteCount = cpu_to_le16(byte_count);
  3111. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3112. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3113. cifs_stats_inc(&tcon->stats.cifs_stats.num_acl_get);
  3114. if (rc) {
  3115. cifs_dbg(FYI, "Send error in Query POSIX ACL = %d\n", rc);
  3116. } else {
  3117. /* decode response */
  3118. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3119. /* BB also check enough total bytes returned */
  3120. if (rc || get_bcc(&pSMBr->hdr) < 2)
  3121. rc = -EIO; /* bad smb */
  3122. else {
  3123. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3124. __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
  3125. rc = cifs_copy_posix_acl(acl_inf,
  3126. (char *)&pSMBr->hdr.Protocol+data_offset,
  3127. buflen, acl_type, count);
  3128. }
  3129. }
  3130. cifs_buf_release(pSMB);
  3131. if (rc == -EAGAIN)
  3132. goto queryAclRetry;
  3133. return rc;
  3134. }
  3135. int
  3136. CIFSSMBSetPosixACL(const unsigned int xid, struct cifs_tcon *tcon,
  3137. const unsigned char *fileName,
  3138. const char *local_acl, const int buflen,
  3139. const int acl_type,
  3140. const struct nls_table *nls_codepage, int remap)
  3141. {
  3142. struct smb_com_transaction2_spi_req *pSMB = NULL;
  3143. struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
  3144. char *parm_data;
  3145. int name_len;
  3146. int rc = 0;
  3147. int bytes_returned = 0;
  3148. __u16 params, byte_count, data_count, param_offset, offset;
  3149. cifs_dbg(FYI, "In SetPosixACL (Unix) for path %s\n", fileName);
  3150. setAclRetry:
  3151. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3152. (void **) &pSMBr);
  3153. if (rc)
  3154. return rc;
  3155. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3156. name_len =
  3157. cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
  3158. PATH_MAX, nls_codepage, remap);
  3159. name_len++; /* trailing null */
  3160. name_len *= 2;
  3161. } else { /* BB improve the check for buffer overruns BB */
  3162. name_len = strnlen(fileName, PATH_MAX);
  3163. name_len++; /* trailing null */
  3164. strncpy(pSMB->FileName, fileName, name_len);
  3165. }
  3166. params = 6 + name_len;
  3167. pSMB->MaxParameterCount = cpu_to_le16(2);
  3168. /* BB find max SMB size from sess */
  3169. pSMB->MaxDataCount = cpu_to_le16(1000);
  3170. pSMB->MaxSetupCount = 0;
  3171. pSMB->Reserved = 0;
  3172. pSMB->Flags = 0;
  3173. pSMB->Timeout = 0;
  3174. pSMB->Reserved2 = 0;
  3175. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  3176. InformationLevel) - 4;
  3177. offset = param_offset + params;
  3178. parm_data = ((char *) &pSMB->hdr.Protocol) + offset;
  3179. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  3180. /* convert to on the wire format for POSIX ACL */
  3181. data_count = ACL_to_cifs_posix(parm_data, local_acl, buflen, acl_type);
  3182. if (data_count == 0) {
  3183. rc = -EOPNOTSUPP;
  3184. goto setACLerrorExit;
  3185. }
  3186. pSMB->DataOffset = cpu_to_le16(offset);
  3187. pSMB->SetupCount = 1;
  3188. pSMB->Reserved3 = 0;
  3189. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  3190. pSMB->InformationLevel = cpu_to_le16(SMB_SET_POSIX_ACL);
  3191. byte_count = 3 /* pad */ + params + data_count;
  3192. pSMB->DataCount = cpu_to_le16(data_count);
  3193. pSMB->TotalDataCount = pSMB->DataCount;
  3194. pSMB->ParameterCount = cpu_to_le16(params);
  3195. pSMB->TotalParameterCount = pSMB->ParameterCount;
  3196. pSMB->Reserved4 = 0;
  3197. inc_rfc1001_len(pSMB, byte_count);
  3198. pSMB->ByteCount = cpu_to_le16(byte_count);
  3199. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3200. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3201. if (rc)
  3202. cifs_dbg(FYI, "Set POSIX ACL returned %d\n", rc);
  3203. setACLerrorExit:
  3204. cifs_buf_release(pSMB);
  3205. if (rc == -EAGAIN)
  3206. goto setAclRetry;
  3207. return rc;
  3208. }
  3209. /* BB fix tabs in this function FIXME BB */
  3210. int
  3211. CIFSGetExtAttr(const unsigned int xid, struct cifs_tcon *tcon,
  3212. const int netfid, __u64 *pExtAttrBits, __u64 *pMask)
  3213. {
  3214. int rc = 0;
  3215. struct smb_t2_qfi_req *pSMB = NULL;
  3216. struct smb_t2_qfi_rsp *pSMBr = NULL;
  3217. int bytes_returned;
  3218. __u16 params, byte_count;
  3219. cifs_dbg(FYI, "In GetExtAttr\n");
  3220. if (tcon == NULL)
  3221. return -ENODEV;
  3222. GetExtAttrRetry:
  3223. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3224. (void **) &pSMBr);
  3225. if (rc)
  3226. return rc;
  3227. params = 2 /* level */ + 2 /* fid */;
  3228. pSMB->t2.TotalDataCount = 0;
  3229. pSMB->t2.MaxParameterCount = cpu_to_le16(4);
  3230. /* BB find exact max data count below from sess structure BB */
  3231. pSMB->t2.MaxDataCount = cpu_to_le16(4000);
  3232. pSMB->t2.MaxSetupCount = 0;
  3233. pSMB->t2.Reserved = 0;
  3234. pSMB->t2.Flags = 0;
  3235. pSMB->t2.Timeout = 0;
  3236. pSMB->t2.Reserved2 = 0;
  3237. pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
  3238. Fid) - 4);
  3239. pSMB->t2.DataCount = 0;
  3240. pSMB->t2.DataOffset = 0;
  3241. pSMB->t2.SetupCount = 1;
  3242. pSMB->t2.Reserved3 = 0;
  3243. pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
  3244. byte_count = params + 1 /* pad */ ;
  3245. pSMB->t2.TotalParameterCount = cpu_to_le16(params);
  3246. pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
  3247. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_ATTR_FLAGS);
  3248. pSMB->Pad = 0;
  3249. pSMB->Fid = netfid;
  3250. inc_rfc1001_len(pSMB, byte_count);
  3251. pSMB->t2.ByteCount = cpu_to_le16(byte_count);
  3252. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3253. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3254. if (rc) {
  3255. cifs_dbg(FYI, "error %d in GetExtAttr\n", rc);
  3256. } else {
  3257. /* decode response */
  3258. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3259. /* BB also check enough total bytes returned */
  3260. if (rc || get_bcc(&pSMBr->hdr) < 2)
  3261. /* If rc should we check for EOPNOSUPP and
  3262. disable the srvino flag? or in caller? */
  3263. rc = -EIO; /* bad smb */
  3264. else {
  3265. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3266. __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
  3267. struct file_chattr_info *pfinfo;
  3268. /* BB Do we need a cast or hash here ? */
  3269. if (count != 16) {
  3270. cifs_dbg(FYI, "Illegal size ret in GetExtAttr\n");
  3271. rc = -EIO;
  3272. goto GetExtAttrOut;
  3273. }
  3274. pfinfo = (struct file_chattr_info *)
  3275. (data_offset + (char *) &pSMBr->hdr.Protocol);
  3276. *pExtAttrBits = le64_to_cpu(pfinfo->mode);
  3277. *pMask = le64_to_cpu(pfinfo->mask);
  3278. }
  3279. }
  3280. GetExtAttrOut:
  3281. cifs_buf_release(pSMB);
  3282. if (rc == -EAGAIN)
  3283. goto GetExtAttrRetry;
  3284. return rc;
  3285. }
  3286. #endif /* CONFIG_POSIX */
  3287. #ifdef CONFIG_CIFS_ACL
  3288. /*
  3289. * Initialize NT TRANSACT SMB into small smb request buffer. This assumes that
  3290. * all NT TRANSACTS that we init here have total parm and data under about 400
  3291. * bytes (to fit in small cifs buffer size), which is the case so far, it
  3292. * easily fits. NB: Setup words themselves and ByteCount MaxSetupCount (size of
  3293. * returned setup area) and MaxParameterCount (returned parms size) must be set
  3294. * by caller
  3295. */
  3296. static int
  3297. smb_init_nttransact(const __u16 sub_command, const int setup_count,
  3298. const int parm_len, struct cifs_tcon *tcon,
  3299. void **ret_buf)
  3300. {
  3301. int rc;
  3302. __u32 temp_offset;
  3303. struct smb_com_ntransact_req *pSMB;
  3304. rc = small_smb_init(SMB_COM_NT_TRANSACT, 19 + setup_count, tcon,
  3305. (void **)&pSMB);
  3306. if (rc)
  3307. return rc;
  3308. *ret_buf = (void *)pSMB;
  3309. pSMB->Reserved = 0;
  3310. pSMB->TotalParameterCount = cpu_to_le32(parm_len);
  3311. pSMB->TotalDataCount = 0;
  3312. pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
  3313. pSMB->ParameterCount = pSMB->TotalParameterCount;
  3314. pSMB->DataCount = pSMB->TotalDataCount;
  3315. temp_offset = offsetof(struct smb_com_ntransact_req, Parms) +
  3316. (setup_count * 2) - 4 /* for rfc1001 length itself */;
  3317. pSMB->ParameterOffset = cpu_to_le32(temp_offset);
  3318. pSMB->DataOffset = cpu_to_le32(temp_offset + parm_len);
  3319. pSMB->SetupCount = setup_count; /* no need to le convert byte fields */
  3320. pSMB->SubCommand = cpu_to_le16(sub_command);
  3321. return 0;
  3322. }
  3323. static int
  3324. validate_ntransact(char *buf, char **ppparm, char **ppdata,
  3325. __u32 *pparmlen, __u32 *pdatalen)
  3326. {
  3327. char *end_of_smb;
  3328. __u32 data_count, data_offset, parm_count, parm_offset;
  3329. struct smb_com_ntransact_rsp *pSMBr;
  3330. u16 bcc;
  3331. *pdatalen = 0;
  3332. *pparmlen = 0;
  3333. if (buf == NULL)
  3334. return -EINVAL;
  3335. pSMBr = (struct smb_com_ntransact_rsp *)buf;
  3336. bcc = get_bcc(&pSMBr->hdr);
  3337. end_of_smb = 2 /* sizeof byte count */ + bcc +
  3338. (char *)&pSMBr->ByteCount;
  3339. data_offset = le32_to_cpu(pSMBr->DataOffset);
  3340. data_count = le32_to_cpu(pSMBr->DataCount);
  3341. parm_offset = le32_to_cpu(pSMBr->ParameterOffset);
  3342. parm_count = le32_to_cpu(pSMBr->ParameterCount);
  3343. *ppparm = (char *)&pSMBr->hdr.Protocol + parm_offset;
  3344. *ppdata = (char *)&pSMBr->hdr.Protocol + data_offset;
  3345. /* should we also check that parm and data areas do not overlap? */
  3346. if (*ppparm > end_of_smb) {
  3347. cifs_dbg(FYI, "parms start after end of smb\n");
  3348. return -EINVAL;
  3349. } else if (parm_count + *ppparm > end_of_smb) {
  3350. cifs_dbg(FYI, "parm end after end of smb\n");
  3351. return -EINVAL;
  3352. } else if (*ppdata > end_of_smb) {
  3353. cifs_dbg(FYI, "data starts after end of smb\n");
  3354. return -EINVAL;
  3355. } else if (data_count + *ppdata > end_of_smb) {
  3356. cifs_dbg(FYI, "data %p + count %d (%p) past smb end %p start %p\n",
  3357. *ppdata, data_count, (data_count + *ppdata),
  3358. end_of_smb, pSMBr);
  3359. return -EINVAL;
  3360. } else if (parm_count + data_count > bcc) {
  3361. cifs_dbg(FYI, "parm count and data count larger than SMB\n");
  3362. return -EINVAL;
  3363. }
  3364. *pdatalen = data_count;
  3365. *pparmlen = parm_count;
  3366. return 0;
  3367. }
  3368. /* Get Security Descriptor (by handle) from remote server for a file or dir */
  3369. int
  3370. CIFSSMBGetCIFSACL(const unsigned int xid, struct cifs_tcon *tcon, __u16 fid,
  3371. struct cifs_ntsd **acl_inf, __u32 *pbuflen)
  3372. {
  3373. int rc = 0;
  3374. int buf_type = 0;
  3375. QUERY_SEC_DESC_REQ *pSMB;
  3376. struct kvec iov[1];
  3377. cifs_dbg(FYI, "GetCifsACL\n");
  3378. *pbuflen = 0;
  3379. *acl_inf = NULL;
  3380. rc = smb_init_nttransact(NT_TRANSACT_QUERY_SECURITY_DESC, 0,
  3381. 8 /* parm len */, tcon, (void **) &pSMB);
  3382. if (rc)
  3383. return rc;
  3384. pSMB->MaxParameterCount = cpu_to_le32(4);
  3385. /* BB TEST with big acls that might need to be e.g. larger than 16K */
  3386. pSMB->MaxSetupCount = 0;
  3387. pSMB->Fid = fid; /* file handle always le */
  3388. pSMB->AclFlags = cpu_to_le32(CIFS_ACL_OWNER | CIFS_ACL_GROUP |
  3389. CIFS_ACL_DACL);
  3390. pSMB->ByteCount = cpu_to_le16(11); /* 3 bytes pad + 8 bytes parm */
  3391. inc_rfc1001_len(pSMB, 11);
  3392. iov[0].iov_base = (char *)pSMB;
  3393. iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
  3394. rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovec */, &buf_type,
  3395. 0);
  3396. cifs_stats_inc(&tcon->stats.cifs_stats.num_acl_get);
  3397. if (rc) {
  3398. cifs_dbg(FYI, "Send error in QuerySecDesc = %d\n", rc);
  3399. } else { /* decode response */
  3400. __le32 *parm;
  3401. __u32 parm_len;
  3402. __u32 acl_len;
  3403. struct smb_com_ntransact_rsp *pSMBr;
  3404. char *pdata;
  3405. /* validate_nttransact */
  3406. rc = validate_ntransact(iov[0].iov_base, (char **)&parm,
  3407. &pdata, &parm_len, pbuflen);
  3408. if (rc)
  3409. goto qsec_out;
  3410. pSMBr = (struct smb_com_ntransact_rsp *)iov[0].iov_base;
  3411. cifs_dbg(FYI, "smb %p parm %p data %p\n",
  3412. pSMBr, parm, *acl_inf);
  3413. if (le32_to_cpu(pSMBr->ParameterCount) != 4) {
  3414. rc = -EIO; /* bad smb */
  3415. *pbuflen = 0;
  3416. goto qsec_out;
  3417. }
  3418. /* BB check that data area is minimum length and as big as acl_len */
  3419. acl_len = le32_to_cpu(*parm);
  3420. if (acl_len != *pbuflen) {
  3421. cifs_dbg(VFS, "acl length %d does not match %d\n",
  3422. acl_len, *pbuflen);
  3423. if (*pbuflen > acl_len)
  3424. *pbuflen = acl_len;
  3425. }
  3426. /* check if buffer is big enough for the acl
  3427. header followed by the smallest SID */
  3428. if ((*pbuflen < sizeof(struct cifs_ntsd) + 8) ||
  3429. (*pbuflen >= 64 * 1024)) {
  3430. cifs_dbg(VFS, "bad acl length %d\n", *pbuflen);
  3431. rc = -EINVAL;
  3432. *pbuflen = 0;
  3433. } else {
  3434. *acl_inf = kmemdup(pdata, *pbuflen, GFP_KERNEL);
  3435. if (*acl_inf == NULL) {
  3436. *pbuflen = 0;
  3437. rc = -ENOMEM;
  3438. }
  3439. }
  3440. }
  3441. qsec_out:
  3442. free_rsp_buf(buf_type, iov[0].iov_base);
  3443. /* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
  3444. return rc;
  3445. }
  3446. int
  3447. CIFSSMBSetCIFSACL(const unsigned int xid, struct cifs_tcon *tcon, __u16 fid,
  3448. struct cifs_ntsd *pntsd, __u32 acllen, int aclflag)
  3449. {
  3450. __u16 byte_count, param_count, data_count, param_offset, data_offset;
  3451. int rc = 0;
  3452. int bytes_returned = 0;
  3453. SET_SEC_DESC_REQ *pSMB = NULL;
  3454. void *pSMBr;
  3455. setCifsAclRetry:
  3456. rc = smb_init(SMB_COM_NT_TRANSACT, 19, tcon, (void **) &pSMB, &pSMBr);
  3457. if (rc)
  3458. return rc;
  3459. pSMB->MaxSetupCount = 0;
  3460. pSMB->Reserved = 0;
  3461. param_count = 8;
  3462. param_offset = offsetof(struct smb_com_transaction_ssec_req, Fid) - 4;
  3463. data_count = acllen;
  3464. data_offset = param_offset + param_count;
  3465. byte_count = 3 /* pad */ + param_count;
  3466. pSMB->DataCount = cpu_to_le32(data_count);
  3467. pSMB->TotalDataCount = pSMB->DataCount;
  3468. pSMB->MaxParameterCount = cpu_to_le32(4);
  3469. pSMB->MaxDataCount = cpu_to_le32(16384);
  3470. pSMB->ParameterCount = cpu_to_le32(param_count);
  3471. pSMB->ParameterOffset = cpu_to_le32(param_offset);
  3472. pSMB->TotalParameterCount = pSMB->ParameterCount;
  3473. pSMB->DataOffset = cpu_to_le32(data_offset);
  3474. pSMB->SetupCount = 0;
  3475. pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_SET_SECURITY_DESC);
  3476. pSMB->ByteCount = cpu_to_le16(byte_count+data_count);
  3477. pSMB->Fid = fid; /* file handle always le */
  3478. pSMB->Reserved2 = 0;
  3479. pSMB->AclFlags = cpu_to_le32(aclflag);
  3480. if (pntsd && acllen) {
  3481. memcpy((char *)pSMBr + offsetof(struct smb_hdr, Protocol) +
  3482. data_offset, pntsd, acllen);
  3483. inc_rfc1001_len(pSMB, byte_count + data_count);
  3484. } else
  3485. inc_rfc1001_len(pSMB, byte_count);
  3486. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3487. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3488. cifs_dbg(FYI, "SetCIFSACL bytes_returned: %d, rc: %d\n",
  3489. bytes_returned, rc);
  3490. if (rc)
  3491. cifs_dbg(FYI, "Set CIFS ACL returned %d\n", rc);
  3492. cifs_buf_release(pSMB);
  3493. if (rc == -EAGAIN)
  3494. goto setCifsAclRetry;
  3495. return (rc);
  3496. }
  3497. #endif /* CONFIG_CIFS_ACL */
  3498. /* Legacy Query Path Information call for lookup to old servers such
  3499. as Win9x/WinME */
  3500. int
  3501. SMBQueryInformation(const unsigned int xid, struct cifs_tcon *tcon,
  3502. const char *search_name, FILE_ALL_INFO *data,
  3503. const struct nls_table *nls_codepage, int remap)
  3504. {
  3505. QUERY_INFORMATION_REQ *pSMB;
  3506. QUERY_INFORMATION_RSP *pSMBr;
  3507. int rc = 0;
  3508. int bytes_returned;
  3509. int name_len;
  3510. cifs_dbg(FYI, "In SMBQPath path %s\n", search_name);
  3511. QInfRetry:
  3512. rc = smb_init(SMB_COM_QUERY_INFORMATION, 0, tcon, (void **) &pSMB,
  3513. (void **) &pSMBr);
  3514. if (rc)
  3515. return rc;
  3516. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3517. name_len =
  3518. cifsConvertToUTF16((__le16 *) pSMB->FileName,
  3519. search_name, PATH_MAX, nls_codepage,
  3520. remap);
  3521. name_len++; /* trailing null */
  3522. name_len *= 2;
  3523. } else {
  3524. name_len = strnlen(search_name, PATH_MAX);
  3525. name_len++; /* trailing null */
  3526. strncpy(pSMB->FileName, search_name, name_len);
  3527. }
  3528. pSMB->BufferFormat = 0x04;
  3529. name_len++; /* account for buffer type byte */
  3530. inc_rfc1001_len(pSMB, (__u16)name_len);
  3531. pSMB->ByteCount = cpu_to_le16(name_len);
  3532. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3533. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3534. if (rc) {
  3535. cifs_dbg(FYI, "Send error in QueryInfo = %d\n", rc);
  3536. } else if (data) {
  3537. struct timespec ts;
  3538. __u32 time = le32_to_cpu(pSMBr->last_write_time);
  3539. /* decode response */
  3540. /* BB FIXME - add time zone adjustment BB */
  3541. memset(data, 0, sizeof(FILE_ALL_INFO));
  3542. ts.tv_nsec = 0;
  3543. ts.tv_sec = time;
  3544. /* decode time fields */
  3545. data->ChangeTime = cpu_to_le64(cifs_UnixTimeToNT(ts));
  3546. data->LastWriteTime = data->ChangeTime;
  3547. data->LastAccessTime = 0;
  3548. data->AllocationSize =
  3549. cpu_to_le64(le32_to_cpu(pSMBr->size));
  3550. data->EndOfFile = data->AllocationSize;
  3551. data->Attributes =
  3552. cpu_to_le32(le16_to_cpu(pSMBr->attr));
  3553. } else
  3554. rc = -EIO; /* bad buffer passed in */
  3555. cifs_buf_release(pSMB);
  3556. if (rc == -EAGAIN)
  3557. goto QInfRetry;
  3558. return rc;
  3559. }
  3560. int
  3561. CIFSSMBQFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
  3562. u16 netfid, FILE_ALL_INFO *pFindData)
  3563. {
  3564. struct smb_t2_qfi_req *pSMB = NULL;
  3565. struct smb_t2_qfi_rsp *pSMBr = NULL;
  3566. int rc = 0;
  3567. int bytes_returned;
  3568. __u16 params, byte_count;
  3569. QFileInfoRetry:
  3570. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3571. (void **) &pSMBr);
  3572. if (rc)
  3573. return rc;
  3574. params = 2 /* level */ + 2 /* fid */;
  3575. pSMB->t2.TotalDataCount = 0;
  3576. pSMB->t2.MaxParameterCount = cpu_to_le16(4);
  3577. /* BB find exact max data count below from sess structure BB */
  3578. pSMB->t2.MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
  3579. pSMB->t2.MaxSetupCount = 0;
  3580. pSMB->t2.Reserved = 0;
  3581. pSMB->t2.Flags = 0;
  3582. pSMB->t2.Timeout = 0;
  3583. pSMB->t2.Reserved2 = 0;
  3584. pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
  3585. Fid) - 4);
  3586. pSMB->t2.DataCount = 0;
  3587. pSMB->t2.DataOffset = 0;
  3588. pSMB->t2.SetupCount = 1;
  3589. pSMB->t2.Reserved3 = 0;
  3590. pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
  3591. byte_count = params + 1 /* pad */ ;
  3592. pSMB->t2.TotalParameterCount = cpu_to_le16(params);
  3593. pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
  3594. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_ALL_INFO);
  3595. pSMB->Pad = 0;
  3596. pSMB->Fid = netfid;
  3597. inc_rfc1001_len(pSMB, byte_count);
  3598. pSMB->t2.ByteCount = cpu_to_le16(byte_count);
  3599. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3600. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3601. if (rc) {
  3602. cifs_dbg(FYI, "Send error in QFileInfo = %d", rc);
  3603. } else { /* decode response */
  3604. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3605. if (rc) /* BB add auto retry on EOPNOTSUPP? */
  3606. rc = -EIO;
  3607. else if (get_bcc(&pSMBr->hdr) < 40)
  3608. rc = -EIO; /* bad smb */
  3609. else if (pFindData) {
  3610. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3611. memcpy((char *) pFindData,
  3612. (char *) &pSMBr->hdr.Protocol +
  3613. data_offset, sizeof(FILE_ALL_INFO));
  3614. } else
  3615. rc = -ENOMEM;
  3616. }
  3617. cifs_buf_release(pSMB);
  3618. if (rc == -EAGAIN)
  3619. goto QFileInfoRetry;
  3620. return rc;
  3621. }
  3622. int
  3623. CIFSSMBQPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
  3624. const char *search_name, FILE_ALL_INFO *data,
  3625. int legacy /* old style infolevel */,
  3626. const struct nls_table *nls_codepage, int remap)
  3627. {
  3628. /* level 263 SMB_QUERY_FILE_ALL_INFO */
  3629. TRANSACTION2_QPI_REQ *pSMB = NULL;
  3630. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  3631. int rc = 0;
  3632. int bytes_returned;
  3633. int name_len;
  3634. __u16 params, byte_count;
  3635. /* cifs_dbg(FYI, "In QPathInfo path %s\n", search_name); */
  3636. QPathInfoRetry:
  3637. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3638. (void **) &pSMBr);
  3639. if (rc)
  3640. return rc;
  3641. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3642. name_len =
  3643. cifsConvertToUTF16((__le16 *) pSMB->FileName, search_name,
  3644. PATH_MAX, nls_codepage, remap);
  3645. name_len++; /* trailing null */
  3646. name_len *= 2;
  3647. } else { /* BB improve the check for buffer overruns BB */
  3648. name_len = strnlen(search_name, PATH_MAX);
  3649. name_len++; /* trailing null */
  3650. strncpy(pSMB->FileName, search_name, name_len);
  3651. }
  3652. params = 2 /* level */ + 4 /* reserved */ + name_len /* includes NUL */;
  3653. pSMB->TotalDataCount = 0;
  3654. pSMB->MaxParameterCount = cpu_to_le16(2);
  3655. /* BB find exact max SMB PDU from sess structure BB */
  3656. pSMB->MaxDataCount = cpu_to_le16(4000);
  3657. pSMB->MaxSetupCount = 0;
  3658. pSMB->Reserved = 0;
  3659. pSMB->Flags = 0;
  3660. pSMB->Timeout = 0;
  3661. pSMB->Reserved2 = 0;
  3662. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  3663. struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
  3664. pSMB->DataCount = 0;
  3665. pSMB->DataOffset = 0;
  3666. pSMB->SetupCount = 1;
  3667. pSMB->Reserved3 = 0;
  3668. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  3669. byte_count = params + 1 /* pad */ ;
  3670. pSMB->TotalParameterCount = cpu_to_le16(params);
  3671. pSMB->ParameterCount = pSMB->TotalParameterCount;
  3672. if (legacy)
  3673. pSMB->InformationLevel = cpu_to_le16(SMB_INFO_STANDARD);
  3674. else
  3675. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_ALL_INFO);
  3676. pSMB->Reserved4 = 0;
  3677. inc_rfc1001_len(pSMB, byte_count);
  3678. pSMB->ByteCount = cpu_to_le16(byte_count);
  3679. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3680. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3681. if (rc) {
  3682. cifs_dbg(FYI, "Send error in QPathInfo = %d\n", rc);
  3683. } else { /* decode response */
  3684. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3685. if (rc) /* BB add auto retry on EOPNOTSUPP? */
  3686. rc = -EIO;
  3687. else if (!legacy && get_bcc(&pSMBr->hdr) < 40)
  3688. rc = -EIO; /* bad smb */
  3689. else if (legacy && get_bcc(&pSMBr->hdr) < 24)
  3690. rc = -EIO; /* 24 or 26 expected but we do not read
  3691. last field */
  3692. else if (data) {
  3693. int size;
  3694. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3695. /*
  3696. * On legacy responses we do not read the last field,
  3697. * EAsize, fortunately since it varies by subdialect and
  3698. * also note it differs on Set vs Get, ie two bytes or 4
  3699. * bytes depending but we don't care here.
  3700. */
  3701. if (legacy)
  3702. size = sizeof(FILE_INFO_STANDARD);
  3703. else
  3704. size = sizeof(FILE_ALL_INFO);
  3705. memcpy((char *) data, (char *) &pSMBr->hdr.Protocol +
  3706. data_offset, size);
  3707. } else
  3708. rc = -ENOMEM;
  3709. }
  3710. cifs_buf_release(pSMB);
  3711. if (rc == -EAGAIN)
  3712. goto QPathInfoRetry;
  3713. return rc;
  3714. }
  3715. int
  3716. CIFSSMBUnixQFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
  3717. u16 netfid, FILE_UNIX_BASIC_INFO *pFindData)
  3718. {
  3719. struct smb_t2_qfi_req *pSMB = NULL;
  3720. struct smb_t2_qfi_rsp *pSMBr = NULL;
  3721. int rc = 0;
  3722. int bytes_returned;
  3723. __u16 params, byte_count;
  3724. UnixQFileInfoRetry:
  3725. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3726. (void **) &pSMBr);
  3727. if (rc)
  3728. return rc;
  3729. params = 2 /* level */ + 2 /* fid */;
  3730. pSMB->t2.TotalDataCount = 0;
  3731. pSMB->t2.MaxParameterCount = cpu_to_le16(4);
  3732. /* BB find exact max data count below from sess structure BB */
  3733. pSMB->t2.MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
  3734. pSMB->t2.MaxSetupCount = 0;
  3735. pSMB->t2.Reserved = 0;
  3736. pSMB->t2.Flags = 0;
  3737. pSMB->t2.Timeout = 0;
  3738. pSMB->t2.Reserved2 = 0;
  3739. pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
  3740. Fid) - 4);
  3741. pSMB->t2.DataCount = 0;
  3742. pSMB->t2.DataOffset = 0;
  3743. pSMB->t2.SetupCount = 1;
  3744. pSMB->t2.Reserved3 = 0;
  3745. pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
  3746. byte_count = params + 1 /* pad */ ;
  3747. pSMB->t2.TotalParameterCount = cpu_to_le16(params);
  3748. pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
  3749. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
  3750. pSMB->Pad = 0;
  3751. pSMB->Fid = netfid;
  3752. inc_rfc1001_len(pSMB, byte_count);
  3753. pSMB->t2.ByteCount = cpu_to_le16(byte_count);
  3754. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3755. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3756. if (rc) {
  3757. cifs_dbg(FYI, "Send error in UnixQFileInfo = %d", rc);
  3758. } else { /* decode response */
  3759. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3760. if (rc || get_bcc(&pSMBr->hdr) < sizeof(FILE_UNIX_BASIC_INFO)) {
  3761. cifs_dbg(VFS, "Malformed FILE_UNIX_BASIC_INFO response. Unix Extensions can be disabled on mount by specifying the nosfu mount option.\n");
  3762. rc = -EIO; /* bad smb */
  3763. } else {
  3764. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3765. memcpy((char *) pFindData,
  3766. (char *) &pSMBr->hdr.Protocol +
  3767. data_offset,
  3768. sizeof(FILE_UNIX_BASIC_INFO));
  3769. }
  3770. }
  3771. cifs_buf_release(pSMB);
  3772. if (rc == -EAGAIN)
  3773. goto UnixQFileInfoRetry;
  3774. return rc;
  3775. }
  3776. int
  3777. CIFSSMBUnixQPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
  3778. const unsigned char *searchName,
  3779. FILE_UNIX_BASIC_INFO *pFindData,
  3780. const struct nls_table *nls_codepage, int remap)
  3781. {
  3782. /* SMB_QUERY_FILE_UNIX_BASIC */
  3783. TRANSACTION2_QPI_REQ *pSMB = NULL;
  3784. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  3785. int rc = 0;
  3786. int bytes_returned = 0;
  3787. int name_len;
  3788. __u16 params, byte_count;
  3789. cifs_dbg(FYI, "In QPathInfo (Unix) the path %s\n", searchName);
  3790. UnixQPathInfoRetry:
  3791. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3792. (void **) &pSMBr);
  3793. if (rc)
  3794. return rc;
  3795. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3796. name_len =
  3797. cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
  3798. PATH_MAX, nls_codepage, remap);
  3799. name_len++; /* trailing null */
  3800. name_len *= 2;
  3801. } else { /* BB improve the check for buffer overruns BB */
  3802. name_len = strnlen(searchName, PATH_MAX);
  3803. name_len++; /* trailing null */
  3804. strncpy(pSMB->FileName, searchName, name_len);
  3805. }
  3806. params = 2 /* level */ + 4 /* reserved */ + name_len /* includes NUL */;
  3807. pSMB->TotalDataCount = 0;
  3808. pSMB->MaxParameterCount = cpu_to_le16(2);
  3809. /* BB find exact max SMB PDU from sess structure BB */
  3810. pSMB->MaxDataCount = cpu_to_le16(4000);
  3811. pSMB->MaxSetupCount = 0;
  3812. pSMB->Reserved = 0;
  3813. pSMB->Flags = 0;
  3814. pSMB->Timeout = 0;
  3815. pSMB->Reserved2 = 0;
  3816. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  3817. struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
  3818. pSMB->DataCount = 0;
  3819. pSMB->DataOffset = 0;
  3820. pSMB->SetupCount = 1;
  3821. pSMB->Reserved3 = 0;
  3822. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  3823. byte_count = params + 1 /* pad */ ;
  3824. pSMB->TotalParameterCount = cpu_to_le16(params);
  3825. pSMB->ParameterCount = pSMB->TotalParameterCount;
  3826. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
  3827. pSMB->Reserved4 = 0;
  3828. inc_rfc1001_len(pSMB, byte_count);
  3829. pSMB->ByteCount = cpu_to_le16(byte_count);
  3830. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3831. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3832. if (rc) {
  3833. cifs_dbg(FYI, "Send error in UnixQPathInfo = %d", rc);
  3834. } else { /* decode response */
  3835. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3836. if (rc || get_bcc(&pSMBr->hdr) < sizeof(FILE_UNIX_BASIC_INFO)) {
  3837. cifs_dbg(VFS, "Malformed FILE_UNIX_BASIC_INFO response. Unix Extensions can be disabled on mount by specifying the nosfu mount option.\n");
  3838. rc = -EIO; /* bad smb */
  3839. } else {
  3840. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  3841. memcpy((char *) pFindData,
  3842. (char *) &pSMBr->hdr.Protocol +
  3843. data_offset,
  3844. sizeof(FILE_UNIX_BASIC_INFO));
  3845. }
  3846. }
  3847. cifs_buf_release(pSMB);
  3848. if (rc == -EAGAIN)
  3849. goto UnixQPathInfoRetry;
  3850. return rc;
  3851. }
  3852. /* xid, tcon, searchName and codepage are input parms, rest are returned */
  3853. int
  3854. CIFSFindFirst(const unsigned int xid, struct cifs_tcon *tcon,
  3855. const char *searchName, struct cifs_sb_info *cifs_sb,
  3856. __u16 *pnetfid, __u16 search_flags,
  3857. struct cifs_search_info *psrch_inf, bool msearch)
  3858. {
  3859. /* level 257 SMB_ */
  3860. TRANSACTION2_FFIRST_REQ *pSMB = NULL;
  3861. TRANSACTION2_FFIRST_RSP *pSMBr = NULL;
  3862. T2_FFIRST_RSP_PARMS *parms;
  3863. int rc = 0;
  3864. int bytes_returned = 0;
  3865. int name_len, remap;
  3866. __u16 params, byte_count;
  3867. struct nls_table *nls_codepage;
  3868. cifs_dbg(FYI, "In FindFirst for %s\n", searchName);
  3869. findFirstRetry:
  3870. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  3871. (void **) &pSMBr);
  3872. if (rc)
  3873. return rc;
  3874. nls_codepage = cifs_sb->local_nls;
  3875. remap = cifs_remap(cifs_sb);
  3876. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  3877. name_len =
  3878. cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
  3879. PATH_MAX, nls_codepage, remap);
  3880. /* We can not add the asterik earlier in case
  3881. it got remapped to 0xF03A as if it were part of the
  3882. directory name instead of a wildcard */
  3883. name_len *= 2;
  3884. if (msearch) {
  3885. pSMB->FileName[name_len] = CIFS_DIR_SEP(cifs_sb);
  3886. pSMB->FileName[name_len+1] = 0;
  3887. pSMB->FileName[name_len+2] = '*';
  3888. pSMB->FileName[name_len+3] = 0;
  3889. name_len += 4; /* now the trailing null */
  3890. /* null terminate just in case */
  3891. pSMB->FileName[name_len] = 0;
  3892. pSMB->FileName[name_len+1] = 0;
  3893. name_len += 2;
  3894. }
  3895. } else { /* BB add check for overrun of SMB buf BB */
  3896. name_len = strnlen(searchName, PATH_MAX);
  3897. /* BB fix here and in unicode clause above ie
  3898. if (name_len > buffersize-header)
  3899. free buffer exit; BB */
  3900. strncpy(pSMB->FileName, searchName, name_len);
  3901. if (msearch) {
  3902. pSMB->FileName[name_len] = CIFS_DIR_SEP(cifs_sb);
  3903. pSMB->FileName[name_len+1] = '*';
  3904. pSMB->FileName[name_len+2] = 0;
  3905. name_len += 3;
  3906. }
  3907. }
  3908. params = 12 + name_len /* includes null */ ;
  3909. pSMB->TotalDataCount = 0; /* no EAs */
  3910. pSMB->MaxParameterCount = cpu_to_le16(10);
  3911. pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize & 0xFFFFFF00);
  3912. pSMB->MaxSetupCount = 0;
  3913. pSMB->Reserved = 0;
  3914. pSMB->Flags = 0;
  3915. pSMB->Timeout = 0;
  3916. pSMB->Reserved2 = 0;
  3917. byte_count = params + 1 /* pad */ ;
  3918. pSMB->TotalParameterCount = cpu_to_le16(params);
  3919. pSMB->ParameterCount = pSMB->TotalParameterCount;
  3920. pSMB->ParameterOffset = cpu_to_le16(
  3921. offsetof(struct smb_com_transaction2_ffirst_req, SearchAttributes)
  3922. - 4);
  3923. pSMB->DataCount = 0;
  3924. pSMB->DataOffset = 0;
  3925. pSMB->SetupCount = 1; /* one byte, no need to make endian neutral */
  3926. pSMB->Reserved3 = 0;
  3927. pSMB->SubCommand = cpu_to_le16(TRANS2_FIND_FIRST);
  3928. pSMB->SearchAttributes =
  3929. cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
  3930. ATTR_DIRECTORY);
  3931. pSMB->SearchCount = cpu_to_le16(CIFSMaxBufSize/sizeof(FILE_UNIX_INFO));
  3932. pSMB->SearchFlags = cpu_to_le16(search_flags);
  3933. pSMB->InformationLevel = cpu_to_le16(psrch_inf->info_level);
  3934. /* BB what should we set StorageType to? Does it matter? BB */
  3935. pSMB->SearchStorageType = 0;
  3936. inc_rfc1001_len(pSMB, byte_count);
  3937. pSMB->ByteCount = cpu_to_le16(byte_count);
  3938. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  3939. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  3940. cifs_stats_inc(&tcon->stats.cifs_stats.num_ffirst);
  3941. if (rc) {/* BB add logic to retry regular search if Unix search
  3942. rejected unexpectedly by server */
  3943. /* BB Add code to handle unsupported level rc */
  3944. cifs_dbg(FYI, "Error in FindFirst = %d\n", rc);
  3945. cifs_buf_release(pSMB);
  3946. /* BB eventually could optimize out free and realloc of buf */
  3947. /* for this case */
  3948. if (rc == -EAGAIN)
  3949. goto findFirstRetry;
  3950. } else { /* decode response */
  3951. /* BB remember to free buffer if error BB */
  3952. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  3953. if (rc == 0) {
  3954. unsigned int lnoff;
  3955. if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
  3956. psrch_inf->unicode = true;
  3957. else
  3958. psrch_inf->unicode = false;
  3959. psrch_inf->ntwrk_buf_start = (char *)pSMBr;
  3960. psrch_inf->smallBuf = 0;
  3961. psrch_inf->srch_entries_start =
  3962. (char *) &pSMBr->hdr.Protocol +
  3963. le16_to_cpu(pSMBr->t2.DataOffset);
  3964. parms = (T2_FFIRST_RSP_PARMS *)((char *) &pSMBr->hdr.Protocol +
  3965. le16_to_cpu(pSMBr->t2.ParameterOffset));
  3966. if (parms->EndofSearch)
  3967. psrch_inf->endOfSearch = true;
  3968. else
  3969. psrch_inf->endOfSearch = false;
  3970. psrch_inf->entries_in_buffer =
  3971. le16_to_cpu(parms->SearchCount);
  3972. psrch_inf->index_of_last_entry = 2 /* skip . and .. */ +
  3973. psrch_inf->entries_in_buffer;
  3974. lnoff = le16_to_cpu(parms->LastNameOffset);
  3975. if (CIFSMaxBufSize < lnoff) {
  3976. cifs_dbg(VFS, "ignoring corrupt resume name\n");
  3977. psrch_inf->last_entry = NULL;
  3978. return rc;
  3979. }
  3980. psrch_inf->last_entry = psrch_inf->srch_entries_start +
  3981. lnoff;
  3982. if (pnetfid)
  3983. *pnetfid = parms->SearchHandle;
  3984. } else {
  3985. cifs_buf_release(pSMB);
  3986. }
  3987. }
  3988. return rc;
  3989. }
  3990. int CIFSFindNext(const unsigned int xid, struct cifs_tcon *tcon,
  3991. __u16 searchHandle, __u16 search_flags,
  3992. struct cifs_search_info *psrch_inf)
  3993. {
  3994. TRANSACTION2_FNEXT_REQ *pSMB = NULL;
  3995. TRANSACTION2_FNEXT_RSP *pSMBr = NULL;
  3996. T2_FNEXT_RSP_PARMS *parms;
  3997. char *response_data;
  3998. int rc = 0;
  3999. int bytes_returned;
  4000. unsigned int name_len;
  4001. __u16 params, byte_count;
  4002. cifs_dbg(FYI, "In FindNext\n");
  4003. if (psrch_inf->endOfSearch)
  4004. return -ENOENT;
  4005. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4006. (void **) &pSMBr);
  4007. if (rc)
  4008. return rc;
  4009. params = 14; /* includes 2 bytes of null string, converted to LE below*/
  4010. byte_count = 0;
  4011. pSMB->TotalDataCount = 0; /* no EAs */
  4012. pSMB->MaxParameterCount = cpu_to_le16(8);
  4013. pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize & 0xFFFFFF00);
  4014. pSMB->MaxSetupCount = 0;
  4015. pSMB->Reserved = 0;
  4016. pSMB->Flags = 0;
  4017. pSMB->Timeout = 0;
  4018. pSMB->Reserved2 = 0;
  4019. pSMB->ParameterOffset = cpu_to_le16(
  4020. offsetof(struct smb_com_transaction2_fnext_req,SearchHandle) - 4);
  4021. pSMB->DataCount = 0;
  4022. pSMB->DataOffset = 0;
  4023. pSMB->SetupCount = 1;
  4024. pSMB->Reserved3 = 0;
  4025. pSMB->SubCommand = cpu_to_le16(TRANS2_FIND_NEXT);
  4026. pSMB->SearchHandle = searchHandle; /* always kept as le */
  4027. pSMB->SearchCount =
  4028. cpu_to_le16(CIFSMaxBufSize / sizeof(FILE_UNIX_INFO));
  4029. pSMB->InformationLevel = cpu_to_le16(psrch_inf->info_level);
  4030. pSMB->ResumeKey = psrch_inf->resume_key;
  4031. pSMB->SearchFlags = cpu_to_le16(search_flags);
  4032. name_len = psrch_inf->resume_name_len;
  4033. params += name_len;
  4034. if (name_len < PATH_MAX) {
  4035. memcpy(pSMB->ResumeFileName, psrch_inf->presume_name, name_len);
  4036. byte_count += name_len;
  4037. /* 14 byte parm len above enough for 2 byte null terminator */
  4038. pSMB->ResumeFileName[name_len] = 0;
  4039. pSMB->ResumeFileName[name_len+1] = 0;
  4040. } else {
  4041. rc = -EINVAL;
  4042. goto FNext2_err_exit;
  4043. }
  4044. byte_count = params + 1 /* pad */ ;
  4045. pSMB->TotalParameterCount = cpu_to_le16(params);
  4046. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4047. inc_rfc1001_len(pSMB, byte_count);
  4048. pSMB->ByteCount = cpu_to_le16(byte_count);
  4049. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4050. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4051. cifs_stats_inc(&tcon->stats.cifs_stats.num_fnext);
  4052. if (rc) {
  4053. if (rc == -EBADF) {
  4054. psrch_inf->endOfSearch = true;
  4055. cifs_buf_release(pSMB);
  4056. rc = 0; /* search probably was closed at end of search*/
  4057. } else
  4058. cifs_dbg(FYI, "FindNext returned = %d\n", rc);
  4059. } else { /* decode response */
  4060. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4061. if (rc == 0) {
  4062. unsigned int lnoff;
  4063. /* BB fixme add lock for file (srch_info) struct here */
  4064. if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
  4065. psrch_inf->unicode = true;
  4066. else
  4067. psrch_inf->unicode = false;
  4068. response_data = (char *) &pSMBr->hdr.Protocol +
  4069. le16_to_cpu(pSMBr->t2.ParameterOffset);
  4070. parms = (T2_FNEXT_RSP_PARMS *)response_data;
  4071. response_data = (char *)&pSMBr->hdr.Protocol +
  4072. le16_to_cpu(pSMBr->t2.DataOffset);
  4073. if (psrch_inf->smallBuf)
  4074. cifs_small_buf_release(
  4075. psrch_inf->ntwrk_buf_start);
  4076. else
  4077. cifs_buf_release(psrch_inf->ntwrk_buf_start);
  4078. psrch_inf->srch_entries_start = response_data;
  4079. psrch_inf->ntwrk_buf_start = (char *)pSMB;
  4080. psrch_inf->smallBuf = 0;
  4081. if (parms->EndofSearch)
  4082. psrch_inf->endOfSearch = true;
  4083. else
  4084. psrch_inf->endOfSearch = false;
  4085. psrch_inf->entries_in_buffer =
  4086. le16_to_cpu(parms->SearchCount);
  4087. psrch_inf->index_of_last_entry +=
  4088. psrch_inf->entries_in_buffer;
  4089. lnoff = le16_to_cpu(parms->LastNameOffset);
  4090. if (CIFSMaxBufSize < lnoff) {
  4091. cifs_dbg(VFS, "ignoring corrupt resume name\n");
  4092. psrch_inf->last_entry = NULL;
  4093. return rc;
  4094. } else
  4095. psrch_inf->last_entry =
  4096. psrch_inf->srch_entries_start + lnoff;
  4097. /* cifs_dbg(FYI, "fnxt2 entries in buf %d index_of_last %d\n",
  4098. psrch_inf->entries_in_buffer, psrch_inf->index_of_last_entry); */
  4099. /* BB fixme add unlock here */
  4100. }
  4101. }
  4102. /* BB On error, should we leave previous search buf (and count and
  4103. last entry fields) intact or free the previous one? */
  4104. /* Note: On -EAGAIN error only caller can retry on handle based calls
  4105. since file handle passed in no longer valid */
  4106. FNext2_err_exit:
  4107. if (rc != 0)
  4108. cifs_buf_release(pSMB);
  4109. return rc;
  4110. }
  4111. int
  4112. CIFSFindClose(const unsigned int xid, struct cifs_tcon *tcon,
  4113. const __u16 searchHandle)
  4114. {
  4115. int rc = 0;
  4116. FINDCLOSE_REQ *pSMB = NULL;
  4117. cifs_dbg(FYI, "In CIFSSMBFindClose\n");
  4118. rc = small_smb_init(SMB_COM_FIND_CLOSE2, 1, tcon, (void **)&pSMB);
  4119. /* no sense returning error if session restarted
  4120. as file handle has been closed */
  4121. if (rc == -EAGAIN)
  4122. return 0;
  4123. if (rc)
  4124. return rc;
  4125. pSMB->FileID = searchHandle;
  4126. pSMB->ByteCount = 0;
  4127. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  4128. if (rc)
  4129. cifs_dbg(VFS, "Send error in FindClose = %d\n", rc);
  4130. cifs_stats_inc(&tcon->stats.cifs_stats.num_fclose);
  4131. /* Since session is dead, search handle closed on server already */
  4132. if (rc == -EAGAIN)
  4133. rc = 0;
  4134. return rc;
  4135. }
  4136. int
  4137. CIFSGetSrvInodeNumber(const unsigned int xid, struct cifs_tcon *tcon,
  4138. const char *search_name, __u64 *inode_number,
  4139. const struct nls_table *nls_codepage, int remap)
  4140. {
  4141. int rc = 0;
  4142. TRANSACTION2_QPI_REQ *pSMB = NULL;
  4143. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  4144. int name_len, bytes_returned;
  4145. __u16 params, byte_count;
  4146. cifs_dbg(FYI, "In GetSrvInodeNum for %s\n", search_name);
  4147. if (tcon == NULL)
  4148. return -ENODEV;
  4149. GetInodeNumberRetry:
  4150. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4151. (void **) &pSMBr);
  4152. if (rc)
  4153. return rc;
  4154. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  4155. name_len =
  4156. cifsConvertToUTF16((__le16 *) pSMB->FileName,
  4157. search_name, PATH_MAX, nls_codepage,
  4158. remap);
  4159. name_len++; /* trailing null */
  4160. name_len *= 2;
  4161. } else { /* BB improve the check for buffer overruns BB */
  4162. name_len = strnlen(search_name, PATH_MAX);
  4163. name_len++; /* trailing null */
  4164. strncpy(pSMB->FileName, search_name, name_len);
  4165. }
  4166. params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
  4167. pSMB->TotalDataCount = 0;
  4168. pSMB->MaxParameterCount = cpu_to_le16(2);
  4169. /* BB find exact max data count below from sess structure BB */
  4170. pSMB->MaxDataCount = cpu_to_le16(4000);
  4171. pSMB->MaxSetupCount = 0;
  4172. pSMB->Reserved = 0;
  4173. pSMB->Flags = 0;
  4174. pSMB->Timeout = 0;
  4175. pSMB->Reserved2 = 0;
  4176. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4177. struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
  4178. pSMB->DataCount = 0;
  4179. pSMB->DataOffset = 0;
  4180. pSMB->SetupCount = 1;
  4181. pSMB->Reserved3 = 0;
  4182. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  4183. byte_count = params + 1 /* pad */ ;
  4184. pSMB->TotalParameterCount = cpu_to_le16(params);
  4185. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4186. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_INTERNAL_INFO);
  4187. pSMB->Reserved4 = 0;
  4188. inc_rfc1001_len(pSMB, byte_count);
  4189. pSMB->ByteCount = cpu_to_le16(byte_count);
  4190. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4191. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4192. if (rc) {
  4193. cifs_dbg(FYI, "error %d in QueryInternalInfo\n", rc);
  4194. } else {
  4195. /* decode response */
  4196. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4197. /* BB also check enough total bytes returned */
  4198. if (rc || get_bcc(&pSMBr->hdr) < 2)
  4199. /* If rc should we check for EOPNOSUPP and
  4200. disable the srvino flag? or in caller? */
  4201. rc = -EIO; /* bad smb */
  4202. else {
  4203. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4204. __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
  4205. struct file_internal_info *pfinfo;
  4206. /* BB Do we need a cast or hash here ? */
  4207. if (count < 8) {
  4208. cifs_dbg(FYI, "Illegal size ret in QryIntrnlInf\n");
  4209. rc = -EIO;
  4210. goto GetInodeNumOut;
  4211. }
  4212. pfinfo = (struct file_internal_info *)
  4213. (data_offset + (char *) &pSMBr->hdr.Protocol);
  4214. *inode_number = le64_to_cpu(pfinfo->UniqueId);
  4215. }
  4216. }
  4217. GetInodeNumOut:
  4218. cifs_buf_release(pSMB);
  4219. if (rc == -EAGAIN)
  4220. goto GetInodeNumberRetry;
  4221. return rc;
  4222. }
  4223. /* parses DFS refferal V3 structure
  4224. * caller is responsible for freeing target_nodes
  4225. * returns:
  4226. * on success - 0
  4227. * on failure - errno
  4228. */
  4229. static int
  4230. parse_DFS_referrals(TRANSACTION2_GET_DFS_REFER_RSP *pSMBr,
  4231. unsigned int *num_of_nodes,
  4232. struct dfs_info3_param **target_nodes,
  4233. const struct nls_table *nls_codepage, int remap,
  4234. const char *searchName)
  4235. {
  4236. int i, rc = 0;
  4237. char *data_end;
  4238. bool is_unicode;
  4239. struct dfs_referral_level_3 *ref;
  4240. if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
  4241. is_unicode = true;
  4242. else
  4243. is_unicode = false;
  4244. *num_of_nodes = le16_to_cpu(pSMBr->NumberOfReferrals);
  4245. if (*num_of_nodes < 1) {
  4246. cifs_dbg(VFS, "num_referrals: must be at least > 0, but we get num_referrals = %d\n",
  4247. *num_of_nodes);
  4248. rc = -EINVAL;
  4249. goto parse_DFS_referrals_exit;
  4250. }
  4251. ref = (struct dfs_referral_level_3 *) &(pSMBr->referrals);
  4252. if (ref->VersionNumber != cpu_to_le16(3)) {
  4253. cifs_dbg(VFS, "Referrals of V%d version are not supported, should be V3\n",
  4254. le16_to_cpu(ref->VersionNumber));
  4255. rc = -EINVAL;
  4256. goto parse_DFS_referrals_exit;
  4257. }
  4258. /* get the upper boundary of the resp buffer */
  4259. data_end = (char *)(&(pSMBr->PathConsumed)) +
  4260. le16_to_cpu(pSMBr->t2.DataCount);
  4261. cifs_dbg(FYI, "num_referrals: %d dfs flags: 0x%x ...\n",
  4262. *num_of_nodes, le32_to_cpu(pSMBr->DFSFlags));
  4263. *target_nodes = kcalloc(*num_of_nodes, sizeof(struct dfs_info3_param),
  4264. GFP_KERNEL);
  4265. if (*target_nodes == NULL) {
  4266. rc = -ENOMEM;
  4267. goto parse_DFS_referrals_exit;
  4268. }
  4269. /* collect necessary data from referrals */
  4270. for (i = 0; i < *num_of_nodes; i++) {
  4271. char *temp;
  4272. int max_len;
  4273. struct dfs_info3_param *node = (*target_nodes)+i;
  4274. node->flags = le32_to_cpu(pSMBr->DFSFlags);
  4275. if (is_unicode) {
  4276. __le16 *tmp = kmalloc(strlen(searchName)*2 + 2,
  4277. GFP_KERNEL);
  4278. if (tmp == NULL) {
  4279. rc = -ENOMEM;
  4280. goto parse_DFS_referrals_exit;
  4281. }
  4282. cifsConvertToUTF16((__le16 *) tmp, searchName,
  4283. PATH_MAX, nls_codepage, remap);
  4284. node->path_consumed = cifs_utf16_bytes(tmp,
  4285. le16_to_cpu(pSMBr->PathConsumed),
  4286. nls_codepage);
  4287. kfree(tmp);
  4288. } else
  4289. node->path_consumed = le16_to_cpu(pSMBr->PathConsumed);
  4290. node->server_type = le16_to_cpu(ref->ServerType);
  4291. node->ref_flag = le16_to_cpu(ref->ReferralEntryFlags);
  4292. /* copy DfsPath */
  4293. temp = (char *)ref + le16_to_cpu(ref->DfsPathOffset);
  4294. max_len = data_end - temp;
  4295. node->path_name = cifs_strndup_from_utf16(temp, max_len,
  4296. is_unicode, nls_codepage);
  4297. if (!node->path_name) {
  4298. rc = -ENOMEM;
  4299. goto parse_DFS_referrals_exit;
  4300. }
  4301. /* copy link target UNC */
  4302. temp = (char *)ref + le16_to_cpu(ref->NetworkAddressOffset);
  4303. max_len = data_end - temp;
  4304. node->node_name = cifs_strndup_from_utf16(temp, max_len,
  4305. is_unicode, nls_codepage);
  4306. if (!node->node_name) {
  4307. rc = -ENOMEM;
  4308. goto parse_DFS_referrals_exit;
  4309. }
  4310. ref++;
  4311. }
  4312. parse_DFS_referrals_exit:
  4313. if (rc) {
  4314. free_dfs_info_array(*target_nodes, *num_of_nodes);
  4315. *target_nodes = NULL;
  4316. *num_of_nodes = 0;
  4317. }
  4318. return rc;
  4319. }
  4320. int
  4321. CIFSGetDFSRefer(const unsigned int xid, struct cifs_ses *ses,
  4322. const char *search_name, struct dfs_info3_param **target_nodes,
  4323. unsigned int *num_of_nodes,
  4324. const struct nls_table *nls_codepage, int remap)
  4325. {
  4326. /* TRANS2_GET_DFS_REFERRAL */
  4327. TRANSACTION2_GET_DFS_REFER_REQ *pSMB = NULL;
  4328. TRANSACTION2_GET_DFS_REFER_RSP *pSMBr = NULL;
  4329. int rc = 0;
  4330. int bytes_returned;
  4331. int name_len;
  4332. __u16 params, byte_count;
  4333. *num_of_nodes = 0;
  4334. *target_nodes = NULL;
  4335. cifs_dbg(FYI, "In GetDFSRefer the path %s\n", search_name);
  4336. if (ses == NULL)
  4337. return -ENODEV;
  4338. getDFSRetry:
  4339. rc = smb_init(SMB_COM_TRANSACTION2, 15, NULL, (void **) &pSMB,
  4340. (void **) &pSMBr);
  4341. if (rc)
  4342. return rc;
  4343. /* server pointer checked in called function,
  4344. but should never be null here anyway */
  4345. pSMB->hdr.Mid = get_next_mid(ses->server);
  4346. pSMB->hdr.Tid = ses->ipc_tid;
  4347. pSMB->hdr.Uid = ses->Suid;
  4348. if (ses->capabilities & CAP_STATUS32)
  4349. pSMB->hdr.Flags2 |= SMBFLG2_ERR_STATUS;
  4350. if (ses->capabilities & CAP_DFS)
  4351. pSMB->hdr.Flags2 |= SMBFLG2_DFS;
  4352. if (ses->capabilities & CAP_UNICODE) {
  4353. pSMB->hdr.Flags2 |= SMBFLG2_UNICODE;
  4354. name_len =
  4355. cifsConvertToUTF16((__le16 *) pSMB->RequestFileName,
  4356. search_name, PATH_MAX, nls_codepage,
  4357. remap);
  4358. name_len++; /* trailing null */
  4359. name_len *= 2;
  4360. } else { /* BB improve the check for buffer overruns BB */
  4361. name_len = strnlen(search_name, PATH_MAX);
  4362. name_len++; /* trailing null */
  4363. strncpy(pSMB->RequestFileName, search_name, name_len);
  4364. }
  4365. if (ses->server->sign)
  4366. pSMB->hdr.Flags2 |= SMBFLG2_SECURITY_SIGNATURE;
  4367. pSMB->hdr.Uid = ses->Suid;
  4368. params = 2 /* level */ + name_len /*includes null */ ;
  4369. pSMB->TotalDataCount = 0;
  4370. pSMB->DataCount = 0;
  4371. pSMB->DataOffset = 0;
  4372. pSMB->MaxParameterCount = 0;
  4373. /* BB find exact max SMB PDU from sess structure BB */
  4374. pSMB->MaxDataCount = cpu_to_le16(4000);
  4375. pSMB->MaxSetupCount = 0;
  4376. pSMB->Reserved = 0;
  4377. pSMB->Flags = 0;
  4378. pSMB->Timeout = 0;
  4379. pSMB->Reserved2 = 0;
  4380. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4381. struct smb_com_transaction2_get_dfs_refer_req, MaxReferralLevel) - 4);
  4382. pSMB->SetupCount = 1;
  4383. pSMB->Reserved3 = 0;
  4384. pSMB->SubCommand = cpu_to_le16(TRANS2_GET_DFS_REFERRAL);
  4385. byte_count = params + 3 /* pad */ ;
  4386. pSMB->ParameterCount = cpu_to_le16(params);
  4387. pSMB->TotalParameterCount = pSMB->ParameterCount;
  4388. pSMB->MaxReferralLevel = cpu_to_le16(3);
  4389. inc_rfc1001_len(pSMB, byte_count);
  4390. pSMB->ByteCount = cpu_to_le16(byte_count);
  4391. rc = SendReceive(xid, ses, (struct smb_hdr *) pSMB,
  4392. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4393. if (rc) {
  4394. cifs_dbg(FYI, "Send error in GetDFSRefer = %d\n", rc);
  4395. goto GetDFSRefExit;
  4396. }
  4397. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4398. /* BB Also check if enough total bytes returned? */
  4399. if (rc || get_bcc(&pSMBr->hdr) < 17) {
  4400. rc = -EIO; /* bad smb */
  4401. goto GetDFSRefExit;
  4402. }
  4403. cifs_dbg(FYI, "Decoding GetDFSRefer response BCC: %d Offset %d\n",
  4404. get_bcc(&pSMBr->hdr), le16_to_cpu(pSMBr->t2.DataOffset));
  4405. /* parse returned result into more usable form */
  4406. rc = parse_DFS_referrals(pSMBr, num_of_nodes,
  4407. target_nodes, nls_codepage, remap,
  4408. search_name);
  4409. GetDFSRefExit:
  4410. cifs_buf_release(pSMB);
  4411. if (rc == -EAGAIN)
  4412. goto getDFSRetry;
  4413. return rc;
  4414. }
  4415. /* Query File System Info such as free space to old servers such as Win 9x */
  4416. int
  4417. SMBOldQFSInfo(const unsigned int xid, struct cifs_tcon *tcon,
  4418. struct kstatfs *FSData)
  4419. {
  4420. /* level 0x01 SMB_QUERY_FILE_SYSTEM_INFO */
  4421. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4422. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4423. FILE_SYSTEM_ALLOC_INFO *response_data;
  4424. int rc = 0;
  4425. int bytes_returned = 0;
  4426. __u16 params, byte_count;
  4427. cifs_dbg(FYI, "OldQFSInfo\n");
  4428. oldQFSInfoRetry:
  4429. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4430. (void **) &pSMBr);
  4431. if (rc)
  4432. return rc;
  4433. params = 2; /* level */
  4434. pSMB->TotalDataCount = 0;
  4435. pSMB->MaxParameterCount = cpu_to_le16(2);
  4436. pSMB->MaxDataCount = cpu_to_le16(1000);
  4437. pSMB->MaxSetupCount = 0;
  4438. pSMB->Reserved = 0;
  4439. pSMB->Flags = 0;
  4440. pSMB->Timeout = 0;
  4441. pSMB->Reserved2 = 0;
  4442. byte_count = params + 1 /* pad */ ;
  4443. pSMB->TotalParameterCount = cpu_to_le16(params);
  4444. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4445. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4446. struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4447. pSMB->DataCount = 0;
  4448. pSMB->DataOffset = 0;
  4449. pSMB->SetupCount = 1;
  4450. pSMB->Reserved3 = 0;
  4451. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4452. pSMB->InformationLevel = cpu_to_le16(SMB_INFO_ALLOCATION);
  4453. inc_rfc1001_len(pSMB, byte_count);
  4454. pSMB->ByteCount = cpu_to_le16(byte_count);
  4455. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4456. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4457. if (rc) {
  4458. cifs_dbg(FYI, "Send error in QFSInfo = %d\n", rc);
  4459. } else { /* decode response */
  4460. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4461. if (rc || get_bcc(&pSMBr->hdr) < 18)
  4462. rc = -EIO; /* bad smb */
  4463. else {
  4464. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4465. cifs_dbg(FYI, "qfsinf resp BCC: %d Offset %d\n",
  4466. get_bcc(&pSMBr->hdr), data_offset);
  4467. response_data = (FILE_SYSTEM_ALLOC_INFO *)
  4468. (((char *) &pSMBr->hdr.Protocol) + data_offset);
  4469. FSData->f_bsize =
  4470. le16_to_cpu(response_data->BytesPerSector) *
  4471. le32_to_cpu(response_data->
  4472. SectorsPerAllocationUnit);
  4473. FSData->f_blocks =
  4474. le32_to_cpu(response_data->TotalAllocationUnits);
  4475. FSData->f_bfree = FSData->f_bavail =
  4476. le32_to_cpu(response_data->FreeAllocationUnits);
  4477. cifs_dbg(FYI, "Blocks: %lld Free: %lld Block size %ld\n",
  4478. (unsigned long long)FSData->f_blocks,
  4479. (unsigned long long)FSData->f_bfree,
  4480. FSData->f_bsize);
  4481. }
  4482. }
  4483. cifs_buf_release(pSMB);
  4484. if (rc == -EAGAIN)
  4485. goto oldQFSInfoRetry;
  4486. return rc;
  4487. }
  4488. int
  4489. CIFSSMBQFSInfo(const unsigned int xid, struct cifs_tcon *tcon,
  4490. struct kstatfs *FSData)
  4491. {
  4492. /* level 0x103 SMB_QUERY_FILE_SYSTEM_INFO */
  4493. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4494. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4495. FILE_SYSTEM_INFO *response_data;
  4496. int rc = 0;
  4497. int bytes_returned = 0;
  4498. __u16 params, byte_count;
  4499. cifs_dbg(FYI, "In QFSInfo\n");
  4500. QFSInfoRetry:
  4501. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4502. (void **) &pSMBr);
  4503. if (rc)
  4504. return rc;
  4505. params = 2; /* level */
  4506. pSMB->TotalDataCount = 0;
  4507. pSMB->MaxParameterCount = cpu_to_le16(2);
  4508. pSMB->MaxDataCount = cpu_to_le16(1000);
  4509. pSMB->MaxSetupCount = 0;
  4510. pSMB->Reserved = 0;
  4511. pSMB->Flags = 0;
  4512. pSMB->Timeout = 0;
  4513. pSMB->Reserved2 = 0;
  4514. byte_count = params + 1 /* pad */ ;
  4515. pSMB->TotalParameterCount = cpu_to_le16(params);
  4516. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4517. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4518. struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4519. pSMB->DataCount = 0;
  4520. pSMB->DataOffset = 0;
  4521. pSMB->SetupCount = 1;
  4522. pSMB->Reserved3 = 0;
  4523. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4524. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_SIZE_INFO);
  4525. inc_rfc1001_len(pSMB, byte_count);
  4526. pSMB->ByteCount = cpu_to_le16(byte_count);
  4527. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4528. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4529. if (rc) {
  4530. cifs_dbg(FYI, "Send error in QFSInfo = %d\n", rc);
  4531. } else { /* decode response */
  4532. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4533. if (rc || get_bcc(&pSMBr->hdr) < 24)
  4534. rc = -EIO; /* bad smb */
  4535. else {
  4536. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4537. response_data =
  4538. (FILE_SYSTEM_INFO
  4539. *) (((char *) &pSMBr->hdr.Protocol) +
  4540. data_offset);
  4541. FSData->f_bsize =
  4542. le32_to_cpu(response_data->BytesPerSector) *
  4543. le32_to_cpu(response_data->
  4544. SectorsPerAllocationUnit);
  4545. FSData->f_blocks =
  4546. le64_to_cpu(response_data->TotalAllocationUnits);
  4547. FSData->f_bfree = FSData->f_bavail =
  4548. le64_to_cpu(response_data->FreeAllocationUnits);
  4549. cifs_dbg(FYI, "Blocks: %lld Free: %lld Block size %ld\n",
  4550. (unsigned long long)FSData->f_blocks,
  4551. (unsigned long long)FSData->f_bfree,
  4552. FSData->f_bsize);
  4553. }
  4554. }
  4555. cifs_buf_release(pSMB);
  4556. if (rc == -EAGAIN)
  4557. goto QFSInfoRetry;
  4558. return rc;
  4559. }
  4560. int
  4561. CIFSSMBQFSAttributeInfo(const unsigned int xid, struct cifs_tcon *tcon)
  4562. {
  4563. /* level 0x105 SMB_QUERY_FILE_SYSTEM_INFO */
  4564. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4565. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4566. FILE_SYSTEM_ATTRIBUTE_INFO *response_data;
  4567. int rc = 0;
  4568. int bytes_returned = 0;
  4569. __u16 params, byte_count;
  4570. cifs_dbg(FYI, "In QFSAttributeInfo\n");
  4571. QFSAttributeRetry:
  4572. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4573. (void **) &pSMBr);
  4574. if (rc)
  4575. return rc;
  4576. params = 2; /* level */
  4577. pSMB->TotalDataCount = 0;
  4578. pSMB->MaxParameterCount = cpu_to_le16(2);
  4579. /* BB find exact max SMB PDU from sess structure BB */
  4580. pSMB->MaxDataCount = cpu_to_le16(1000);
  4581. pSMB->MaxSetupCount = 0;
  4582. pSMB->Reserved = 0;
  4583. pSMB->Flags = 0;
  4584. pSMB->Timeout = 0;
  4585. pSMB->Reserved2 = 0;
  4586. byte_count = params + 1 /* pad */ ;
  4587. pSMB->TotalParameterCount = cpu_to_le16(params);
  4588. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4589. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4590. struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4591. pSMB->DataCount = 0;
  4592. pSMB->DataOffset = 0;
  4593. pSMB->SetupCount = 1;
  4594. pSMB->Reserved3 = 0;
  4595. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4596. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_ATTRIBUTE_INFO);
  4597. inc_rfc1001_len(pSMB, byte_count);
  4598. pSMB->ByteCount = cpu_to_le16(byte_count);
  4599. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4600. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4601. if (rc) {
  4602. cifs_dbg(VFS, "Send error in QFSAttributeInfo = %d\n", rc);
  4603. } else { /* decode response */
  4604. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4605. if (rc || get_bcc(&pSMBr->hdr) < 13) {
  4606. /* BB also check if enough bytes returned */
  4607. rc = -EIO; /* bad smb */
  4608. } else {
  4609. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4610. response_data =
  4611. (FILE_SYSTEM_ATTRIBUTE_INFO
  4612. *) (((char *) &pSMBr->hdr.Protocol) +
  4613. data_offset);
  4614. memcpy(&tcon->fsAttrInfo, response_data,
  4615. sizeof(FILE_SYSTEM_ATTRIBUTE_INFO));
  4616. }
  4617. }
  4618. cifs_buf_release(pSMB);
  4619. if (rc == -EAGAIN)
  4620. goto QFSAttributeRetry;
  4621. return rc;
  4622. }
  4623. int
  4624. CIFSSMBQFSDeviceInfo(const unsigned int xid, struct cifs_tcon *tcon)
  4625. {
  4626. /* level 0x104 SMB_QUERY_FILE_SYSTEM_INFO */
  4627. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4628. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4629. FILE_SYSTEM_DEVICE_INFO *response_data;
  4630. int rc = 0;
  4631. int bytes_returned = 0;
  4632. __u16 params, byte_count;
  4633. cifs_dbg(FYI, "In QFSDeviceInfo\n");
  4634. QFSDeviceRetry:
  4635. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4636. (void **) &pSMBr);
  4637. if (rc)
  4638. return rc;
  4639. params = 2; /* level */
  4640. pSMB->TotalDataCount = 0;
  4641. pSMB->MaxParameterCount = cpu_to_le16(2);
  4642. /* BB find exact max SMB PDU from sess structure BB */
  4643. pSMB->MaxDataCount = cpu_to_le16(1000);
  4644. pSMB->MaxSetupCount = 0;
  4645. pSMB->Reserved = 0;
  4646. pSMB->Flags = 0;
  4647. pSMB->Timeout = 0;
  4648. pSMB->Reserved2 = 0;
  4649. byte_count = params + 1 /* pad */ ;
  4650. pSMB->TotalParameterCount = cpu_to_le16(params);
  4651. pSMB->ParameterCount = pSMB->TotalParameterCount;
  4652. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  4653. struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4654. pSMB->DataCount = 0;
  4655. pSMB->DataOffset = 0;
  4656. pSMB->SetupCount = 1;
  4657. pSMB->Reserved3 = 0;
  4658. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4659. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_DEVICE_INFO);
  4660. inc_rfc1001_len(pSMB, byte_count);
  4661. pSMB->ByteCount = cpu_to_le16(byte_count);
  4662. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4663. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4664. if (rc) {
  4665. cifs_dbg(FYI, "Send error in QFSDeviceInfo = %d\n", rc);
  4666. } else { /* decode response */
  4667. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4668. if (rc || get_bcc(&pSMBr->hdr) <
  4669. sizeof(FILE_SYSTEM_DEVICE_INFO))
  4670. rc = -EIO; /* bad smb */
  4671. else {
  4672. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4673. response_data =
  4674. (FILE_SYSTEM_DEVICE_INFO *)
  4675. (((char *) &pSMBr->hdr.Protocol) +
  4676. data_offset);
  4677. memcpy(&tcon->fsDevInfo, response_data,
  4678. sizeof(FILE_SYSTEM_DEVICE_INFO));
  4679. }
  4680. }
  4681. cifs_buf_release(pSMB);
  4682. if (rc == -EAGAIN)
  4683. goto QFSDeviceRetry;
  4684. return rc;
  4685. }
  4686. int
  4687. CIFSSMBQFSUnixInfo(const unsigned int xid, struct cifs_tcon *tcon)
  4688. {
  4689. /* level 0x200 SMB_QUERY_CIFS_UNIX_INFO */
  4690. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4691. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4692. FILE_SYSTEM_UNIX_INFO *response_data;
  4693. int rc = 0;
  4694. int bytes_returned = 0;
  4695. __u16 params, byte_count;
  4696. cifs_dbg(FYI, "In QFSUnixInfo\n");
  4697. QFSUnixRetry:
  4698. rc = smb_init_no_reconnect(SMB_COM_TRANSACTION2, 15, tcon,
  4699. (void **) &pSMB, (void **) &pSMBr);
  4700. if (rc)
  4701. return rc;
  4702. params = 2; /* level */
  4703. pSMB->TotalDataCount = 0;
  4704. pSMB->DataCount = 0;
  4705. pSMB->DataOffset = 0;
  4706. pSMB->MaxParameterCount = cpu_to_le16(2);
  4707. /* BB find exact max SMB PDU from sess structure BB */
  4708. pSMB->MaxDataCount = cpu_to_le16(100);
  4709. pSMB->MaxSetupCount = 0;
  4710. pSMB->Reserved = 0;
  4711. pSMB->Flags = 0;
  4712. pSMB->Timeout = 0;
  4713. pSMB->Reserved2 = 0;
  4714. byte_count = params + 1 /* pad */ ;
  4715. pSMB->ParameterCount = cpu_to_le16(params);
  4716. pSMB->TotalParameterCount = pSMB->ParameterCount;
  4717. pSMB->ParameterOffset = cpu_to_le16(offsetof(struct
  4718. smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4719. pSMB->SetupCount = 1;
  4720. pSMB->Reserved3 = 0;
  4721. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4722. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_CIFS_UNIX_INFO);
  4723. inc_rfc1001_len(pSMB, byte_count);
  4724. pSMB->ByteCount = cpu_to_le16(byte_count);
  4725. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4726. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4727. if (rc) {
  4728. cifs_dbg(VFS, "Send error in QFSUnixInfo = %d\n", rc);
  4729. } else { /* decode response */
  4730. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4731. if (rc || get_bcc(&pSMBr->hdr) < 13) {
  4732. rc = -EIO; /* bad smb */
  4733. } else {
  4734. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4735. response_data =
  4736. (FILE_SYSTEM_UNIX_INFO
  4737. *) (((char *) &pSMBr->hdr.Protocol) +
  4738. data_offset);
  4739. memcpy(&tcon->fsUnixInfo, response_data,
  4740. sizeof(FILE_SYSTEM_UNIX_INFO));
  4741. }
  4742. }
  4743. cifs_buf_release(pSMB);
  4744. if (rc == -EAGAIN)
  4745. goto QFSUnixRetry;
  4746. return rc;
  4747. }
  4748. int
  4749. CIFSSMBSetFSUnixInfo(const unsigned int xid, struct cifs_tcon *tcon, __u64 cap)
  4750. {
  4751. /* level 0x200 SMB_SET_CIFS_UNIX_INFO */
  4752. TRANSACTION2_SETFSI_REQ *pSMB = NULL;
  4753. TRANSACTION2_SETFSI_RSP *pSMBr = NULL;
  4754. int rc = 0;
  4755. int bytes_returned = 0;
  4756. __u16 params, param_offset, offset, byte_count;
  4757. cifs_dbg(FYI, "In SETFSUnixInfo\n");
  4758. SETFSUnixRetry:
  4759. /* BB switch to small buf init to save memory */
  4760. rc = smb_init_no_reconnect(SMB_COM_TRANSACTION2, 15, tcon,
  4761. (void **) &pSMB, (void **) &pSMBr);
  4762. if (rc)
  4763. return rc;
  4764. params = 4; /* 2 bytes zero followed by info level. */
  4765. pSMB->MaxSetupCount = 0;
  4766. pSMB->Reserved = 0;
  4767. pSMB->Flags = 0;
  4768. pSMB->Timeout = 0;
  4769. pSMB->Reserved2 = 0;
  4770. param_offset = offsetof(struct smb_com_transaction2_setfsi_req, FileNum)
  4771. - 4;
  4772. offset = param_offset + params;
  4773. pSMB->MaxParameterCount = cpu_to_le16(4);
  4774. /* BB find exact max SMB PDU from sess structure BB */
  4775. pSMB->MaxDataCount = cpu_to_le16(100);
  4776. pSMB->SetupCount = 1;
  4777. pSMB->Reserved3 = 0;
  4778. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FS_INFORMATION);
  4779. byte_count = 1 /* pad */ + params + 12;
  4780. pSMB->DataCount = cpu_to_le16(12);
  4781. pSMB->ParameterCount = cpu_to_le16(params);
  4782. pSMB->TotalDataCount = pSMB->DataCount;
  4783. pSMB->TotalParameterCount = pSMB->ParameterCount;
  4784. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  4785. pSMB->DataOffset = cpu_to_le16(offset);
  4786. /* Params. */
  4787. pSMB->FileNum = 0;
  4788. pSMB->InformationLevel = cpu_to_le16(SMB_SET_CIFS_UNIX_INFO);
  4789. /* Data. */
  4790. pSMB->ClientUnixMajor = cpu_to_le16(CIFS_UNIX_MAJOR_VERSION);
  4791. pSMB->ClientUnixMinor = cpu_to_le16(CIFS_UNIX_MINOR_VERSION);
  4792. pSMB->ClientUnixCap = cpu_to_le64(cap);
  4793. inc_rfc1001_len(pSMB, byte_count);
  4794. pSMB->ByteCount = cpu_to_le16(byte_count);
  4795. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4796. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4797. if (rc) {
  4798. cifs_dbg(VFS, "Send error in SETFSUnixInfo = %d\n", rc);
  4799. } else { /* decode response */
  4800. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4801. if (rc)
  4802. rc = -EIO; /* bad smb */
  4803. }
  4804. cifs_buf_release(pSMB);
  4805. if (rc == -EAGAIN)
  4806. goto SETFSUnixRetry;
  4807. return rc;
  4808. }
  4809. int
  4810. CIFSSMBQFSPosixInfo(const unsigned int xid, struct cifs_tcon *tcon,
  4811. struct kstatfs *FSData)
  4812. {
  4813. /* level 0x201 SMB_QUERY_CIFS_POSIX_INFO */
  4814. TRANSACTION2_QFSI_REQ *pSMB = NULL;
  4815. TRANSACTION2_QFSI_RSP *pSMBr = NULL;
  4816. FILE_SYSTEM_POSIX_INFO *response_data;
  4817. int rc = 0;
  4818. int bytes_returned = 0;
  4819. __u16 params, byte_count;
  4820. cifs_dbg(FYI, "In QFSPosixInfo\n");
  4821. QFSPosixRetry:
  4822. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4823. (void **) &pSMBr);
  4824. if (rc)
  4825. return rc;
  4826. params = 2; /* level */
  4827. pSMB->TotalDataCount = 0;
  4828. pSMB->DataCount = 0;
  4829. pSMB->DataOffset = 0;
  4830. pSMB->MaxParameterCount = cpu_to_le16(2);
  4831. /* BB find exact max SMB PDU from sess structure BB */
  4832. pSMB->MaxDataCount = cpu_to_le16(100);
  4833. pSMB->MaxSetupCount = 0;
  4834. pSMB->Reserved = 0;
  4835. pSMB->Flags = 0;
  4836. pSMB->Timeout = 0;
  4837. pSMB->Reserved2 = 0;
  4838. byte_count = params + 1 /* pad */ ;
  4839. pSMB->ParameterCount = cpu_to_le16(params);
  4840. pSMB->TotalParameterCount = pSMB->ParameterCount;
  4841. pSMB->ParameterOffset = cpu_to_le16(offsetof(struct
  4842. smb_com_transaction2_qfsi_req, InformationLevel) - 4);
  4843. pSMB->SetupCount = 1;
  4844. pSMB->Reserved3 = 0;
  4845. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
  4846. pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_POSIX_FS_INFO);
  4847. inc_rfc1001_len(pSMB, byte_count);
  4848. pSMB->ByteCount = cpu_to_le16(byte_count);
  4849. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4850. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4851. if (rc) {
  4852. cifs_dbg(FYI, "Send error in QFSUnixInfo = %d\n", rc);
  4853. } else { /* decode response */
  4854. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  4855. if (rc || get_bcc(&pSMBr->hdr) < 13) {
  4856. rc = -EIO; /* bad smb */
  4857. } else {
  4858. __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  4859. response_data =
  4860. (FILE_SYSTEM_POSIX_INFO
  4861. *) (((char *) &pSMBr->hdr.Protocol) +
  4862. data_offset);
  4863. FSData->f_bsize =
  4864. le32_to_cpu(response_data->BlockSize);
  4865. FSData->f_blocks =
  4866. le64_to_cpu(response_data->TotalBlocks);
  4867. FSData->f_bfree =
  4868. le64_to_cpu(response_data->BlocksAvail);
  4869. if (response_data->UserBlocksAvail == cpu_to_le64(-1)) {
  4870. FSData->f_bavail = FSData->f_bfree;
  4871. } else {
  4872. FSData->f_bavail =
  4873. le64_to_cpu(response_data->UserBlocksAvail);
  4874. }
  4875. if (response_data->TotalFileNodes != cpu_to_le64(-1))
  4876. FSData->f_files =
  4877. le64_to_cpu(response_data->TotalFileNodes);
  4878. if (response_data->FreeFileNodes != cpu_to_le64(-1))
  4879. FSData->f_ffree =
  4880. le64_to_cpu(response_data->FreeFileNodes);
  4881. }
  4882. }
  4883. cifs_buf_release(pSMB);
  4884. if (rc == -EAGAIN)
  4885. goto QFSPosixRetry;
  4886. return rc;
  4887. }
  4888. /*
  4889. * We can not use write of zero bytes trick to set file size due to need for
  4890. * large file support. Also note that this SetPathInfo is preferred to
  4891. * SetFileInfo based method in next routine which is only needed to work around
  4892. * a sharing violation bugin Samba which this routine can run into.
  4893. */
  4894. int
  4895. CIFSSMBSetEOF(const unsigned int xid, struct cifs_tcon *tcon,
  4896. const char *file_name, __u64 size, struct cifs_sb_info *cifs_sb,
  4897. bool set_allocation)
  4898. {
  4899. struct smb_com_transaction2_spi_req *pSMB = NULL;
  4900. struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
  4901. struct file_end_of_file_info *parm_data;
  4902. int name_len;
  4903. int rc = 0;
  4904. int bytes_returned = 0;
  4905. int remap = cifs_remap(cifs_sb);
  4906. __u16 params, byte_count, data_count, param_offset, offset;
  4907. cifs_dbg(FYI, "In SetEOF\n");
  4908. SetEOFRetry:
  4909. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  4910. (void **) &pSMBr);
  4911. if (rc)
  4912. return rc;
  4913. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  4914. name_len =
  4915. cifsConvertToUTF16((__le16 *) pSMB->FileName, file_name,
  4916. PATH_MAX, cifs_sb->local_nls, remap);
  4917. name_len++; /* trailing null */
  4918. name_len *= 2;
  4919. } else { /* BB improve the check for buffer overruns BB */
  4920. name_len = strnlen(file_name, PATH_MAX);
  4921. name_len++; /* trailing null */
  4922. strncpy(pSMB->FileName, file_name, name_len);
  4923. }
  4924. params = 6 + name_len;
  4925. data_count = sizeof(struct file_end_of_file_info);
  4926. pSMB->MaxParameterCount = cpu_to_le16(2);
  4927. pSMB->MaxDataCount = cpu_to_le16(4100);
  4928. pSMB->MaxSetupCount = 0;
  4929. pSMB->Reserved = 0;
  4930. pSMB->Flags = 0;
  4931. pSMB->Timeout = 0;
  4932. pSMB->Reserved2 = 0;
  4933. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  4934. InformationLevel) - 4;
  4935. offset = param_offset + params;
  4936. if (set_allocation) {
  4937. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  4938. pSMB->InformationLevel =
  4939. cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO2);
  4940. else
  4941. pSMB->InformationLevel =
  4942. cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO);
  4943. } else /* Set File Size */ {
  4944. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  4945. pSMB->InformationLevel =
  4946. cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO2);
  4947. else
  4948. pSMB->InformationLevel =
  4949. cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO);
  4950. }
  4951. parm_data =
  4952. (struct file_end_of_file_info *) (((char *) &pSMB->hdr.Protocol) +
  4953. offset);
  4954. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  4955. pSMB->DataOffset = cpu_to_le16(offset);
  4956. pSMB->SetupCount = 1;
  4957. pSMB->Reserved3 = 0;
  4958. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  4959. byte_count = 3 /* pad */ + params + data_count;
  4960. pSMB->DataCount = cpu_to_le16(data_count);
  4961. pSMB->TotalDataCount = pSMB->DataCount;
  4962. pSMB->ParameterCount = cpu_to_le16(params);
  4963. pSMB->TotalParameterCount = pSMB->ParameterCount;
  4964. pSMB->Reserved4 = 0;
  4965. inc_rfc1001_len(pSMB, byte_count);
  4966. parm_data->FileSize = cpu_to_le64(size);
  4967. pSMB->ByteCount = cpu_to_le16(byte_count);
  4968. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  4969. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  4970. if (rc)
  4971. cifs_dbg(FYI, "SetPathInfo (file size) returned %d\n", rc);
  4972. cifs_buf_release(pSMB);
  4973. if (rc == -EAGAIN)
  4974. goto SetEOFRetry;
  4975. return rc;
  4976. }
  4977. int
  4978. CIFSSMBSetFileSize(const unsigned int xid, struct cifs_tcon *tcon,
  4979. struct cifsFileInfo *cfile, __u64 size, bool set_allocation)
  4980. {
  4981. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  4982. struct file_end_of_file_info *parm_data;
  4983. int rc = 0;
  4984. __u16 params, param_offset, offset, byte_count, count;
  4985. cifs_dbg(FYI, "SetFileSize (via SetFileInfo) %lld\n",
  4986. (long long)size);
  4987. rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
  4988. if (rc)
  4989. return rc;
  4990. pSMB->hdr.Pid = cpu_to_le16((__u16)cfile->pid);
  4991. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(cfile->pid >> 16));
  4992. params = 6;
  4993. pSMB->MaxSetupCount = 0;
  4994. pSMB->Reserved = 0;
  4995. pSMB->Flags = 0;
  4996. pSMB->Timeout = 0;
  4997. pSMB->Reserved2 = 0;
  4998. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  4999. offset = param_offset + params;
  5000. count = sizeof(struct file_end_of_file_info);
  5001. pSMB->MaxParameterCount = cpu_to_le16(2);
  5002. /* BB find exact max SMB PDU from sess structure BB */
  5003. pSMB->MaxDataCount = cpu_to_le16(1000);
  5004. pSMB->SetupCount = 1;
  5005. pSMB->Reserved3 = 0;
  5006. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  5007. byte_count = 3 /* pad */ + params + count;
  5008. pSMB->DataCount = cpu_to_le16(count);
  5009. pSMB->ParameterCount = cpu_to_le16(params);
  5010. pSMB->TotalDataCount = pSMB->DataCount;
  5011. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5012. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5013. parm_data =
  5014. (struct file_end_of_file_info *) (((char *) &pSMB->hdr.Protocol)
  5015. + offset);
  5016. pSMB->DataOffset = cpu_to_le16(offset);
  5017. parm_data->FileSize = cpu_to_le64(size);
  5018. pSMB->Fid = cfile->fid.netfid;
  5019. if (set_allocation) {
  5020. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  5021. pSMB->InformationLevel =
  5022. cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO2);
  5023. else
  5024. pSMB->InformationLevel =
  5025. cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO);
  5026. } else /* Set File Size */ {
  5027. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  5028. pSMB->InformationLevel =
  5029. cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO2);
  5030. else
  5031. pSMB->InformationLevel =
  5032. cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO);
  5033. }
  5034. pSMB->Reserved4 = 0;
  5035. inc_rfc1001_len(pSMB, byte_count);
  5036. pSMB->ByteCount = cpu_to_le16(byte_count);
  5037. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  5038. if (rc) {
  5039. cifs_dbg(FYI, "Send error in SetFileInfo (SetFileSize) = %d\n",
  5040. rc);
  5041. }
  5042. /* Note: On -EAGAIN error only caller can retry on handle based calls
  5043. since file handle passed in no longer valid */
  5044. return rc;
  5045. }
  5046. /* Some legacy servers such as NT4 require that the file times be set on
  5047. an open handle, rather than by pathname - this is awkward due to
  5048. potential access conflicts on the open, but it is unavoidable for these
  5049. old servers since the only other choice is to go from 100 nanosecond DCE
  5050. time and resort to the original setpathinfo level which takes the ancient
  5051. DOS time format with 2 second granularity */
  5052. int
  5053. CIFSSMBSetFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
  5054. const FILE_BASIC_INFO *data, __u16 fid, __u32 pid_of_opener)
  5055. {
  5056. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  5057. char *data_offset;
  5058. int rc = 0;
  5059. __u16 params, param_offset, offset, byte_count, count;
  5060. cifs_dbg(FYI, "Set Times (via SetFileInfo)\n");
  5061. rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
  5062. if (rc)
  5063. return rc;
  5064. pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
  5065. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
  5066. params = 6;
  5067. pSMB->MaxSetupCount = 0;
  5068. pSMB->Reserved = 0;
  5069. pSMB->Flags = 0;
  5070. pSMB->Timeout = 0;
  5071. pSMB->Reserved2 = 0;
  5072. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  5073. offset = param_offset + params;
  5074. data_offset = (char *)pSMB +
  5075. offsetof(struct smb_hdr, Protocol) + offset;
  5076. count = sizeof(FILE_BASIC_INFO);
  5077. pSMB->MaxParameterCount = cpu_to_le16(2);
  5078. /* BB find max SMB PDU from sess */
  5079. pSMB->MaxDataCount = cpu_to_le16(1000);
  5080. pSMB->SetupCount = 1;
  5081. pSMB->Reserved3 = 0;
  5082. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  5083. byte_count = 3 /* pad */ + params + count;
  5084. pSMB->DataCount = cpu_to_le16(count);
  5085. pSMB->ParameterCount = cpu_to_le16(params);
  5086. pSMB->TotalDataCount = pSMB->DataCount;
  5087. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5088. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5089. pSMB->DataOffset = cpu_to_le16(offset);
  5090. pSMB->Fid = fid;
  5091. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  5092. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO2);
  5093. else
  5094. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO);
  5095. pSMB->Reserved4 = 0;
  5096. inc_rfc1001_len(pSMB, byte_count);
  5097. pSMB->ByteCount = cpu_to_le16(byte_count);
  5098. memcpy(data_offset, data, sizeof(FILE_BASIC_INFO));
  5099. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  5100. if (rc)
  5101. cifs_dbg(FYI, "Send error in Set Time (SetFileInfo) = %d\n",
  5102. rc);
  5103. /* Note: On -EAGAIN error only caller can retry on handle based calls
  5104. since file handle passed in no longer valid */
  5105. return rc;
  5106. }
  5107. int
  5108. CIFSSMBSetFileDisposition(const unsigned int xid, struct cifs_tcon *tcon,
  5109. bool delete_file, __u16 fid, __u32 pid_of_opener)
  5110. {
  5111. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  5112. char *data_offset;
  5113. int rc = 0;
  5114. __u16 params, param_offset, offset, byte_count, count;
  5115. cifs_dbg(FYI, "Set File Disposition (via SetFileInfo)\n");
  5116. rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
  5117. if (rc)
  5118. return rc;
  5119. pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
  5120. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
  5121. params = 6;
  5122. pSMB->MaxSetupCount = 0;
  5123. pSMB->Reserved = 0;
  5124. pSMB->Flags = 0;
  5125. pSMB->Timeout = 0;
  5126. pSMB->Reserved2 = 0;
  5127. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  5128. offset = param_offset + params;
  5129. data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
  5130. count = 1;
  5131. pSMB->MaxParameterCount = cpu_to_le16(2);
  5132. /* BB find max SMB PDU from sess */
  5133. pSMB->MaxDataCount = cpu_to_le16(1000);
  5134. pSMB->SetupCount = 1;
  5135. pSMB->Reserved3 = 0;
  5136. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  5137. byte_count = 3 /* pad */ + params + count;
  5138. pSMB->DataCount = cpu_to_le16(count);
  5139. pSMB->ParameterCount = cpu_to_le16(params);
  5140. pSMB->TotalDataCount = pSMB->DataCount;
  5141. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5142. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5143. pSMB->DataOffset = cpu_to_le16(offset);
  5144. pSMB->Fid = fid;
  5145. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_DISPOSITION_INFO);
  5146. pSMB->Reserved4 = 0;
  5147. inc_rfc1001_len(pSMB, byte_count);
  5148. pSMB->ByteCount = cpu_to_le16(byte_count);
  5149. *data_offset = delete_file ? 1 : 0;
  5150. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  5151. if (rc)
  5152. cifs_dbg(FYI, "Send error in SetFileDisposition = %d\n", rc);
  5153. return rc;
  5154. }
  5155. int
  5156. CIFSSMBSetPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
  5157. const char *fileName, const FILE_BASIC_INFO *data,
  5158. const struct nls_table *nls_codepage, int remap)
  5159. {
  5160. TRANSACTION2_SPI_REQ *pSMB = NULL;
  5161. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  5162. int name_len;
  5163. int rc = 0;
  5164. int bytes_returned = 0;
  5165. char *data_offset;
  5166. __u16 params, param_offset, offset, byte_count, count;
  5167. cifs_dbg(FYI, "In SetTimes\n");
  5168. SetTimesRetry:
  5169. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  5170. (void **) &pSMBr);
  5171. if (rc)
  5172. return rc;
  5173. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  5174. name_len =
  5175. cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
  5176. PATH_MAX, nls_codepage, remap);
  5177. name_len++; /* trailing null */
  5178. name_len *= 2;
  5179. } else { /* BB improve the check for buffer overruns BB */
  5180. name_len = strnlen(fileName, PATH_MAX);
  5181. name_len++; /* trailing null */
  5182. strncpy(pSMB->FileName, fileName, name_len);
  5183. }
  5184. params = 6 + name_len;
  5185. count = sizeof(FILE_BASIC_INFO);
  5186. pSMB->MaxParameterCount = cpu_to_le16(2);
  5187. /* BB find max SMB PDU from sess structure BB */
  5188. pSMB->MaxDataCount = cpu_to_le16(1000);
  5189. pSMB->MaxSetupCount = 0;
  5190. pSMB->Reserved = 0;
  5191. pSMB->Flags = 0;
  5192. pSMB->Timeout = 0;
  5193. pSMB->Reserved2 = 0;
  5194. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  5195. InformationLevel) - 4;
  5196. offset = param_offset + params;
  5197. data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
  5198. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5199. pSMB->DataOffset = cpu_to_le16(offset);
  5200. pSMB->SetupCount = 1;
  5201. pSMB->Reserved3 = 0;
  5202. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  5203. byte_count = 3 /* pad */ + params + count;
  5204. pSMB->DataCount = cpu_to_le16(count);
  5205. pSMB->ParameterCount = cpu_to_le16(params);
  5206. pSMB->TotalDataCount = pSMB->DataCount;
  5207. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5208. if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
  5209. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO2);
  5210. else
  5211. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO);
  5212. pSMB->Reserved4 = 0;
  5213. inc_rfc1001_len(pSMB, byte_count);
  5214. memcpy(data_offset, data, sizeof(FILE_BASIC_INFO));
  5215. pSMB->ByteCount = cpu_to_le16(byte_count);
  5216. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5217. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  5218. if (rc)
  5219. cifs_dbg(FYI, "SetPathInfo (times) returned %d\n", rc);
  5220. cifs_buf_release(pSMB);
  5221. if (rc == -EAGAIN)
  5222. goto SetTimesRetry;
  5223. return rc;
  5224. }
  5225. /* Can not be used to set time stamps yet (due to old DOS time format) */
  5226. /* Can be used to set attributes */
  5227. #if 0 /* Possibly not needed - since it turns out that strangely NT4 has a bug
  5228. handling it anyway and NT4 was what we thought it would be needed for
  5229. Do not delete it until we prove whether needed for Win9x though */
  5230. int
  5231. CIFSSMBSetAttrLegacy(unsigned int xid, struct cifs_tcon *tcon, char *fileName,
  5232. __u16 dos_attrs, const struct nls_table *nls_codepage)
  5233. {
  5234. SETATTR_REQ *pSMB = NULL;
  5235. SETATTR_RSP *pSMBr = NULL;
  5236. int rc = 0;
  5237. int bytes_returned;
  5238. int name_len;
  5239. cifs_dbg(FYI, "In SetAttrLegacy\n");
  5240. SetAttrLgcyRetry:
  5241. rc = smb_init(SMB_COM_SETATTR, 8, tcon, (void **) &pSMB,
  5242. (void **) &pSMBr);
  5243. if (rc)
  5244. return rc;
  5245. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  5246. name_len =
  5247. ConvertToUTF16((__le16 *) pSMB->fileName, fileName,
  5248. PATH_MAX, nls_codepage);
  5249. name_len++; /* trailing null */
  5250. name_len *= 2;
  5251. } else { /* BB improve the check for buffer overruns BB */
  5252. name_len = strnlen(fileName, PATH_MAX);
  5253. name_len++; /* trailing null */
  5254. strncpy(pSMB->fileName, fileName, name_len);
  5255. }
  5256. pSMB->attr = cpu_to_le16(dos_attrs);
  5257. pSMB->BufferFormat = 0x04;
  5258. inc_rfc1001_len(pSMB, name_len + 1);
  5259. pSMB->ByteCount = cpu_to_le16(name_len + 1);
  5260. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5261. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  5262. if (rc)
  5263. cifs_dbg(FYI, "Error in LegacySetAttr = %d\n", rc);
  5264. cifs_buf_release(pSMB);
  5265. if (rc == -EAGAIN)
  5266. goto SetAttrLgcyRetry;
  5267. return rc;
  5268. }
  5269. #endif /* temporarily unneeded SetAttr legacy function */
  5270. static void
  5271. cifs_fill_unix_set_info(FILE_UNIX_BASIC_INFO *data_offset,
  5272. const struct cifs_unix_set_info_args *args)
  5273. {
  5274. u64 uid = NO_CHANGE_64, gid = NO_CHANGE_64;
  5275. u64 mode = args->mode;
  5276. if (uid_valid(args->uid))
  5277. uid = from_kuid(&init_user_ns, args->uid);
  5278. if (gid_valid(args->gid))
  5279. gid = from_kgid(&init_user_ns, args->gid);
  5280. /*
  5281. * Samba server ignores set of file size to zero due to bugs in some
  5282. * older clients, but we should be precise - we use SetFileSize to
  5283. * set file size and do not want to truncate file size to zero
  5284. * accidentally as happened on one Samba server beta by putting
  5285. * zero instead of -1 here
  5286. */
  5287. data_offset->EndOfFile = cpu_to_le64(NO_CHANGE_64);
  5288. data_offset->NumOfBytes = cpu_to_le64(NO_CHANGE_64);
  5289. data_offset->LastStatusChange = cpu_to_le64(args->ctime);
  5290. data_offset->LastAccessTime = cpu_to_le64(args->atime);
  5291. data_offset->LastModificationTime = cpu_to_le64(args->mtime);
  5292. data_offset->Uid = cpu_to_le64(uid);
  5293. data_offset->Gid = cpu_to_le64(gid);
  5294. /* better to leave device as zero when it is */
  5295. data_offset->DevMajor = cpu_to_le64(MAJOR(args->device));
  5296. data_offset->DevMinor = cpu_to_le64(MINOR(args->device));
  5297. data_offset->Permissions = cpu_to_le64(mode);
  5298. if (S_ISREG(mode))
  5299. data_offset->Type = cpu_to_le32(UNIX_FILE);
  5300. else if (S_ISDIR(mode))
  5301. data_offset->Type = cpu_to_le32(UNIX_DIR);
  5302. else if (S_ISLNK(mode))
  5303. data_offset->Type = cpu_to_le32(UNIX_SYMLINK);
  5304. else if (S_ISCHR(mode))
  5305. data_offset->Type = cpu_to_le32(UNIX_CHARDEV);
  5306. else if (S_ISBLK(mode))
  5307. data_offset->Type = cpu_to_le32(UNIX_BLOCKDEV);
  5308. else if (S_ISFIFO(mode))
  5309. data_offset->Type = cpu_to_le32(UNIX_FIFO);
  5310. else if (S_ISSOCK(mode))
  5311. data_offset->Type = cpu_to_le32(UNIX_SOCKET);
  5312. }
  5313. int
  5314. CIFSSMBUnixSetFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
  5315. const struct cifs_unix_set_info_args *args,
  5316. u16 fid, u32 pid_of_opener)
  5317. {
  5318. struct smb_com_transaction2_sfi_req *pSMB = NULL;
  5319. char *data_offset;
  5320. int rc = 0;
  5321. u16 params, param_offset, offset, byte_count, count;
  5322. cifs_dbg(FYI, "Set Unix Info (via SetFileInfo)\n");
  5323. rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
  5324. if (rc)
  5325. return rc;
  5326. pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
  5327. pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
  5328. params = 6;
  5329. pSMB->MaxSetupCount = 0;
  5330. pSMB->Reserved = 0;
  5331. pSMB->Flags = 0;
  5332. pSMB->Timeout = 0;
  5333. pSMB->Reserved2 = 0;
  5334. param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
  5335. offset = param_offset + params;
  5336. data_offset = (char *)pSMB +
  5337. offsetof(struct smb_hdr, Protocol) + offset;
  5338. count = sizeof(FILE_UNIX_BASIC_INFO);
  5339. pSMB->MaxParameterCount = cpu_to_le16(2);
  5340. /* BB find max SMB PDU from sess */
  5341. pSMB->MaxDataCount = cpu_to_le16(1000);
  5342. pSMB->SetupCount = 1;
  5343. pSMB->Reserved3 = 0;
  5344. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
  5345. byte_count = 3 /* pad */ + params + count;
  5346. pSMB->DataCount = cpu_to_le16(count);
  5347. pSMB->ParameterCount = cpu_to_le16(params);
  5348. pSMB->TotalDataCount = pSMB->DataCount;
  5349. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5350. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5351. pSMB->DataOffset = cpu_to_le16(offset);
  5352. pSMB->Fid = fid;
  5353. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_BASIC);
  5354. pSMB->Reserved4 = 0;
  5355. inc_rfc1001_len(pSMB, byte_count);
  5356. pSMB->ByteCount = cpu_to_le16(byte_count);
  5357. cifs_fill_unix_set_info((FILE_UNIX_BASIC_INFO *)data_offset, args);
  5358. rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
  5359. if (rc)
  5360. cifs_dbg(FYI, "Send error in Set Time (SetFileInfo) = %d\n",
  5361. rc);
  5362. /* Note: On -EAGAIN error only caller can retry on handle based calls
  5363. since file handle passed in no longer valid */
  5364. return rc;
  5365. }
  5366. int
  5367. CIFSSMBUnixSetPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
  5368. const char *file_name,
  5369. const struct cifs_unix_set_info_args *args,
  5370. const struct nls_table *nls_codepage, int remap)
  5371. {
  5372. TRANSACTION2_SPI_REQ *pSMB = NULL;
  5373. TRANSACTION2_SPI_RSP *pSMBr = NULL;
  5374. int name_len;
  5375. int rc = 0;
  5376. int bytes_returned = 0;
  5377. FILE_UNIX_BASIC_INFO *data_offset;
  5378. __u16 params, param_offset, offset, count, byte_count;
  5379. cifs_dbg(FYI, "In SetUID/GID/Mode\n");
  5380. setPermsRetry:
  5381. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  5382. (void **) &pSMBr);
  5383. if (rc)
  5384. return rc;
  5385. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  5386. name_len =
  5387. cifsConvertToUTF16((__le16 *) pSMB->FileName, file_name,
  5388. PATH_MAX, nls_codepage, remap);
  5389. name_len++; /* trailing null */
  5390. name_len *= 2;
  5391. } else { /* BB improve the check for buffer overruns BB */
  5392. name_len = strnlen(file_name, PATH_MAX);
  5393. name_len++; /* trailing null */
  5394. strncpy(pSMB->FileName, file_name, name_len);
  5395. }
  5396. params = 6 + name_len;
  5397. count = sizeof(FILE_UNIX_BASIC_INFO);
  5398. pSMB->MaxParameterCount = cpu_to_le16(2);
  5399. /* BB find max SMB PDU from sess structure BB */
  5400. pSMB->MaxDataCount = cpu_to_le16(1000);
  5401. pSMB->MaxSetupCount = 0;
  5402. pSMB->Reserved = 0;
  5403. pSMB->Flags = 0;
  5404. pSMB->Timeout = 0;
  5405. pSMB->Reserved2 = 0;
  5406. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  5407. InformationLevel) - 4;
  5408. offset = param_offset + params;
  5409. data_offset =
  5410. (FILE_UNIX_BASIC_INFO *) ((char *) &pSMB->hdr.Protocol +
  5411. offset);
  5412. memset(data_offset, 0, count);
  5413. pSMB->DataOffset = cpu_to_le16(offset);
  5414. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5415. pSMB->SetupCount = 1;
  5416. pSMB->Reserved3 = 0;
  5417. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  5418. byte_count = 3 /* pad */ + params + count;
  5419. pSMB->ParameterCount = cpu_to_le16(params);
  5420. pSMB->DataCount = cpu_to_le16(count);
  5421. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5422. pSMB->TotalDataCount = pSMB->DataCount;
  5423. pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_BASIC);
  5424. pSMB->Reserved4 = 0;
  5425. inc_rfc1001_len(pSMB, byte_count);
  5426. cifs_fill_unix_set_info(data_offset, args);
  5427. pSMB->ByteCount = cpu_to_le16(byte_count);
  5428. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5429. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  5430. if (rc)
  5431. cifs_dbg(FYI, "SetPathInfo (perms) returned %d\n", rc);
  5432. cifs_buf_release(pSMB);
  5433. if (rc == -EAGAIN)
  5434. goto setPermsRetry;
  5435. return rc;
  5436. }
  5437. #ifdef CONFIG_CIFS_XATTR
  5438. /*
  5439. * Do a path-based QUERY_ALL_EAS call and parse the result. This is a common
  5440. * function used by listxattr and getxattr type calls. When ea_name is set,
  5441. * it looks for that attribute name and stuffs that value into the EAData
  5442. * buffer. When ea_name is NULL, it stuffs a list of attribute names into the
  5443. * buffer. In both cases, the return value is either the length of the
  5444. * resulting data or a negative error code. If EAData is a NULL pointer then
  5445. * the data isn't copied to it, but the length is returned.
  5446. */
  5447. ssize_t
  5448. CIFSSMBQAllEAs(const unsigned int xid, struct cifs_tcon *tcon,
  5449. const unsigned char *searchName, const unsigned char *ea_name,
  5450. char *EAData, size_t buf_size,
  5451. const struct nls_table *nls_codepage, int remap)
  5452. {
  5453. /* BB assumes one setup word */
  5454. TRANSACTION2_QPI_REQ *pSMB = NULL;
  5455. TRANSACTION2_QPI_RSP *pSMBr = NULL;
  5456. int rc = 0;
  5457. int bytes_returned;
  5458. int list_len;
  5459. struct fealist *ea_response_data;
  5460. struct fea *temp_fea;
  5461. char *temp_ptr;
  5462. char *end_of_smb;
  5463. __u16 params, byte_count, data_offset;
  5464. unsigned int ea_name_len = ea_name ? strlen(ea_name) : 0;
  5465. cifs_dbg(FYI, "In Query All EAs path %s\n", searchName);
  5466. QAllEAsRetry:
  5467. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  5468. (void **) &pSMBr);
  5469. if (rc)
  5470. return rc;
  5471. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  5472. list_len =
  5473. cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
  5474. PATH_MAX, nls_codepage, remap);
  5475. list_len++; /* trailing null */
  5476. list_len *= 2;
  5477. } else { /* BB improve the check for buffer overruns BB */
  5478. list_len = strnlen(searchName, PATH_MAX);
  5479. list_len++; /* trailing null */
  5480. strncpy(pSMB->FileName, searchName, list_len);
  5481. }
  5482. params = 2 /* level */ + 4 /* reserved */ + list_len /* includes NUL */;
  5483. pSMB->TotalDataCount = 0;
  5484. pSMB->MaxParameterCount = cpu_to_le16(2);
  5485. /* BB find exact max SMB PDU from sess structure BB */
  5486. pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
  5487. pSMB->MaxSetupCount = 0;
  5488. pSMB->Reserved = 0;
  5489. pSMB->Flags = 0;
  5490. pSMB->Timeout = 0;
  5491. pSMB->Reserved2 = 0;
  5492. pSMB->ParameterOffset = cpu_to_le16(offsetof(
  5493. struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
  5494. pSMB->DataCount = 0;
  5495. pSMB->DataOffset = 0;
  5496. pSMB->SetupCount = 1;
  5497. pSMB->Reserved3 = 0;
  5498. pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
  5499. byte_count = params + 1 /* pad */ ;
  5500. pSMB->TotalParameterCount = cpu_to_le16(params);
  5501. pSMB->ParameterCount = pSMB->TotalParameterCount;
  5502. pSMB->InformationLevel = cpu_to_le16(SMB_INFO_QUERY_ALL_EAS);
  5503. pSMB->Reserved4 = 0;
  5504. inc_rfc1001_len(pSMB, byte_count);
  5505. pSMB->ByteCount = cpu_to_le16(byte_count);
  5506. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5507. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  5508. if (rc) {
  5509. cifs_dbg(FYI, "Send error in QueryAllEAs = %d\n", rc);
  5510. goto QAllEAsOut;
  5511. }
  5512. /* BB also check enough total bytes returned */
  5513. /* BB we need to improve the validity checking
  5514. of these trans2 responses */
  5515. rc = validate_t2((struct smb_t2_rsp *)pSMBr);
  5516. if (rc || get_bcc(&pSMBr->hdr) < 4) {
  5517. rc = -EIO; /* bad smb */
  5518. goto QAllEAsOut;
  5519. }
  5520. /* check that length of list is not more than bcc */
  5521. /* check that each entry does not go beyond length
  5522. of list */
  5523. /* check that each element of each entry does not
  5524. go beyond end of list */
  5525. /* validate_trans2_offsets() */
  5526. /* BB check if start of smb + data_offset > &bcc+ bcc */
  5527. data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
  5528. ea_response_data = (struct fealist *)
  5529. (((char *) &pSMBr->hdr.Protocol) + data_offset);
  5530. list_len = le32_to_cpu(ea_response_data->list_len);
  5531. cifs_dbg(FYI, "ea length %d\n", list_len);
  5532. if (list_len <= 8) {
  5533. cifs_dbg(FYI, "empty EA list returned from server\n");
  5534. /* didn't find the named attribute */
  5535. if (ea_name)
  5536. rc = -ENODATA;
  5537. goto QAllEAsOut;
  5538. }
  5539. /* make sure list_len doesn't go past end of SMB */
  5540. end_of_smb = (char *)pByteArea(&pSMBr->hdr) + get_bcc(&pSMBr->hdr);
  5541. if ((char *)ea_response_data + list_len > end_of_smb) {
  5542. cifs_dbg(FYI, "EA list appears to go beyond SMB\n");
  5543. rc = -EIO;
  5544. goto QAllEAsOut;
  5545. }
  5546. /* account for ea list len */
  5547. list_len -= 4;
  5548. temp_fea = ea_response_data->list;
  5549. temp_ptr = (char *)temp_fea;
  5550. while (list_len > 0) {
  5551. unsigned int name_len;
  5552. __u16 value_len;
  5553. list_len -= 4;
  5554. temp_ptr += 4;
  5555. /* make sure we can read name_len and value_len */
  5556. if (list_len < 0) {
  5557. cifs_dbg(FYI, "EA entry goes beyond length of list\n");
  5558. rc = -EIO;
  5559. goto QAllEAsOut;
  5560. }
  5561. name_len = temp_fea->name_len;
  5562. value_len = le16_to_cpu(temp_fea->value_len);
  5563. list_len -= name_len + 1 + value_len;
  5564. if (list_len < 0) {
  5565. cifs_dbg(FYI, "EA entry goes beyond length of list\n");
  5566. rc = -EIO;
  5567. goto QAllEAsOut;
  5568. }
  5569. if (ea_name) {
  5570. if (ea_name_len == name_len &&
  5571. memcmp(ea_name, temp_ptr, name_len) == 0) {
  5572. temp_ptr += name_len + 1;
  5573. rc = value_len;
  5574. if (buf_size == 0)
  5575. goto QAllEAsOut;
  5576. if ((size_t)value_len > buf_size) {
  5577. rc = -ERANGE;
  5578. goto QAllEAsOut;
  5579. }
  5580. memcpy(EAData, temp_ptr, value_len);
  5581. goto QAllEAsOut;
  5582. }
  5583. } else {
  5584. /* account for prefix user. and trailing null */
  5585. rc += (5 + 1 + name_len);
  5586. if (rc < (int) buf_size) {
  5587. memcpy(EAData, "user.", 5);
  5588. EAData += 5;
  5589. memcpy(EAData, temp_ptr, name_len);
  5590. EAData += name_len;
  5591. /* null terminate name */
  5592. *EAData = 0;
  5593. ++EAData;
  5594. } else if (buf_size == 0) {
  5595. /* skip copy - calc size only */
  5596. } else {
  5597. /* stop before overrun buffer */
  5598. rc = -ERANGE;
  5599. break;
  5600. }
  5601. }
  5602. temp_ptr += name_len + 1 + value_len;
  5603. temp_fea = (struct fea *)temp_ptr;
  5604. }
  5605. /* didn't find the named attribute */
  5606. if (ea_name)
  5607. rc = -ENODATA;
  5608. QAllEAsOut:
  5609. cifs_buf_release(pSMB);
  5610. if (rc == -EAGAIN)
  5611. goto QAllEAsRetry;
  5612. return (ssize_t)rc;
  5613. }
  5614. int
  5615. CIFSSMBSetEA(const unsigned int xid, struct cifs_tcon *tcon,
  5616. const char *fileName, const char *ea_name, const void *ea_value,
  5617. const __u16 ea_value_len, const struct nls_table *nls_codepage,
  5618. int remap)
  5619. {
  5620. struct smb_com_transaction2_spi_req *pSMB = NULL;
  5621. struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
  5622. struct fealist *parm_data;
  5623. int name_len;
  5624. int rc = 0;
  5625. int bytes_returned = 0;
  5626. __u16 params, param_offset, byte_count, offset, count;
  5627. cifs_dbg(FYI, "In SetEA\n");
  5628. SetEARetry:
  5629. rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
  5630. (void **) &pSMBr);
  5631. if (rc)
  5632. return rc;
  5633. if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
  5634. name_len =
  5635. cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
  5636. PATH_MAX, nls_codepage, remap);
  5637. name_len++; /* trailing null */
  5638. name_len *= 2;
  5639. } else { /* BB improve the check for buffer overruns BB */
  5640. name_len = strnlen(fileName, PATH_MAX);
  5641. name_len++; /* trailing null */
  5642. strncpy(pSMB->FileName, fileName, name_len);
  5643. }
  5644. params = 6 + name_len;
  5645. /* done calculating parms using name_len of file name,
  5646. now use name_len to calculate length of ea name
  5647. we are going to create in the inode xattrs */
  5648. if (ea_name == NULL)
  5649. name_len = 0;
  5650. else
  5651. name_len = strnlen(ea_name, 255);
  5652. count = sizeof(*parm_data) + ea_value_len + name_len;
  5653. pSMB->MaxParameterCount = cpu_to_le16(2);
  5654. /* BB find max SMB PDU from sess */
  5655. pSMB->MaxDataCount = cpu_to_le16(1000);
  5656. pSMB->MaxSetupCount = 0;
  5657. pSMB->Reserved = 0;
  5658. pSMB->Flags = 0;
  5659. pSMB->Timeout = 0;
  5660. pSMB->Reserved2 = 0;
  5661. param_offset = offsetof(struct smb_com_transaction2_spi_req,
  5662. InformationLevel) - 4;
  5663. offset = param_offset + params;
  5664. pSMB->InformationLevel =
  5665. cpu_to_le16(SMB_SET_FILE_EA);
  5666. parm_data = (void *)pSMB + offsetof(struct smb_hdr, Protocol) + offset;
  5667. pSMB->ParameterOffset = cpu_to_le16(param_offset);
  5668. pSMB->DataOffset = cpu_to_le16(offset);
  5669. pSMB->SetupCount = 1;
  5670. pSMB->Reserved3 = 0;
  5671. pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
  5672. byte_count = 3 /* pad */ + params + count;
  5673. pSMB->DataCount = cpu_to_le16(count);
  5674. parm_data->list_len = cpu_to_le32(count);
  5675. parm_data->list[0].EA_flags = 0;
  5676. /* we checked above that name len is less than 255 */
  5677. parm_data->list[0].name_len = (__u8)name_len;
  5678. /* EA names are always ASCII */
  5679. if (ea_name)
  5680. strncpy(parm_data->list[0].name, ea_name, name_len);
  5681. parm_data->list[0].name[name_len] = 0;
  5682. parm_data->list[0].value_len = cpu_to_le16(ea_value_len);
  5683. /* caller ensures that ea_value_len is less than 64K but
  5684. we need to ensure that it fits within the smb */
  5685. /*BB add length check to see if it would fit in
  5686. negotiated SMB buffer size BB */
  5687. /* if (ea_value_len > buffer_size - 512 (enough for header)) */
  5688. if (ea_value_len)
  5689. memcpy(parm_data->list[0].name+name_len+1,
  5690. ea_value, ea_value_len);
  5691. pSMB->TotalDataCount = pSMB->DataCount;
  5692. pSMB->ParameterCount = cpu_to_le16(params);
  5693. pSMB->TotalParameterCount = pSMB->ParameterCount;
  5694. pSMB->Reserved4 = 0;
  5695. inc_rfc1001_len(pSMB, byte_count);
  5696. pSMB->ByteCount = cpu_to_le16(byte_count);
  5697. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5698. (struct smb_hdr *) pSMBr, &bytes_returned, 0);
  5699. if (rc)
  5700. cifs_dbg(FYI, "SetPathInfo (EA) returned %d\n", rc);
  5701. cifs_buf_release(pSMB);
  5702. if (rc == -EAGAIN)
  5703. goto SetEARetry;
  5704. return rc;
  5705. }
  5706. #endif
  5707. #ifdef CONFIG_CIFS_DNOTIFY_EXPERIMENTAL /* BB unused temporarily */
  5708. /*
  5709. * Years ago the kernel added a "dnotify" function for Samba server,
  5710. * to allow network clients (such as Windows) to display updated
  5711. * lists of files in directory listings automatically when
  5712. * files are added by one user when another user has the
  5713. * same directory open on their desktop. The Linux cifs kernel
  5714. * client hooked into the kernel side of this interface for
  5715. * the same reason, but ironically when the VFS moved from
  5716. * "dnotify" to "inotify" it became harder to plug in Linux
  5717. * network file system clients (the most obvious use case
  5718. * for notify interfaces is when multiple users can update
  5719. * the contents of the same directory - exactly what network
  5720. * file systems can do) although the server (Samba) could
  5721. * still use it. For the short term we leave the worker
  5722. * function ifdeffed out (below) until inotify is fixed
  5723. * in the VFS to make it easier to plug in network file
  5724. * system clients. If inotify turns out to be permanently
  5725. * incompatible for network fs clients, we could instead simply
  5726. * expose this config flag by adding a future cifs (and smb2) notify ioctl.
  5727. */
  5728. int CIFSSMBNotify(const unsigned int xid, struct cifs_tcon *tcon,
  5729. const int notify_subdirs, const __u16 netfid,
  5730. __u32 filter, struct file *pfile, int multishot,
  5731. const struct nls_table *nls_codepage)
  5732. {
  5733. int rc = 0;
  5734. struct smb_com_transaction_change_notify_req *pSMB = NULL;
  5735. struct smb_com_ntransaction_change_notify_rsp *pSMBr = NULL;
  5736. struct dir_notify_req *dnotify_req;
  5737. int bytes_returned;
  5738. cifs_dbg(FYI, "In CIFSSMBNotify for file handle %d\n", (int)netfid);
  5739. rc = smb_init(SMB_COM_NT_TRANSACT, 23, tcon, (void **) &pSMB,
  5740. (void **) &pSMBr);
  5741. if (rc)
  5742. return rc;
  5743. pSMB->TotalParameterCount = 0 ;
  5744. pSMB->TotalDataCount = 0;
  5745. pSMB->MaxParameterCount = cpu_to_le32(2);
  5746. pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
  5747. pSMB->MaxSetupCount = 4;
  5748. pSMB->Reserved = 0;
  5749. pSMB->ParameterOffset = 0;
  5750. pSMB->DataCount = 0;
  5751. pSMB->DataOffset = 0;
  5752. pSMB->SetupCount = 4; /* single byte does not need le conversion */
  5753. pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_NOTIFY_CHANGE);
  5754. pSMB->ParameterCount = pSMB->TotalParameterCount;
  5755. if (notify_subdirs)
  5756. pSMB->WatchTree = 1; /* one byte - no le conversion needed */
  5757. pSMB->Reserved2 = 0;
  5758. pSMB->CompletionFilter = cpu_to_le32(filter);
  5759. pSMB->Fid = netfid; /* file handle always le */
  5760. pSMB->ByteCount = 0;
  5761. rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
  5762. (struct smb_hdr *)pSMBr, &bytes_returned,
  5763. CIFS_ASYNC_OP);
  5764. if (rc) {
  5765. cifs_dbg(FYI, "Error in Notify = %d\n", rc);
  5766. } else {
  5767. /* Add file to outstanding requests */
  5768. /* BB change to kmem cache alloc */
  5769. dnotify_req = kmalloc(
  5770. sizeof(struct dir_notify_req),
  5771. GFP_KERNEL);
  5772. if (dnotify_req) {
  5773. dnotify_req->Pid = pSMB->hdr.Pid;
  5774. dnotify_req->PidHigh = pSMB->hdr.PidHigh;
  5775. dnotify_req->Mid = pSMB->hdr.Mid;
  5776. dnotify_req->Tid = pSMB->hdr.Tid;
  5777. dnotify_req->Uid = pSMB->hdr.Uid;
  5778. dnotify_req->netfid = netfid;
  5779. dnotify_req->pfile = pfile;
  5780. dnotify_req->filter = filter;
  5781. dnotify_req->multishot = multishot;
  5782. spin_lock(&GlobalMid_Lock);
  5783. list_add_tail(&dnotify_req->lhead,
  5784. &GlobalDnotifyReqList);
  5785. spin_unlock(&GlobalMid_Lock);
  5786. } else
  5787. rc = -ENOMEM;
  5788. }
  5789. cifs_buf_release(pSMB);
  5790. return rc;
  5791. }
  5792. #endif /* was needed for dnotify, and will be needed for inotify when VFS fix */