rtw_mlme_ext.c 479 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203102041020510206102071020810209102101021110212102131021410215102161021710218102191022010221102221022310224102251022610227102281022910230102311023210233102341023510236102371023810239102401024110242102431024410245102461024710248102491025010251102521025310254102551025610257102581025910260102611026210263102641026510266102671026810269102701027110272102731027410275102761027710278102791028010281102821028310284102851028610287102881028910290102911029210293102941029510296102971029810299103001030110302103031030410305103061030710308103091031010311103121031310314103151031610317103181031910320103211032210323103241032510326103271032810329103301033110332103331033410335103361033710338103391034010341103421034310344103451034610347103481034910350103511035210353103541035510356103571035810359103601036110362103631036410365103661036710368103691037010371103721037310374103751037610377103781037910380103811038210383103841038510386103871038810389103901039110392103931039410395103961039710398103991040010401104021040310404104051040610407104081040910410104111041210413104141041510416104171041810419104201042110422104231042410425104261042710428104291043010431104321043310434104351043610437104381043910440104411044210443104441044510446104471044810449104501045110452104531045410455104561045710458104591046010461104621046310464104651046610467104681046910470104711047210473104741047510476104771047810479104801048110482104831048410485104861048710488104891049010491104921049310494104951049610497104981049910500105011050210503105041050510506105071050810509105101051110512105131051410515105161051710518105191052010521105221052310524105251052610527105281052910530105311053210533105341053510536105371053810539105401054110542105431054410545105461054710548105491055010551105521055310554105551055610557105581055910560105611056210563105641056510566105671056810569105701057110572105731057410575105761057710578105791058010581105821058310584105851058610587105881058910590105911059210593105941059510596105971059810599106001060110602106031060410605106061060710608106091061010611106121061310614106151061610617106181061910620106211062210623106241062510626106271062810629106301063110632106331063410635106361063710638106391064010641106421064310644106451064610647106481064910650106511065210653106541065510656106571065810659106601066110662106631066410665106661066710668106691067010671106721067310674106751067610677106781067910680106811068210683106841068510686106871068810689106901069110692106931069410695106961069710698106991070010701107021070310704107051070610707107081070910710107111071210713107141071510716107171071810719107201072110722107231072410725107261072710728107291073010731107321073310734107351073610737107381073910740107411074210743107441074510746107471074810749107501075110752107531075410755107561075710758107591076010761107621076310764107651076610767107681076910770107711077210773107741077510776107771077810779107801078110782107831078410785107861078710788107891079010791107921079310794107951079610797107981079910800108011080210803108041080510806108071080810809108101081110812108131081410815108161081710818108191082010821108221082310824108251082610827108281082910830108311083210833108341083510836108371083810839108401084110842108431084410845108461084710848108491085010851108521085310854108551085610857108581085910860108611086210863108641086510866108671086810869108701087110872108731087410875108761087710878108791088010881108821088310884108851088610887108881088910890108911089210893108941089510896108971089810899109001090110902109031090410905109061090710908109091091010911109121091310914109151091610917109181091910920109211092210923109241092510926109271092810929109301093110932109331093410935109361093710938109391094010941109421094310944109451094610947109481094910950109511095210953109541095510956109571095810959109601096110962109631096410965109661096710968109691097010971109721097310974109751097610977109781097910980109811098210983109841098510986109871098810989109901099110992109931099410995109961099710998109991100011001110021100311004110051100611007110081100911010110111101211013110141101511016110171101811019110201102111022110231102411025110261102711028110291103011031110321103311034110351103611037110381103911040110411104211043110441104511046110471104811049110501105111052110531105411055110561105711058110591106011061110621106311064110651106611067110681106911070110711107211073110741107511076110771107811079110801108111082110831108411085110861108711088110891109011091110921109311094110951109611097110981109911100111011110211103111041110511106111071110811109111101111111112111131111411115111161111711118111191112011121111221112311124111251112611127111281112911130111311113211133111341113511136111371113811139111401114111142111431114411145111461114711148111491115011151111521115311154111551115611157111581115911160111611116211163111641116511166111671116811169111701117111172111731117411175111761117711178111791118011181111821118311184111851118611187111881118911190111911119211193111941119511196111971119811199112001120111202112031120411205112061120711208112091121011211112121121311214112151121611217112181121911220112211122211223112241122511226112271122811229112301123111232112331123411235112361123711238112391124011241112421124311244112451124611247112481124911250112511125211253112541125511256112571125811259112601126111262112631126411265112661126711268112691127011271112721127311274112751127611277112781127911280112811128211283112841128511286112871128811289112901129111292112931129411295112961129711298112991130011301113021130311304113051130611307113081130911310113111131211313113141131511316113171131811319113201132111322113231132411325113261132711328113291133011331113321133311334113351133611337113381133911340113411134211343113441134511346113471134811349113501135111352113531135411355113561135711358113591136011361113621136311364113651136611367113681136911370113711137211373113741137511376113771137811379113801138111382113831138411385113861138711388113891139011391113921139311394113951139611397113981139911400114011140211403114041140511406114071140811409114101141111412114131141411415114161141711418114191142011421114221142311424114251142611427114281142911430114311143211433114341143511436114371143811439114401144111442114431144411445114461144711448114491145011451114521145311454114551145611457114581145911460114611146211463114641146511466114671146811469114701147111472114731147411475114761147711478114791148011481114821148311484114851148611487114881148911490114911149211493114941149511496114971149811499115001150111502115031150411505115061150711508115091151011511115121151311514115151151611517115181151911520115211152211523115241152511526115271152811529115301153111532115331153411535115361153711538115391154011541115421154311544115451154611547115481154911550115511155211553115541155511556115571155811559115601156111562115631156411565115661156711568115691157011571115721157311574115751157611577115781157911580115811158211583115841158511586115871158811589115901159111592115931159411595115961159711598115991160011601116021160311604116051160611607116081160911610116111161211613116141161511616116171161811619116201162111622116231162411625116261162711628116291163011631116321163311634116351163611637116381163911640116411164211643116441164511646116471164811649116501165111652116531165411655116561165711658116591166011661116621166311664116651166611667116681166911670116711167211673116741167511676116771167811679116801168111682116831168411685116861168711688116891169011691116921169311694116951169611697116981169911700117011170211703117041170511706117071170811709117101171111712117131171411715117161171711718117191172011721117221172311724117251172611727117281172911730117311173211733117341173511736117371173811739117401174111742117431174411745117461174711748117491175011751117521175311754117551175611757117581175911760117611176211763117641176511766117671176811769117701177111772117731177411775117761177711778117791178011781117821178311784117851178611787117881178911790117911179211793117941179511796117971179811799118001180111802118031180411805118061180711808118091181011811118121181311814118151181611817118181181911820118211182211823118241182511826118271182811829118301183111832118331183411835118361183711838118391184011841118421184311844118451184611847118481184911850118511185211853118541185511856118571185811859118601186111862118631186411865118661186711868118691187011871118721187311874118751187611877118781187911880118811188211883118841188511886118871188811889118901189111892118931189411895118961189711898118991190011901119021190311904119051190611907119081190911910119111191211913119141191511916119171191811919119201192111922119231192411925119261192711928119291193011931119321193311934119351193611937119381193911940119411194211943119441194511946119471194811949119501195111952119531195411955119561195711958119591196011961119621196311964119651196611967119681196911970119711197211973119741197511976119771197811979119801198111982119831198411985119861198711988119891199011991119921199311994119951199611997119981199912000120011200212003120041200512006120071200812009120101201112012120131201412015120161201712018120191202012021120221202312024120251202612027120281202912030120311203212033120341203512036120371203812039120401204112042120431204412045120461204712048120491205012051120521205312054120551205612057120581205912060120611206212063120641206512066120671206812069120701207112072120731207412075120761207712078120791208012081120821208312084120851208612087120881208912090120911209212093120941209512096120971209812099121001210112102121031210412105121061210712108121091211012111121121211312114121151211612117121181211912120121211212212123121241212512126121271212812129121301213112132121331213412135121361213712138121391214012141121421214312144121451214612147121481214912150121511215212153121541215512156121571215812159121601216112162121631216412165121661216712168121691217012171121721217312174121751217612177121781217912180121811218212183121841218512186121871218812189121901219112192121931219412195121961219712198121991220012201122021220312204122051220612207122081220912210122111221212213122141221512216122171221812219122201222112222122231222412225122261222712228122291223012231122321223312234122351223612237122381223912240122411224212243122441224512246122471224812249122501225112252122531225412255122561225712258122591226012261122621226312264122651226612267122681226912270122711227212273122741227512276122771227812279122801228112282122831228412285122861228712288122891229012291122921229312294122951229612297122981229912300123011230212303123041230512306123071230812309123101231112312123131231412315123161231712318123191232012321123221232312324123251232612327123281232912330123311233212333123341233512336123371233812339123401234112342123431234412345123461234712348123491235012351123521235312354123551235612357123581235912360123611236212363123641236512366123671236812369123701237112372123731237412375123761237712378123791238012381123821238312384123851238612387123881238912390123911239212393123941239512396123971239812399124001240112402124031240412405124061240712408124091241012411124121241312414124151241612417124181241912420124211242212423124241242512426124271242812429124301243112432124331243412435124361243712438124391244012441124421244312444124451244612447124481244912450124511245212453124541245512456124571245812459124601246112462124631246412465124661246712468124691247012471124721247312474124751247612477124781247912480124811248212483124841248512486124871248812489124901249112492124931249412495124961249712498124991250012501125021250312504125051250612507125081250912510125111251212513125141251512516125171251812519125201252112522125231252412525125261252712528125291253012531125321253312534125351253612537125381253912540125411254212543125441254512546125471254812549125501255112552125531255412555125561255712558125591256012561125621256312564125651256612567125681256912570125711257212573125741257512576125771257812579125801258112582125831258412585125861258712588125891259012591125921259312594125951259612597125981259912600126011260212603126041260512606126071260812609126101261112612126131261412615126161261712618126191262012621126221262312624126251262612627126281262912630126311263212633126341263512636126371263812639126401264112642126431264412645126461264712648126491265012651126521265312654126551265612657126581265912660126611266212663126641266512666126671266812669126701267112672126731267412675126761267712678126791268012681126821268312684126851268612687126881268912690126911269212693126941269512696126971269812699127001270112702127031270412705127061270712708127091271012711127121271312714127151271612717127181271912720127211272212723127241272512726127271272812729127301273112732127331273412735127361273712738127391274012741127421274312744127451274612747127481274912750127511275212753127541275512756127571275812759127601276112762127631276412765127661276712768127691277012771127721277312774127751277612777127781277912780127811278212783127841278512786127871278812789127901279112792127931279412795127961279712798127991280012801128021280312804128051280612807128081280912810128111281212813128141281512816128171281812819128201282112822128231282412825128261282712828128291283012831128321283312834128351283612837128381283912840128411284212843128441284512846128471284812849128501285112852128531285412855128561285712858128591286012861128621286312864128651286612867128681286912870128711287212873128741287512876128771287812879128801288112882128831288412885128861288712888128891289012891128921289312894128951289612897128981289912900129011290212903129041290512906129071290812909129101291112912129131291412915129161291712918129191292012921129221292312924129251292612927129281292912930129311293212933129341293512936129371293812939129401294112942129431294412945129461294712948129491295012951129521295312954129551295612957129581295912960129611296212963129641296512966129671296812969129701297112972129731297412975129761297712978129791298012981129821298312984129851298612987129881298912990129911299212993129941299512996129971299812999130001300113002130031300413005130061300713008130091301013011130121301313014130151301613017130181301913020130211302213023130241302513026130271302813029130301303113032130331303413035130361303713038130391304013041130421304313044130451304613047130481304913050130511305213053130541305513056130571305813059130601306113062130631306413065130661306713068130691307013071130721307313074130751307613077130781307913080130811308213083130841308513086130871308813089130901309113092130931309413095130961309713098130991310013101131021310313104131051310613107131081310913110131111311213113131141311513116131171311813119131201312113122131231312413125131261312713128131291313013131131321313313134131351313613137131381313913140131411314213143131441314513146131471314813149131501315113152131531315413155131561315713158131591316013161131621316313164131651316613167131681316913170131711317213173131741317513176131771317813179131801318113182131831318413185131861318713188131891319013191131921319313194131951319613197131981319913200132011320213203132041320513206132071320813209132101321113212132131321413215132161321713218132191322013221132221322313224132251322613227132281322913230132311323213233132341323513236132371323813239132401324113242132431324413245132461324713248132491325013251132521325313254132551325613257132581325913260132611326213263132641326513266132671326813269132701327113272132731327413275132761327713278132791328013281132821328313284132851328613287132881328913290132911329213293132941329513296132971329813299133001330113302133031330413305133061330713308133091331013311133121331313314133151331613317133181331913320133211332213323133241332513326133271332813329133301333113332133331333413335133361333713338133391334013341133421334313344133451334613347133481334913350133511335213353133541335513356133571335813359133601336113362133631336413365133661336713368133691337013371133721337313374133751337613377133781337913380133811338213383133841338513386133871338813389133901339113392133931339413395133961339713398133991340013401134021340313404134051340613407134081340913410134111341213413134141341513416134171341813419134201342113422134231342413425134261342713428134291343013431134321343313434134351343613437134381343913440134411344213443134441344513446134471344813449134501345113452134531345413455134561345713458134591346013461134621346313464134651346613467134681346913470134711347213473134741347513476134771347813479134801348113482134831348413485134861348713488134891349013491134921349313494134951349613497134981349913500135011350213503135041350513506135071350813509135101351113512135131351413515135161351713518135191352013521135221352313524135251352613527135281352913530135311353213533135341353513536135371353813539135401354113542135431354413545135461354713548135491355013551135521355313554135551355613557135581355913560135611356213563135641356513566135671356813569135701357113572135731357413575135761357713578135791358013581135821358313584135851358613587135881358913590135911359213593135941359513596135971359813599136001360113602136031360413605136061360713608136091361013611136121361313614136151361613617136181361913620136211362213623136241362513626136271362813629136301363113632136331363413635136361363713638136391364013641136421364313644136451364613647136481364913650136511365213653136541365513656136571365813659136601366113662136631366413665136661366713668136691367013671136721367313674136751367613677136781367913680136811368213683136841368513686136871368813689136901369113692136931369413695136961369713698136991370013701137021370313704137051370613707137081370913710137111371213713137141371513716137171371813719137201372113722137231372413725137261372713728137291373013731137321373313734137351373613737137381373913740137411374213743137441374513746137471374813749137501375113752137531375413755137561375713758137591376013761137621376313764137651376613767137681376913770137711377213773137741377513776137771377813779137801378113782137831378413785137861378713788137891379013791137921379313794137951379613797137981379913800138011380213803138041380513806138071380813809138101381113812138131381413815138161381713818138191382013821138221382313824138251382613827138281382913830138311383213833138341383513836138371383813839138401384113842138431384413845138461384713848138491385013851138521385313854138551385613857138581385913860138611386213863138641386513866138671386813869138701387113872138731387413875138761387713878138791388013881138821388313884138851388613887138881388913890138911389213893138941389513896138971389813899139001390113902139031390413905139061390713908139091391013911139121391313914139151391613917139181391913920139211392213923139241392513926139271392813929139301393113932139331393413935139361393713938139391394013941139421394313944139451394613947139481394913950139511395213953139541395513956139571395813959139601396113962139631396413965139661396713968139691397013971139721397313974139751397613977139781397913980139811398213983139841398513986139871398813989139901399113992139931399413995139961399713998139991400014001140021400314004140051400614007140081400914010140111401214013140141401514016140171401814019140201402114022140231402414025140261402714028140291403014031140321403314034140351403614037140381403914040140411404214043140441404514046140471404814049140501405114052140531405414055140561405714058140591406014061140621406314064140651406614067140681406914070140711407214073140741407514076140771407814079140801408114082140831408414085140861408714088140891409014091140921409314094140951409614097140981409914100141011410214103141041410514106141071410814109141101411114112141131411414115141161411714118141191412014121141221412314124141251412614127141281412914130141311413214133141341413514136141371413814139141401414114142141431414414145141461414714148141491415014151141521415314154141551415614157141581415914160141611416214163141641416514166141671416814169141701417114172141731417414175141761417714178141791418014181141821418314184141851418614187141881418914190141911419214193141941419514196141971419814199142001420114202142031420414205142061420714208142091421014211142121421314214142151421614217142181421914220142211422214223142241422514226142271422814229142301423114232142331423414235142361423714238142391424014241142421424314244142451424614247142481424914250142511425214253142541425514256142571425814259142601426114262142631426414265142661426714268142691427014271142721427314274142751427614277142781427914280142811428214283142841428514286142871428814289142901429114292142931429414295142961429714298142991430014301143021430314304143051430614307143081430914310143111431214313143141431514316143171431814319143201432114322143231432414325143261432714328143291433014331143321433314334143351433614337143381433914340143411434214343143441434514346143471434814349143501435114352143531435414355143561435714358143591436014361143621436314364143651436614367143681436914370143711437214373143741437514376143771437814379143801438114382143831438414385143861438714388143891439014391143921439314394143951439614397143981439914400144011440214403144041440514406144071440814409144101441114412144131441414415144161441714418144191442014421144221442314424144251442614427144281442914430144311443214433144341443514436144371443814439144401444114442144431444414445144461444714448144491445014451144521445314454144551445614457144581445914460144611446214463144641446514466144671446814469144701447114472144731447414475144761447714478144791448014481144821448314484144851448614487144881448914490144911449214493144941449514496144971449814499145001450114502145031450414505145061450714508145091451014511145121451314514145151451614517145181451914520145211452214523145241452514526145271452814529145301453114532145331453414535145361453714538145391454014541145421454314544145451454614547145481454914550145511455214553145541455514556145571455814559145601456114562145631456414565145661456714568145691457014571145721457314574145751457614577145781457914580145811458214583145841458514586145871458814589145901459114592145931459414595145961459714598145991460014601146021460314604146051460614607146081460914610146111461214613146141461514616146171461814619146201462114622146231462414625146261462714628146291463014631146321463314634146351463614637146381463914640146411464214643146441464514646146471464814649146501465114652146531465414655146561465714658146591466014661146621466314664146651466614667146681466914670146711467214673146741467514676146771467814679146801468114682146831468414685146861468714688146891469014691146921469314694146951469614697146981469914700147011470214703147041470514706147071470814709147101471114712147131471414715147161471714718147191472014721147221472314724147251472614727147281472914730147311473214733147341473514736147371473814739147401474114742147431474414745147461474714748147491475014751147521475314754147551475614757147581475914760147611476214763147641476514766147671476814769147701477114772147731477414775147761477714778147791478014781147821478314784147851478614787147881478914790147911479214793147941479514796147971479814799148001480114802148031480414805148061480714808148091481014811148121481314814148151481614817148181481914820148211482214823148241482514826148271482814829148301483114832148331483414835148361483714838148391484014841148421484314844148451484614847148481484914850148511485214853148541485514856148571485814859148601486114862148631486414865148661486714868148691487014871148721487314874148751487614877148781487914880148811488214883148841488514886148871488814889148901489114892148931489414895148961489714898148991490014901149021490314904149051490614907149081490914910149111491214913149141491514916149171491814919149201492114922149231492414925149261492714928149291493014931149321493314934149351493614937149381493914940149411494214943149441494514946149471494814949149501495114952149531495414955149561495714958149591496014961149621496314964149651496614967149681496914970149711497214973149741497514976149771497814979149801498114982149831498414985149861498714988149891499014991149921499314994149951499614997149981499915000150011500215003150041500515006150071500815009150101501115012150131501415015150161501715018150191502015021150221502315024150251502615027150281502915030150311503215033150341503515036150371503815039150401504115042150431504415045150461504715048150491505015051150521505315054150551505615057150581505915060150611506215063150641506515066150671506815069150701507115072150731507415075150761507715078150791508015081150821508315084150851508615087150881508915090150911509215093150941509515096150971509815099151001510115102151031510415105151061510715108151091511015111151121511315114151151511615117151181511915120151211512215123151241512515126151271512815129151301513115132151331513415135151361513715138151391514015141151421514315144151451514615147151481514915150151511515215153151541515515156151571515815159151601516115162151631516415165151661516715168151691517015171151721517315174151751517615177151781517915180151811518215183151841518515186151871518815189151901519115192151931519415195151961519715198151991520015201152021520315204152051520615207152081520915210152111521215213152141521515216152171521815219152201522115222152231522415225152261522715228152291523015231152321523315234152351523615237152381523915240152411524215243152441524515246152471524815249152501525115252152531525415255152561525715258152591526015261152621526315264152651526615267152681526915270152711527215273152741527515276152771527815279152801528115282152831528415285152861528715288152891529015291152921529315294152951529615297152981529915300153011530215303153041530515306153071530815309153101531115312153131531415315153161531715318153191532015321153221532315324153251532615327153281532915330153311533215333153341533515336153371533815339153401534115342153431534415345153461534715348153491535015351153521535315354153551535615357153581535915360153611536215363153641536515366153671536815369153701537115372153731537415375153761537715378153791538015381153821538315384153851538615387153881538915390153911539215393153941539515396153971539815399154001540115402154031540415405154061540715408154091541015411154121541315414154151541615417154181541915420154211542215423154241542515426154271542815429154301543115432154331543415435154361543715438154391544015441154421544315444154451544615447154481544915450154511545215453154541545515456154571545815459154601546115462154631546415465154661546715468154691547015471154721547315474154751547615477154781547915480154811548215483154841548515486154871548815489154901549115492154931549415495154961549715498154991550015501155021550315504155051550615507155081550915510155111551215513155141551515516155171551815519155201552115522155231552415525155261552715528155291553015531155321553315534155351553615537155381553915540155411554215543155441554515546155471554815549155501555115552155531555415555155561555715558155591556015561155621556315564155651556615567155681556915570155711557215573155741557515576155771557815579155801558115582155831558415585155861558715588155891559015591155921559315594155951559615597155981559915600156011560215603156041560515606156071560815609156101561115612156131561415615156161561715618156191562015621156221562315624156251562615627156281562915630156311563215633156341563515636156371563815639156401564115642156431564415645156461564715648156491565015651156521565315654156551565615657156581565915660156611566215663156641566515666156671566815669156701567115672156731567415675156761567715678156791568015681156821568315684156851568615687156881568915690156911569215693156941569515696156971569815699157001570115702157031570415705157061570715708157091571015711157121571315714157151571615717157181571915720157211572215723157241572515726157271572815729157301573115732157331573415735157361573715738157391574015741157421574315744157451574615747157481574915750157511575215753157541575515756157571575815759157601576115762157631576415765157661576715768157691577015771157721577315774157751577615777157781577915780157811578215783157841578515786157871578815789157901579115792157931579415795157961579715798157991580015801158021580315804158051580615807158081580915810158111581215813158141581515816158171581815819158201582115822158231582415825158261582715828158291583015831158321583315834158351583615837158381583915840158411584215843158441584515846158471584815849158501585115852158531585415855158561585715858158591586015861158621586315864158651586615867158681586915870158711587215873158741587515876158771587815879158801588115882158831588415885158861588715888158891589015891158921589315894158951589615897158981589915900159011590215903159041590515906159071590815909159101591115912159131591415915159161591715918159191592015921159221592315924159251592615927159281592915930159311593215933159341593515936159371593815939159401594115942159431594415945159461594715948159491595015951159521595315954159551595615957159581595915960159611596215963159641596515966159671596815969159701597115972159731597415975159761597715978159791598015981159821598315984159851598615987159881598915990159911599215993159941599515996159971599815999160001600116002160031600416005160061600716008160091601016011160121601316014160151601616017160181601916020160211602216023160241602516026160271602816029160301603116032160331603416035160361603716038160391604016041160421604316044160451604616047160481604916050160511605216053160541605516056160571605816059160601606116062160631606416065160661606716068160691607016071160721607316074160751607616077160781607916080160811608216083160841608516086160871608816089160901609116092160931609416095160961609716098160991610016101161021610316104161051610616107161081610916110161111611216113161141611516116161171611816119161201612116122161231612416125161261612716128161291613016131161321613316134161351613616137161381613916140161411614216143161441614516146161471614816149161501615116152161531615416155161561615716158161591616016161161621616316164161651616616167161681616916170161711617216173161741617516176161771617816179161801618116182161831618416185161861618716188161891619016191161921619316194161951619616197161981619916200162011620216203162041620516206162071620816209162101621116212162131621416215162161621716218162191622016221162221622316224162251622616227162281622916230162311623216233162341623516236162371623816239162401624116242162431624416245162461624716248162491625016251162521625316254162551625616257162581625916260162611626216263162641626516266162671626816269162701627116272162731627416275162761627716278162791628016281162821628316284162851628616287162881628916290162911629216293162941629516296162971629816299163001630116302163031630416305163061630716308163091631016311163121631316314163151631616317163181631916320163211632216323163241632516326163271632816329163301633116332163331633416335163361633716338163391634016341163421634316344163451634616347163481634916350163511635216353163541635516356163571635816359163601636116362163631636416365163661636716368163691637016371163721637316374163751637616377163781637916380163811638216383163841638516386163871638816389163901639116392163931639416395163961639716398163991640016401164021640316404164051640616407164081640916410164111641216413164141641516416164171641816419164201642116422164231642416425164261642716428164291643016431164321643316434164351643616437164381643916440164411644216443164441644516446164471644816449164501645116452164531645416455164561645716458164591646016461164621646316464164651646616467164681646916470164711647216473164741647516476164771647816479164801648116482164831648416485164861648716488
  1. /******************************************************************************
  2. *
  3. * Copyright(c) 2007 - 2017 Realtek Corporation.
  4. *
  5. * This program is free software; you can redistribute it and/or modify it
  6. * under the terms of version 2 of the GNU General Public License as
  7. * published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful, but WITHOUT
  10. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  12. * more details.
  13. *
  14. *****************************************************************************/
  15. #define _RTW_MLME_EXT_C_
  16. #include <drv_types.h>
  17. #ifdef CONFIG_IOCTL_CFG80211
  18. #include <rtw_wifi_regd.h>
  19. #endif /* CONFIG_IOCTL_CFG80211 */
  20. #include <hal_data.h>
  21. struct mlme_handler mlme_sta_tbl[] = {
  22. {WIFI_ASSOCREQ, "OnAssocReq", &OnAssocReq},
  23. {WIFI_ASSOCRSP, "OnAssocRsp", &OnAssocRsp},
  24. {WIFI_REASSOCREQ, "OnReAssocReq", &OnAssocReq},
  25. {WIFI_REASSOCRSP, "OnReAssocRsp", &OnAssocRsp},
  26. {WIFI_PROBEREQ, "OnProbeReq", &OnProbeReq},
  27. {WIFI_PROBERSP, "OnProbeRsp", &OnProbeRsp},
  28. /*----------------------------------------------------------
  29. below 2 are reserved
  30. -----------------------------------------------------------*/
  31. {0, "DoReserved", &DoReserved},
  32. {0, "DoReserved", &DoReserved},
  33. {WIFI_BEACON, "OnBeacon", &OnBeacon},
  34. {WIFI_ATIM, "OnATIM", &OnAtim},
  35. {WIFI_DISASSOC, "OnDisassoc", &OnDisassoc},
  36. {WIFI_AUTH, "OnAuth", &OnAuthClient},
  37. {WIFI_DEAUTH, "OnDeAuth", &OnDeAuth},
  38. {WIFI_ACTION, "OnAction", &OnAction},
  39. {WIFI_ACTION_NOACK, "OnActionNoAck", &OnAction},
  40. };
  41. #ifdef _CONFIG_NATIVEAP_MLME_
  42. struct mlme_handler mlme_ap_tbl[] = {
  43. {WIFI_ASSOCREQ, "OnAssocReq", &OnAssocReq},
  44. {WIFI_ASSOCRSP, "OnAssocRsp", &OnAssocRsp},
  45. {WIFI_REASSOCREQ, "OnReAssocReq", &OnAssocReq},
  46. {WIFI_REASSOCRSP, "OnReAssocRsp", &OnAssocRsp},
  47. {WIFI_PROBEREQ, "OnProbeReq", &OnProbeReq},
  48. {WIFI_PROBERSP, "OnProbeRsp", &OnProbeRsp},
  49. /*----------------------------------------------------------
  50. below 2 are reserved
  51. -----------------------------------------------------------*/
  52. {0, "DoReserved", &DoReserved},
  53. {0, "DoReserved", &DoReserved},
  54. {WIFI_BEACON, "OnBeacon", &OnBeacon},
  55. {WIFI_ATIM, "OnATIM", &OnAtim},
  56. {WIFI_DISASSOC, "OnDisassoc", &OnDisassoc},
  57. {WIFI_AUTH, "OnAuth", &OnAuth},
  58. {WIFI_DEAUTH, "OnDeAuth", &OnDeAuth},
  59. {WIFI_ACTION, "OnAction", &OnAction},
  60. {WIFI_ACTION_NOACK, "OnActionNoAck", &OnAction},
  61. };
  62. #endif
  63. struct action_handler OnAction_tbl[] = {
  64. {RTW_WLAN_CATEGORY_SPECTRUM_MGMT, "ACTION_SPECTRUM_MGMT", on_action_spct},
  65. {RTW_WLAN_CATEGORY_QOS, "ACTION_QOS", &OnAction_qos},
  66. {RTW_WLAN_CATEGORY_DLS, "ACTION_DLS", &OnAction_dls},
  67. {RTW_WLAN_CATEGORY_BACK, "ACTION_BACK", &OnAction_back},
  68. {RTW_WLAN_CATEGORY_PUBLIC, "ACTION_PUBLIC", on_action_public},
  69. {RTW_WLAN_CATEGORY_RADIO_MEAS, "ACTION_RADIO_MEAS", &on_action_rm},
  70. {RTW_WLAN_CATEGORY_FT, "ACTION_FT", &OnAction_ft},
  71. {RTW_WLAN_CATEGORY_HT, "ACTION_HT", &OnAction_ht},
  72. #ifdef CONFIG_IEEE80211W
  73. {RTW_WLAN_CATEGORY_SA_QUERY, "ACTION_SA_QUERY", &OnAction_sa_query},
  74. #else
  75. {RTW_WLAN_CATEGORY_SA_QUERY, "ACTION_SA_QUERY", &DoReserved},
  76. #endif /* CONFIG_IEEE80211W */
  77. #ifdef CONFIG_RTW_WNM
  78. {RTW_WLAN_CATEGORY_WNM, "ACTION_WNM", &on_action_wnm},
  79. #endif
  80. {RTW_WLAN_CATEGORY_UNPROTECTED_WNM, "ACTION_UNPROTECTED_WNM", &DoReserved},
  81. #ifdef CONFIG_RTW_MESH
  82. {RTW_WLAN_CATEGORY_MESH, "ACTION_MESH", &on_action_mesh},
  83. {RTW_WLAN_CATEGORY_SELF_PROTECTED, "ACTION_SELF_PROTECTED", &on_action_self_protected},
  84. #endif
  85. {RTW_WLAN_CATEGORY_WMM, "ACTION_WMM", &OnAction_wmm},
  86. {RTW_WLAN_CATEGORY_VHT, "ACTION_VHT", &OnAction_vht},
  87. {RTW_WLAN_CATEGORY_P2P, "ACTION_P2P", &OnAction_p2p},
  88. };
  89. u8 null_addr[ETH_ALEN] = {0, 0, 0, 0, 0, 0};
  90. /**************************************************
  91. OUI definitions for the vendor specific IE
  92. ***************************************************/
  93. unsigned char RTW_WPA_OUI[] = {0x00, 0x50, 0xf2, 0x01};
  94. unsigned char WMM_OUI[] = {0x00, 0x50, 0xf2, 0x02};
  95. unsigned char WPS_OUI[] = {0x00, 0x50, 0xf2, 0x04};
  96. unsigned char P2P_OUI[] = {0x50, 0x6F, 0x9A, 0x09};
  97. unsigned char WFD_OUI[] = {0x50, 0x6F, 0x9A, 0x0A};
  98. unsigned char WMM_INFO_OUI[] = {0x00, 0x50, 0xf2, 0x02, 0x00, 0x01};
  99. unsigned char WMM_PARA_OUI[] = {0x00, 0x50, 0xf2, 0x02, 0x01, 0x01};
  100. unsigned char WPA_TKIP_CIPHER[4] = {0x00, 0x50, 0xf2, 0x02};
  101. unsigned char RSN_TKIP_CIPHER[4] = {0x00, 0x0f, 0xac, 0x02};
  102. extern unsigned char REALTEK_96B_IE[];
  103. static void init_channel_list(_adapter *padapter, RT_CHANNEL_INFO *channel_set
  104. , struct p2p_channels *channel_list)
  105. {
  106. struct registry_priv *regsty = adapter_to_regsty(padapter);
  107. struct p2p_oper_class_map op_class[] = {
  108. { IEEE80211G, 81, 1, 13, 1, BW20 },
  109. { IEEE80211G, 82, 14, 14, 1, BW20 },
  110. #if 0 /* Do not enable HT40 on 2 GHz */
  111. { IEEE80211G, 83, 1, 9, 1, BW40PLUS },
  112. { IEEE80211G, 84, 5, 13, 1, BW40MINUS },
  113. #endif
  114. { IEEE80211A, 115, 36, 48, 4, BW20 },
  115. { IEEE80211A, 116, 36, 44, 8, BW40PLUS },
  116. { IEEE80211A, 117, 40, 48, 8, BW40MINUS },
  117. { IEEE80211A, 124, 149, 161, 4, BW20 },
  118. { IEEE80211A, 125, 149, 169, 4, BW20 },
  119. { IEEE80211A, 126, 149, 157, 8, BW40PLUS },
  120. { IEEE80211A, 127, 153, 161, 8, BW40MINUS },
  121. { -1, 0, 0, 0, 0, BW20 }
  122. };
  123. int cla, op;
  124. cla = 0;
  125. for (op = 0; op_class[op].op_class; op++) {
  126. u8 ch;
  127. struct p2p_oper_class_map *o = &op_class[op];
  128. struct p2p_reg_class *reg = NULL;
  129. for (ch = o->min_chan; ch <= o->max_chan; ch += o->inc) {
  130. if (rtw_chset_search_ch(channel_set, ch) == -1)
  131. continue;
  132. #if defined(CONFIG_80211N_HT) || defined(CONFIG_80211AC_VHT)
  133. if ((padapter->registrypriv.ht_enable == 0) && (o->inc == 8))
  134. continue;
  135. if ((REGSTY_IS_BW_5G_SUPPORT(regsty, CHANNEL_WIDTH_40)) &&
  136. ((o->bw == BW40MINUS) || (o->bw == BW40PLUS)))
  137. continue;
  138. #endif
  139. if (reg == NULL) {
  140. reg = &channel_list->reg_class[cla];
  141. cla++;
  142. reg->reg_class = o->op_class;
  143. reg->channels = 0;
  144. }
  145. reg->channel[reg->channels] = ch;
  146. reg->channels++;
  147. }
  148. }
  149. channel_list->reg_classes = cla;
  150. }
  151. #ifdef CONFIG_TXPWR_LIMIT
  152. void rtw_txpwr_init_regd(struct rf_ctl_t *rfctl)
  153. {
  154. u8 regd;
  155. struct regd_exc_ent *exc;
  156. struct txpwr_lmt_ent *ent;
  157. _irqL irqL;
  158. _enter_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
  159. rfctl->regd_name = NULL;
  160. if (rfctl->txpwr_regd_num == 0) {
  161. RTW_PRINT("there is no any txpwr_regd\n");
  162. goto release_lock;
  163. }
  164. /* search from exception mapping */
  165. exc = _rtw_regd_exc_search(rfctl
  166. , rfctl->country_ent ? rfctl->country_ent->alpha2 : NULL
  167. , rfctl->ChannelPlan);
  168. if (exc) {
  169. u8 has_country = (exc->country[0] == '\0' && exc->country[1] == '\0') ? 0 : 1;
  170. if (strcmp(exc->regd_name, regd_str(TXPWR_LMT_NONE)) == 0)
  171. rfctl->regd_name = regd_str(TXPWR_LMT_NONE);
  172. else if (strcmp(exc->regd_name, regd_str(TXPWR_LMT_WW)) == 0)
  173. rfctl->regd_name = regd_str(TXPWR_LMT_WW);
  174. else {
  175. ent = _rtw_txpwr_lmt_get_by_name(rfctl, exc->regd_name);
  176. if (ent)
  177. rfctl->regd_name = ent->regd_name;
  178. }
  179. RTW_PRINT("exception mapping country:%c%c domain:0x%02x to%s regd_name:%s\n"
  180. , has_country ? exc->country[0] : '0'
  181. , has_country ? exc->country[1] : '0'
  182. , exc->domain
  183. , rfctl->regd_name ? "" : " unknown"
  184. , exc->regd_name
  185. );
  186. if (rfctl->regd_name)
  187. goto release_lock;
  188. }
  189. /* follow default channel plan mapping */
  190. regd = rtw_chplan_get_default_regd(rfctl->ChannelPlan);
  191. if (regd == TXPWR_LMT_NONE)
  192. rfctl->regd_name = regd_str(TXPWR_LMT_NONE);
  193. else if (regd == TXPWR_LMT_WW)
  194. rfctl->regd_name = regd_str(TXPWR_LMT_WW);
  195. else {
  196. ent = _rtw_txpwr_lmt_get_by_name(rfctl, regd_str(regd));
  197. if (ent)
  198. rfctl->regd_name = ent->regd_name;
  199. }
  200. RTW_PRINT("default mapping domain:0x%02x to%s regd_name:%s\n"
  201. , rfctl->ChannelPlan
  202. , rfctl->regd_name ? "" : " unknown"
  203. , regd_str(regd)
  204. );
  205. if (rfctl->regd_name)
  206. goto release_lock;
  207. switch (regd) {
  208. /*
  209. * To support older chips without new predefined regd:
  210. * - use FCC if IC or CHILE not found
  211. * - use ETSI if KCC or ACMA not found
  212. */
  213. case TXPWR_LMT_IC:
  214. case TXPWR_LMT_KCC:
  215. case TXPWR_LMT_ACMA:
  216. case TXPWR_LMT_CHILE:
  217. if (regd == TXPWR_LMT_IC || regd == TXPWR_LMT_CHILE)
  218. regd = TXPWR_LMT_FCC;
  219. else if (regd == TXPWR_LMT_KCC || regd == TXPWR_LMT_ACMA)
  220. regd = TXPWR_LMT_ETSI;
  221. ent = _rtw_txpwr_lmt_get_by_name(rfctl, regd_str(regd));
  222. if (ent)
  223. rfctl->regd_name = ent->regd_name;
  224. RTW_PRINT("alternate regd_name:%s %s\n"
  225. , regd_str(regd)
  226. , rfctl->regd_name ? "is used" : "not found"
  227. );
  228. if (rfctl->regd_name)
  229. break;
  230. __attribute__((__fallthrough__));
  231. default:
  232. rfctl->regd_name = regd_str(TXPWR_LMT_WW);
  233. RTW_PRINT("assign %s for default case\n", regd_str(TXPWR_LMT_WW));
  234. break;
  235. };
  236. release_lock:
  237. _exit_critical_mutex(&rfctl->txpwr_lmt_mutex, &irqL);
  238. }
  239. #endif /* CONFIG_TXPWR_LIMIT */
  240. void rtw_rfctl_init(_adapter *adapter)
  241. {
  242. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  243. rfctl->max_chan_nums = init_channel_set(adapter, rfctl->ChannelPlan, rfctl->channel_set);
  244. init_channel_list(adapter, rfctl->channel_set, &rfctl->channel_list);
  245. _rtw_mutex_init(&rfctl->offch_mutex);
  246. #ifdef CONFIG_TXPWR_LIMIT
  247. _rtw_mutex_init(&rfctl->txpwr_lmt_mutex);
  248. _rtw_init_listhead(&rfctl->reg_exc_list);
  249. _rtw_init_listhead(&rfctl->txpwr_lmt_list);
  250. #endif
  251. rfctl->ch_sel_same_band_prefer = 1;
  252. #ifdef CONFIG_DFS_MASTER
  253. rfctl->cac_start_time = rfctl->cac_end_time = RTW_CAC_STOPPED;
  254. rtw_init_timer(&(rfctl->radar_detect_timer), adapter, rtw_dfs_rd_timer_hdl, rfctl);
  255. #endif
  256. #ifdef CONFIG_DFS_SLAVE_WITH_RADAR_DETECT
  257. rfctl->dfs_slave_with_rd = 1;
  258. #endif
  259. }
  260. void rtw_rfctl_deinit(_adapter *adapter)
  261. {
  262. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  263. _rtw_mutex_free(&rfctl->offch_mutex);
  264. #ifdef CONFIG_TXPWR_LIMIT
  265. rtw_regd_exc_list_free(rfctl);
  266. rtw_txpwr_lmt_list_free(rfctl);
  267. _rtw_mutex_free(&rfctl->txpwr_lmt_mutex);
  268. #endif
  269. }
  270. #ifdef CONFIG_DFS_MASTER
  271. /*
  272. * called in rtw_dfs_rd_enable()
  273. * assume the request channel coverage is DFS range
  274. * base on the current status and the request channel coverage to check if need to reset complete CAC time
  275. */
  276. bool rtw_is_cac_reset_needed(struct rf_ctl_t *rfctl, u8 ch, u8 bw, u8 offset)
  277. {
  278. bool needed = _FALSE;
  279. u32 cur_hi, cur_lo, hi, lo;
  280. if (rfctl->radar_detected == 1) {
  281. needed = _TRUE;
  282. goto exit;
  283. }
  284. if (rfctl->radar_detect_ch == 0) {
  285. needed = _TRUE;
  286. goto exit;
  287. }
  288. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE) {
  289. RTW_ERR("request detection range ch:%u, bw:%u, offset:%u\n", ch, bw, offset);
  290. rtw_warn_on(1);
  291. }
  292. if (rtw_chbw_to_freq_range(rfctl->radar_detect_ch, rfctl->radar_detect_bw, rfctl->radar_detect_offset, &cur_hi, &cur_lo) == _FALSE) {
  293. RTW_ERR("cur detection range ch:%u, bw:%u, offset:%u\n", rfctl->radar_detect_ch, rfctl->radar_detect_bw, rfctl->radar_detect_offset);
  294. rtw_warn_on(1);
  295. }
  296. if (hi <= lo || cur_hi <= cur_lo) {
  297. RTW_ERR("hi:%u, lo:%u, cur_hi:%u, cur_lo:%u\n", hi, lo, cur_hi, cur_lo);
  298. rtw_warn_on(1);
  299. }
  300. if (rtw_is_range_a_in_b(hi, lo, cur_hi, cur_lo)) {
  301. /* request is in current detect range */
  302. goto exit;
  303. }
  304. /* check if request channel coverage has new range and the new range is in DFS range */
  305. if (!rtw_is_range_overlap(hi, lo, cur_hi, cur_lo)) {
  306. /* request has no overlap with current */
  307. needed = _TRUE;
  308. } else if (rtw_is_range_a_in_b(cur_hi, cur_lo, hi, lo)) {
  309. /* request is supper set of current */
  310. if ((hi != cur_hi && rtw_is_dfs_range(hi, cur_hi)) || (lo != cur_lo && rtw_is_dfs_range(cur_lo, lo)))
  311. needed = _TRUE;
  312. } else {
  313. /* request is not supper set of current, but has overlap */
  314. if ((lo < cur_lo && rtw_is_dfs_range(cur_lo, lo)) || (hi > cur_hi && rtw_is_dfs_range(hi, cur_hi)))
  315. needed = _TRUE;
  316. }
  317. exit:
  318. return needed;
  319. }
  320. bool _rtw_rfctl_overlap_radar_detect_ch(struct rf_ctl_t *rfctl, u8 ch, u8 bw, u8 offset)
  321. {
  322. bool ret = _FALSE;
  323. u32 hi = 0, lo = 0;
  324. u32 r_hi = 0, r_lo = 0;
  325. int i;
  326. if (rfctl->radar_detect_by_others)
  327. goto exit;
  328. if (rfctl->radar_detect_ch == 0)
  329. goto exit;
  330. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE) {
  331. rtw_warn_on(1);
  332. goto exit;
  333. }
  334. if (rtw_chbw_to_freq_range(rfctl->radar_detect_ch
  335. , rfctl->radar_detect_bw, rfctl->radar_detect_offset
  336. , &r_hi, &r_lo) == _FALSE) {
  337. rtw_warn_on(1);
  338. goto exit;
  339. }
  340. if (rtw_is_range_overlap(hi, lo, r_hi, r_lo))
  341. ret = _TRUE;
  342. exit:
  343. return ret;
  344. }
  345. bool rtw_rfctl_overlap_radar_detect_ch(struct rf_ctl_t *rfctl)
  346. {
  347. return _rtw_rfctl_overlap_radar_detect_ch(rfctl
  348. , rfctl_to_dvobj(rfctl)->oper_channel
  349. , rfctl_to_dvobj(rfctl)->oper_bwmode
  350. , rfctl_to_dvobj(rfctl)->oper_ch_offset);
  351. }
  352. bool rtw_rfctl_is_tx_blocked_by_ch_waiting(struct rf_ctl_t *rfctl)
  353. {
  354. return rtw_rfctl_overlap_radar_detect_ch(rfctl) && IS_CH_WAITING(rfctl);
  355. }
  356. bool rtw_chset_is_chbw_non_ocp(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset)
  357. {
  358. bool ret = _FALSE;
  359. u32 hi = 0, lo = 0;
  360. int i;
  361. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE)
  362. goto exit;
  363. for (i = 0; i < MAX_CHANNEL_NUM && ch_set[i].ChannelNum != 0; i++) {
  364. if (!rtw_ch2freq(ch_set[i].ChannelNum)) {
  365. rtw_warn_on(1);
  366. continue;
  367. }
  368. if (!CH_IS_NON_OCP(&ch_set[i]))
  369. continue;
  370. if (lo <= rtw_ch2freq(ch_set[i].ChannelNum)
  371. && rtw_ch2freq(ch_set[i].ChannelNum) <= hi
  372. ) {
  373. ret = _TRUE;
  374. break;
  375. }
  376. }
  377. exit:
  378. return ret;
  379. }
  380. bool rtw_chset_is_ch_non_ocp(RT_CHANNEL_INFO *ch_set, u8 ch)
  381. {
  382. return rtw_chset_is_chbw_non_ocp(ch_set, ch, CHANNEL_WIDTH_20, HAL_PRIME_CHNL_OFFSET_DONT_CARE);
  383. }
  384. u32 rtw_chset_get_ch_non_ocp_ms(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset)
  385. {
  386. int ms = 0;
  387. systime current_time;
  388. u32 hi = 0, lo = 0;
  389. int i;
  390. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE)
  391. goto exit;
  392. current_time = rtw_get_current_time();
  393. for (i = 0; i < MAX_CHANNEL_NUM && ch_set[i].ChannelNum != 0; i++) {
  394. if (!rtw_ch2freq(ch_set[i].ChannelNum)) {
  395. rtw_warn_on(1);
  396. continue;
  397. }
  398. if (!CH_IS_NON_OCP(&ch_set[i]))
  399. continue;
  400. if (lo <= rtw_ch2freq(ch_set[i].ChannelNum)
  401. && rtw_ch2freq(ch_set[i].ChannelNum) <= hi
  402. ) {
  403. if (rtw_systime_to_ms(ch_set[i].non_ocp_end_time - current_time) > ms)
  404. ms = rtw_systime_to_ms(ch_set[i].non_ocp_end_time - current_time);
  405. }
  406. }
  407. exit:
  408. return ms;
  409. }
  410. /**
  411. * rtw_chset_update_non_ocp - update non_ocp_end_time according to the given @ch, @bw, @offset into @ch_set
  412. * @ch_set: the given channel set
  413. * @ch: channel number on which radar is detected
  414. * @bw: bandwidth on which radar is detected
  415. * @offset: bandwidth offset on which radar is detected
  416. * @ms: ms to add from now to update non_ocp_end_time, ms < 0 means use NON_OCP_TIME_MS
  417. */
  418. static void _rtw_chset_update_non_ocp(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset, int ms)
  419. {
  420. u32 hi = 0, lo = 0;
  421. int i;
  422. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE)
  423. goto exit;
  424. for (i = 0; i < MAX_CHANNEL_NUM && ch_set[i].ChannelNum != 0; i++) {
  425. if (!rtw_ch2freq(ch_set[i].ChannelNum)) {
  426. rtw_warn_on(1);
  427. continue;
  428. }
  429. if (lo <= rtw_ch2freq(ch_set[i].ChannelNum)
  430. && rtw_ch2freq(ch_set[i].ChannelNum) <= hi
  431. ) {
  432. if (ms >= 0)
  433. ch_set[i].non_ocp_end_time = rtw_get_current_time() + rtw_ms_to_systime(ms);
  434. else
  435. ch_set[i].non_ocp_end_time = rtw_get_current_time() + rtw_ms_to_systime(NON_OCP_TIME_MS);
  436. }
  437. }
  438. exit:
  439. return;
  440. }
  441. inline void rtw_chset_update_non_ocp(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset)
  442. {
  443. _rtw_chset_update_non_ocp(ch_set, ch, bw, offset, -1);
  444. }
  445. inline void rtw_chset_update_non_ocp_ms(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset, int ms)
  446. {
  447. _rtw_chset_update_non_ocp(ch_set, ch, bw, offset, ms);
  448. }
  449. u32 rtw_get_ch_waiting_ms(struct rf_ctl_t *rfctl, u8 ch, u8 bw, u8 offset, u32 *r_non_ocp_ms, u32 *r_cac_ms)
  450. {
  451. struct dvobj_priv *dvobj = rfctl_to_dvobj(rfctl);
  452. u32 non_ocp_ms;
  453. u32 cac_ms;
  454. u8 in_rd_range = 0; /* if in current radar detection range*/
  455. if (rtw_chset_is_chbw_non_ocp(rfctl->channel_set, ch, bw, offset))
  456. non_ocp_ms = rtw_chset_get_ch_non_ocp_ms(rfctl->channel_set, ch, bw, offset);
  457. else
  458. non_ocp_ms = 0;
  459. if (rfctl->radar_detect_enabled) {
  460. u32 cur_hi, cur_lo, hi, lo;
  461. if (rtw_chbw_to_freq_range(ch, bw, offset, &hi, &lo) == _FALSE) {
  462. RTW_ERR("input range ch:%u, bw:%u, offset:%u\n", ch, bw, offset);
  463. rtw_warn_on(1);
  464. }
  465. if (rtw_chbw_to_freq_range(rfctl->radar_detect_ch, rfctl->radar_detect_bw, rfctl->radar_detect_offset, &cur_hi, &cur_lo) == _FALSE) {
  466. RTW_ERR("cur detection range ch:%u, bw:%u, offset:%u\n", rfctl->radar_detect_ch, rfctl->radar_detect_bw, rfctl->radar_detect_offset);
  467. rtw_warn_on(1);
  468. }
  469. if (rtw_is_range_a_in_b(hi, lo, cur_hi, cur_lo))
  470. in_rd_range = 1;
  471. }
  472. if (!rtw_is_dfs_chbw(ch, bw, offset))
  473. cac_ms = 0;
  474. else if (in_rd_range && !non_ocp_ms) {
  475. if (IS_CH_WAITING(rfctl))
  476. cac_ms = rtw_systime_to_ms(rfctl->cac_end_time - rtw_get_current_time());
  477. else
  478. cac_ms = 0;
  479. } else if (rtw_is_long_cac_ch(ch, bw, offset, rtw_odm_get_dfs_domain(dvobj)))
  480. cac_ms = CAC_TIME_CE_MS;
  481. else
  482. cac_ms = CAC_TIME_MS;
  483. if (r_non_ocp_ms)
  484. *r_non_ocp_ms = non_ocp_ms;
  485. if (r_cac_ms)
  486. *r_cac_ms = cac_ms;
  487. return non_ocp_ms + cac_ms;
  488. }
  489. void rtw_reset_cac(struct rf_ctl_t *rfctl, u8 ch, u8 bw, u8 offset)
  490. {
  491. u32 non_ocp_ms;
  492. u32 cac_ms;
  493. rtw_get_ch_waiting_ms(rfctl
  494. , ch
  495. , bw
  496. , offset
  497. , &non_ocp_ms
  498. , &cac_ms
  499. );
  500. rfctl->cac_start_time = rtw_get_current_time() + rtw_ms_to_systime(non_ocp_ms);
  501. rfctl->cac_end_time = rfctl->cac_start_time + rtw_ms_to_systime(cac_ms);
  502. /* skip special value */
  503. if (rfctl->cac_start_time == RTW_CAC_STOPPED) {
  504. rfctl->cac_start_time++;
  505. rfctl->cac_end_time++;
  506. }
  507. if (rfctl->cac_end_time == RTW_CAC_STOPPED)
  508. rfctl->cac_end_time++;
  509. }
  510. u32 rtw_force_stop_cac(struct rf_ctl_t *rfctl, u32 timeout_ms)
  511. {
  512. struct dvobj_priv *dvobj = rfctl_to_dvobj(rfctl);
  513. systime start;
  514. u32 pass_ms;
  515. start = rtw_get_current_time();
  516. rfctl->cac_force_stop = 1;
  517. while (rtw_get_passing_time_ms(start) <= timeout_ms
  518. && IS_UNDER_CAC(rfctl)
  519. ) {
  520. if (dev_is_surprise_removed(dvobj) || dev_is_drv_stopped(dvobj))
  521. break;
  522. rtw_msleep_os(20);
  523. }
  524. if (IS_UNDER_CAC(rfctl)) {
  525. if (!dev_is_surprise_removed(dvobj) && !dev_is_drv_stopped(dvobj))
  526. RTW_INFO("%s waiting for cac stop timeout!\n", __func__);
  527. }
  528. rfctl->cac_force_stop = 0;
  529. pass_ms = rtw_get_passing_time_ms(start);
  530. return pass_ms;
  531. }
  532. #endif /* CONFIG_DFS_MASTER */
  533. /* choose channel with shortest waiting (non ocp + cac) time */
  534. bool rtw_choose_shortest_waiting_ch(struct rf_ctl_t *rfctl, u8 sel_ch, u8 max_bw
  535. , u8 *dec_ch, u8 *dec_bw, u8 *dec_offset
  536. , u8 d_flags, u8 cur_ch, u8 same_band_prefer, u8 mesh_only)
  537. {
  538. #ifndef DBG_CHOOSE_SHORTEST_WAITING_CH
  539. #define DBG_CHOOSE_SHORTEST_WAITING_CH 0
  540. #endif
  541. struct dvobj_priv *dvobj = rfctl_to_dvobj(rfctl);
  542. struct registry_priv *regsty = dvobj_to_regsty(dvobj);
  543. u8 ch, bw, offset;
  544. u8 ch_c = 0, bw_c = 0, offset_c = 0;
  545. int i;
  546. u32 min_waiting_ms = 0;
  547. if (!dec_ch || !dec_bw || !dec_offset) {
  548. rtw_warn_on(1);
  549. return _FALSE;
  550. }
  551. /* full search and narrow bw judegement first to avoid potetial judegement timing issue */
  552. for (bw = CHANNEL_WIDTH_20; bw <= max_bw; bw++) {
  553. if (!hal_is_bw_support(dvobj_get_primary_adapter(dvobj), bw))
  554. continue;
  555. for (i = 0; i < rfctl->max_chan_nums; i++) {
  556. u32 non_ocp_ms = 0;
  557. u32 cac_ms = 0;
  558. u32 waiting_ms = 0;
  559. ch = rfctl->channel_set[i].ChannelNum;
  560. if (sel_ch > 0 && ch != sel_ch)
  561. continue;
  562. if ((d_flags & RTW_CHF_2G) && ch <= 14)
  563. continue;
  564. if ((d_flags & RTW_CHF_5G) && ch > 14)
  565. continue;
  566. if (ch > 14) {
  567. if (bw > REGSTY_BW_5G(regsty))
  568. continue;
  569. } else {
  570. if (bw > REGSTY_BW_2G(regsty))
  571. continue;
  572. }
  573. if (mesh_only && ch >= 5 && ch <= 9 && bw > CHANNEL_WIDTH_20)
  574. continue;
  575. if (!rtw_get_offset_by_chbw(ch, bw, &offset))
  576. continue;
  577. if (!rtw_chset_is_chbw_valid(rfctl->channel_set, ch, bw, offset))
  578. continue;
  579. if ((d_flags & RTW_CHF_NON_OCP) && rtw_chset_is_chbw_non_ocp(rfctl->channel_set, ch, bw, offset))
  580. continue;
  581. if ((d_flags & RTW_CHF_DFS) && rtw_is_dfs_chbw(ch, bw, offset))
  582. continue;
  583. if ((d_flags & RTW_CHF_LONG_CAC) && rtw_is_long_cac_ch(ch, bw, offset, rtw_odm_get_dfs_domain(dvobj)))
  584. continue;
  585. if ((d_flags & RTW_CHF_NON_DFS) && !rtw_is_dfs_chbw(ch, bw, offset))
  586. continue;
  587. if ((d_flags & RTW_CHF_NON_LONG_CAC) && !rtw_is_long_cac_ch(ch, bw, offset, rtw_odm_get_dfs_domain(dvobj)))
  588. continue;
  589. #ifdef CONFIG_DFS_MASTER
  590. waiting_ms = rtw_get_ch_waiting_ms(rfctl, ch, bw, offset, &non_ocp_ms, &cac_ms);
  591. #endif
  592. if (DBG_CHOOSE_SHORTEST_WAITING_CH)
  593. RTW_INFO("%s:%u,%u,%u %u(non_ocp:%u, cac:%u)\n"
  594. , __func__, ch, bw, offset, waiting_ms, non_ocp_ms, cac_ms);
  595. if (ch_c == 0
  596. /* first: smaller wating time */
  597. || min_waiting_ms > waiting_ms
  598. /* then: wider bw */
  599. || (min_waiting_ms == waiting_ms && bw > bw_c)
  600. /* then: same band if requested */
  601. || (same_band_prefer && min_waiting_ms == waiting_ms && bw == bw_c
  602. && !rtw_is_same_band(cur_ch, ch_c) && rtw_is_same_band(cur_ch, ch))
  603. ) {
  604. ch_c = ch;
  605. bw_c = bw;
  606. offset_c = offset;
  607. min_waiting_ms = waiting_ms;
  608. }
  609. }
  610. }
  611. if (ch_c != 0) {
  612. RTW_INFO("%s: d_flags:0x%02x cur_ch:%u sb_prefer:%u%s %u,%u,%u waiting_ms:%u\n"
  613. , __func__, d_flags, cur_ch, same_band_prefer
  614. , mesh_only ? " mesh_only" : ""
  615. , ch_c, bw_c, offset_c, min_waiting_ms);
  616. *dec_ch = ch_c;
  617. *dec_bw = bw_c;
  618. *dec_offset = offset_c;
  619. return _TRUE;
  620. }
  621. if (d_flags == 0) {
  622. RTW_INFO("%s: sel_ch:%u max_bw:%u d_flags:0x%02x cur_ch:%u sb_prefer:%u%s\n"
  623. , __func__, sel_ch, max_bw, d_flags, cur_ch, same_band_prefer
  624. , mesh_only ? " mesh_only" : "");
  625. rtw_warn_on(1);
  626. }
  627. return _FALSE;
  628. }
  629. void dump_chset(void *sel, RT_CHANNEL_INFO *ch_set)
  630. {
  631. u8 i;
  632. for (i = 0; i < MAX_CHANNEL_NUM && ch_set[i].ChannelNum != 0; i++) {
  633. RTW_PRINT_SEL(sel, "ch:%3u, freq:%u, scan_type:%d"
  634. , ch_set[i].ChannelNum, rtw_ch2freq(ch_set[i].ChannelNum), ch_set[i].ScanType);
  635. #ifdef CONFIG_FIND_BEST_CHANNEL
  636. _RTW_PRINT_SEL(sel, ", rx_count:%u", ch_set[i].rx_count);
  637. #endif
  638. #ifdef CONFIG_DFS_MASTER
  639. if (rtw_is_dfs_ch(ch_set[i].ChannelNum)) {
  640. if (CH_IS_NON_OCP(&ch_set[i]))
  641. _RTW_PRINT_SEL(sel, ", non_ocp:%d"
  642. , rtw_systime_to_ms(ch_set[i].non_ocp_end_time - rtw_get_current_time()));
  643. else
  644. _RTW_PRINT_SEL(sel, ", non_ocp:N/A");
  645. }
  646. #endif
  647. _RTW_PRINT_SEL(sel, "\n");
  648. }
  649. RTW_PRINT_SEL(sel, "total ch number:%d\n", i);
  650. }
  651. void dump_cur_chset(void *sel, struct rf_ctl_t *rfctl)
  652. {
  653. struct dvobj_priv *dvobj = rfctl_to_dvobj(rfctl);
  654. struct registry_priv *regsty = dvobj_to_regsty(dvobj);
  655. int i;
  656. if (rfctl->country_ent)
  657. dump_country_chplan(sel, rfctl->country_ent);
  658. else
  659. RTW_PRINT_SEL(sel, "chplan:0x%02X\n", rfctl->ChannelPlan);
  660. #ifdef CONFIG_TXPWR_LIMIT
  661. RTW_PRINT_SEL(sel, "PLS regd:%s\n", rfctl->regd_name);
  662. #endif
  663. #ifdef CONFIG_DFS_MASTER
  664. RTW_PRINT_SEL(sel, "dfs_domain:%u\n", rtw_odm_get_dfs_domain(dvobj));
  665. #endif
  666. for (i = 0; i < MAX_CHANNEL_NUM; i++)
  667. if (regsty->excl_chs[i] != 0)
  668. break;
  669. if (i < MAX_CHANNEL_NUM) {
  670. RTW_PRINT_SEL(sel, "excl_chs:");
  671. for (i = 0; i < MAX_CHANNEL_NUM; i++) {
  672. if (regsty->excl_chs[i] == 0)
  673. break;
  674. _RTW_PRINT_SEL(sel, "%u ", regsty->excl_chs[i]);
  675. }
  676. _RTW_PRINT_SEL(sel, "\n");
  677. }
  678. dump_chset(sel, rfctl->channel_set);
  679. }
  680. /*
  681. * Search the @param ch in given @param ch_set
  682. * @ch_set: the given channel set
  683. * @ch: the given channel number
  684. *
  685. * return the index of channel_num in channel_set, -1 if not found
  686. */
  687. int rtw_chset_search_ch(RT_CHANNEL_INFO *ch_set, const u32 ch)
  688. {
  689. int i;
  690. if (ch == 0)
  691. return -1;
  692. for (i = 0; i < MAX_CHANNEL_NUM && ch_set[i].ChannelNum != 0; i++) {
  693. if (ch == ch_set[i].ChannelNum)
  694. return i;
  695. }
  696. return -1;
  697. }
  698. /*
  699. * Check if the @param ch, bw, offset is valid for the given @param ch_set
  700. * @ch_set: the given channel set
  701. * @ch: the given channel number
  702. * @bw: the given bandwidth
  703. * @offset: the given channel offset
  704. *
  705. * return valid (1) or not (0)
  706. */
  707. u8 rtw_chset_is_chbw_valid(RT_CHANNEL_INFO *ch_set, u8 ch, u8 bw, u8 offset)
  708. {
  709. u8 cch;
  710. u8 *op_chs;
  711. u8 op_ch_num;
  712. u8 valid = 0;
  713. int i;
  714. cch = rtw_get_center_ch(ch, bw, offset);
  715. if (!rtw_get_op_chs_by_cch_bw(cch, bw, &op_chs, &op_ch_num))
  716. goto exit;
  717. for (i = 0; i < op_ch_num; i++) {
  718. if (0)
  719. RTW_INFO("%u,%u,%u - cch:%u, bw:%u, op_ch:%u\n", ch, bw, offset, cch, bw, *(op_chs + i));
  720. if (rtw_chset_search_ch(ch_set, *(op_chs + i)) == -1)
  721. break;
  722. }
  723. if (op_ch_num != 0 && i == op_ch_num)
  724. valid = 1;
  725. exit:
  726. return valid;
  727. }
  728. /**
  729. * rtw_chset_sync_chbw - obey g_ch, adjust g_bw, g_offset, bw, offset to fit in channel plan
  730. * @ch_set: channel plan to check
  731. * @req_ch: pointer of the request ch, may be modified further
  732. * @req_bw: pointer of the request bw, may be modified further
  733. * @req_offset: pointer of the request offset, may be modified further
  734. * @g_ch: pointer of the ongoing group ch
  735. * @g_bw: pointer of the ongoing group bw, may be modified further
  736. * @g_offset: pointer of the ongoing group offset, may be modified further
  737. */
  738. void rtw_chset_sync_chbw(RT_CHANNEL_INFO *ch_set, u8 *req_ch, u8 *req_bw, u8 *req_offset
  739. , u8 *g_ch, u8 *g_bw, u8 *g_offset)
  740. {
  741. u8 r_ch, r_bw, r_offset;
  742. u8 u_ch, u_bw, u_offset;
  743. u8 cur_bw = *req_bw;
  744. while (1) {
  745. r_ch = *req_ch;
  746. r_bw = cur_bw;
  747. r_offset = *req_offset;
  748. u_ch = *g_ch;
  749. u_bw = *g_bw;
  750. u_offset = *g_offset;
  751. rtw_sync_chbw(&r_ch, &r_bw, &r_offset, &u_ch, &u_bw, &u_offset);
  752. if (rtw_chset_is_chbw_valid(ch_set, r_ch, r_bw, r_offset))
  753. break;
  754. if (cur_bw == CHANNEL_WIDTH_20) {
  755. rtw_warn_on(1);
  756. break;
  757. }
  758. cur_bw--;
  759. };
  760. *req_ch = r_ch;
  761. *req_bw = r_bw;
  762. *req_offset = r_offset;
  763. *g_ch = u_ch;
  764. *g_bw = u_bw;
  765. *g_offset = u_offset;
  766. }
  767. /*
  768. * Check the @param ch is fit with setband setting of @param adapter
  769. * @adapter: the given adapter
  770. * @ch: the given channel number
  771. *
  772. * return _TRUE when check valid, _FALSE not valid
  773. */
  774. bool rtw_mlme_band_check(_adapter *adapter, const u32 ch)
  775. {
  776. if (adapter->setband == WIFI_FREQUENCY_BAND_AUTO /* 2.4G and 5G */
  777. || (adapter->setband == WIFI_FREQUENCY_BAND_2GHZ && ch < 35) /* 2.4G only */
  778. || (adapter->setband == WIFI_FREQUENCY_BAND_5GHZ && ch > 35) /* 5G only */
  779. )
  780. return _TRUE;
  781. return _FALSE;
  782. }
  783. inline void RTW_SET_SCAN_BAND_SKIP(_adapter *padapter, int skip_band)
  784. {
  785. int bs = ATOMIC_READ(&padapter->bandskip);
  786. bs |= skip_band;
  787. ATOMIC_SET(&padapter->bandskip, bs);
  788. }
  789. inline void RTW_CLR_SCAN_BAND_SKIP(_adapter *padapter, int skip_band)
  790. {
  791. int bs = ATOMIC_READ(&padapter->bandskip);
  792. bs &= ~(skip_band);
  793. ATOMIC_SET(&padapter->bandskip, bs);
  794. }
  795. inline int RTW_GET_SCAN_BAND_SKIP(_adapter *padapter)
  796. {
  797. return ATOMIC_READ(&padapter->bandskip);
  798. }
  799. #define RTW_IS_SCAN_BAND_SKIP(padapter, skip_band) (ATOMIC_READ(&padapter->bandskip) & (skip_band))
  800. bool rtw_mlme_ignore_chan(_adapter *adapter, const u32 ch)
  801. {
  802. if (RTW_IS_SCAN_BAND_SKIP(adapter, BAND_24G) && ch < 35) /* SKIP 2.4G Band channel */
  803. return _TRUE;
  804. if (RTW_IS_SCAN_BAND_SKIP(adapter, BAND_5G) && ch > 35) /* SKIP 5G Band channel */
  805. return _TRUE;
  806. return _FALSE;
  807. }
  808. /****************************************************************************
  809. Following are the initialization functions for WiFi MLME
  810. *****************************************************************************/
  811. int init_hw_mlme_ext(_adapter *padapter)
  812. {
  813. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  814. HAL_DATA_TYPE *pHalData = GET_HAL_DATA(padapter);
  815. u8 rx_bar_enble = _TRUE;
  816. /*
  817. * Sync driver status and hardware setting
  818. */
  819. /* Modify to make sure first time change channel(band) would be done properly */
  820. pHalData->current_channel = 0;
  821. pHalData->current_channel_bw = CHANNEL_WIDTH_MAX;
  822. pHalData->current_band_type = BAND_MAX;
  823. /* set_opmode_cmd(padapter, infra_client_with_mlme); */ /* removed */
  824. rtw_hal_set_hwreg(padapter, HW_VAR_ENABLE_RX_BAR, &rx_bar_enble);
  825. set_channel_bwmode(padapter, pmlmeext->cur_channel, pmlmeext->cur_ch_offset, pmlmeext->cur_bwmode);
  826. return _SUCCESS;
  827. }
  828. void init_mlme_default_rate_set(_adapter *padapter)
  829. {
  830. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  831. unsigned char end_set[1] = {0xff};
  832. u8 offset_datarate = 0;
  833. u8 offset_basicrate = 0;
  834. #ifdef CONFIG_80211N_HT
  835. unsigned char supported_mcs_set[16] = {0xff, 0xff, 0xff, 0x00, 0x00, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0};
  836. #endif
  837. if (IsSupportedTxCCK(padapter->registrypriv.wireless_mode)) {
  838. unsigned char datarate_b[B_MODE_RATE_NUM] ={_1M_RATE_, _2M_RATE_, _5M_RATE_, _11M_RATE_};
  839. _rtw_memcpy(pmlmeext->datarate, datarate_b, B_MODE_RATE_NUM);
  840. _rtw_memcpy(pmlmeext->basicrate, datarate_b, B_MODE_RATE_NUM);
  841. offset_datarate += B_MODE_RATE_NUM;
  842. offset_basicrate += B_MODE_RATE_NUM;
  843. RTW_INFO("%s: support CCK\n", __func__);
  844. }
  845. if(IsSupportedTxOFDM(padapter->registrypriv.wireless_mode)) {
  846. unsigned char datarate_g[G_MODE_RATE_NUM] ={_6M_RATE_, _9M_RATE_, _12M_RATE_, _18M_RATE_,_24M_RATE_, _36M_RATE_, _48M_RATE_, _54M_RATE_};
  847. unsigned char basicrate_g[G_MODE_BASIC_RATE_NUM] = {_6M_RATE_, _12M_RATE_, _24M_RATE_};
  848. _rtw_memcpy(pmlmeext->datarate + offset_datarate, datarate_g, G_MODE_RATE_NUM);
  849. _rtw_memcpy(pmlmeext->basicrate + offset_basicrate,basicrate_g, G_MODE_BASIC_RATE_NUM);
  850. offset_datarate += G_MODE_RATE_NUM;
  851. offset_basicrate += G_MODE_BASIC_RATE_NUM;
  852. RTW_INFO("%s: support OFDM\n", __func__);
  853. }
  854. _rtw_memcpy(pmlmeext->datarate + offset_datarate, end_set, 1);
  855. _rtw_memcpy(pmlmeext->basicrate + offset_basicrate, end_set, 1);
  856. #ifdef CONFIG_80211N_HT
  857. if( padapter->registrypriv.ht_enable && is_supported_ht(padapter->registrypriv.wireless_mode))
  858. _rtw_memcpy(pmlmeext->default_supported_mcs_set, supported_mcs_set, sizeof(pmlmeext->default_supported_mcs_set));
  859. #endif
  860. }
  861. static void init_mlme_ext_priv_value(_adapter *padapter)
  862. {
  863. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  864. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  865. ATOMIC_SET(&pmlmeext->event_seq, 0);
  866. pmlmeext->mgnt_seq = 0;/* reset to zero when disconnect at client mode */
  867. #ifdef CONFIG_IEEE80211W
  868. pmlmeext->sa_query_seq = 0;
  869. #endif
  870. pmlmeext->cur_channel = padapter->registrypriv.channel;
  871. pmlmeext->cur_bwmode = CHANNEL_WIDTH_20;
  872. pmlmeext->cur_ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  873. pmlmeext->retry = 0;
  874. pmlmeext->cur_wireless_mode = padapter->registrypriv.wireless_mode;
  875. init_mlme_default_rate_set(padapter);
  876. if (pmlmeext->cur_channel > 14)
  877. pmlmeext->tx_rate = IEEE80211_OFDM_RATE_6MB;
  878. else
  879. pmlmeext->tx_rate = IEEE80211_CCK_RATE_1MB;
  880. mlmeext_set_scan_state(pmlmeext, SCAN_DISABLE);
  881. pmlmeext->sitesurvey_res.channel_idx = 0;
  882. pmlmeext->sitesurvey_res.bss_cnt = 0;
  883. pmlmeext->sitesurvey_res.scan_ch_ms = SURVEY_TO;
  884. pmlmeext->sitesurvey_res.rx_ampdu_accept = RX_AMPDU_ACCEPT_INVALID;
  885. pmlmeext->sitesurvey_res.rx_ampdu_size = RX_AMPDU_SIZE_INVALID;
  886. #ifdef CONFIG_SCAN_BACKOP
  887. mlmeext_assign_scan_backop_flags_sta(pmlmeext, /*SS_BACKOP_EN|*/SS_BACKOP_PS_ANNC | SS_BACKOP_TX_RESUME);
  888. #ifdef CONFIG_AP_MODE
  889. mlmeext_assign_scan_backop_flags_ap(pmlmeext, SS_BACKOP_EN | SS_BACKOP_PS_ANNC | SS_BACKOP_TX_RESUME);
  890. #endif
  891. #ifdef CONFIG_RTW_MESH
  892. mlmeext_assign_scan_backop_flags_mesh(pmlmeext, /*SS_BACKOP_EN | */SS_BACKOP_PS_ANNC | SS_BACKOP_TX_RESUME);
  893. #endif
  894. pmlmeext->sitesurvey_res.scan_cnt = 0;
  895. pmlmeext->sitesurvey_res.scan_cnt_max = RTW_SCAN_NUM_OF_CH;
  896. pmlmeext->sitesurvey_res.backop_ms = RTW_BACK_OP_CH_MS;
  897. #endif
  898. #if defined(CONFIG_ANTENNA_DIVERSITY) || defined(DBG_SCAN_SW_ANTDIV_BL)
  899. pmlmeext->sitesurvey_res.is_sw_antdiv_bl_scan = 0;
  900. #endif
  901. pmlmeext->scan_abort = _FALSE;
  902. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  903. pmlmeinfo->reauth_count = 0;
  904. pmlmeinfo->reassoc_count = 0;
  905. pmlmeinfo->link_count = 0;
  906. pmlmeinfo->auth_seq = 0;
  907. pmlmeinfo->auth_algo = dot11AuthAlgrthm_Open;
  908. pmlmeinfo->key_index = 0;
  909. pmlmeinfo->iv = 0;
  910. pmlmeinfo->enc_algo = _NO_PRIVACY_;
  911. pmlmeinfo->authModeToggle = 0;
  912. _rtw_memset(pmlmeinfo->chg_txt, 0, 128);
  913. pmlmeinfo->slotTime = SHORT_SLOT_TIME;
  914. pmlmeinfo->preamble_mode = PREAMBLE_AUTO;
  915. pmlmeinfo->dialogToken = 0;
  916. pmlmeext->action_public_rxseq = 0xffff;
  917. pmlmeext->action_public_dialog_token = 0xff;
  918. #ifdef ROKU_PRIVATE
  919. /*infra mode, used to store AP's info*/
  920. _rtw_memset(pmlmeinfo->SupportedRates_infra_ap, 0, NDIS_802_11_LENGTH_RATES_EX);
  921. pmlmeinfo->ht_vht_received = 0;
  922. #endif /* ROKU_PRIVATE */
  923. }
  924. void init_mlme_ext_timer(_adapter *padapter)
  925. {
  926. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  927. rtw_init_timer(&pmlmeext->survey_timer, padapter, survey_timer_hdl, padapter);
  928. rtw_init_timer(&pmlmeext->link_timer, padapter, link_timer_hdl, padapter);
  929. #ifdef CONFIG_RTW_80211R
  930. rtw_init_timer(&pmlmeext->ft_link_timer, padapter, rtw_ft_link_timer_hdl, padapter);
  931. rtw_init_timer(&pmlmeext->ft_roam_timer, padapter, rtw_ft_roam_timer_hdl, padapter);
  932. #endif
  933. #ifdef CONFIG_RTW_REPEATER_SON
  934. rtw_init_timer(&pmlmeext->rson_scan_timer, padapter, rson_timer_hdl, padapter);
  935. #endif
  936. }
  937. int init_mlme_ext_priv(_adapter *padapter)
  938. {
  939. int res = _SUCCESS;
  940. struct registry_priv *pregistrypriv = &padapter->registrypriv;
  941. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  942. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  943. /* We don't need to memset padapter->XXX to zero, because adapter is allocated by rtw_zvmalloc(). */
  944. /* _rtw_memset((u8 *)pmlmeext, 0, sizeof(struct mlme_ext_priv)); */
  945. pmlmeext->padapter = padapter;
  946. /* fill_fwpriv(padapter, &(pmlmeext->fwpriv)); */
  947. init_mlme_ext_priv_value(padapter);
  948. pmlmeinfo->bAcceptAddbaReq = pregistrypriv->bAcceptAddbaReq;
  949. init_mlme_ext_timer(padapter);
  950. #ifdef CONFIG_AP_MODE
  951. init_mlme_ap_info(padapter);
  952. #endif
  953. pmlmeext->last_scan_time = 0;
  954. pmlmeext->mlmeext_init = _TRUE;
  955. #ifdef CONFIG_ACTIVE_KEEP_ALIVE_CHECK
  956. pmlmeext->active_keep_alive_check = _TRUE;
  957. #else
  958. pmlmeext->active_keep_alive_check = _FALSE;
  959. #endif
  960. #ifdef DBG_FIXED_CHAN
  961. pmlmeext->fixed_chan = 0xFF;
  962. #endif
  963. pmlmeext->tsf_update_pause_factor = pregistrypriv->tsf_update_pause_factor;
  964. pmlmeext->tsf_update_restore_factor = pregistrypriv->tsf_update_restore_factor;
  965. return res;
  966. }
  967. void free_mlme_ext_priv(struct mlme_ext_priv *pmlmeext)
  968. {
  969. _adapter *padapter = pmlmeext->padapter;
  970. if (!padapter)
  971. return;
  972. if (rtw_is_drv_stopped(padapter)) {
  973. _cancel_timer_ex(&pmlmeext->survey_timer);
  974. _cancel_timer_ex(&pmlmeext->link_timer);
  975. }
  976. }
  977. #ifdef CONFIG_PATCH_JOIN_WRONG_CHANNEL
  978. static u8 cmp_pkt_chnl_diff(_adapter *padapter, u8 *pframe, uint packet_len)
  979. {
  980. /* if the channel is same, return 0. else return channel differential */
  981. uint len;
  982. u8 channel;
  983. u8 *p;
  984. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + _BEACON_IE_OFFSET_, _DSSET_IE_, &len, packet_len - _BEACON_IE_OFFSET_);
  985. if (p) {
  986. channel = *(p + 2);
  987. if (padapter->mlmeextpriv.cur_channel >= channel)
  988. return padapter->mlmeextpriv.cur_channel - channel;
  989. else
  990. return channel - padapter->mlmeextpriv.cur_channel;
  991. } else
  992. return 0;
  993. }
  994. #endif /* CONFIG_PATCH_JOIN_WRONG_CHANNEL */
  995. static void _mgt_dispatcher(_adapter *padapter, struct mlme_handler *ptable, union recv_frame *precv_frame)
  996. {
  997. u8 bc_addr[ETH_ALEN] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  998. u8 *pframe = precv_frame->u.hdr.rx_data;
  999. if (ptable->func) {
  1000. /* receive the frames that ra(a1) is my address or ra(a1) is bc address. */
  1001. if (!_rtw_memcmp(GetAddr1Ptr(pframe), adapter_mac_addr(padapter), ETH_ALEN) &&
  1002. !_rtw_memcmp(GetAddr1Ptr(pframe), bc_addr, ETH_ALEN))
  1003. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  1004. {
  1005. struct rtw_wdev_priv *pwdev_priv = adapter_wdev_data(padapter);
  1006. if (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) != _TRUE)
  1007. return;
  1008. if (check_fwstate(&padapter->mlmepriv, _FW_LINKED) == _TRUE)
  1009. return;
  1010. if ( pwdev_priv->pno_mac_addr[0] == 0xFF)
  1011. return;
  1012. if (!_rtw_memcmp(GetAddr1Ptr(pframe), adapter_pno_mac_addr(padapter), ETH_ALEN))
  1013. return;
  1014. }
  1015. #else
  1016. return;
  1017. #endif
  1018. ptable->func(padapter, precv_frame);
  1019. }
  1020. }
  1021. void mgt_dispatcher(_adapter *padapter, union recv_frame *precv_frame)
  1022. {
  1023. int index;
  1024. struct mlme_handler *ptable;
  1025. u8 bc_addr[ETH_ALEN] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  1026. u8 *pframe = precv_frame->u.hdr.rx_data;
  1027. struct sta_info *psta = rtw_get_stainfo(&padapter->stapriv, get_addr2_ptr(pframe));
  1028. struct recv_priv *precvpriv = &padapter->recvpriv;
  1029. #if 0
  1030. {
  1031. u8 *pbuf;
  1032. pbuf = GetAddr1Ptr(pframe);
  1033. RTW_INFO("A1-%x:%x:%x:%x:%x:%x\n", *pbuf, *(pbuf + 1), *(pbuf + 2), *(pbuf + 3), *(pbuf + 4), *(pbuf + 5));
  1034. pbuf = get_addr2_ptr(pframe);
  1035. RTW_INFO("A2-%x:%x:%x:%x:%x:%x\n", *pbuf, *(pbuf + 1), *(pbuf + 2), *(pbuf + 3), *(pbuf + 4), *(pbuf + 5));
  1036. pbuf = GetAddr3Ptr(pframe);
  1037. RTW_INFO("A3-%x:%x:%x:%x:%x:%x\n", *pbuf, *(pbuf + 1), *(pbuf + 2), *(pbuf + 3), *(pbuf + 4), *(pbuf + 5));
  1038. }
  1039. #endif
  1040. if (GetFrameType(pframe) != WIFI_MGT_TYPE) {
  1041. return;
  1042. }
  1043. /* receive the frames that ra(a1) is my address or ra(a1) is bc address. */
  1044. if (!_rtw_memcmp(GetAddr1Ptr(pframe), adapter_mac_addr(padapter), ETH_ALEN) &&
  1045. !_rtw_memcmp(GetAddr1Ptr(pframe), bc_addr, ETH_ALEN))
  1046. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  1047. {
  1048. struct rtw_wdev_priv *pwdev_priv = adapter_wdev_data(padapter);
  1049. if (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) != _TRUE)
  1050. return;
  1051. if (check_fwstate(&padapter->mlmepriv, _FW_LINKED) == _TRUE)
  1052. return;
  1053. if ( pwdev_priv->pno_mac_addr[0] == 0xFF)
  1054. return;
  1055. if (!_rtw_memcmp(GetAddr1Ptr(pframe), adapter_pno_mac_addr(padapter), ETH_ALEN))
  1056. return;
  1057. }
  1058. #else
  1059. return;
  1060. #endif
  1061. ptable = mlme_sta_tbl;
  1062. index = get_frame_sub_type(pframe) >> 4;
  1063. #ifdef CONFIG_TDLS
  1064. if ((index << 4) == WIFI_ACTION) {
  1065. /* category==public (4), action==TDLS_DISCOVERY_RESPONSE */
  1066. if (*(pframe + 24) == RTW_WLAN_CATEGORY_PUBLIC && *(pframe + 25) == TDLS_DISCOVERY_RESPONSE) {
  1067. RTW_INFO("[TDLS] Recv %s from "MAC_FMT"\n", rtw_tdls_action_txt(TDLS_DISCOVERY_RESPONSE), MAC_ARG(get_addr2_ptr(pframe)));
  1068. On_TDLS_Dis_Rsp(padapter, precv_frame);
  1069. }
  1070. }
  1071. #endif /* CONFIG_TDLS */
  1072. if (index >= (sizeof(mlme_sta_tbl) / sizeof(struct mlme_handler))) {
  1073. return;
  1074. }
  1075. ptable += index;
  1076. #if 1
  1077. if (psta != NULL) {
  1078. if (GetRetry(pframe)) {
  1079. if (precv_frame->u.hdr.attrib.seq_num == psta->RxMgmtFrameSeqNum) {
  1080. /* drop the duplicate management frame */
  1081. precvpriv->dbg_rx_dup_mgt_frame_drop_count++;
  1082. RTW_INFO("Drop duplicate management frame with seq_num = %d.\n", precv_frame->u.hdr.attrib.seq_num);
  1083. return;
  1084. }
  1085. }
  1086. psta->RxMgmtFrameSeqNum = precv_frame->u.hdr.attrib.seq_num;
  1087. }
  1088. #else
  1089. if (GetRetry(pframe)) {
  1090. /* return; */
  1091. }
  1092. #endif
  1093. #ifdef CONFIG_AP_MODE
  1094. switch (get_frame_sub_type(pframe)) {
  1095. case WIFI_AUTH:
  1096. if (MLME_IS_AP(padapter) || MLME_IS_MESH(padapter))
  1097. ptable->func = &OnAuth;
  1098. else
  1099. ptable->func = &OnAuthClient;
  1100. __attribute__((__fallthrough__));
  1101. case WIFI_ASSOCREQ:
  1102. case WIFI_REASSOCREQ:
  1103. _mgt_dispatcher(padapter, ptable, precv_frame);
  1104. #ifdef CONFIG_HOSTAPD_MLME
  1105. if (MLME_IS_AP(padapter))
  1106. rtw_hostapd_mlme_rx(padapter, precv_frame);
  1107. #endif
  1108. break;
  1109. case WIFI_PROBEREQ:
  1110. _mgt_dispatcher(padapter, ptable, precv_frame);
  1111. #ifdef CONFIG_HOSTAPD_MLME
  1112. if (MLME_IS_AP(padapter))
  1113. rtw_hostapd_mlme_rx(padapter, precv_frame);
  1114. #endif
  1115. break;
  1116. case WIFI_BEACON:
  1117. _mgt_dispatcher(padapter, ptable, precv_frame);
  1118. break;
  1119. case WIFI_ACTION:
  1120. _mgt_dispatcher(padapter, ptable, precv_frame);
  1121. break;
  1122. default:
  1123. _mgt_dispatcher(padapter, ptable, precv_frame);
  1124. #ifdef CONFIG_HOSTAPD_MLME
  1125. if (MLME_IS_AP(padapter))
  1126. rtw_hostapd_mlme_rx(padapter, precv_frame);
  1127. #endif
  1128. break;
  1129. }
  1130. #else
  1131. _mgt_dispatcher(padapter, ptable, precv_frame);
  1132. #endif
  1133. }
  1134. #ifdef CONFIG_P2P
  1135. u32 p2p_listen_state_process(_adapter *padapter, unsigned char *da)
  1136. {
  1137. bool response = _TRUE;
  1138. #ifdef CONFIG_IOCTL_CFG80211
  1139. if (padapter->wdinfo.driver_interface == DRIVER_CFG80211) {
  1140. if (rtw_cfg80211_get_is_roch(padapter) == _FALSE
  1141. || rtw_get_oper_ch(padapter) != padapter->wdinfo.listen_channel
  1142. || adapter_wdev_data(padapter)->p2p_enabled == _FALSE
  1143. || padapter->mlmepriv.wps_probe_resp_ie == NULL
  1144. || padapter->mlmepriv.p2p_probe_resp_ie == NULL
  1145. ) {
  1146. #ifdef CONFIG_DEBUG_CFG80211
  1147. RTW_INFO(ADPT_FMT" DON'T issue_probersp_p2p: p2p_enabled:%d, wps_probe_resp_ie:%p, p2p_probe_resp_ie:%p\n"
  1148. , ADPT_ARG(padapter)
  1149. , adapter_wdev_data(padapter)->p2p_enabled
  1150. , padapter->mlmepriv.wps_probe_resp_ie
  1151. , padapter->mlmepriv.p2p_probe_resp_ie);
  1152. RTW_INFO(ADPT_FMT" DON'T issue_probersp_p2p: is_ro_ch:%d, op_ch:%d, p2p_listen_channel:%d\n"
  1153. , ADPT_ARG(padapter)
  1154. , rtw_cfg80211_get_is_roch(padapter)
  1155. , rtw_get_oper_ch(padapter)
  1156. , padapter->wdinfo.listen_channel);
  1157. #endif
  1158. response = _FALSE;
  1159. }
  1160. } else
  1161. #endif /* CONFIG_IOCTL_CFG80211 */
  1162. if (padapter->wdinfo.driver_interface == DRIVER_WEXT) {
  1163. /* do nothing if the device name is empty */
  1164. if (!padapter->wdinfo.device_name_len)
  1165. response = _FALSE;
  1166. }
  1167. if (response == _TRUE)
  1168. issue_probersp_p2p(padapter, da);
  1169. return _SUCCESS;
  1170. }
  1171. #endif /* CONFIG_P2P */
  1172. /****************************************************************************
  1173. Following are the callback functions for each subtype of the management frames
  1174. *****************************************************************************/
  1175. unsigned int OnProbeReq(_adapter *padapter, union recv_frame *precv_frame)
  1176. {
  1177. unsigned int ielen;
  1178. unsigned char *p;
  1179. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  1180. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1181. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  1182. WLAN_BSSID_EX *cur = &(pmlmeinfo->network);
  1183. u8 *pframe = precv_frame->u.hdr.rx_data;
  1184. uint len = precv_frame->u.hdr.len;
  1185. u8 is_valid_p2p_probereq = _FALSE;
  1186. #ifdef CONFIG_ATMEL_RC_PATCH
  1187. u8 *target_ie = NULL, *wps_ie = NULL;
  1188. u8 *start;
  1189. uint search_len = 0, wps_ielen = 0, target_ielen = 0;
  1190. struct sta_info *psta;
  1191. struct sta_priv *pstapriv = &padapter->stapriv;
  1192. #endif
  1193. #ifdef CONFIG_P2P
  1194. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  1195. struct rx_pkt_attrib *pattrib = &precv_frame->u.hdr.attrib;
  1196. u8 wifi_test_chk_rate = 1;
  1197. #ifdef CONFIG_IOCTL_CFG80211
  1198. if ((pwdinfo->driver_interface == DRIVER_CFG80211)
  1199. && !rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE)
  1200. && (GET_CFG80211_REPORT_MGMT(adapter_wdev_data(padapter), IEEE80211_STYPE_PROBE_REQ) == _TRUE)
  1201. ) {
  1202. rtw_cfg80211_rx_probe_request(padapter, precv_frame);
  1203. return _SUCCESS;
  1204. }
  1205. #endif /* CONFIG_IOCTL_CFG80211 */
  1206. if (!rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE) &&
  1207. !rtw_p2p_chk_state(pwdinfo, P2P_STATE_IDLE) &&
  1208. !rtw_p2p_chk_role(pwdinfo, P2P_ROLE_CLIENT) &&
  1209. !rtw_p2p_chk_state(pwdinfo, P2P_STATE_FIND_PHASE_SEARCH) &&
  1210. !rtw_p2p_chk_state(pwdinfo, P2P_STATE_SCAN)
  1211. ) {
  1212. /* Commented by Albert 2011/03/17 */
  1213. /* mcs_rate = 0->CCK 1M rate */
  1214. /* mcs_rate = 1->CCK 2M rate */
  1215. /* mcs_rate = 2->CCK 5.5M rate */
  1216. /* mcs_rate = 3->CCK 11M rate */
  1217. /* In the P2P mode, the driver should not support the CCK rate */
  1218. /* Commented by Kurt 2012/10/16 */
  1219. /* IOT issue: Google Nexus7 use 1M rate to send p2p_probe_req after GO nego completed and Nexus7 is client */
  1220. if (padapter->registrypriv.wifi_spec == 1) {
  1221. if (pattrib->data_rate <= DESC_RATE11M)
  1222. wifi_test_chk_rate = 0;
  1223. }
  1224. if (wifi_test_chk_rate == 1) {
  1225. is_valid_p2p_probereq = process_probe_req_p2p_ie(pwdinfo, pframe, len);
  1226. if (is_valid_p2p_probereq == _TRUE) {
  1227. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_DEVICE)) {
  1228. /* FIXME */
  1229. if (padapter->wdinfo.driver_interface == DRIVER_WEXT)
  1230. report_survey_event(padapter, precv_frame);
  1231. p2p_listen_state_process(padapter, get_sa(pframe));
  1232. return _SUCCESS;
  1233. }
  1234. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO))
  1235. goto _continue;
  1236. }
  1237. }
  1238. }
  1239. _continue:
  1240. #endif /* CONFIG_P2P */
  1241. if (check_fwstate(pmlmepriv, WIFI_STATION_STATE))
  1242. return _SUCCESS;
  1243. if (check_fwstate(pmlmepriv, _FW_LINKED) == _FALSE &&
  1244. check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE | WIFI_AP_STATE | WIFI_MESH_STATE) == _FALSE)
  1245. return _SUCCESS;
  1246. /* RTW_INFO("+OnProbeReq\n"); */
  1247. #ifdef CONFIG_ATMEL_RC_PATCH
  1248. wps_ie = rtw_get_wps_ie(
  1249. pframe + WLAN_HDR_A3_LEN + _PROBEREQ_IE_OFFSET_,
  1250. len - WLAN_HDR_A3_LEN - _PROBEREQ_IE_OFFSET_,
  1251. NULL, &wps_ielen);
  1252. if (wps_ie)
  1253. target_ie = rtw_get_wps_attr_content(wps_ie, wps_ielen, WPS_ATTR_MANUFACTURER, NULL, &target_ielen);
  1254. if ((target_ie && (target_ielen == 4)) && (_TRUE == _rtw_memcmp((void *)target_ie, "Ozmo", 4))) {
  1255. /* psta->flag_atmel_rc = 1; */
  1256. unsigned char *sa_addr = get_sa(pframe);
  1257. printk("%s: Find Ozmo RC -- %02x:%02x:%02x:%02x:%02x:%02x \n\n",
  1258. __func__, *sa_addr, *(sa_addr + 1), *(sa_addr + 2), *(sa_addr + 3), *(sa_addr + 4), *(sa_addr + 5));
  1259. _rtw_memcpy(pstapriv->atmel_rc_pattern, get_sa(pframe), ETH_ALEN);
  1260. }
  1261. #endif
  1262. #ifdef CONFIG_AUTO_AP_MODE
  1263. if (check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE &&
  1264. pmlmepriv->cur_network.join_res == _TRUE) {
  1265. _irqL irqL;
  1266. struct sta_info *psta;
  1267. u8 *mac_addr, *peer_addr;
  1268. struct sta_priv *pstapriv = &padapter->stapriv;
  1269. u8 RC_OUI[4] = {0x00, 0xE0, 0x4C, 0x0A};
  1270. /* EID[1] + EID_LEN[1] + RC_OUI[4] + MAC[6] + PairingID[2] + ChannelNum[2] */
  1271. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + _PROBEREQ_IE_OFFSET_, _VENDOR_SPECIFIC_IE_, (int *)&ielen,
  1272. len - WLAN_HDR_A3_LEN - _PROBEREQ_IE_OFFSET_);
  1273. if (!p || ielen != 14)
  1274. goto _non_rc_device;
  1275. if (!_rtw_memcmp(p + 2, RC_OUI, sizeof(RC_OUI)))
  1276. goto _non_rc_device;
  1277. if (!_rtw_memcmp(p + 6, get_sa(pframe), ETH_ALEN)) {
  1278. RTW_INFO("%s, do rc pairing ("MAC_FMT"), but mac addr mismatch!("MAC_FMT")\n", __FUNCTION__,
  1279. MAC_ARG(get_sa(pframe)), MAC_ARG(p + 6));
  1280. goto _non_rc_device;
  1281. }
  1282. RTW_INFO("%s, got the pairing device("MAC_FMT")\n", __FUNCTION__, MAC_ARG(get_sa(pframe)));
  1283. /* new a station */
  1284. psta = rtw_get_stainfo(pstapriv, get_sa(pframe));
  1285. if (psta == NULL) {
  1286. /* allocate a new one */
  1287. RTW_INFO("going to alloc stainfo for rc="MAC_FMT"\n", MAC_ARG(get_sa(pframe)));
  1288. psta = rtw_alloc_stainfo(pstapriv, get_sa(pframe));
  1289. if (psta == NULL) {
  1290. /* TODO: */
  1291. RTW_INFO(" Exceed the upper limit of supported clients...\n");
  1292. return _SUCCESS;
  1293. }
  1294. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  1295. if (rtw_is_list_empty(&psta->asoc_list)) {
  1296. psta->expire_to = pstapriv->expire_to;
  1297. rtw_list_insert_tail(&psta->asoc_list, &pstapriv->asoc_list);
  1298. pstapriv->asoc_list_cnt++;
  1299. }
  1300. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  1301. /* generate pairing ID */
  1302. mac_addr = adapter_mac_addr(padapter);
  1303. peer_addr = psta->cmn.mac_addr;
  1304. psta->pid = (u16)(((mac_addr[4] << 8) + mac_addr[5]) + ((peer_addr[4] << 8) + peer_addr[5]));
  1305. /* update peer stainfo */
  1306. psta->isrc = _TRUE;
  1307. /* AID assignment */
  1308. if (psta->cmn.aid > 0)
  1309. RTW_INFO(FUNC_ADPT_FMT" old AID=%d\n", FUNC_ADPT_ARG(padapter), psta->cmn.aid);
  1310. else {
  1311. if (!rtw_aid_alloc(padapter, psta)) {
  1312. RTW_INFO(FUNC_ADPT_FMT" no room for more AIDs\n", FUNC_ADPT_ARG(padapter));
  1313. return _SUCCESS;
  1314. }
  1315. RTW_INFO(FUNC_ADPT_FMT" allocate new AID=%d\n", FUNC_ADPT_ARG(padapter), psta->cmn.aid);
  1316. }
  1317. psta->qos_option = 1;
  1318. psta->cmn.bw_mode = CHANNEL_WIDTH_20;
  1319. psta->ieee8021x_blocked = _FALSE;
  1320. #ifdef CONFIG_80211N_HT
  1321. if(padapter->registrypriv.ht_enable &&
  1322. is_supported_ht(padapter->registrypriv.wireless_mode)) {
  1323. psta->htpriv.ht_option = _TRUE;
  1324. psta->htpriv.ampdu_enable = _FALSE;
  1325. psta->htpriv.sgi_20m = _FALSE;
  1326. psta->htpriv.sgi_40m = _FALSE;
  1327. psta->htpriv.ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  1328. psta->htpriv.agg_enable_bitmap = 0x0;/* reset */
  1329. psta->htpriv.candidate_tid_bitmap = 0x0;/* reset */
  1330. }
  1331. #endif
  1332. rtw_hal_set_odm_var(padapter, HAL_ODM_STA_INFO, psta, _TRUE);
  1333. _rtw_memset((void *)&psta->sta_stats, 0, sizeof(struct stainfo_stats));
  1334. _enter_critical_bh(&psta->lock, &irqL);
  1335. psta->state |= _FW_LINKED;
  1336. _exit_critical_bh(&psta->lock, &irqL);
  1337. report_add_sta_event(padapter, psta->cmn.mac_addr);
  1338. }
  1339. issue_probersp(padapter, get_sa(pframe), _FALSE);
  1340. return _SUCCESS;
  1341. }
  1342. _non_rc_device:
  1343. return _SUCCESS;
  1344. #endif /* CONFIG_AUTO_AP_MODE */
  1345. #ifdef CONFIG_CONCURRENT_MODE
  1346. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) &&
  1347. rtw_mi_buddy_check_fwstate(padapter, _FW_UNDER_LINKING | _FW_UNDER_SURVEY)) {
  1348. /* don't process probe req */
  1349. return _SUCCESS;
  1350. }
  1351. #endif
  1352. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + _PROBEREQ_IE_OFFSET_, _SSID_IE_, (int *)&ielen,
  1353. len - WLAN_HDR_A3_LEN - _PROBEREQ_IE_OFFSET_);
  1354. /* check (wildcard) SSID */
  1355. if (p != NULL) {
  1356. if (is_valid_p2p_probereq == _TRUE)
  1357. goto _issue_probersp;
  1358. if ((ielen != 0 && _FALSE == _rtw_memcmp((void *)(p + 2), (void *)cur->Ssid.Ssid, cur->Ssid.SsidLength))
  1359. || (ielen == 0 && pmlmeinfo->hidden_ssid_mode))
  1360. goto exit;
  1361. #ifdef CONFIG_RTW_MESH
  1362. if (MLME_IS_MESH(padapter)) {
  1363. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + _PROBEREQ_IE_OFFSET_, WLAN_EID_MESH_ID, (int *)&ielen,
  1364. len - WLAN_HDR_A3_LEN - _PROBEREQ_IE_OFFSET_);
  1365. if (!p)
  1366. goto exit;
  1367. if (ielen != 0 && _rtw_memcmp((void *)(p + 2), (void *)cur->mesh_id.Ssid, cur->mesh_id.SsidLength) == _FALSE)
  1368. goto exit;
  1369. }
  1370. #endif
  1371. _issue_probersp:
  1372. if (((check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE &&
  1373. pmlmepriv->cur_network.join_res == _TRUE)) || check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) {
  1374. /* RTW_INFO("+issue_probersp during ap mode\n"); */
  1375. issue_probersp(padapter, get_sa(pframe), is_valid_p2p_probereq);
  1376. }
  1377. }
  1378. exit:
  1379. return _SUCCESS;
  1380. }
  1381. unsigned int OnProbeRsp(_adapter *padapter, union recv_frame *precv_frame)
  1382. {
  1383. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1384. u8 *pframe = precv_frame->u.hdr.rx_data;
  1385. #ifdef CONFIG_P2P
  1386. struct wifidirect_info *pwdinfo = &padapter->wdinfo;
  1387. #endif
  1388. #ifdef CONFIG_P2P
  1389. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_TX_PROVISION_DIS_REQ)) {
  1390. if (_TRUE == pwdinfo->tx_prov_disc_info.benable) {
  1391. if (_rtw_memcmp(pwdinfo->tx_prov_disc_info.peerIFAddr, get_addr2_ptr(pframe), ETH_ALEN)) {
  1392. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_CLIENT)) {
  1393. pwdinfo->tx_prov_disc_info.benable = _FALSE;
  1394. issue_p2p_provision_request(padapter,
  1395. pwdinfo->tx_prov_disc_info.ssid.Ssid,
  1396. pwdinfo->tx_prov_disc_info.ssid.SsidLength,
  1397. pwdinfo->tx_prov_disc_info.peerDevAddr);
  1398. } else if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_DEVICE) || rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  1399. pwdinfo->tx_prov_disc_info.benable = _FALSE;
  1400. issue_p2p_provision_request(padapter,
  1401. NULL,
  1402. 0,
  1403. pwdinfo->tx_prov_disc_info.peerDevAddr);
  1404. }
  1405. }
  1406. }
  1407. return _SUCCESS;
  1408. } else if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_GONEGO_ING)) {
  1409. if (_TRUE == pwdinfo->nego_req_info.benable) {
  1410. RTW_INFO("[%s] P2P State is GONEGO ING!\n", __FUNCTION__);
  1411. if (_rtw_memcmp(pwdinfo->nego_req_info.peerDevAddr, get_addr2_ptr(pframe), ETH_ALEN)) {
  1412. pwdinfo->nego_req_info.benable = _FALSE;
  1413. issue_p2p_GO_request(padapter, pwdinfo->nego_req_info.peerDevAddr);
  1414. }
  1415. }
  1416. } else if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_TX_INVITE_REQ)) {
  1417. if (_TRUE == pwdinfo->invitereq_info.benable) {
  1418. RTW_INFO("[%s] P2P_STATE_TX_INVITE_REQ!\n", __FUNCTION__);
  1419. if (_rtw_memcmp(pwdinfo->invitereq_info.peer_macaddr, get_addr2_ptr(pframe), ETH_ALEN)) {
  1420. pwdinfo->invitereq_info.benable = _FALSE;
  1421. issue_p2p_invitation_request(padapter, pwdinfo->invitereq_info.peer_macaddr);
  1422. }
  1423. }
  1424. }
  1425. #endif
  1426. if ((mlmeext_chk_scan_state(pmlmeext, SCAN_PROCESS))
  1427. || (MLME_IS_MESH(padapter) && check_fwstate(&padapter->mlmepriv, WIFI_ASOC_STATE))
  1428. #ifdef CONFIG_RTW_REPEATER_SON
  1429. || (padapter->rtw_rson_scanstage == RSON_SCAN_PROCESS)
  1430. #endif
  1431. ) {
  1432. rtw_mi_report_survey_event(padapter, precv_frame);
  1433. return _SUCCESS;
  1434. }
  1435. #if 0 /* move to validate_recv_mgnt_frame */
  1436. if (_rtw_memcmp(GetAddr3Ptr(pframe), get_my_bssid(&pmlmeinfo->network), ETH_ALEN)) {
  1437. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS) {
  1438. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  1439. if (psta != NULL)
  1440. psta->sta_stats.rx_mgnt_pkts++;
  1441. }
  1442. }
  1443. #endif
  1444. return _SUCCESS;
  1445. }
  1446. /* for 11n Logo 4.2.31/4.2.32 */
  1447. static void rtw_check_legacy_ap(_adapter *padapter, u8 *pframe, u32 len)
  1448. {
  1449. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1450. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  1451. if (!padapter->registrypriv.wifi_spec)
  1452. return;
  1453. if(!MLME_IS_AP(padapter))
  1454. return;
  1455. if (pmlmeext->bstart_bss == _TRUE) {
  1456. int left;
  1457. unsigned char *pos;
  1458. struct rtw_ieee802_11_elems elems;
  1459. #ifdef CONFIG_80211N_HT
  1460. u16 cur_op_mode;
  1461. #endif
  1462. /* checking IEs */
  1463. left = len - sizeof(struct rtw_ieee80211_hdr_3addr) - _BEACON_IE_OFFSET_;
  1464. pos = pframe + sizeof(struct rtw_ieee80211_hdr_3addr) + _BEACON_IE_OFFSET_;
  1465. if (rtw_ieee802_11_parse_elems(pos, left, &elems, 1) == ParseFailed) {
  1466. RTW_INFO("%s: parse fail for "MAC_FMT"\n", __func__, MAC_ARG(GetAddr3Ptr(pframe)));
  1467. return;
  1468. }
  1469. #ifdef CONFIG_80211N_HT
  1470. cur_op_mode = pmlmepriv->ht_op_mode & HT_INFO_OPERATION_MODE_OP_MODE_MASK;
  1471. #endif
  1472. /* for legacy ap */
  1473. if (elems.ht_capabilities == NULL && elems.ht_capabilities_len == 0) {
  1474. if (0)
  1475. RTW_INFO("%s: "MAC_FMT" is legacy ap\n", __func__, MAC_ARG(GetAddr3Ptr(pframe)));
  1476. ATOMIC_SET(&pmlmepriv->olbc, _TRUE);
  1477. ATOMIC_SET(&pmlmepriv->olbc_ht, _TRUE);
  1478. }
  1479. }
  1480. }
  1481. unsigned int OnBeacon(_adapter *padapter, union recv_frame *precv_frame)
  1482. {
  1483. struct sta_info *psta;
  1484. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1485. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  1486. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  1487. struct sta_priv *pstapriv = &padapter->stapriv;
  1488. u8 *pframe = precv_frame->u.hdr.rx_data;
  1489. uint len = precv_frame->u.hdr.len;
  1490. WLAN_BSSID_EX *pbss;
  1491. int ret = _SUCCESS;
  1492. #ifdef CONFIG_TDLS
  1493. struct sta_info *ptdls_sta;
  1494. struct tdls_info *ptdlsinfo = &padapter->tdlsinfo;
  1495. #ifdef CONFIG_TDLS_CH_SW
  1496. struct tdls_ch_switch *pchsw_info = &padapter->tdlsinfo.chsw_info;
  1497. #endif
  1498. #endif /* CONFIG_TDLS */
  1499. if (validate_beacon_len(pframe, len) == _FALSE)
  1500. return _SUCCESS;
  1501. if (mlmeext_chk_scan_state(pmlmeext, SCAN_PROCESS)
  1502. || (MLME_IS_MESH(padapter) && check_fwstate(pmlmepriv, WIFI_ASOC_STATE))
  1503. ) {
  1504. rtw_mi_report_survey_event(padapter, precv_frame);
  1505. return _SUCCESS;
  1506. }
  1507. #ifdef CONFIG_RTW_REPEATER_SON
  1508. if (padapter->rtw_rson_scanstage == RSON_SCAN_PROCESS)
  1509. rtw_mi_report_survey_event(padapter, precv_frame);
  1510. #endif
  1511. rtw_check_legacy_ap(padapter, pframe, len);
  1512. if (_rtw_memcmp(GetAddr3Ptr(pframe), get_my_bssid(&pmlmeinfo->network), ETH_ALEN)) {
  1513. if ((pmlmeinfo->state & WIFI_FW_AUTH_NULL)
  1514. && (rtw_sta_linking_test_wait_done() || pmlmeext->join_abort)
  1515. ) {
  1516. if (rtw_sta_linking_test_force_fail() || pmlmeext->join_abort) {
  1517. set_link_timer(pmlmeext, 1);
  1518. return _SUCCESS;
  1519. }
  1520. /* we should update current network before auth, or some IE is wrong */
  1521. pbss = (WLAN_BSSID_EX *)rtw_malloc(sizeof(WLAN_BSSID_EX));
  1522. if (pbss) {
  1523. if (collect_bss_info(padapter, precv_frame, pbss) == _SUCCESS) {
  1524. struct beacon_keys recv_beacon;
  1525. update_network(&(pmlmepriv->cur_network.network), pbss, padapter, _TRUE);
  1526. rtw_get_bcn_info(&(pmlmepriv->cur_network));
  1527. /* update bcn keys */
  1528. if (rtw_get_bcn_keys(padapter, pframe, len, &recv_beacon) == _TRUE) {
  1529. RTW_INFO("%s: beacon keys ready\n", __func__);
  1530. _rtw_memcpy(&pmlmepriv->cur_beacon_keys,
  1531. &recv_beacon, sizeof(recv_beacon));
  1532. } else {
  1533. RTW_ERR("%s: get beacon keys failed\n", __func__);
  1534. _rtw_memset(&pmlmepriv->cur_beacon_keys, 0, sizeof(recv_beacon));
  1535. }
  1536. #ifdef CONFIG_BCN_CNT_CONFIRM_HDL
  1537. pmlmepriv->new_beacon_cnts = 0;
  1538. #endif
  1539. }
  1540. rtw_mfree((u8 *)pbss, sizeof(WLAN_BSSID_EX));
  1541. }
  1542. /* check the vendor of the assoc AP */
  1543. pmlmeinfo->assoc_AP_vendor = check_assoc_AP(pframe + sizeof(struct rtw_ieee80211_hdr_3addr), len - sizeof(struct rtw_ieee80211_hdr_3addr));
  1544. /* update TSF Value */
  1545. update_TSF(pmlmeext, pframe, len);
  1546. pmlmeext->bcn_cnt = 0;
  1547. pmlmeext->last_bcn_cnt = 0;
  1548. #ifdef CONFIG_P2P_PS
  1549. /* Comment by YiWei , in wifi p2p spec the "3.3 P2P Power Management" , "These mechanisms are available in a P2P Group in which only P2P Devices are associated." */
  1550. /* process_p2p_ps_ie(padapter, (pframe + WLAN_HDR_A3_LEN), (len - WLAN_HDR_A3_LEN)); */
  1551. #endif /* CONFIG_P2P_PS */
  1552. #if defined(CONFIG_P2P) && defined(CONFIG_CONCURRENT_MODE)
  1553. if (padapter->registrypriv.wifi_spec) {
  1554. if (process_p2p_cross_connect_ie(padapter, (pframe + WLAN_HDR_A3_LEN), (len - WLAN_HDR_A3_LEN)) == _FALSE) {
  1555. if (rtw_mi_buddy_check_mlmeinfo_state(padapter, WIFI_FW_AP_STATE)) {
  1556. RTW_PRINT("no issue auth, P2P cross-connect does not permit\n ");
  1557. return _SUCCESS;
  1558. }
  1559. }
  1560. }
  1561. #endif /* CONFIG_P2P CONFIG_P2P and CONFIG_CONCURRENT_MODE */
  1562. /* start auth */
  1563. start_clnt_auth(padapter);
  1564. return _SUCCESS;
  1565. }
  1566. if (((pmlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE) && (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)) {
  1567. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  1568. if (psta != NULL) {
  1569. #ifdef CONFIG_PATCH_JOIN_WRONG_CHANNEL
  1570. /* Merge from 8712 FW code */
  1571. if (cmp_pkt_chnl_diff(padapter, pframe, len) != 0) {
  1572. /* join wrong channel, deauth and reconnect */
  1573. issue_deauth(padapter, (&(pmlmeinfo->network))->MacAddress, WLAN_REASON_DEAUTH_LEAVING);
  1574. report_del_sta_event(padapter, (&(pmlmeinfo->network))->MacAddress, WLAN_REASON_JOIN_WRONG_CHANNEL, _TRUE, _FALSE);
  1575. pmlmeinfo->state &= (~WIFI_FW_ASSOC_SUCCESS);
  1576. return _SUCCESS;
  1577. }
  1578. #endif /* CONFIG_PATCH_JOIN_WRONG_CHANNEL */
  1579. #ifdef CONFIG_RTW_80211R
  1580. rtw_ft_update_bcn(padapter, precv_frame);
  1581. #endif
  1582. ret = rtw_check_bcn_info(padapter, pframe, len);
  1583. if (!ret) {
  1584. RTW_PRINT("ap has changed, disconnect now\n ");
  1585. receive_disconnect(padapter, pmlmeinfo->network.MacAddress , 0, _FALSE);
  1586. return _SUCCESS;
  1587. }
  1588. /* update WMM, ERP in the beacon */
  1589. /* todo: the timer is used instead of the number of the beacon received */
  1590. if ((sta_rx_pkts(psta) & 0xf) == 0) {
  1591. /* RTW_INFO("update_bcn_info\n"); */
  1592. update_beacon_info(padapter, pframe, len, psta);
  1593. }
  1594. pmlmepriv->cur_network_scanned->network.Rssi = precv_frame->u.hdr.attrib.phy_info.recv_signal_power;
  1595. pmlmeext->bcn_cnt++;
  1596. #ifdef CONFIG_BCN_RECV_TIME
  1597. rtw_rx_bcn_time_update(padapter, len, precv_frame->u.hdr.attrib.data_rate);
  1598. #endif
  1599. #ifdef CONFIG_TDLS
  1600. #ifdef CONFIG_TDLS_CH_SW
  1601. if (rtw_tdls_is_chsw_allowed(padapter) == _TRUE) {
  1602. /* Send TDLS Channel Switch Request when receiving Beacon */
  1603. if ((padapter->tdlsinfo.chsw_info.ch_sw_state & TDLS_CH_SW_INITIATOR_STATE) && (ATOMIC_READ(&pchsw_info->chsw_on) == _TRUE)
  1604. && (pmlmeext->cur_channel == rtw_get_oper_ch(padapter))) {
  1605. ptdls_sta = rtw_get_stainfo(&padapter->stapriv, padapter->tdlsinfo.chsw_info.addr);
  1606. if (ptdls_sta != NULL) {
  1607. if (ptdls_sta->tdls_sta_state | TDLS_LINKED_STATE)
  1608. _set_timer(&ptdls_sta->stay_on_base_chnl_timer, TDLS_CH_SW_STAY_ON_BASE_CHNL_TIMEOUT);
  1609. }
  1610. }
  1611. }
  1612. #endif
  1613. #endif /* CONFIG_TDLS */
  1614. #ifdef CONFIG_DFS
  1615. process_csa_ie(padapter
  1616. , pframe + WLAN_HDR_A3_LEN + _BEACON_IE_OFFSET_
  1617. , len - (WLAN_HDR_A3_LEN + _BEACON_IE_OFFSET_));
  1618. #endif
  1619. #ifdef CONFIG_P2P_PS
  1620. process_p2p_ps_ie(padapter, (pframe + WLAN_HDR_A3_LEN), (len - WLAN_HDR_A3_LEN));
  1621. #endif /* CONFIG_P2P_PS */
  1622. if (pmlmeext->tsf_update_required && pmlmeext->en_hw_update_tsf)
  1623. rtw_enable_hw_update_tsf_cmd(padapter);
  1624. #if 0 /* move to validate_recv_mgnt_frame */
  1625. psta->sta_stats.rx_mgnt_pkts++;
  1626. #endif
  1627. }
  1628. } else if ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) {
  1629. u8 rate_set[16];
  1630. u8 rate_num = 0;
  1631. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  1632. if (psta != NULL) {
  1633. /*
  1634. * update WMM, ERP in the beacon
  1635. * todo: the timer is used instead of the number of the beacon received
  1636. */
  1637. if ((sta_rx_pkts(psta) & 0xf) == 0)
  1638. update_beacon_info(padapter, pframe, len, psta);
  1639. if (pmlmeext->tsf_update_required && pmlmeext->en_hw_update_tsf)
  1640. rtw_enable_hw_update_tsf_cmd(padapter);
  1641. } else {
  1642. rtw_ies_get_supported_rate(pframe + WLAN_HDR_A3_LEN + _BEACON_IE_OFFSET_, len - WLAN_HDR_A3_LEN - _BEACON_IE_OFFSET_, rate_set, &rate_num);
  1643. if (rate_num == 0) {
  1644. RTW_INFO(FUNC_ADPT_FMT" RX beacon with no supported rate\n", FUNC_ADPT_ARG(padapter));
  1645. goto _END_ONBEACON_;
  1646. }
  1647. psta = rtw_alloc_stainfo(pstapriv, get_addr2_ptr(pframe));
  1648. if (psta == NULL) {
  1649. RTW_INFO(FUNC_ADPT_FMT" Exceed the upper limit of supported clients\n", FUNC_ADPT_ARG(padapter));
  1650. goto _END_ONBEACON_;
  1651. }
  1652. psta->expire_to = pstapriv->adhoc_expire_to;
  1653. _rtw_memcpy(psta->bssrateset, rate_set, rate_num);
  1654. psta->bssratelen = rate_num;
  1655. /* update TSF Value */
  1656. update_TSF(pmlmeext, pframe, len);
  1657. /* report sta add event */
  1658. report_add_sta_event(padapter, get_addr2_ptr(pframe));
  1659. }
  1660. }
  1661. }
  1662. _END_ONBEACON_:
  1663. return _SUCCESS;
  1664. }
  1665. unsigned int OnAuth(_adapter *padapter, union recv_frame *precv_frame)
  1666. {
  1667. #ifdef CONFIG_AP_MODE
  1668. _irqL irqL;
  1669. unsigned int auth_mode, seq, ie_len;
  1670. unsigned char *sa, *p;
  1671. u16 algorithm;
  1672. int status;
  1673. static struct sta_info stat;
  1674. struct sta_info *pstat = NULL;
  1675. struct sta_priv *pstapriv = &padapter->stapriv;
  1676. struct security_priv *psecuritypriv = &padapter->securitypriv;
  1677. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1678. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  1679. u8 *pframe = precv_frame->u.hdr.rx_data;
  1680. uint len = precv_frame->u.hdr.len;
  1681. u8 offset = 0;
  1682. #ifdef CONFIG_CONCURRENT_MODE
  1683. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) &&
  1684. rtw_mi_buddy_check_fwstate(padapter, _FW_UNDER_LINKING | _FW_UNDER_SURVEY)) {
  1685. /* don't process auth request; */
  1686. return _SUCCESS;
  1687. }
  1688. #endif /* CONFIG_CONCURRENT_MODE */
  1689. if ((pmlmeinfo->state & 0x03) != WIFI_FW_AP_STATE)
  1690. return _FAIL;
  1691. if (!MLME_IS_ASOC(padapter))
  1692. return _SUCCESS;
  1693. #if defined(CONFIG_IOCTL_CFG80211) && defined(CONFIG_RTW_MESH)
  1694. if (MLME_IS_MESH(padapter))
  1695. return rtw_mesh_on_auth(padapter, precv_frame);
  1696. #endif
  1697. RTW_INFO("+OnAuth\n");
  1698. sa = get_addr2_ptr(pframe);
  1699. auth_mode = psecuritypriv->dot11AuthAlgrthm;
  1700. if (GetPrivacy(pframe)) {
  1701. u8 *iv;
  1702. struct rx_pkt_attrib *prxattrib = &(precv_frame->u.hdr.attrib);
  1703. prxattrib->hdrlen = WLAN_HDR_A3_LEN;
  1704. prxattrib->encrypt = _WEP40_;
  1705. iv = pframe + prxattrib->hdrlen;
  1706. prxattrib->key_index = ((iv[3] >> 6) & 0x3);
  1707. prxattrib->iv_len = 4;
  1708. prxattrib->icv_len = 4;
  1709. rtw_wep_decrypt(padapter, (u8 *)precv_frame);
  1710. offset = 4;
  1711. }
  1712. algorithm = le16_to_cpu(*(u16 *)((SIZE_PTR)pframe + WLAN_HDR_A3_LEN + offset));
  1713. seq = le16_to_cpu(*(u16 *)((SIZE_PTR)pframe + WLAN_HDR_A3_LEN + offset + 2));
  1714. RTW_INFO("auth alg=%x, seq=%X\n", algorithm, seq);
  1715. if (rtw_ap_linking_test_force_auth_fail()) {
  1716. status = rtw_ap_linking_test_force_auth_fail();
  1717. RTW_INFO(FUNC_ADPT_FMT" force auth fail with status:%u\n"
  1718. , FUNC_ADPT_ARG(padapter), status);
  1719. goto auth_fail;
  1720. }
  1721. if (auth_mode == 2 &&
  1722. psecuritypriv->dot11PrivacyAlgrthm != _WEP40_ &&
  1723. psecuritypriv->dot11PrivacyAlgrthm != _WEP104_)
  1724. auth_mode = 0;
  1725. if ((algorithm > 0 && auth_mode == 0) || /* rx a shared-key auth but shared not enabled */
  1726. (algorithm == 0 && auth_mode == 1)) { /* rx a open-system auth but shared-key is enabled */
  1727. RTW_INFO("auth rejected due to bad alg [alg=%d, auth_mib=%d] %02X%02X%02X%02X%02X%02X\n",
  1728. algorithm, auth_mode, sa[0], sa[1], sa[2], sa[3], sa[4], sa[5]);
  1729. status = _STATS_NO_SUPP_ALG_;
  1730. goto auth_fail;
  1731. }
  1732. #if CONFIG_RTW_MACADDR_ACL
  1733. if (rtw_access_ctrl(padapter, sa) == _FALSE) {
  1734. status = _STATS_UNABLE_HANDLE_STA_;
  1735. goto auth_fail;
  1736. }
  1737. #endif
  1738. pstat = rtw_get_stainfo(pstapriv, sa);
  1739. if (pstat == NULL) {
  1740. /* allocate a new one */
  1741. RTW_INFO("going to alloc stainfo for sa="MAC_FMT"\n", MAC_ARG(sa));
  1742. pstat = rtw_alloc_stainfo(pstapriv, sa);
  1743. if (pstat == NULL) {
  1744. RTW_INFO(" Exceed the upper limit of supported clients...\n");
  1745. status = _STATS_UNABLE_HANDLE_STA_;
  1746. goto auth_fail;
  1747. }
  1748. pstat->state = WIFI_FW_AUTH_NULL;
  1749. pstat->auth_seq = 0;
  1750. /* pstat->flags = 0; */
  1751. /* pstat->capability = 0; */
  1752. } else {
  1753. #ifdef CONFIG_IEEE80211W
  1754. if (pstat->bpairwise_key_installed != _TRUE && !(pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1755. #endif /* CONFIG_IEEE80211W */
  1756. {
  1757. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  1758. if (rtw_is_list_empty(&pstat->asoc_list) == _FALSE) {
  1759. rtw_list_delete(&pstat->asoc_list);
  1760. pstapriv->asoc_list_cnt--;
  1761. if (pstat->expire_to > 0)
  1762. ;/* TODO: STA re_auth within expire_to */
  1763. }
  1764. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  1765. if (seq == 1)
  1766. ; /* TODO: STA re_auth and auth timeout */
  1767. }
  1768. }
  1769. #ifdef CONFIG_IEEE80211W
  1770. if (pstat->bpairwise_key_installed != _TRUE && !(pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1771. #endif /* CONFIG_IEEE80211W */
  1772. {
  1773. _enter_critical_bh(&pstapriv->auth_list_lock, &irqL);
  1774. if (rtw_is_list_empty(&pstat->auth_list)) {
  1775. rtw_list_insert_tail(&pstat->auth_list, &pstapriv->auth_list);
  1776. pstapriv->auth_list_cnt++;
  1777. }
  1778. _exit_critical_bh(&pstapriv->auth_list_lock, &irqL);
  1779. }
  1780. if (pstat->auth_seq == 0)
  1781. pstat->expire_to = pstapriv->auth_to;
  1782. if ((pstat->auth_seq + 1) != seq) {
  1783. RTW_INFO("(1)auth rejected because out of seq [rx_seq=%d, exp_seq=%d]!\n",
  1784. seq, pstat->auth_seq + 1);
  1785. status = _STATS_OUT_OF_AUTH_SEQ_;
  1786. goto auth_fail;
  1787. }
  1788. if (algorithm == 0 && (auth_mode == 0 || auth_mode == 2 || auth_mode == 3)) {
  1789. if (seq == 1) {
  1790. #ifdef CONFIG_IEEE80211W
  1791. if (pstat->bpairwise_key_installed != _TRUE && !(pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1792. #endif /* CONFIG_IEEE80211W */
  1793. {
  1794. pstat->state &= ~WIFI_FW_AUTH_NULL;
  1795. pstat->state |= WIFI_FW_AUTH_SUCCESS;
  1796. pstat->expire_to = pstapriv->assoc_to;
  1797. }
  1798. pstat->authalg = algorithm;
  1799. } else {
  1800. RTW_INFO("(2)auth rejected because out of seq [rx_seq=%d, exp_seq=%d]!\n",
  1801. seq, pstat->auth_seq + 1);
  1802. status = _STATS_OUT_OF_AUTH_SEQ_;
  1803. goto auth_fail;
  1804. }
  1805. } else { /* shared system or auto authentication */
  1806. if (seq == 1) {
  1807. /* prepare for the challenging txt... */
  1808. /* get_random_bytes((void *)pstat->chg_txt, 128); */ /* TODO: */
  1809. _rtw_memset((void *)pstat->chg_txt, 78, 128);
  1810. #ifdef CONFIG_IEEE80211W
  1811. if (pstat->bpairwise_key_installed != _TRUE && !(pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1812. #endif /* CONFIG_IEEE80211W */
  1813. {
  1814. pstat->state &= ~WIFI_FW_AUTH_NULL;
  1815. pstat->state |= WIFI_FW_AUTH_STATE;
  1816. }
  1817. pstat->authalg = algorithm;
  1818. pstat->auth_seq = 2;
  1819. } else if (seq == 3) {
  1820. /* checking for challenging txt... */
  1821. RTW_INFO("checking for challenging txt...\n");
  1822. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + 4 + _AUTH_IE_OFFSET_ , _CHLGETXT_IE_, (int *)&ie_len,
  1823. len - WLAN_HDR_A3_LEN - _AUTH_IE_OFFSET_ - 4);
  1824. if ((p == NULL) || (ie_len <= 0)) {
  1825. RTW_INFO("auth rejected because challenge failure!(1)\n");
  1826. status = _STATS_CHALLENGE_FAIL_;
  1827. goto auth_fail;
  1828. }
  1829. if (_rtw_memcmp((void *)(p + 2), pstat->chg_txt, 128)) {
  1830. #ifdef CONFIG_IEEE80211W
  1831. if (pstat->bpairwise_key_installed != _TRUE && !(pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1832. #endif /* CONFIG_IEEE80211W */
  1833. {
  1834. pstat->state &= (~WIFI_FW_AUTH_STATE);
  1835. pstat->state |= WIFI_FW_AUTH_SUCCESS;
  1836. /* challenging txt is correct... */
  1837. pstat->expire_to = pstapriv->assoc_to;
  1838. }
  1839. } else {
  1840. RTW_INFO("auth rejected because challenge failure!\n");
  1841. status = _STATS_CHALLENGE_FAIL_;
  1842. goto auth_fail;
  1843. }
  1844. } else {
  1845. RTW_INFO("(3)auth rejected because out of seq [rx_seq=%d, exp_seq=%d]!\n",
  1846. seq, pstat->auth_seq + 1);
  1847. status = _STATS_OUT_OF_AUTH_SEQ_;
  1848. goto auth_fail;
  1849. }
  1850. }
  1851. /* Now, we are going to issue_auth... */
  1852. pstat->auth_seq = seq + 1;
  1853. #ifdef CONFIG_NATIVEAP_MLME
  1854. issue_auth(padapter, pstat, (unsigned short)(_STATS_SUCCESSFUL_));
  1855. #endif
  1856. if ((pstat->state & WIFI_FW_AUTH_SUCCESS) || (pstat->state & WIFI_FW_ASSOC_SUCCESS))
  1857. pstat->auth_seq = 0;
  1858. return _SUCCESS;
  1859. auth_fail:
  1860. if (pstat)
  1861. rtw_free_stainfo(padapter , pstat);
  1862. pstat = &stat;
  1863. _rtw_memset((char *)pstat, '\0', sizeof(stat));
  1864. pstat->auth_seq = 2;
  1865. _rtw_memcpy(pstat->cmn.mac_addr, sa, ETH_ALEN);
  1866. #ifdef CONFIG_NATIVEAP_MLME
  1867. issue_auth(padapter, pstat, (unsigned short)status);
  1868. #endif
  1869. #endif
  1870. return _FAIL;
  1871. }
  1872. unsigned int OnAuthClient(_adapter *padapter, union recv_frame *precv_frame)
  1873. {
  1874. unsigned int seq, len, status, algthm, offset;
  1875. unsigned char *p;
  1876. unsigned int go2asoc = 0;
  1877. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1878. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  1879. u8 *pframe = precv_frame->u.hdr.rx_data;
  1880. uint pkt_len = precv_frame->u.hdr.len;
  1881. RTW_INFO("%s\n", __FUNCTION__);
  1882. /* check A1 matches or not */
  1883. if (!_rtw_memcmp(adapter_mac_addr(padapter), get_da(pframe), ETH_ALEN))
  1884. return _SUCCESS;
  1885. if (!(pmlmeinfo->state & WIFI_FW_AUTH_STATE) || pmlmeext->join_abort)
  1886. return _SUCCESS;
  1887. offset = (GetPrivacy(pframe)) ? 4 : 0;
  1888. algthm = le16_to_cpu(*(unsigned short *)((SIZE_PTR)pframe + WLAN_HDR_A3_LEN + offset));
  1889. seq = le16_to_cpu(*(unsigned short *)((SIZE_PTR)pframe + WLAN_HDR_A3_LEN + offset + 2));
  1890. status = le16_to_cpu(*(unsigned short *)((SIZE_PTR)pframe + WLAN_HDR_A3_LEN + offset + 4));
  1891. if (status != 0) {
  1892. RTW_INFO("clnt auth fail, status: %d\n", status);
  1893. if (status == 13) { /* && pmlmeinfo->auth_algo == dot11AuthAlgrthm_Auto) */
  1894. if (pmlmeinfo->auth_algo == dot11AuthAlgrthm_Shared)
  1895. pmlmeinfo->auth_algo = dot11AuthAlgrthm_Open;
  1896. else
  1897. pmlmeinfo->auth_algo = dot11AuthAlgrthm_Shared;
  1898. /* pmlmeinfo->reauth_count = 0; */
  1899. }
  1900. pmlmeinfo->auth_status = status;
  1901. set_link_timer(pmlmeext, 1);
  1902. goto authclnt_fail;
  1903. }
  1904. if (seq == 2) {
  1905. if (pmlmeinfo->auth_algo == dot11AuthAlgrthm_Shared) {
  1906. /* legendary shared system */
  1907. p = rtw_get_ie(pframe + WLAN_HDR_A3_LEN + _AUTH_IE_OFFSET_, _CHLGETXT_IE_, (int *)&len,
  1908. pkt_len - WLAN_HDR_A3_LEN - _AUTH_IE_OFFSET_);
  1909. if (p == NULL) {
  1910. /* RTW_INFO("marc: no challenge text?\n"); */
  1911. goto authclnt_fail;
  1912. }
  1913. _rtw_memcpy((void *)(pmlmeinfo->chg_txt), (void *)(p + 2), len);
  1914. pmlmeinfo->auth_seq = 3;
  1915. issue_auth(padapter, NULL, 0);
  1916. set_link_timer(pmlmeext, REAUTH_TO);
  1917. return _SUCCESS;
  1918. } else {
  1919. /* open, or 802.11r FTAA system */
  1920. go2asoc = 1;
  1921. }
  1922. } else if (seq == 4) {
  1923. if (pmlmeinfo->auth_algo == dot11AuthAlgrthm_Shared)
  1924. go2asoc = 1;
  1925. else
  1926. goto authclnt_fail;
  1927. } else {
  1928. /* this is also illegal */
  1929. /* RTW_INFO("marc: clnt auth failed due to illegal seq=%x\n", seq); */
  1930. goto authclnt_fail;
  1931. }
  1932. if (go2asoc) {
  1933. #ifdef CONFIG_RTW_80211R
  1934. if (rtw_ft_update_auth_rsp_ies(padapter, pframe, pkt_len))
  1935. return _SUCCESS;
  1936. #endif
  1937. RTW_PRINT("auth success, start assoc\n");
  1938. start_clnt_assoc(padapter);
  1939. return _SUCCESS;
  1940. }
  1941. authclnt_fail:
  1942. /* pmlmeinfo->state &= ~(WIFI_FW_AUTH_STATE); */
  1943. return _FAIL;
  1944. }
  1945. unsigned int OnAssocReq(_adapter *padapter, union recv_frame *precv_frame)
  1946. {
  1947. #ifdef CONFIG_AP_MODE
  1948. _irqL irqL;
  1949. u16 listen_interval;
  1950. struct rtw_ieee802_11_elems elems;
  1951. struct sta_info *pstat;
  1952. unsigned char reassoc, *pos;
  1953. int left;
  1954. unsigned short status = _STATS_SUCCESSFUL_;
  1955. unsigned short frame_type, ie_offset = 0;
  1956. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  1957. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  1958. WLAN_BSSID_EX *cur = &(pmlmeinfo->network);
  1959. struct sta_priv *pstapriv = &padapter->stapriv;
  1960. u8 *pframe = precv_frame->u.hdr.rx_data;
  1961. uint pkt_len = precv_frame->u.hdr.len;
  1962. #ifdef CONFIG_P2P
  1963. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  1964. u8 p2p_status_code = P2P_STATUS_SUCCESS;
  1965. u8 *p2pie;
  1966. u32 p2pielen = 0;
  1967. #endif /* CONFIG_P2P */
  1968. #ifdef CONFIG_CONCURRENT_MODE
  1969. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) &&
  1970. rtw_mi_buddy_check_fwstate(padapter, _FW_UNDER_LINKING | _FW_UNDER_SURVEY)) {
  1971. /* don't process assoc request; */
  1972. return _SUCCESS;
  1973. }
  1974. #endif /* CONFIG_CONCURRENT_MODE */
  1975. if ((pmlmeinfo->state & 0x03) != WIFI_FW_AP_STATE)
  1976. return _FAIL;
  1977. frame_type = get_frame_sub_type(pframe);
  1978. if (frame_type == WIFI_ASSOCREQ) {
  1979. reassoc = 0;
  1980. ie_offset = _ASOCREQ_IE_OFFSET_;
  1981. } else { /* WIFI_REASSOCREQ */
  1982. reassoc = 1;
  1983. ie_offset = _REASOCREQ_IE_OFFSET_;
  1984. }
  1985. if (pkt_len < IEEE80211_3ADDR_LEN + ie_offset) {
  1986. RTW_INFO("handle_assoc(reassoc=%d) - too short payload (len=%lu)"
  1987. "\n", reassoc, (unsigned long)pkt_len);
  1988. return _FAIL;
  1989. }
  1990. pstat = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  1991. if (pstat == (struct sta_info *)NULL) {
  1992. status = _RSON_CLS2_;
  1993. goto asoc_class2_error;
  1994. }
  1995. RTW_INFO("%s\n", __FUNCTION__);
  1996. /* check if this stat has been successfully authenticated/assocated */
  1997. if (!((pstat->state) & WIFI_FW_AUTH_SUCCESS)) {
  1998. if (!((pstat->state) & WIFI_FW_ASSOC_SUCCESS)) {
  1999. status = _RSON_CLS2_;
  2000. goto asoc_class2_error;
  2001. } else {
  2002. pstat->state &= (~WIFI_FW_ASSOC_SUCCESS);
  2003. pstat->state |= WIFI_FW_ASSOC_STATE;
  2004. }
  2005. } else {
  2006. pstat->state &= (~WIFI_FW_AUTH_SUCCESS);
  2007. pstat->state |= WIFI_FW_ASSOC_STATE;
  2008. }
  2009. #if 0/* todo:tkip_countermeasures */
  2010. if (hapd->tkip_countermeasures) {
  2011. resp = WLAN_REASON_MICHAEL_MIC_FAILURE;
  2012. goto fail;
  2013. }
  2014. #endif
  2015. if (rtw_ap_linking_test_force_asoc_fail()) {
  2016. status = rtw_ap_linking_test_force_asoc_fail();
  2017. RTW_INFO(FUNC_ADPT_FMT" force asoc fail with status:%u\n"
  2018. , FUNC_ADPT_ARG(padapter), status);
  2019. goto OnAssocReqFail;
  2020. }
  2021. /* now parse all ieee802_11 ie to point to elems */
  2022. left = pkt_len - (IEEE80211_3ADDR_LEN + ie_offset);
  2023. pos = pframe + (IEEE80211_3ADDR_LEN + ie_offset);
  2024. if (rtw_ieee802_11_parse_elems(pos, left, &elems, 1) == ParseFailed) {
  2025. RTW_INFO("STA " MAC_FMT " sent invalid association request\n",
  2026. MAC_ARG(pstat->cmn.mac_addr));
  2027. status = _STATS_FAILURE_;
  2028. goto OnAssocReqFail;
  2029. }
  2030. rtw_ap_parse_sta_capability(padapter, pstat, pframe + WLAN_HDR_A3_LEN);
  2031. listen_interval = RTW_GET_LE16(pframe + WLAN_HDR_A3_LEN + 2);
  2032. #if 0/* todo: */
  2033. /* check listen_interval */
  2034. if (listen_interval > hapd->conf->max_listen_interval) {
  2035. hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
  2036. HOSTAPD_LEVEL_DEBUG,
  2037. "Too large Listen Interval (%d)",
  2038. listen_interval);
  2039. resp = WLAN_STATUS_ASSOC_DENIED_LISTEN_INT_TOO_LARGE;
  2040. goto fail;
  2041. }
  2042. pstat->listen_interval = listen_interval;
  2043. #endif
  2044. /* now we should check all the fields... */
  2045. /* checking SSID */
  2046. if (elems.ssid == NULL
  2047. || elems.ssid_len == 0
  2048. || elems.ssid_len != cur->Ssid.SsidLength
  2049. || _rtw_memcmp(elems.ssid, cur->Ssid.Ssid, cur->Ssid.SsidLength) == _FALSE
  2050. ) {
  2051. status = _STATS_FAILURE_;
  2052. goto OnAssocReqFail;
  2053. }
  2054. /* (Extended) Supported rates */
  2055. status = rtw_ap_parse_sta_supported_rates(padapter, pstat
  2056. , pframe + WLAN_HDR_A3_LEN + ie_offset, pkt_len - WLAN_HDR_A3_LEN - ie_offset);
  2057. if (status != _STATS_SUCCESSFUL_)
  2058. goto OnAssocReqFail;
  2059. /* check RSN/WPA/WPS */
  2060. status = rtw_ap_parse_sta_security_ie(padapter, pstat, &elems);
  2061. if (status != _STATS_SUCCESSFUL_)
  2062. goto OnAssocReqFail;
  2063. /* check if there is WMM IE & support WWM-PS */
  2064. rtw_ap_parse_sta_wmm_ie(padapter, pstat
  2065. , pframe + WLAN_HDR_A3_LEN + ie_offset, pkt_len - WLAN_HDR_A3_LEN - ie_offset);
  2066. rtw_ap_parse_sta_ht_ie(padapter, pstat, &elems);
  2067. rtw_ap_parse_sta_vht_ie(padapter, pstat, &elems);
  2068. if (((pstat->flags & WLAN_STA_HT) || (pstat->flags & WLAN_STA_VHT)) &&
  2069. ((pstat->wpa2_pairwise_cipher & WPA_CIPHER_TKIP) ||
  2070. (pstat->wpa_pairwise_cipher & WPA_CIPHER_TKIP))) {
  2071. RTW_INFO("(V)HT: " MAC_FMT " tried to use TKIP with (V)HT association\n", MAC_ARG(pstat->cmn.mac_addr));
  2072. pstat->flags &= ~WLAN_STA_HT;
  2073. pstat->flags &= ~WLAN_STA_VHT;
  2074. /*status = WLAN_STATUS_CIPHER_REJECTED_PER_POLICY;
  2075. * goto OnAssocReqFail;
  2076. */
  2077. }
  2078. if (status != _STATS_SUCCESSFUL_)
  2079. goto OnAssocReqFail;
  2080. #ifdef CONFIG_P2P
  2081. pstat->is_p2p_device = _FALSE;
  2082. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  2083. p2pie = rtw_get_p2p_ie(pframe + WLAN_HDR_A3_LEN + ie_offset , pkt_len - WLAN_HDR_A3_LEN - ie_offset , NULL, &p2pielen);
  2084. if (p2pie) {
  2085. pstat->is_p2p_device = _TRUE;
  2086. p2p_status_code = (u8)process_assoc_req_p2p_ie(pwdinfo, pframe, pkt_len, pstat);
  2087. if (p2p_status_code > 0) {
  2088. pstat->p2p_status_code = p2p_status_code;
  2089. status = _STATS_CAP_FAIL_;
  2090. goto OnAssocReqFail;
  2091. }
  2092. }
  2093. #ifdef CONFIG_WFD
  2094. rtw_process_wfd_ies(padapter, pframe + WLAN_HDR_A3_LEN + ie_offset, pkt_len - WLAN_HDR_A3_LEN - ie_offset, __func__);
  2095. #endif
  2096. }
  2097. pstat->p2p_status_code = p2p_status_code;
  2098. #endif /* CONFIG_P2P */
  2099. #ifdef CONFIG_RTW_REPEATER_SON
  2100. if (rtw_rson_ap_check_sta(padapter, pframe, pkt_len, ie_offset))
  2101. goto OnAssocReqFail;
  2102. #endif
  2103. /* TODO: identify_proprietary_vendor_ie(); */
  2104. /* Realtek proprietary IE */
  2105. /* identify if this is Broadcom sta */
  2106. /* identify if this is ralink sta */
  2107. /* Customer proprietary IE */
  2108. #ifdef CONFIG_RTW_80211K
  2109. rtw_ap_parse_sta_rm_en_cap(padapter, pstat, &elems);
  2110. #endif
  2111. /* AID assignment */
  2112. if (pstat->cmn.aid > 0)
  2113. RTW_INFO(FUNC_ADPT_FMT" old AID=%d\n", FUNC_ADPT_ARG(padapter), pstat->cmn.aid);
  2114. else {
  2115. if (!rtw_aid_alloc(padapter, pstat)) {
  2116. RTW_INFO(FUNC_ADPT_FMT" no room for more AIDs\n", FUNC_ADPT_ARG(padapter));
  2117. status = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
  2118. goto OnAssocReqFail;
  2119. }
  2120. RTW_INFO(FUNC_ADPT_FMT" allocate new AID=%d\n", FUNC_ADPT_ARG(padapter), pstat->cmn.aid);
  2121. }
  2122. pstat->state &= (~WIFI_FW_ASSOC_STATE);
  2123. pstat->state |= WIFI_FW_ASSOC_SUCCESS;
  2124. /* RTW_INFO("==================%s, %d, (%x), bpairwise_key_installed=%d, MAC:"MAC_FMT"\n"
  2125. , __func__, __LINE__, pstat->state, pstat->bpairwise_key_installed, MAC_ARG(pstat->cmn.mac_addr)); */
  2126. #ifdef CONFIG_IEEE80211W
  2127. if (pstat->bpairwise_key_installed != _TRUE)
  2128. #endif /* CONFIG_IEEE80211W */
  2129. {
  2130. _enter_critical_bh(&pstapriv->auth_list_lock, &irqL);
  2131. if (!rtw_is_list_empty(&pstat->auth_list)) {
  2132. rtw_list_delete(&pstat->auth_list);
  2133. pstapriv->auth_list_cnt--;
  2134. }
  2135. _exit_critical_bh(&pstapriv->auth_list_lock, &irqL);
  2136. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2137. if (rtw_is_list_empty(&pstat->asoc_list)) {
  2138. pstat->expire_to = pstapriv->expire_to;
  2139. rtw_list_insert_tail(&pstat->asoc_list, &pstapriv->asoc_list);
  2140. pstapriv->asoc_list_cnt++;
  2141. }
  2142. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2143. }
  2144. /* now the station is qualified to join our BSS... */
  2145. if (pstat && (pstat->state & WIFI_FW_ASSOC_SUCCESS) && (_STATS_SUCCESSFUL_ == status)) {
  2146. #ifdef CONFIG_NATIVEAP_MLME
  2147. #ifdef CONFIG_IEEE80211W
  2148. if (pstat->bpairwise_key_installed != _TRUE)
  2149. #endif /* CONFIG_IEEE80211W */
  2150. {
  2151. /* .1 bss_cap_update & sta_info_update */
  2152. bss_cap_update_on_sta_join(padapter, pstat);
  2153. sta_info_update(padapter, pstat);
  2154. }
  2155. #ifdef CONFIG_IEEE80211W
  2156. if (pstat->bpairwise_key_installed == _TRUE)
  2157. status = _STATS_REFUSED_TEMPORARILY_;
  2158. #endif /* CONFIG_IEEE80211W */
  2159. /* .2 issue assoc rsp before notify station join event. */
  2160. if (frame_type == WIFI_ASSOCREQ)
  2161. issue_asocrsp(padapter, status, pstat, WIFI_ASSOCRSP);
  2162. else
  2163. issue_asocrsp(padapter, status, pstat, WIFI_REASSOCRSP);
  2164. #ifdef CONFIG_IOCTL_CFG80211
  2165. _enter_critical_bh(&pstat->lock, &irqL);
  2166. if (pstat->passoc_req) {
  2167. rtw_mfree(pstat->passoc_req, pstat->assoc_req_len);
  2168. pstat->passoc_req = NULL;
  2169. pstat->assoc_req_len = 0;
  2170. }
  2171. pstat->passoc_req = rtw_zmalloc(pkt_len);
  2172. if (pstat->passoc_req) {
  2173. _rtw_memcpy(pstat->passoc_req, pframe, pkt_len);
  2174. pstat->assoc_req_len = pkt_len;
  2175. }
  2176. _exit_critical_bh(&pstat->lock, &irqL);
  2177. #endif /* CONFIG_IOCTL_CFG80211 */
  2178. #ifdef CONFIG_IEEE80211W
  2179. if (pstat->bpairwise_key_installed != _TRUE)
  2180. #endif /* CONFIG_IEEE80211W */
  2181. {
  2182. /* .3-(1) report sta add event */
  2183. report_add_sta_event(padapter, pstat->cmn.mac_addr);
  2184. }
  2185. #ifdef CONFIG_IEEE80211W
  2186. if (pstat->bpairwise_key_installed == _TRUE && SEC_IS_BIP_KEY_INSTALLED(&padapter->securitypriv) == _TRUE) {
  2187. RTW_INFO(MAC_FMT"\n", MAC_ARG(pstat->cmn.mac_addr));
  2188. issue_action_SA_Query(padapter, pstat->cmn.mac_addr, 0, 0, IEEE80211W_RIGHT_KEY);
  2189. }
  2190. #endif /* CONFIG_IEEE80211W */
  2191. #endif /* CONFIG_NATIVEAP_MLME */
  2192. }
  2193. return _SUCCESS;
  2194. asoc_class2_error:
  2195. #ifdef CONFIG_NATIVEAP_MLME
  2196. issue_deauth(padapter, (void *)get_addr2_ptr(pframe), status);
  2197. #endif
  2198. return _FAIL;
  2199. OnAssocReqFail:
  2200. #ifdef CONFIG_NATIVEAP_MLME
  2201. pstat->cmn.aid = 0;
  2202. if (frame_type == WIFI_ASSOCREQ)
  2203. issue_asocrsp(padapter, status, pstat, WIFI_ASSOCRSP);
  2204. else
  2205. issue_asocrsp(padapter, status, pstat, WIFI_REASSOCRSP);
  2206. #endif
  2207. #endif /* CONFIG_AP_MODE */
  2208. return _FAIL;
  2209. }
  2210. #if defined(CONFIG_LAYER2_ROAMING) && defined(CONFIG_RTW_80211K)
  2211. void rtw_roam_nb_discover(_adapter *padapter, u8 bfroce)
  2212. {
  2213. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  2214. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  2215. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  2216. struct sta_priv *pstapriv = &padapter->stapriv;
  2217. struct sta_info *psta;
  2218. u8 nb_req_issue = _FALSE;
  2219. if (!check_fwstate(pmlmepriv, _FW_LINKED))
  2220. return;
  2221. if (!rtw_chk_roam_flags(padapter, RTW_ROAM_ACTIVE))
  2222. return;
  2223. psta = rtw_get_stainfo(pstapriv, pmlmeinfo->network.MacAddress);
  2224. if (!psta)
  2225. return;
  2226. if (bfroce || (!pmlmepriv->nb_info.nb_rpt_is_same))
  2227. nb_req_issue = _TRUE;
  2228. if (nb_req_issue && (psta->rm_en_cap[0] & RTW_RRM_NB_RPT_EN))
  2229. rm_add_nb_req(padapter, psta);
  2230. }
  2231. #endif
  2232. unsigned int OnAssocRsp(_adapter *padapter, union recv_frame *precv_frame)
  2233. {
  2234. uint i;
  2235. int res;
  2236. unsigned short status;
  2237. PNDIS_802_11_VARIABLE_IEs pIE;
  2238. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  2239. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  2240. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  2241. /* WLAN_BSSID_EX *cur_network = &(pmlmeinfo->network); */
  2242. u8 *pframe = precv_frame->u.hdr.rx_data;
  2243. uint pkt_len = precv_frame->u.hdr.len;
  2244. #ifdef CONFIG_WAPI_SUPPORT
  2245. PNDIS_802_11_VARIABLE_IEs pWapiIE = NULL;
  2246. #endif
  2247. RTW_INFO("%s\n", __FUNCTION__);
  2248. /* check A1 matches or not */
  2249. if (!_rtw_memcmp(adapter_mac_addr(padapter), get_da(pframe), ETH_ALEN))
  2250. return _SUCCESS;
  2251. if (!(pmlmeinfo->state & (WIFI_FW_AUTH_SUCCESS | WIFI_FW_ASSOC_STATE)) || pmlmeext->join_abort)
  2252. return _SUCCESS;
  2253. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)
  2254. return _SUCCESS;
  2255. _cancel_timer_ex(&pmlmeext->link_timer);
  2256. /* status */
  2257. status = le16_to_cpu(*(unsigned short *)(pframe + WLAN_HDR_A3_LEN + 2));
  2258. if (status > 0) {
  2259. RTW_INFO("assoc reject, status code: %d\n", status);
  2260. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  2261. res = -4;
  2262. goto report_assoc_result;
  2263. }
  2264. /* get capabilities */
  2265. pmlmeinfo->capability = le16_to_cpu(*(unsigned short *)(pframe + WLAN_HDR_A3_LEN));
  2266. /* set slot time */
  2267. pmlmeinfo->slotTime = (pmlmeinfo->capability & BIT(10)) ? 9 : 20;
  2268. /* AID */
  2269. res = pmlmeinfo->aid = (int)(le16_to_cpu(*(unsigned short *)(pframe + WLAN_HDR_A3_LEN + 4)) & 0x3fff);
  2270. /* following are moved to join event callback function */
  2271. /* to handle HT, WMM, rate adaptive, update MAC reg */
  2272. /* for not to handle the synchronous IO in the tasklet */
  2273. for (i = (6 + WLAN_HDR_A3_LEN); i < pkt_len;) {
  2274. pIE = (PNDIS_802_11_VARIABLE_IEs)(pframe + i);
  2275. switch (pIE->ElementID) {
  2276. case _VENDOR_SPECIFIC_IE_:
  2277. if (_rtw_memcmp(pIE->data, WMM_PARA_OUI, 6)) /* WMM */
  2278. WMM_param_handler(padapter, pIE);
  2279. #if defined(CONFIG_P2P) && defined(CONFIG_WFD)
  2280. else if (_rtw_memcmp(pIE->data, WFD_OUI, 4)) /* WFD */
  2281. rtw_process_wfd_ie(padapter, (u8 *)pIE, pIE->Length, __func__);
  2282. #endif
  2283. break;
  2284. #ifdef CONFIG_WAPI_SUPPORT
  2285. case _WAPI_IE_:
  2286. pWapiIE = pIE;
  2287. break;
  2288. #endif
  2289. case _HT_CAPABILITY_IE_: /* HT caps */
  2290. HT_caps_handler(padapter, pIE);
  2291. #ifdef ROKU_PRIVATE
  2292. HT_caps_handler_infra_ap(padapter, pIE);
  2293. #endif /* ROKU_PRIVATE */
  2294. break;
  2295. case _HT_EXTRA_INFO_IE_: /* HT info */
  2296. HT_info_handler(padapter, pIE);
  2297. break;
  2298. #ifdef CONFIG_80211AC_VHT
  2299. case EID_VHTCapability:
  2300. VHT_caps_handler(padapter, pIE);
  2301. #ifdef ROKU_PRIVATE
  2302. VHT_caps_handler_infra_ap(padapter, pIE);
  2303. #endif /* ROKU_PRIVATE */
  2304. break;
  2305. case EID_VHTOperation:
  2306. VHT_operation_handler(padapter, pIE);
  2307. break;
  2308. #endif
  2309. case _ERPINFO_IE_:
  2310. ERP_IE_handler(padapter, pIE);
  2311. break;
  2312. #ifdef CONFIG_TDLS
  2313. case _EXT_CAP_IE_:
  2314. if (check_ap_tdls_prohibited(pIE->data, pIE->Length) == _TRUE)
  2315. padapter->tdlsinfo.ap_prohibited = _TRUE;
  2316. if (check_ap_tdls_ch_switching_prohibited(pIE->data, pIE->Length) == _TRUE)
  2317. padapter->tdlsinfo.ch_switch_prohibited = _TRUE;
  2318. break;
  2319. #endif /* CONFIG_TDLS */
  2320. #ifdef CONFIG_RTW_80211K
  2321. case _EID_RRM_EN_CAP_IE_:
  2322. RM_IE_handler(padapter, pIE);
  2323. break;
  2324. #endif
  2325. #ifdef ROKU_PRIVATE
  2326. /* Infra mode, used to store AP's info , Parse the supported rates from AssocRsp */
  2327. case _SUPPORTEDRATES_IE_:
  2328. Supported_rate_infra_ap(padapter, pIE);
  2329. break;
  2330. case _EXT_SUPPORTEDRATES_IE_:
  2331. Extended_Supported_rate_infra_ap(padapter, pIE);
  2332. break;
  2333. #endif /* ROKU_PRIVATE */
  2334. default:
  2335. break;
  2336. }
  2337. i += (pIE->Length + 2);
  2338. }
  2339. #ifdef CONFIG_WAPI_SUPPORT
  2340. rtw_wapi_on_assoc_ok(padapter, pIE);
  2341. #endif
  2342. pmlmeinfo->state &= (~WIFI_FW_ASSOC_STATE);
  2343. pmlmeinfo->state |= WIFI_FW_ASSOC_SUCCESS;
  2344. /* Update Basic Rate Table for spec, 2010-12-28 , by thomas */
  2345. UpdateBrateTbl(padapter, pmlmeinfo->network.SupportedRates);
  2346. report_assoc_result:
  2347. if (res > 0)
  2348. rtw_buf_update(&pmlmepriv->assoc_rsp, &pmlmepriv->assoc_rsp_len, pframe, pkt_len);
  2349. else
  2350. rtw_buf_free(&pmlmepriv->assoc_rsp, &pmlmepriv->assoc_rsp_len);
  2351. report_join_res(padapter, res, status);
  2352. #if defined(CONFIG_LAYER2_ROAMING) && defined(CONFIG_RTW_80211K)
  2353. rtw_roam_nb_discover(padapter, _TRUE);
  2354. #endif
  2355. return _SUCCESS;
  2356. }
  2357. unsigned int OnDeAuth(_adapter *padapter, union recv_frame *precv_frame)
  2358. {
  2359. unsigned short reason;
  2360. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  2361. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  2362. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  2363. u8 *pframe = precv_frame->u.hdr.rx_data;
  2364. #ifdef CONFIG_P2P
  2365. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  2366. #endif /* CONFIG_P2P */
  2367. /* check A3 */
  2368. if (!(_rtw_memcmp(GetAddr3Ptr(pframe), get_my_bssid(&pmlmeinfo->network), ETH_ALEN)))
  2369. return _SUCCESS;
  2370. RTW_INFO(FUNC_ADPT_FMT" - Start to Disconnect\n", FUNC_ADPT_ARG(padapter));
  2371. #ifdef CONFIG_P2P
  2372. if (pwdinfo->rx_invitereq_info.scan_op_ch_only) {
  2373. _cancel_timer_ex(&pwdinfo->reset_ch_sitesurvey);
  2374. _set_timer(&pwdinfo->reset_ch_sitesurvey, 10);
  2375. }
  2376. #endif /* CONFIG_P2P */
  2377. reason = le16_to_cpu(*(unsigned short *)(pframe + WLAN_HDR_A3_LEN));
  2378. #ifdef CONFIG_AP_MODE
  2379. if (MLME_IS_AP(padapter)) {
  2380. _irqL irqL;
  2381. struct sta_info *psta;
  2382. struct sta_priv *pstapriv = &padapter->stapriv;
  2383. /* _enter_critical_bh(&(pstapriv->sta_hash_lock), &irqL); */
  2384. /* rtw_free_stainfo(padapter, psta); */
  2385. /* _exit_critical_bh(&(pstapriv->sta_hash_lock), &irqL); */
  2386. RTW_PRINT(FUNC_ADPT_FMT" reason=%u, ta=%pM\n"
  2387. , FUNC_ADPT_ARG(padapter), reason, get_addr2_ptr(pframe));
  2388. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  2389. if (psta) {
  2390. u8 updated = _FALSE;
  2391. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2392. if (rtw_is_list_empty(&psta->asoc_list) == _FALSE) {
  2393. rtw_list_delete(&psta->asoc_list);
  2394. pstapriv->asoc_list_cnt--;
  2395. updated = ap_free_sta(padapter, psta, _FALSE, reason, _TRUE);
  2396. }
  2397. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2398. associated_clients_update(padapter, updated, STA_INFO_UPDATE_ALL);
  2399. }
  2400. return _SUCCESS;
  2401. } else
  2402. #endif
  2403. if (!MLME_IS_MESH(padapter)) {
  2404. int ignore_received_deauth = 0;
  2405. /* Commented by Albert 20130604 */
  2406. /* Before sending the auth frame to start the STA/GC mode connection with AP/GO, */
  2407. /* we will send the deauth first. */
  2408. /* However, the Win8.1 with BRCM Wi-Fi will send the deauth with reason code 6 to us after receieving our deauth. */
  2409. /* Added the following code to avoid this case. */
  2410. if ((pmlmeinfo->state & WIFI_FW_AUTH_STATE) ||
  2411. (pmlmeinfo->state & WIFI_FW_ASSOC_STATE)) {
  2412. if (reason == WLAN_REASON_CLASS2_FRAME_FROM_NONAUTH_STA)
  2413. ignore_received_deauth = 1;
  2414. else if (WLAN_REASON_PREV_AUTH_NOT_VALID == reason) {
  2415. /* TODO: 802.11r */
  2416. ignore_received_deauth = 1;
  2417. }
  2418. }
  2419. RTW_PRINT(FUNC_ADPT_FMT" reason=%u, ta=%pM, ignore=%d\n"
  2420. , FUNC_ADPT_ARG(padapter), reason, get_addr2_ptr(pframe), ignore_received_deauth);
  2421. if (0 == ignore_received_deauth)
  2422. receive_disconnect(padapter, get_addr2_ptr(pframe), reason, _FALSE);
  2423. }
  2424. pmlmepriv->LinkDetectInfo.bBusyTraffic = _FALSE;
  2425. return _SUCCESS;
  2426. }
  2427. unsigned int OnDisassoc(_adapter *padapter, union recv_frame *precv_frame)
  2428. {
  2429. unsigned short reason;
  2430. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  2431. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  2432. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  2433. u8 *pframe = precv_frame->u.hdr.rx_data;
  2434. #ifdef CONFIG_P2P
  2435. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  2436. #endif /* CONFIG_P2P */
  2437. /* check A3 */
  2438. if (!(_rtw_memcmp(GetAddr3Ptr(pframe), get_my_bssid(&pmlmeinfo->network), ETH_ALEN)))
  2439. return _SUCCESS;
  2440. RTW_INFO(FUNC_ADPT_FMT" - Start to Disconnect\n", FUNC_ADPT_ARG(padapter));
  2441. #ifdef CONFIG_P2P
  2442. if (pwdinfo->rx_invitereq_info.scan_op_ch_only) {
  2443. _cancel_timer_ex(&pwdinfo->reset_ch_sitesurvey);
  2444. _set_timer(&pwdinfo->reset_ch_sitesurvey, 10);
  2445. }
  2446. #endif /* CONFIG_P2P */
  2447. reason = le16_to_cpu(*(unsigned short *)(pframe + WLAN_HDR_A3_LEN));
  2448. #ifdef CONFIG_AP_MODE
  2449. if (MLME_IS_AP(padapter)) {
  2450. _irqL irqL;
  2451. struct sta_info *psta;
  2452. struct sta_priv *pstapriv = &padapter->stapriv;
  2453. /* _enter_critical_bh(&(pstapriv->sta_hash_lock), &irqL); */
  2454. /* rtw_free_stainfo(padapter, psta); */
  2455. /* _exit_critical_bh(&(pstapriv->sta_hash_lock), &irqL); */
  2456. RTW_PRINT(FUNC_ADPT_FMT" reason=%u, ta=%pM\n"
  2457. , FUNC_ADPT_ARG(padapter), reason, get_addr2_ptr(pframe));
  2458. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  2459. if (psta) {
  2460. u8 updated = _FALSE;
  2461. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2462. if (rtw_is_list_empty(&psta->asoc_list) == _FALSE) {
  2463. rtw_list_delete(&psta->asoc_list);
  2464. pstapriv->asoc_list_cnt--;
  2465. updated = ap_free_sta(padapter, psta, _FALSE, reason, _TRUE);
  2466. }
  2467. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2468. associated_clients_update(padapter, updated, STA_INFO_UPDATE_ALL);
  2469. }
  2470. return _SUCCESS;
  2471. } else
  2472. #endif
  2473. if (!MLME_IS_MESH(padapter)) {
  2474. RTW_PRINT(FUNC_ADPT_FMT" reason=%u, ta=%pM\n"
  2475. , FUNC_ADPT_ARG(padapter), reason, get_addr2_ptr(pframe));
  2476. receive_disconnect(padapter, get_addr2_ptr(pframe), reason, _FALSE);
  2477. }
  2478. pmlmepriv->LinkDetectInfo.bBusyTraffic = _FALSE;
  2479. return _SUCCESS;
  2480. }
  2481. unsigned int OnAtim(_adapter *padapter, union recv_frame *precv_frame)
  2482. {
  2483. RTW_INFO("%s\n", __FUNCTION__);
  2484. return _SUCCESS;
  2485. }
  2486. unsigned int on_action_spct_ch_switch(_adapter *padapter, struct sta_info *psta, u8 *ies, uint ies_len)
  2487. {
  2488. unsigned int ret = _FAIL;
  2489. struct mlme_ext_priv *mlmeext = &padapter->mlmeextpriv;
  2490. struct mlme_ext_info *pmlmeinfo = &(mlmeext->mlmext_info);
  2491. if (!(pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)) {
  2492. ret = _SUCCESS;
  2493. goto exit;
  2494. }
  2495. if ((pmlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE) {
  2496. int ch_switch_mode = -1, ch = -1, ch_switch_cnt = -1;
  2497. int ch_offset = -1;
  2498. u8 bwmode;
  2499. struct ieee80211_info_element *ie;
  2500. RTW_INFO(FUNC_NDEV_FMT" from "MAC_FMT"\n",
  2501. FUNC_NDEV_ARG(padapter->pnetdev), MAC_ARG(psta->cmn.mac_addr));
  2502. for_each_ie(ie, ies, ies_len) {
  2503. if (ie->id == WLAN_EID_CHANNEL_SWITCH) {
  2504. ch_switch_mode = ie->data[0];
  2505. ch = ie->data[1];
  2506. ch_switch_cnt = ie->data[2];
  2507. RTW_INFO("ch_switch_mode:%d, ch:%d, ch_switch_cnt:%d\n",
  2508. ch_switch_mode, ch, ch_switch_cnt);
  2509. } else if (ie->id == WLAN_EID_SECONDARY_CHANNEL_OFFSET) {
  2510. ch_offset = secondary_ch_offset_to_hal_ch_offset(ie->data[0]);
  2511. RTW_INFO("ch_offset:%d\n", ch_offset);
  2512. }
  2513. }
  2514. if (ch == -1)
  2515. return _SUCCESS;
  2516. if (ch_offset == -1)
  2517. bwmode = mlmeext->cur_bwmode;
  2518. else
  2519. bwmode = (ch_offset == HAL_PRIME_CHNL_OFFSET_DONT_CARE) ?
  2520. CHANNEL_WIDTH_20 : CHANNEL_WIDTH_40;
  2521. ch_offset = (ch_offset == -1) ? mlmeext->cur_ch_offset : ch_offset;
  2522. /* todo:
  2523. * 1. the decision of channel switching
  2524. * 2. things after channel switching
  2525. */
  2526. ret = rtw_set_chbw_cmd(padapter, ch, bwmode, ch_offset, 0);
  2527. }
  2528. exit:
  2529. return ret;
  2530. }
  2531. unsigned int on_action_spct(_adapter *padapter, union recv_frame *precv_frame)
  2532. {
  2533. unsigned int ret = _FAIL;
  2534. struct sta_info *psta = NULL;
  2535. struct sta_priv *pstapriv = &padapter->stapriv;
  2536. u8 *pframe = precv_frame->u.hdr.rx_data;
  2537. uint frame_len = precv_frame->u.hdr.len;
  2538. u8 *frame_body = (u8 *)(pframe + sizeof(struct rtw_ieee80211_hdr_3addr));
  2539. u8 category;
  2540. u8 action;
  2541. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  2542. if (!psta)
  2543. goto exit;
  2544. category = frame_body[0];
  2545. if (category != RTW_WLAN_CATEGORY_SPECTRUM_MGMT)
  2546. goto exit;
  2547. action = frame_body[1];
  2548. RTW_INFO(FUNC_ADPT_FMT" action:%u\n", FUNC_ADPT_ARG(padapter), action);
  2549. switch (action) {
  2550. case RTW_WLAN_ACTION_SPCT_MSR_REQ:
  2551. case RTW_WLAN_ACTION_SPCT_MSR_RPRT:
  2552. case RTW_WLAN_ACTION_SPCT_TPC_REQ:
  2553. case RTW_WLAN_ACTION_SPCT_TPC_RPRT:
  2554. break;
  2555. case RTW_WLAN_ACTION_SPCT_CHL_SWITCH:
  2556. #ifdef CONFIG_SPCT_CH_SWITCH
  2557. ret = on_action_spct_ch_switch(padapter, psta
  2558. , frame_body + 2, frame_len - (frame_body - pframe) - 2);
  2559. #elif defined(CONFIG_DFS)
  2560. if (MLME_IS_STA(padapter) && MLME_IS_ASOC(padapter)) {
  2561. process_csa_ie(padapter
  2562. , frame_body + 2, frame_len - (frame_body - pframe) - 2);
  2563. }
  2564. #endif
  2565. break;
  2566. default:
  2567. break;
  2568. }
  2569. exit:
  2570. return ret;
  2571. }
  2572. unsigned int OnAction_qos(_adapter *padapter, union recv_frame *precv_frame)
  2573. {
  2574. return _SUCCESS;
  2575. }
  2576. unsigned int OnAction_dls(_adapter *padapter, union recv_frame *precv_frame)
  2577. {
  2578. return _SUCCESS;
  2579. }
  2580. #ifdef CONFIG_RTW_WNM
  2581. unsigned int on_action_wnm(_adapter *adapter, union recv_frame *rframe)
  2582. {
  2583. unsigned int ret = _FAIL;
  2584. struct sta_info *sta = NULL;
  2585. struct mlme_priv *pmlmepriv = &(adapter->mlmepriv);
  2586. struct sta_priv *stapriv = &(adapter->stapriv);
  2587. u8 *frame = rframe->u.hdr.rx_data;
  2588. u32 frame_len = rframe->u.hdr.len;
  2589. u8 *frame_body = (u8 *)(frame + sizeof(struct rtw_ieee80211_hdr_3addr));
  2590. u32 frame_body_len = frame_len - sizeof(struct rtw_ieee80211_hdr_3addr);
  2591. u8 category, action;
  2592. int cnt = 0;
  2593. char msg[16];
  2594. sta = rtw_get_stainfo(stapriv, get_addr2_ptr(frame));
  2595. if (!sta)
  2596. goto exit;
  2597. category = frame_body[0];
  2598. if (category != RTW_WLAN_CATEGORY_WNM)
  2599. goto exit;
  2600. action = frame_body[1];
  2601. switch (action) {
  2602. #ifdef CONFIG_RTW_80211R
  2603. case RTW_WLAN_ACTION_WNM_BTM_REQ:
  2604. if (check_fwstate(pmlmepriv, WIFI_STATION_STATE) == _TRUE) {
  2605. RTW_INFO("WNM: RTW_WLAN_ACTION_WNM_BTM_REQ recv.\n");
  2606. rtw_wnm_process_btm_req(adapter, frame_body, frame_body_len);
  2607. }
  2608. ret = _SUCCESS;
  2609. break;
  2610. #endif
  2611. default:
  2612. #ifdef CONFIG_IOCTL_CFG80211
  2613. cnt += sprintf((msg + cnt), "ACT_WNM %u", action);
  2614. rtw_cfg80211_rx_action(adapter, rframe, msg);
  2615. #endif
  2616. ret = _SUCCESS;
  2617. break;
  2618. }
  2619. exit:
  2620. return ret;
  2621. }
  2622. #endif /* CONFIG_RTW_WNM */
  2623. /**
  2624. * rtw_rx_ampdu_size - Get the target RX AMPDU buffer size for the specific @adapter
  2625. * @adapter: the adapter to get target RX AMPDU buffer size
  2626. *
  2627. * Returns: the target RX AMPDU buffer size
  2628. */
  2629. u8 rtw_rx_ampdu_size(_adapter *adapter)
  2630. {
  2631. u8 size;
  2632. HT_CAP_AMPDU_FACTOR max_rx_ampdu_factor;
  2633. #ifdef CONFIG_BT_COEXIST
  2634. if (rtw_btcoex_IsBTCoexCtrlAMPDUSize(adapter) == _TRUE) {
  2635. size = rtw_btcoex_GetAMPDUSize(adapter);
  2636. goto exit;
  2637. }
  2638. #endif
  2639. /* for scan */
  2640. if (!mlmeext_chk_scan_state(&adapter->mlmeextpriv, SCAN_DISABLE)
  2641. && !mlmeext_chk_scan_state(&adapter->mlmeextpriv, SCAN_COMPLETE)
  2642. && adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_size != RX_AMPDU_SIZE_INVALID
  2643. ) {
  2644. size = adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_size;
  2645. goto exit;
  2646. }
  2647. /* default value based on max_rx_ampdu_factor */
  2648. if (adapter->driver_rx_ampdu_factor != 0xFF)
  2649. max_rx_ampdu_factor = (HT_CAP_AMPDU_FACTOR)adapter->driver_rx_ampdu_factor;
  2650. else
  2651. rtw_hal_get_def_var(adapter, HW_VAR_MAX_RX_AMPDU_FACTOR, &max_rx_ampdu_factor);
  2652. /* In Maximum A-MPDU Length Exponent subfield of A-MPDU Parameters field of HT Capabilities element,
  2653. the unit of max_rx_ampdu_factor are octets. 8K, 16K, 32K, 64K is right.
  2654. But the buffer size subfield of Block Ack Parameter Set field in ADDBA action frame indicates
  2655. the number of buffers available for this particular TID. Each buffer is equal to max. size of
  2656. MSDU or AMSDU.
  2657. The size variable means how many MSDUs or AMSDUs, it's not Kbytes.
  2658. */
  2659. if (MAX_AMPDU_FACTOR_64K == max_rx_ampdu_factor)
  2660. size = 64;
  2661. else if (MAX_AMPDU_FACTOR_32K == max_rx_ampdu_factor)
  2662. size = 32;
  2663. else if (MAX_AMPDU_FACTOR_16K == max_rx_ampdu_factor)
  2664. size = 16;
  2665. else if (MAX_AMPDU_FACTOR_8K == max_rx_ampdu_factor)
  2666. size = 8;
  2667. else
  2668. size = 64;
  2669. exit:
  2670. if (size > 127)
  2671. size = 127;
  2672. return size;
  2673. }
  2674. /**
  2675. * rtw_rx_ampdu_is_accept - Get the permission if RX AMPDU should be set up for the specific @adapter
  2676. * @adapter: the adapter to get the permission if RX AMPDU should be set up
  2677. *
  2678. * Returns: accept or not
  2679. */
  2680. bool rtw_rx_ampdu_is_accept(_adapter *adapter)
  2681. {
  2682. bool accept;
  2683. if (adapter->fix_rx_ampdu_accept != RX_AMPDU_ACCEPT_INVALID) {
  2684. accept = adapter->fix_rx_ampdu_accept;
  2685. goto exit;
  2686. }
  2687. #ifdef CONFIG_BT_COEXIST
  2688. if (rtw_btcoex_IsBTCoexRejectAMPDU(adapter) == _TRUE) {
  2689. accept = _FALSE;
  2690. goto exit;
  2691. }
  2692. #endif
  2693. /* for scan */
  2694. if (!mlmeext_chk_scan_state(&adapter->mlmeextpriv, SCAN_DISABLE)
  2695. && !mlmeext_chk_scan_state(&adapter->mlmeextpriv, SCAN_COMPLETE)
  2696. && adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_accept != RX_AMPDU_ACCEPT_INVALID
  2697. ) {
  2698. accept = adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_accept;
  2699. goto exit;
  2700. }
  2701. /* default value for other cases */
  2702. accept = adapter->mlmeextpriv.mlmext_info.bAcceptAddbaReq;
  2703. exit:
  2704. return accept;
  2705. }
  2706. /**
  2707. * rtw_rx_ampdu_set_size - Set the target RX AMPDU buffer size for the specific @adapter and specific @reason
  2708. * @adapter: the adapter to set target RX AMPDU buffer size
  2709. * @size: the target RX AMPDU buffer size to set
  2710. * @reason: reason for the target RX AMPDU buffer size setting
  2711. *
  2712. * Returns: whether the target RX AMPDU buffer size is changed
  2713. */
  2714. bool rtw_rx_ampdu_set_size(_adapter *adapter, u8 size, u8 reason)
  2715. {
  2716. bool is_adj = _FALSE;
  2717. struct mlme_ext_priv *mlmeext;
  2718. struct mlme_ext_info *mlmeinfo;
  2719. mlmeext = &adapter->mlmeextpriv;
  2720. mlmeinfo = &mlmeext->mlmext_info;
  2721. if (reason == RX_AMPDU_DRV_FIXED) {
  2722. if (adapter->fix_rx_ampdu_size != size) {
  2723. adapter->fix_rx_ampdu_size = size;
  2724. is_adj = _TRUE;
  2725. RTW_INFO(FUNC_ADPT_FMT" fix_rx_ampdu_size:%u\n", FUNC_ADPT_ARG(adapter), size);
  2726. }
  2727. } else if (reason == RX_AMPDU_DRV_SCAN) {
  2728. struct ss_res *ss = &adapter->mlmeextpriv.sitesurvey_res;
  2729. if (ss->rx_ampdu_size != size) {
  2730. ss->rx_ampdu_size = size;
  2731. is_adj = _TRUE;
  2732. RTW_INFO(FUNC_ADPT_FMT" ss.rx_ampdu_size:%u\n", FUNC_ADPT_ARG(adapter), size);
  2733. }
  2734. }
  2735. return is_adj;
  2736. }
  2737. /**
  2738. * rtw_rx_ampdu_set_accept - Set the permission if RX AMPDU should be set up for the specific @adapter and specific @reason
  2739. * @adapter: the adapter to set if RX AMPDU should be set up
  2740. * @accept: if RX AMPDU should be set up
  2741. * @reason: reason for the permission if RX AMPDU should be set up
  2742. *
  2743. * Returns: whether the permission if RX AMPDU should be set up is changed
  2744. */
  2745. bool rtw_rx_ampdu_set_accept(_adapter *adapter, u8 accept, u8 reason)
  2746. {
  2747. bool is_adj = _FALSE;
  2748. struct mlme_ext_priv *mlmeext;
  2749. struct mlme_ext_info *mlmeinfo;
  2750. mlmeext = &adapter->mlmeextpriv;
  2751. mlmeinfo = &mlmeext->mlmext_info;
  2752. if (reason == RX_AMPDU_DRV_FIXED) {
  2753. if (adapter->fix_rx_ampdu_accept != accept) {
  2754. adapter->fix_rx_ampdu_accept = accept;
  2755. is_adj = _TRUE;
  2756. RTW_INFO(FUNC_ADPT_FMT" fix_rx_ampdu_accept:%u\n", FUNC_ADPT_ARG(adapter), accept);
  2757. }
  2758. } else if (reason == RX_AMPDU_DRV_SCAN) {
  2759. if (adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_accept != accept) {
  2760. adapter->mlmeextpriv.sitesurvey_res.rx_ampdu_accept = accept;
  2761. is_adj = _TRUE;
  2762. RTW_INFO(FUNC_ADPT_FMT" ss.rx_ampdu_accept:%u\n", FUNC_ADPT_ARG(adapter), accept);
  2763. }
  2764. }
  2765. return is_adj;
  2766. }
  2767. /**
  2768. * rx_ampdu_apply_sta_tid - Apply RX AMPDU setting to the specific @sta and @tid
  2769. * @adapter: the adapter to which @sta belongs
  2770. * @sta: the sta to be checked
  2771. * @tid: the tid to be checked
  2772. * @accept: the target permission if RX AMPDU should be set up
  2773. * @size: the target RX AMPDU buffer size
  2774. *
  2775. * Returns:
  2776. * 0: no canceled
  2777. * 1: canceled by no permission
  2778. * 2: canceled by different buffer size
  2779. * 3: canceled by potential mismatched status
  2780. *
  2781. * Blocking function, may sleep
  2782. */
  2783. u8 rx_ampdu_apply_sta_tid(_adapter *adapter, struct sta_info *sta, u8 tid, u8 accept, u8 size)
  2784. {
  2785. u8 ret = 0;
  2786. struct recv_reorder_ctrl *reorder_ctl = &sta->recvreorder_ctrl[tid];
  2787. if (reorder_ctl->enable == _FALSE) {
  2788. if (reorder_ctl->ampdu_size != RX_AMPDU_SIZE_INVALID) {
  2789. send_delba_sta_tid_wait_ack(adapter, 0, sta, tid, 1);
  2790. ret = 3;
  2791. }
  2792. goto exit;
  2793. }
  2794. if (accept == _FALSE) {
  2795. send_delba_sta_tid_wait_ack(adapter, 0, sta, tid, 0);
  2796. ret = 1;
  2797. } else if (reorder_ctl->ampdu_size != size) {
  2798. send_delba_sta_tid_wait_ack(adapter, 0, sta, tid, 0);
  2799. ret = 2;
  2800. }
  2801. exit:
  2802. return ret;
  2803. }
  2804. u8 rx_ampdu_size_sta_limit(_adapter *adapter, struct sta_info *sta)
  2805. {
  2806. u8 sz_limit = 0xFF;
  2807. #ifdef CONFIG_80211N_HT
  2808. struct registry_priv *regsty = adapter_to_regsty(adapter);
  2809. struct mlme_priv *mlme = &adapter->mlmepriv;
  2810. struct mlme_ext_info *mlmeinfo = &adapter->mlmeextpriv.mlmext_info;
  2811. s8 nss = -1;
  2812. u8 bw = rtw_min(sta->cmn.bw_mode, adapter->mlmeextpriv.cur_bwmode);
  2813. #ifdef CONFIG_80211AC_VHT
  2814. if (is_supported_vht(sta->wireless_mode)) {
  2815. nss = rtw_min(rtw_vht_mcsmap_to_nss(mlme->vhtpriv.vht_mcs_map)
  2816. , rtw_vht_mcsmap_to_nss(sta->vhtpriv.vht_mcs_map));
  2817. } else
  2818. #endif
  2819. if (is_supported_ht(sta->wireless_mode)) {
  2820. nss = rtw_min(rtw_ht_mcsset_to_nss(mlmeinfo->HT_caps.u.HT_cap_element.MCS_rate)
  2821. , rtw_ht_mcsset_to_nss(sta->htpriv.ht_cap.supp_mcs_set));
  2822. }
  2823. if (nss >= 1)
  2824. sz_limit = regsty->rx_ampdu_sz_limit_by_nss_bw[nss - 1][bw];
  2825. #endif /* CONFIG_80211N_HT */
  2826. return sz_limit;
  2827. }
  2828. /**
  2829. * rx_ampdu_apply_sta - Apply RX AMPDU setting to the specific @sta
  2830. * @adapter: the adapter to which @sta belongs
  2831. * @sta: the sta to be checked
  2832. * @accept: the target permission if RX AMPDU should be set up
  2833. * @size: the target RX AMPDU buffer size
  2834. *
  2835. * Returns: number of the RX AMPDU assciation canceled for applying current target setting
  2836. *
  2837. * Blocking function, may sleep
  2838. */
  2839. u8 rx_ampdu_apply_sta(_adapter *adapter, struct sta_info *sta, u8 accept, u8 size)
  2840. {
  2841. u8 change_cnt = 0;
  2842. int i;
  2843. for (i = 0; i < TID_NUM; i++) {
  2844. if (rx_ampdu_apply_sta_tid(adapter, sta, i, accept, size) != 0)
  2845. change_cnt++;
  2846. }
  2847. return change_cnt;
  2848. }
  2849. /**
  2850. * rtw_rx_ampdu_apply - Apply the current target RX AMPDU setting for the specific @adapter
  2851. * @adapter: the adapter to be applied
  2852. *
  2853. * Returns: number of the RX AMPDU assciation canceled for applying current target setting
  2854. */
  2855. u16 rtw_rx_ampdu_apply(_adapter *adapter)
  2856. {
  2857. u16 adj_cnt = 0;
  2858. struct sta_info *sta;
  2859. u8 accept = rtw_rx_ampdu_is_accept(adapter);
  2860. u8 size;
  2861. if (adapter->fix_rx_ampdu_size != RX_AMPDU_SIZE_INVALID)
  2862. size = adapter->fix_rx_ampdu_size;
  2863. else
  2864. size = rtw_rx_ampdu_size(adapter);
  2865. if (MLME_IS_STA(adapter)) {
  2866. sta = rtw_get_stainfo(&adapter->stapriv, get_bssid(&adapter->mlmepriv));
  2867. if (sta) {
  2868. u8 sta_size = size;
  2869. if (adapter->fix_rx_ampdu_size == RX_AMPDU_SIZE_INVALID)
  2870. sta_size = rtw_min(size, rx_ampdu_size_sta_limit(adapter, sta));
  2871. adj_cnt += rx_ampdu_apply_sta(adapter, sta, accept, sta_size);
  2872. }
  2873. /* TODO: TDLS peer */
  2874. } else if (MLME_IS_AP(adapter) || MLME_IS_MESH(adapter)) {
  2875. _irqL irqL;
  2876. _list *phead, *plist;
  2877. u8 peer_num = 0;
  2878. char peers[NUM_STA];
  2879. struct sta_priv *pstapriv = &adapter->stapriv;
  2880. int i;
  2881. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2882. phead = &pstapriv->asoc_list;
  2883. plist = get_next(phead);
  2884. while ((rtw_end_of_queue_search(phead, plist)) == _FALSE) {
  2885. int stainfo_offset;
  2886. sta = LIST_CONTAINOR(plist, struct sta_info, asoc_list);
  2887. plist = get_next(plist);
  2888. stainfo_offset = rtw_stainfo_offset(pstapriv, sta);
  2889. if (stainfo_offset_valid(stainfo_offset))
  2890. peers[peer_num++] = stainfo_offset;
  2891. }
  2892. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  2893. for (i = 0; i < peer_num; i++) {
  2894. sta = rtw_get_stainfo_by_offset(pstapriv, peers[i]);
  2895. if (sta) {
  2896. u8 sta_size = size;
  2897. if (adapter->fix_rx_ampdu_size == RX_AMPDU_SIZE_INVALID)
  2898. sta_size = rtw_min(size, rx_ampdu_size_sta_limit(adapter, sta));
  2899. adj_cnt += rx_ampdu_apply_sta(adapter, sta, accept, sta_size);
  2900. }
  2901. }
  2902. }
  2903. /* TODO: ADHOC */
  2904. return adj_cnt;
  2905. }
  2906. unsigned int OnAction_back(_adapter *padapter, union recv_frame *precv_frame)
  2907. {
  2908. u8 *addr;
  2909. struct sta_info *psta = NULL;
  2910. struct recv_reorder_ctrl *preorder_ctrl;
  2911. unsigned char *frame_body;
  2912. unsigned char category, action;
  2913. unsigned short tid, status, reason_code = 0;
  2914. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  2915. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  2916. u8 *pframe = precv_frame->u.hdr.rx_data;
  2917. struct sta_priv *pstapriv = &padapter->stapriv;
  2918. struct registry_priv *pregpriv = &padapter->registrypriv;
  2919. #ifdef CONFIG_80211N_HT
  2920. RTW_INFO("%s\n", __FUNCTION__);
  2921. /* check RA matches or not */
  2922. if (!_rtw_memcmp(adapter_mac_addr(padapter), GetAddr1Ptr(pframe), ETH_ALEN))
  2923. return _SUCCESS;
  2924. #if 0
  2925. /* check A1 matches or not */
  2926. if (!_rtw_memcmp(adapter_mac_addr(padapter), get_da(pframe), ETH_ALEN))
  2927. return _SUCCESS;
  2928. #endif
  2929. if ((pmlmeinfo->state & 0x03) != WIFI_FW_AP_STATE)
  2930. if (!(pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS))
  2931. return _SUCCESS;
  2932. addr = get_addr2_ptr(pframe);
  2933. psta = rtw_get_stainfo(pstapriv, addr);
  2934. if (psta == NULL)
  2935. return _SUCCESS;
  2936. frame_body = (unsigned char *)(pframe + sizeof(struct rtw_ieee80211_hdr_3addr));
  2937. category = frame_body[0];
  2938. if (category == RTW_WLAN_CATEGORY_BACK) { /* representing Block Ack */
  2939. #ifdef CONFIG_TDLS
  2940. if ((psta->tdls_sta_state & TDLS_LINKED_STATE) &&
  2941. (psta->htpriv.ht_option == _TRUE) &&
  2942. (psta->htpriv.ampdu_enable == _TRUE))
  2943. RTW_INFO("Recv [%s] from direc link\n", __FUNCTION__);
  2944. else
  2945. #endif /* CONFIG_TDLS */
  2946. if (!pmlmeinfo->HT_enable)
  2947. return _SUCCESS;
  2948. action = frame_body[1];
  2949. RTW_INFO("%s, action=%d\n", __FUNCTION__, action);
  2950. switch (action) {
  2951. case RTW_WLAN_ACTION_ADDBA_REQ: /* ADDBA request */
  2952. _rtw_memcpy(&(pmlmeinfo->ADDBA_req), &(frame_body[2]), sizeof(struct ADDBA_request));
  2953. /* process_addba_req(padapter, (u8*)&(pmlmeinfo->ADDBA_req), GetAddr3Ptr(pframe)); */
  2954. process_addba_req(padapter, (u8 *)&(pmlmeinfo->ADDBA_req), addr);
  2955. break;
  2956. case RTW_WLAN_ACTION_ADDBA_RESP: /* ADDBA response */
  2957. /* status = frame_body[3] | (frame_body[4] << 8); */ /* endian issue */
  2958. status = RTW_GET_LE16(&frame_body[3]);
  2959. tid = ((frame_body[5] >> 2) & 0x7);
  2960. if (status == 0) {
  2961. /* successful */
  2962. RTW_INFO("agg_enable for TID=%d\n", tid);
  2963. psta->htpriv.agg_enable_bitmap |= 1 << tid;
  2964. psta->htpriv.candidate_tid_bitmap &= ~BIT(tid);
  2965. /* amsdu in ampdu */
  2966. if (pregpriv->tx_ampdu_amsdu == 0)
  2967. psta->htpriv.tx_amsdu_enable = _FALSE;
  2968. else if (pregpriv->tx_ampdu_amsdu == 1)
  2969. psta->htpriv.tx_amsdu_enable = _TRUE;
  2970. else {
  2971. if (frame_body[5] & 1)
  2972. psta->htpriv.tx_amsdu_enable = _TRUE;
  2973. }
  2974. } else
  2975. psta->htpriv.agg_enable_bitmap &= ~BIT(tid);
  2976. if (psta->state & WIFI_STA_ALIVE_CHK_STATE) {
  2977. RTW_INFO("%s alive check - rx ADDBA response\n", __func__);
  2978. psta->htpriv.agg_enable_bitmap &= ~BIT(tid);
  2979. psta->expire_to = pstapriv->expire_to;
  2980. psta->state ^= WIFI_STA_ALIVE_CHK_STATE;
  2981. }
  2982. /* RTW_INFO("marc: ADDBA RSP: %x\n", pmlmeinfo->agg_enable_bitmap); */
  2983. break;
  2984. case RTW_WLAN_ACTION_DELBA: /* DELBA */
  2985. if ((frame_body[3] & BIT(3)) == 0) {
  2986. psta->htpriv.agg_enable_bitmap &= ~(1 << ((frame_body[3] >> 4) & 0xf));
  2987. psta->htpriv.candidate_tid_bitmap &= ~(1 << ((frame_body[3] >> 4) & 0xf));
  2988. /* reason_code = frame_body[4] | (frame_body[5] << 8); */
  2989. reason_code = RTW_GET_LE16(&frame_body[4]);
  2990. } else if ((frame_body[3] & BIT(3)) == BIT(3)) {
  2991. tid = (frame_body[3] >> 4) & 0x0F;
  2992. preorder_ctrl = &psta->recvreorder_ctrl[tid];
  2993. preorder_ctrl->enable = _FALSE;
  2994. preorder_ctrl->ampdu_size = RX_AMPDU_SIZE_INVALID;
  2995. }
  2996. RTW_INFO("%s(): DELBA: %x(%x)\n", __FUNCTION__, pmlmeinfo->agg_enable_bitmap, reason_code);
  2997. /* todo: how to notify the host while receiving DELETE BA */
  2998. break;
  2999. default:
  3000. break;
  3001. }
  3002. }
  3003. #endif /* CONFIG_80211N_HT */
  3004. return _SUCCESS;
  3005. }
  3006. #ifdef CONFIG_P2P
  3007. int get_reg_classes_full_count(struct p2p_channels *channel_list)
  3008. {
  3009. int cnt = 0;
  3010. int i;
  3011. for (i = 0; i < channel_list->reg_classes; i++)
  3012. cnt += channel_list->reg_class[i].channels;
  3013. return cnt;
  3014. }
  3015. void issue_p2p_GO_request(_adapter *padapter, u8 *raddr)
  3016. {
  3017. struct p2p_channels *ch_list = &(adapter_to_rfctl(padapter)->channel_list);
  3018. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  3019. u8 action = P2P_PUB_ACTION_ACTION;
  3020. u32 p2poui = cpu_to_be32(P2POUI);
  3021. u8 oui_subtype = P2P_GO_NEGO_REQ;
  3022. u8 wpsie[255] = { 0x00 }, p2pie[255] = { 0x00 };
  3023. u8 wpsielen = 0, p2pielen = 0;
  3024. u16 len_channellist_attr = 0;
  3025. #ifdef CONFIG_WFD
  3026. u32 wfdielen = 0;
  3027. #endif
  3028. struct xmit_frame *pmgntframe;
  3029. struct pkt_attrib *pattrib;
  3030. unsigned char *pframe;
  3031. struct rtw_ieee80211_hdr *pwlanhdr;
  3032. unsigned short *fctrl;
  3033. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  3034. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  3035. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  3036. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  3037. if (pmgntframe == NULL)
  3038. return;
  3039. RTW_INFO("[%s] In\n", __FUNCTION__);
  3040. /* update attribute */
  3041. pattrib = &pmgntframe->attrib;
  3042. update_mgntframe_attrib(padapter, pattrib);
  3043. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  3044. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  3045. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  3046. fctrl = &(pwlanhdr->frame_ctl);
  3047. *(fctrl) = 0;
  3048. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  3049. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  3050. _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(padapter), ETH_ALEN);
  3051. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  3052. pmlmeext->mgnt_seq++;
  3053. set_frame_sub_type(pframe, WIFI_ACTION);
  3054. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  3055. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  3056. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  3057. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  3058. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  3059. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  3060. pwdinfo->negotiation_dialog_token = 1; /* Initialize the dialog value */
  3061. pframe = rtw_set_fixed_ie(pframe, 1, &pwdinfo->negotiation_dialog_token, &(pattrib->pktlen));
  3062. /* WPS Section */
  3063. wpsielen = 0;
  3064. /* WPS OUI */
  3065. *(u32 *)(wpsie) = cpu_to_be32(WPSOUI);
  3066. wpsielen += 4;
  3067. /* WPS version */
  3068. /* Type: */
  3069. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_VER1);
  3070. wpsielen += 2;
  3071. /* Length: */
  3072. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  3073. wpsielen += 2;
  3074. /* Value: */
  3075. wpsie[wpsielen++] = WPS_VERSION_1; /* Version 1.0 */
  3076. /* Device Password ID */
  3077. /* Type: */
  3078. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_DEVICE_PWID);
  3079. wpsielen += 2;
  3080. /* Length: */
  3081. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  3082. wpsielen += 2;
  3083. /* Value: */
  3084. if (pwdinfo->ui_got_wps_info == P2P_GOT_WPSINFO_PEER_DISPLAY_PIN)
  3085. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_USER_SPEC);
  3086. else if (pwdinfo->ui_got_wps_info == P2P_GOT_WPSINFO_SELF_DISPLAY_PIN)
  3087. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_REGISTRAR_SPEC);
  3088. else if (pwdinfo->ui_got_wps_info == P2P_GOT_WPSINFO_PBC)
  3089. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_PBC);
  3090. wpsielen += 2;
  3091. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, wpsielen, (unsigned char *) wpsie, &pattrib->pktlen);
  3092. /* P2P IE Section. */
  3093. /* P2P OUI */
  3094. p2pielen = 0;
  3095. p2pie[p2pielen++] = 0x50;
  3096. p2pie[p2pielen++] = 0x6F;
  3097. p2pie[p2pielen++] = 0x9A;
  3098. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  3099. /* Commented by Albert 20110306 */
  3100. /* According to the P2P Specification, the group negoitation request frame should contain 9 P2P attributes */
  3101. /* 1. P2P Capability */
  3102. /* 2. Group Owner Intent */
  3103. /* 3. Configuration Timeout */
  3104. /* 4. Listen Channel */
  3105. /* 5. Extended Listen Timing */
  3106. /* 6. Intended P2P Interface Address */
  3107. /* 7. Channel List */
  3108. /* 8. P2P Device Info */
  3109. /* 9. Operating Channel */
  3110. /* P2P Capability */
  3111. /* Type: */
  3112. p2pie[p2pielen++] = P2P_ATTR_CAPABILITY;
  3113. /* Length: */
  3114. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3115. p2pielen += 2;
  3116. /* Value: */
  3117. /* Device Capability Bitmap, 1 byte */
  3118. p2pie[p2pielen++] = DMP_P2P_DEVCAP_SUPPORT;
  3119. /* Group Capability Bitmap, 1 byte */
  3120. if (pwdinfo->persistent_supported)
  3121. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN | P2P_GRPCAP_PERSISTENT_GROUP;
  3122. else
  3123. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN;
  3124. /* Group Owner Intent */
  3125. /* Type: */
  3126. p2pie[p2pielen++] = P2P_ATTR_GO_INTENT;
  3127. /* Length: */
  3128. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  3129. p2pielen += 2;
  3130. /* Value: */
  3131. /* Todo the tie breaker bit. */
  3132. p2pie[p2pielen++] = ((pwdinfo->intent << 1) & 0xFE);
  3133. /* Configuration Timeout */
  3134. /* Type: */
  3135. p2pie[p2pielen++] = P2P_ATTR_CONF_TIMEOUT;
  3136. /* Length: */
  3137. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3138. p2pielen += 2;
  3139. /* Value: */
  3140. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P GO */
  3141. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P Client */
  3142. /* Listen Channel */
  3143. /* Type: */
  3144. p2pie[p2pielen++] = P2P_ATTR_LISTEN_CH;
  3145. /* Length: */
  3146. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  3147. p2pielen += 2;
  3148. /* Value: */
  3149. /* Country String */
  3150. p2pie[p2pielen++] = 'X';
  3151. p2pie[p2pielen++] = 'X';
  3152. /* The third byte should be set to 0x04. */
  3153. /* Described in the "Operating Channel Attribute" section. */
  3154. p2pie[p2pielen++] = 0x04;
  3155. /* Operating Class */
  3156. p2pie[p2pielen++] = 0x51; /* Copy from SD7 */
  3157. /* Channel Number */
  3158. p2pie[p2pielen++] = pwdinfo->listen_channel; /* listening channel number */
  3159. /* Extended Listen Timing ATTR */
  3160. /* Type: */
  3161. p2pie[p2pielen++] = P2P_ATTR_EX_LISTEN_TIMING;
  3162. /* Length: */
  3163. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0004);
  3164. p2pielen += 2;
  3165. /* Value: */
  3166. /* Availability Period */
  3167. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  3168. p2pielen += 2;
  3169. /* Availability Interval */
  3170. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  3171. p2pielen += 2;
  3172. /* Intended P2P Interface Address */
  3173. /* Type: */
  3174. p2pie[p2pielen++] = P2P_ATTR_INTENDED_IF_ADDR;
  3175. /* Length: */
  3176. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN);
  3177. p2pielen += 2;
  3178. /* Value: */
  3179. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  3180. p2pielen += ETH_ALEN;
  3181. /* Channel List */
  3182. /* Type: */
  3183. p2pie[p2pielen++] = P2P_ATTR_CH_LIST;
  3184. /* Length: */
  3185. /* Country String(3) */
  3186. /* + ( Operating Class (1) + Number of Channels(1) ) * Operation Classes (?) */
  3187. /* + number of channels in all classes */
  3188. len_channellist_attr = 3
  3189. + (1 + 1) * (u16)(ch_list->reg_classes)
  3190. + get_reg_classes_full_count(ch_list);
  3191. #ifdef CONFIG_CONCURRENT_MODE
  3192. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0)
  3193. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(5 + 1);
  3194. else
  3195. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3196. #else
  3197. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3198. #endif
  3199. p2pielen += 2;
  3200. /* Value: */
  3201. /* Country String */
  3202. p2pie[p2pielen++] = 'X';
  3203. p2pie[p2pielen++] = 'X';
  3204. /* The third byte should be set to 0x04. */
  3205. /* Described in the "Operating Channel Attribute" section. */
  3206. p2pie[p2pielen++] = 0x04;
  3207. /* Channel Entry List */
  3208. #ifdef CONFIG_CONCURRENT_MODE
  3209. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0) {
  3210. u8 union_ch = rtw_mi_get_union_chan(padapter);
  3211. /* Operating Class */
  3212. if (union_ch > 14) {
  3213. if (union_ch >= 149)
  3214. p2pie[p2pielen++] = 0x7c;
  3215. else
  3216. p2pie[p2pielen++] = 0x73;
  3217. } else
  3218. p2pie[p2pielen++] = 0x51;
  3219. /* Number of Channels */
  3220. /* Just support 1 channel and this channel is AP's channel */
  3221. p2pie[p2pielen++] = 1;
  3222. /* Channel List */
  3223. p2pie[p2pielen++] = union_ch;
  3224. } else
  3225. #endif /* CONFIG_CONCURRENT_MODE */
  3226. {
  3227. int i, j;
  3228. for (j = 0; j < ch_list->reg_classes; j++) {
  3229. /* Operating Class */
  3230. p2pie[p2pielen++] = ch_list->reg_class[j].reg_class;
  3231. /* Number of Channels */
  3232. p2pie[p2pielen++] = ch_list->reg_class[j].channels;
  3233. /* Channel List */
  3234. for (i = 0; i < ch_list->reg_class[j].channels; i++)
  3235. p2pie[p2pielen++] = ch_list->reg_class[j].channel[i];
  3236. }
  3237. }
  3238. /* Device Info */
  3239. /* Type: */
  3240. p2pie[p2pielen++] = P2P_ATTR_DEVICE_INFO;
  3241. /* Length: */
  3242. /* 21->P2P Device Address (6bytes) + Config Methods (2bytes) + Primary Device Type (8bytes) */
  3243. /* + NumofSecondDevType (1byte) + WPS Device Name ID field (2bytes) + WPS Device Name Len field (2bytes) */
  3244. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(21 + pwdinfo->device_name_len);
  3245. p2pielen += 2;
  3246. /* Value: */
  3247. /* P2P Device Address */
  3248. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  3249. p2pielen += ETH_ALEN;
  3250. /* Config Method */
  3251. /* This field should be big endian. Noted by P2P specification. */
  3252. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->supported_wps_cm);
  3253. p2pielen += 2;
  3254. /* Primary Device Type */
  3255. /* Category ID */
  3256. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  3257. p2pielen += 2;
  3258. /* OUI */
  3259. *(u32 *)(p2pie + p2pielen) = cpu_to_be32(WPSOUI);
  3260. p2pielen += 4;
  3261. /* Sub Category ID */
  3262. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  3263. p2pielen += 2;
  3264. /* Number of Secondary Device Types */
  3265. p2pie[p2pielen++] = 0x00; /* No Secondary Device Type List */
  3266. /* Device Name */
  3267. /* Type: */
  3268. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  3269. p2pielen += 2;
  3270. /* Length: */
  3271. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->device_name_len);
  3272. p2pielen += 2;
  3273. /* Value: */
  3274. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_name , pwdinfo->device_name_len);
  3275. p2pielen += pwdinfo->device_name_len;
  3276. /* Operating Channel */
  3277. /* Type: */
  3278. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  3279. /* Length: */
  3280. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  3281. p2pielen += 2;
  3282. /* Value: */
  3283. /* Country String */
  3284. p2pie[p2pielen++] = 'X';
  3285. p2pie[p2pielen++] = 'X';
  3286. /* The third byte should be set to 0x04. */
  3287. /* Described in the "Operating Channel Attribute" section. */
  3288. p2pie[p2pielen++] = 0x04;
  3289. /* Operating Class */
  3290. if (pwdinfo->operating_channel <= 14) {
  3291. /* Operating Class */
  3292. p2pie[p2pielen++] = 0x51;
  3293. } else if ((pwdinfo->operating_channel >= 36) && (pwdinfo->operating_channel <= 48)) {
  3294. /* Operating Class */
  3295. p2pie[p2pielen++] = 0x73;
  3296. } else {
  3297. /* Operating Class */
  3298. p2pie[p2pielen++] = 0x7c;
  3299. }
  3300. /* Channel Number */
  3301. p2pie[p2pielen++] = pwdinfo->operating_channel; /* operating channel number */
  3302. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  3303. #ifdef CONFIG_WFD
  3304. wfdielen = build_nego_req_wfd_ie(pwdinfo, pframe);
  3305. pframe += wfdielen;
  3306. pattrib->pktlen += wfdielen;
  3307. #endif
  3308. pattrib->last_txcmdsz = pattrib->pktlen;
  3309. dump_mgntframe(padapter, pmgntframe);
  3310. return;
  3311. }
  3312. void issue_p2p_GO_response(_adapter *padapter, u8 *raddr, u8 *frame_body, uint len, u8 result)
  3313. {
  3314. struct p2p_channels *ch_list = &(adapter_to_rfctl(padapter)->channel_list);
  3315. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  3316. u8 action = P2P_PUB_ACTION_ACTION;
  3317. u32 p2poui = cpu_to_be32(P2POUI);
  3318. u8 oui_subtype = P2P_GO_NEGO_RESP;
  3319. u8 wpsie[255] = { 0x00 }, p2pie[255] = { 0x00 };
  3320. u8 p2pielen = 0;
  3321. uint wpsielen = 0;
  3322. u16 wps_devicepassword_id = 0x0000;
  3323. uint wps_devicepassword_id_len = 0;
  3324. u16 len_channellist_attr = 0;
  3325. struct xmit_frame *pmgntframe;
  3326. struct pkt_attrib *pattrib;
  3327. unsigned char *pframe;
  3328. struct rtw_ieee80211_hdr *pwlanhdr;
  3329. unsigned short *fctrl;
  3330. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  3331. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  3332. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  3333. #ifdef CONFIG_WFD
  3334. u32 wfdielen = 0;
  3335. #endif
  3336. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  3337. if (pmgntframe == NULL)
  3338. return;
  3339. RTW_INFO("[%s] In, result = %d\n", __FUNCTION__, result);
  3340. /* update attribute */
  3341. pattrib = &pmgntframe->attrib;
  3342. update_mgntframe_attrib(padapter, pattrib);
  3343. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  3344. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  3345. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  3346. fctrl = &(pwlanhdr->frame_ctl);
  3347. *(fctrl) = 0;
  3348. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  3349. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  3350. _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(padapter), ETH_ALEN);
  3351. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  3352. pmlmeext->mgnt_seq++;
  3353. set_frame_sub_type(pframe, WIFI_ACTION);
  3354. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  3355. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  3356. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  3357. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  3358. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  3359. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  3360. pwdinfo->negotiation_dialog_token = frame_body[7]; /* The Dialog Token of provisioning discovery request frame. */
  3361. pframe = rtw_set_fixed_ie(pframe, 1, &(pwdinfo->negotiation_dialog_token), &(pattrib->pktlen));
  3362. /* Commented by Albert 20110328 */
  3363. /* Try to get the device password ID from the WPS IE of group negotiation request frame */
  3364. /* WiFi Direct test plan 5.1.15 */
  3365. rtw_get_wps_ie(frame_body + _PUBLIC_ACTION_IE_OFFSET_, len - _PUBLIC_ACTION_IE_OFFSET_, wpsie, &wpsielen);
  3366. rtw_get_wps_attr_content(wpsie, wpsielen, WPS_ATTR_DEVICE_PWID, (u8 *) &wps_devicepassword_id, &wps_devicepassword_id_len);
  3367. wps_devicepassword_id = be16_to_cpu(wps_devicepassword_id);
  3368. _rtw_memset(wpsie, 0x00, 255);
  3369. wpsielen = 0;
  3370. /* WPS Section */
  3371. wpsielen = 0;
  3372. /* WPS OUI */
  3373. *(u32 *)(wpsie) = cpu_to_be32(WPSOUI);
  3374. wpsielen += 4;
  3375. /* WPS version */
  3376. /* Type: */
  3377. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_VER1);
  3378. wpsielen += 2;
  3379. /* Length: */
  3380. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  3381. wpsielen += 2;
  3382. /* Value: */
  3383. wpsie[wpsielen++] = WPS_VERSION_1; /* Version 1.0 */
  3384. /* Device Password ID */
  3385. /* Type: */
  3386. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_DEVICE_PWID);
  3387. wpsielen += 2;
  3388. /* Length: */
  3389. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  3390. wpsielen += 2;
  3391. /* Value: */
  3392. if (wps_devicepassword_id == WPS_DPID_USER_SPEC)
  3393. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_REGISTRAR_SPEC);
  3394. else if (wps_devicepassword_id == WPS_DPID_REGISTRAR_SPEC)
  3395. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_USER_SPEC);
  3396. else
  3397. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_PBC);
  3398. wpsielen += 2;
  3399. /* Commented by Kurt 20120113 */
  3400. /* If some device wants to do p2p handshake without sending prov_disc_req */
  3401. /* We have to get peer_req_cm from here. */
  3402. if (_rtw_memcmp(pwdinfo->rx_prov_disc_info.strconfig_method_desc_of_prov_disc_req, "000", 3)) {
  3403. if (wps_devicepassword_id == WPS_DPID_USER_SPEC)
  3404. _rtw_memcpy(pwdinfo->rx_prov_disc_info.strconfig_method_desc_of_prov_disc_req, "dis", 3);
  3405. else if (wps_devicepassword_id == WPS_DPID_REGISTRAR_SPEC)
  3406. _rtw_memcpy(pwdinfo->rx_prov_disc_info.strconfig_method_desc_of_prov_disc_req, "pad", 3);
  3407. else
  3408. _rtw_memcpy(pwdinfo->rx_prov_disc_info.strconfig_method_desc_of_prov_disc_req, "pbc", 3);
  3409. }
  3410. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, wpsielen, (unsigned char *) wpsie, &pattrib->pktlen);
  3411. /* P2P IE Section. */
  3412. /* P2P OUI */
  3413. p2pielen = 0;
  3414. p2pie[p2pielen++] = 0x50;
  3415. p2pie[p2pielen++] = 0x6F;
  3416. p2pie[p2pielen++] = 0x9A;
  3417. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  3418. /* Commented by Albert 20100908 */
  3419. /* According to the P2P Specification, the group negoitation response frame should contain 9 P2P attributes */
  3420. /* 1. Status */
  3421. /* 2. P2P Capability */
  3422. /* 3. Group Owner Intent */
  3423. /* 4. Configuration Timeout */
  3424. /* 5. Operating Channel */
  3425. /* 6. Intended P2P Interface Address */
  3426. /* 7. Channel List */
  3427. /* 8. Device Info */
  3428. /* 9. Group ID ( Only GO ) */
  3429. /* ToDo: */
  3430. /* P2P Status */
  3431. /* Type: */
  3432. p2pie[p2pielen++] = P2P_ATTR_STATUS;
  3433. /* Length: */
  3434. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  3435. p2pielen += 2;
  3436. /* Value: */
  3437. p2pie[p2pielen++] = result;
  3438. /* P2P Capability */
  3439. /* Type: */
  3440. p2pie[p2pielen++] = P2P_ATTR_CAPABILITY;
  3441. /* Length: */
  3442. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3443. p2pielen += 2;
  3444. /* Value: */
  3445. /* Device Capability Bitmap, 1 byte */
  3446. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_CLIENT)) {
  3447. /* Commented by Albert 2011/03/08 */
  3448. /* According to the P2P specification */
  3449. /* if the sending device will be client, the P2P Capability should be reserved of group negotation response frame */
  3450. p2pie[p2pielen++] = 0;
  3451. } else {
  3452. /* Be group owner or meet the error case */
  3453. p2pie[p2pielen++] = DMP_P2P_DEVCAP_SUPPORT;
  3454. }
  3455. /* Group Capability Bitmap, 1 byte */
  3456. if (pwdinfo->persistent_supported)
  3457. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN | P2P_GRPCAP_PERSISTENT_GROUP;
  3458. else
  3459. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN;
  3460. /* Group Owner Intent */
  3461. /* Type: */
  3462. p2pie[p2pielen++] = P2P_ATTR_GO_INTENT;
  3463. /* Length: */
  3464. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  3465. p2pielen += 2;
  3466. /* Value: */
  3467. if (pwdinfo->peer_intent & 0x01) {
  3468. /* Peer's tie breaker bit is 1, our tie breaker bit should be 0 */
  3469. p2pie[p2pielen++] = (pwdinfo->intent << 1);
  3470. } else {
  3471. /* Peer's tie breaker bit is 0, our tie breaker bit should be 1 */
  3472. p2pie[p2pielen++] = ((pwdinfo->intent << 1) | BIT(0));
  3473. }
  3474. /* Configuration Timeout */
  3475. /* Type: */
  3476. p2pie[p2pielen++] = P2P_ATTR_CONF_TIMEOUT;
  3477. /* Length: */
  3478. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3479. p2pielen += 2;
  3480. /* Value: */
  3481. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P GO */
  3482. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P Client */
  3483. /* Operating Channel */
  3484. /* Type: */
  3485. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  3486. /* Length: */
  3487. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  3488. p2pielen += 2;
  3489. /* Value: */
  3490. /* Country String */
  3491. p2pie[p2pielen++] = 'X';
  3492. p2pie[p2pielen++] = 'X';
  3493. /* The third byte should be set to 0x04. */
  3494. /* Described in the "Operating Channel Attribute" section. */
  3495. p2pie[p2pielen++] = 0x04;
  3496. /* Operating Class */
  3497. if (pwdinfo->operating_channel <= 14) {
  3498. /* Operating Class */
  3499. p2pie[p2pielen++] = 0x51;
  3500. } else if ((pwdinfo->operating_channel >= 36) && (pwdinfo->operating_channel <= 48)) {
  3501. /* Operating Class */
  3502. p2pie[p2pielen++] = 0x73;
  3503. } else {
  3504. /* Operating Class */
  3505. p2pie[p2pielen++] = 0x7c;
  3506. }
  3507. /* Channel Number */
  3508. p2pie[p2pielen++] = pwdinfo->operating_channel; /* operating channel number */
  3509. /* Intended P2P Interface Address */
  3510. /* Type: */
  3511. p2pie[p2pielen++] = P2P_ATTR_INTENDED_IF_ADDR;
  3512. /* Length: */
  3513. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN);
  3514. p2pielen += 2;
  3515. /* Value: */
  3516. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  3517. p2pielen += ETH_ALEN;
  3518. /* Channel List */
  3519. /* Type: */
  3520. p2pie[p2pielen++] = P2P_ATTR_CH_LIST;
  3521. /* Country String(3) */
  3522. /* + ( Operating Class (1) + Number of Channels(1) ) * Operation Classes (?) */
  3523. /* + number of channels in all classes */
  3524. len_channellist_attr = 3
  3525. + (1 + 1) * (u16)ch_list->reg_classes
  3526. + get_reg_classes_full_count(ch_list);
  3527. #ifdef CONFIG_CONCURRENT_MODE
  3528. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0)
  3529. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(5 + 1);
  3530. else
  3531. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3532. #else
  3533. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3534. #endif
  3535. p2pielen += 2;
  3536. /* Value: */
  3537. /* Country String */
  3538. p2pie[p2pielen++] = 'X';
  3539. p2pie[p2pielen++] = 'X';
  3540. /* The third byte should be set to 0x04. */
  3541. /* Described in the "Operating Channel Attribute" section. */
  3542. p2pie[p2pielen++] = 0x04;
  3543. /* Channel Entry List */
  3544. #ifdef CONFIG_CONCURRENT_MODE
  3545. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0) {
  3546. u8 union_chan = rtw_mi_get_union_chan(padapter);
  3547. /*Operating Class*/
  3548. if (union_chan > 14) {
  3549. if (union_chan >= 149)
  3550. p2pie[p2pielen++] = 0x7c;
  3551. else
  3552. p2pie[p2pielen++] = 0x73;
  3553. } else
  3554. p2pie[p2pielen++] = 0x51;
  3555. /* Number of Channels
  3556. Just support 1 channel and this channel is AP's channel*/
  3557. p2pie[p2pielen++] = 1;
  3558. /*Channel List*/
  3559. p2pie[p2pielen++] = union_chan;
  3560. } else
  3561. #endif /* CONFIG_CONCURRENT_MODE */
  3562. {
  3563. int i, j;
  3564. for (j = 0; j < ch_list->reg_classes; j++) {
  3565. /* Operating Class */
  3566. p2pie[p2pielen++] = ch_list->reg_class[j].reg_class;
  3567. /* Number of Channels */
  3568. p2pie[p2pielen++] = ch_list->reg_class[j].channels;
  3569. /* Channel List */
  3570. for (i = 0; i < ch_list->reg_class[j].channels; i++)
  3571. p2pie[p2pielen++] = ch_list->reg_class[j].channel[i];
  3572. }
  3573. }
  3574. /* Device Info */
  3575. /* Type: */
  3576. p2pie[p2pielen++] = P2P_ATTR_DEVICE_INFO;
  3577. /* Length: */
  3578. /* 21->P2P Device Address (6bytes) + Config Methods (2bytes) + Primary Device Type (8bytes) */
  3579. /* + NumofSecondDevType (1byte) + WPS Device Name ID field (2bytes) + WPS Device Name Len field (2bytes) */
  3580. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(21 + pwdinfo->device_name_len);
  3581. p2pielen += 2;
  3582. /* Value: */
  3583. /* P2P Device Address */
  3584. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  3585. p2pielen += ETH_ALEN;
  3586. /* Config Method */
  3587. /* This field should be big endian. Noted by P2P specification. */
  3588. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->supported_wps_cm);
  3589. p2pielen += 2;
  3590. /* Primary Device Type */
  3591. /* Category ID */
  3592. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  3593. p2pielen += 2;
  3594. /* OUI */
  3595. *(u32 *)(p2pie + p2pielen) = cpu_to_be32(WPSOUI);
  3596. p2pielen += 4;
  3597. /* Sub Category ID */
  3598. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  3599. p2pielen += 2;
  3600. /* Number of Secondary Device Types */
  3601. p2pie[p2pielen++] = 0x00; /* No Secondary Device Type List */
  3602. /* Device Name */
  3603. /* Type: */
  3604. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  3605. p2pielen += 2;
  3606. /* Length: */
  3607. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->device_name_len);
  3608. p2pielen += 2;
  3609. /* Value: */
  3610. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_name , pwdinfo->device_name_len);
  3611. p2pielen += pwdinfo->device_name_len;
  3612. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  3613. /* Group ID Attribute */
  3614. /* Type: */
  3615. p2pie[p2pielen++] = P2P_ATTR_GROUP_ID;
  3616. /* Length: */
  3617. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN + pwdinfo->nego_ssidlen);
  3618. p2pielen += 2;
  3619. /* Value: */
  3620. /* p2P Device Address */
  3621. _rtw_memcpy(p2pie + p2pielen , pwdinfo->device_addr, ETH_ALEN);
  3622. p2pielen += ETH_ALEN;
  3623. /* SSID */
  3624. _rtw_memcpy(p2pie + p2pielen, pwdinfo->nego_ssid, pwdinfo->nego_ssidlen);
  3625. p2pielen += pwdinfo->nego_ssidlen;
  3626. }
  3627. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  3628. #ifdef CONFIG_WFD
  3629. wfdielen = build_nego_resp_wfd_ie(pwdinfo, pframe);
  3630. pframe += wfdielen;
  3631. pattrib->pktlen += wfdielen;
  3632. #endif
  3633. pattrib->last_txcmdsz = pattrib->pktlen;
  3634. dump_mgntframe(padapter, pmgntframe);
  3635. return;
  3636. }
  3637. void issue_p2p_GO_confirm(_adapter *padapter, u8 *raddr, u8 result)
  3638. {
  3639. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  3640. u8 action = P2P_PUB_ACTION_ACTION;
  3641. u32 p2poui = cpu_to_be32(P2POUI);
  3642. u8 oui_subtype = P2P_GO_NEGO_CONF;
  3643. u8 p2pie[255] = { 0x00 };
  3644. u8 p2pielen = 0;
  3645. struct xmit_frame *pmgntframe;
  3646. struct pkt_attrib *pattrib;
  3647. unsigned char *pframe;
  3648. struct rtw_ieee80211_hdr *pwlanhdr;
  3649. unsigned short *fctrl;
  3650. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  3651. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  3652. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  3653. #ifdef CONFIG_WFD
  3654. u32 wfdielen = 0;
  3655. #endif
  3656. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  3657. if (pmgntframe == NULL)
  3658. return;
  3659. RTW_INFO("[%s] In\n", __FUNCTION__);
  3660. /* update attribute */
  3661. pattrib = &pmgntframe->attrib;
  3662. update_mgntframe_attrib(padapter, pattrib);
  3663. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  3664. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  3665. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  3666. fctrl = &(pwlanhdr->frame_ctl);
  3667. *(fctrl) = 0;
  3668. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  3669. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  3670. _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(padapter), ETH_ALEN);
  3671. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  3672. pmlmeext->mgnt_seq++;
  3673. set_frame_sub_type(pframe, WIFI_ACTION);
  3674. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  3675. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  3676. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  3677. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  3678. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  3679. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  3680. pframe = rtw_set_fixed_ie(pframe, 1, &(pwdinfo->negotiation_dialog_token), &(pattrib->pktlen));
  3681. /* P2P IE Section. */
  3682. /* P2P OUI */
  3683. p2pielen = 0;
  3684. p2pie[p2pielen++] = 0x50;
  3685. p2pie[p2pielen++] = 0x6F;
  3686. p2pie[p2pielen++] = 0x9A;
  3687. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  3688. /* Commented by Albert 20110306 */
  3689. /* According to the P2P Specification, the group negoitation request frame should contain 5 P2P attributes */
  3690. /* 1. Status */
  3691. /* 2. P2P Capability */
  3692. /* 3. Operating Channel */
  3693. /* 4. Channel List */
  3694. /* 5. Group ID ( if this WiFi is GO ) */
  3695. /* P2P Status */
  3696. /* Type: */
  3697. p2pie[p2pielen++] = P2P_ATTR_STATUS;
  3698. /* Length: */
  3699. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  3700. p2pielen += 2;
  3701. /* Value: */
  3702. p2pie[p2pielen++] = result;
  3703. /* P2P Capability */
  3704. /* Type: */
  3705. p2pie[p2pielen++] = P2P_ATTR_CAPABILITY;
  3706. /* Length: */
  3707. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3708. p2pielen += 2;
  3709. /* Value: */
  3710. /* Device Capability Bitmap, 1 byte */
  3711. p2pie[p2pielen++] = DMP_P2P_DEVCAP_SUPPORT;
  3712. /* Group Capability Bitmap, 1 byte */
  3713. if (pwdinfo->persistent_supported)
  3714. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN | P2P_GRPCAP_PERSISTENT_GROUP;
  3715. else
  3716. p2pie[p2pielen++] = P2P_GRPCAP_CROSS_CONN;
  3717. /* Operating Channel */
  3718. /* Type: */
  3719. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  3720. /* Length: */
  3721. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  3722. p2pielen += 2;
  3723. /* Value: */
  3724. /* Country String */
  3725. p2pie[p2pielen++] = 'X';
  3726. p2pie[p2pielen++] = 'X';
  3727. /* The third byte should be set to 0x04. */
  3728. /* Described in the "Operating Channel Attribute" section. */
  3729. p2pie[p2pielen++] = 0x04;
  3730. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_CLIENT)) {
  3731. if (pwdinfo->peer_operating_ch <= 14) {
  3732. /* Operating Class */
  3733. p2pie[p2pielen++] = 0x51;
  3734. } else if ((pwdinfo->peer_operating_ch >= 36) && (pwdinfo->peer_operating_ch <= 48)) {
  3735. /* Operating Class */
  3736. p2pie[p2pielen++] = 0x73;
  3737. } else {
  3738. /* Operating Class */
  3739. p2pie[p2pielen++] = 0x7c;
  3740. }
  3741. p2pie[p2pielen++] = pwdinfo->peer_operating_ch;
  3742. } else {
  3743. if (pwdinfo->operating_channel <= 14) {
  3744. /* Operating Class */
  3745. p2pie[p2pielen++] = 0x51;
  3746. } else if ((pwdinfo->operating_channel >= 36) && (pwdinfo->operating_channel <= 48)) {
  3747. /* Operating Class */
  3748. p2pie[p2pielen++] = 0x73;
  3749. } else {
  3750. /* Operating Class */
  3751. p2pie[p2pielen++] = 0x7c;
  3752. }
  3753. /* Channel Number */
  3754. p2pie[p2pielen++] = pwdinfo->operating_channel; /* Use the listen channel as the operating channel */
  3755. }
  3756. /* Channel List */
  3757. /* Type: */
  3758. p2pie[p2pielen++] = P2P_ATTR_CH_LIST;
  3759. *(u16 *)(p2pie + p2pielen) = 6;
  3760. p2pielen += 2;
  3761. /* Country String */
  3762. p2pie[p2pielen++] = 'X';
  3763. p2pie[p2pielen++] = 'X';
  3764. /* The third byte should be set to 0x04. */
  3765. /* Described in the "Operating Channel Attribute" section. */
  3766. p2pie[p2pielen++] = 0x04;
  3767. /* Value: */
  3768. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_CLIENT)) {
  3769. if (pwdinfo->peer_operating_ch <= 14) {
  3770. /* Operating Class */
  3771. p2pie[p2pielen++] = 0x51;
  3772. } else if ((pwdinfo->peer_operating_ch >= 36) && (pwdinfo->peer_operating_ch <= 48)) {
  3773. /* Operating Class */
  3774. p2pie[p2pielen++] = 0x73;
  3775. } else {
  3776. /* Operating Class */
  3777. p2pie[p2pielen++] = 0x7c;
  3778. }
  3779. p2pie[p2pielen++] = 1;
  3780. p2pie[p2pielen++] = pwdinfo->peer_operating_ch;
  3781. } else {
  3782. if (pwdinfo->operating_channel <= 14) {
  3783. /* Operating Class */
  3784. p2pie[p2pielen++] = 0x51;
  3785. } else if ((pwdinfo->operating_channel >= 36) && (pwdinfo->operating_channel <= 48)) {
  3786. /* Operating Class */
  3787. p2pie[p2pielen++] = 0x73;
  3788. } else {
  3789. /* Operating Class */
  3790. p2pie[p2pielen++] = 0x7c;
  3791. }
  3792. /* Channel Number */
  3793. p2pie[p2pielen++] = 1;
  3794. p2pie[p2pielen++] = pwdinfo->operating_channel; /* Use the listen channel as the operating channel */
  3795. }
  3796. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  3797. /* Group ID Attribute */
  3798. /* Type: */
  3799. p2pie[p2pielen++] = P2P_ATTR_GROUP_ID;
  3800. /* Length: */
  3801. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN + pwdinfo->nego_ssidlen);
  3802. p2pielen += 2;
  3803. /* Value: */
  3804. /* p2P Device Address */
  3805. _rtw_memcpy(p2pie + p2pielen , pwdinfo->device_addr, ETH_ALEN);
  3806. p2pielen += ETH_ALEN;
  3807. /* SSID */
  3808. _rtw_memcpy(p2pie + p2pielen, pwdinfo->nego_ssid, pwdinfo->nego_ssidlen);
  3809. p2pielen += pwdinfo->nego_ssidlen;
  3810. }
  3811. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  3812. #ifdef CONFIG_WFD
  3813. wfdielen = build_nego_confirm_wfd_ie(pwdinfo, pframe);
  3814. pframe += wfdielen;
  3815. pattrib->pktlen += wfdielen;
  3816. #endif
  3817. pattrib->last_txcmdsz = pattrib->pktlen;
  3818. dump_mgntframe(padapter, pmgntframe);
  3819. return;
  3820. }
  3821. void issue_p2p_invitation_request(_adapter *padapter, u8 *raddr)
  3822. {
  3823. struct p2p_channels *ch_list = &(adapter_to_rfctl(padapter)->channel_list);
  3824. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  3825. u8 action = P2P_PUB_ACTION_ACTION;
  3826. u32 p2poui = cpu_to_be32(P2POUI);
  3827. u8 oui_subtype = P2P_INVIT_REQ;
  3828. u8 p2pie[255] = { 0x00 };
  3829. u8 p2pielen = 0;
  3830. u8 dialogToken = 3;
  3831. u16 len_channellist_attr = 0;
  3832. #ifdef CONFIG_WFD
  3833. u32 wfdielen = 0;
  3834. #endif
  3835. struct xmit_frame *pmgntframe;
  3836. struct pkt_attrib *pattrib;
  3837. unsigned char *pframe;
  3838. struct rtw_ieee80211_hdr *pwlanhdr;
  3839. unsigned short *fctrl;
  3840. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  3841. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  3842. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  3843. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  3844. if (pmgntframe == NULL)
  3845. return;
  3846. /* update attribute */
  3847. pattrib = &pmgntframe->attrib;
  3848. update_mgntframe_attrib(padapter, pattrib);
  3849. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  3850. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  3851. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  3852. fctrl = &(pwlanhdr->frame_ctl);
  3853. *(fctrl) = 0;
  3854. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  3855. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  3856. _rtw_memcpy(pwlanhdr->addr3, raddr, ETH_ALEN);
  3857. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  3858. pmlmeext->mgnt_seq++;
  3859. set_frame_sub_type(pframe, WIFI_ACTION);
  3860. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  3861. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  3862. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  3863. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  3864. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  3865. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  3866. pframe = rtw_set_fixed_ie(pframe, 1, &(dialogToken), &(pattrib->pktlen));
  3867. /* P2P IE Section. */
  3868. /* P2P OUI */
  3869. p2pielen = 0;
  3870. p2pie[p2pielen++] = 0x50;
  3871. p2pie[p2pielen++] = 0x6F;
  3872. p2pie[p2pielen++] = 0x9A;
  3873. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  3874. /* Commented by Albert 20101011 */
  3875. /* According to the P2P Specification, the P2P Invitation request frame should contain 7 P2P attributes */
  3876. /* 1. Configuration Timeout */
  3877. /* 2. Invitation Flags */
  3878. /* 3. Operating Channel ( Only GO ) */
  3879. /* 4. P2P Group BSSID ( Should be included if I am the GO ) */
  3880. /* 5. Channel List */
  3881. /* 6. P2P Group ID */
  3882. /* 7. P2P Device Info */
  3883. /* Configuration Timeout */
  3884. /* Type: */
  3885. p2pie[p2pielen++] = P2P_ATTR_CONF_TIMEOUT;
  3886. /* Length: */
  3887. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  3888. p2pielen += 2;
  3889. /* Value: */
  3890. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P GO */
  3891. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P Client */
  3892. /* Invitation Flags */
  3893. /* Type: */
  3894. p2pie[p2pielen++] = P2P_ATTR_INVITATION_FLAGS;
  3895. /* Length: */
  3896. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  3897. p2pielen += 2;
  3898. /* Value: */
  3899. p2pie[p2pielen++] = P2P_INVITATION_FLAGS_PERSISTENT;
  3900. /* Operating Channel */
  3901. /* Type: */
  3902. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  3903. /* Length: */
  3904. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  3905. p2pielen += 2;
  3906. /* Value: */
  3907. /* Country String */
  3908. p2pie[p2pielen++] = 'X';
  3909. p2pie[p2pielen++] = 'X';
  3910. /* The third byte should be set to 0x04. */
  3911. /* Described in the "Operating Channel Attribute" section. */
  3912. p2pie[p2pielen++] = 0x04;
  3913. /* Operating Class */
  3914. if (pwdinfo->invitereq_info.operating_ch <= 14)
  3915. p2pie[p2pielen++] = 0x51;
  3916. else if ((pwdinfo->invitereq_info.operating_ch >= 36) && (pwdinfo->invitereq_info.operating_ch <= 48))
  3917. p2pie[p2pielen++] = 0x73;
  3918. else
  3919. p2pie[p2pielen++] = 0x7c;
  3920. /* Channel Number */
  3921. p2pie[p2pielen++] = pwdinfo->invitereq_info.operating_ch; /* operating channel number */
  3922. if (_rtw_memcmp(adapter_mac_addr(padapter), pwdinfo->invitereq_info.go_bssid, ETH_ALEN)) {
  3923. /* P2P Group BSSID */
  3924. /* Type: */
  3925. p2pie[p2pielen++] = P2P_ATTR_GROUP_BSSID;
  3926. /* Length: */
  3927. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN);
  3928. p2pielen += 2;
  3929. /* Value: */
  3930. /* P2P Device Address for GO */
  3931. _rtw_memcpy(p2pie + p2pielen, pwdinfo->invitereq_info.go_bssid, ETH_ALEN);
  3932. p2pielen += ETH_ALEN;
  3933. }
  3934. /* Channel List */
  3935. /* Type: */
  3936. p2pie[p2pielen++] = P2P_ATTR_CH_LIST;
  3937. /* Length: */
  3938. /* Country String(3) */
  3939. /* + ( Operating Class (1) + Number of Channels(1) ) * Operation Classes (?) */
  3940. /* + number of channels in all classes */
  3941. len_channellist_attr = 3
  3942. + (1 + 1) * (u16)ch_list->reg_classes
  3943. + get_reg_classes_full_count(ch_list);
  3944. #ifdef CONFIG_CONCURRENT_MODE
  3945. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0)
  3946. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(5 + 1);
  3947. else
  3948. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3949. #else
  3950. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  3951. #endif
  3952. p2pielen += 2;
  3953. /* Value: */
  3954. /* Country String */
  3955. p2pie[p2pielen++] = 'X';
  3956. p2pie[p2pielen++] = 'X';
  3957. /* The third byte should be set to 0x04. */
  3958. /* Described in the "Operating Channel Attribute" section. */
  3959. p2pie[p2pielen++] = 0x04;
  3960. /* Channel Entry List */
  3961. #ifdef CONFIG_CONCURRENT_MODE
  3962. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0) {
  3963. u8 union_ch = rtw_mi_get_union_chan(padapter);
  3964. /* Operating Class */
  3965. if (union_ch > 14) {
  3966. if (union_ch >= 149)
  3967. p2pie[p2pielen++] = 0x7c;
  3968. else
  3969. p2pie[p2pielen++] = 0x73;
  3970. } else
  3971. p2pie[p2pielen++] = 0x51;
  3972. /* Number of Channels */
  3973. /* Just support 1 channel and this channel is AP's channel */
  3974. p2pie[p2pielen++] = 1;
  3975. /* Channel List */
  3976. p2pie[p2pielen++] = union_ch;
  3977. } else
  3978. #endif /* CONFIG_CONCURRENT_MODE */
  3979. {
  3980. int i, j;
  3981. for (j = 0; j < ch_list->reg_classes; j++) {
  3982. /* Operating Class */
  3983. p2pie[p2pielen++] = ch_list->reg_class[j].reg_class;
  3984. /* Number of Channels */
  3985. p2pie[p2pielen++] = ch_list->reg_class[j].channels;
  3986. /* Channel List */
  3987. for (i = 0; i < ch_list->reg_class[j].channels; i++)
  3988. p2pie[p2pielen++] = ch_list->reg_class[j].channel[i];
  3989. }
  3990. }
  3991. /* P2P Group ID */
  3992. /* Type: */
  3993. p2pie[p2pielen++] = P2P_ATTR_GROUP_ID;
  3994. /* Length: */
  3995. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(6 + pwdinfo->invitereq_info.ssidlen);
  3996. p2pielen += 2;
  3997. /* Value: */
  3998. /* P2P Device Address for GO */
  3999. _rtw_memcpy(p2pie + p2pielen, pwdinfo->invitereq_info.go_bssid, ETH_ALEN);
  4000. p2pielen += ETH_ALEN;
  4001. /* SSID */
  4002. _rtw_memcpy(p2pie + p2pielen, pwdinfo->invitereq_info.go_ssid, pwdinfo->invitereq_info.ssidlen);
  4003. p2pielen += pwdinfo->invitereq_info.ssidlen;
  4004. /* Device Info */
  4005. /* Type: */
  4006. p2pie[p2pielen++] = P2P_ATTR_DEVICE_INFO;
  4007. /* Length: */
  4008. /* 21->P2P Device Address (6bytes) + Config Methods (2bytes) + Primary Device Type (8bytes) */
  4009. /* + NumofSecondDevType (1byte) + WPS Device Name ID field (2bytes) + WPS Device Name Len field (2bytes) */
  4010. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(21 + pwdinfo->device_name_len);
  4011. p2pielen += 2;
  4012. /* Value: */
  4013. /* P2P Device Address */
  4014. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  4015. p2pielen += ETH_ALEN;
  4016. /* Config Method */
  4017. /* This field should be big endian. Noted by P2P specification. */
  4018. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_CONFIG_METHOD_DISPLAY);
  4019. p2pielen += 2;
  4020. /* Primary Device Type */
  4021. /* Category ID */
  4022. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  4023. p2pielen += 2;
  4024. /* OUI */
  4025. *(u32 *)(p2pie + p2pielen) = cpu_to_be32(WPSOUI);
  4026. p2pielen += 4;
  4027. /* Sub Category ID */
  4028. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  4029. p2pielen += 2;
  4030. /* Number of Secondary Device Types */
  4031. p2pie[p2pielen++] = 0x00; /* No Secondary Device Type List */
  4032. /* Device Name */
  4033. /* Type: */
  4034. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  4035. p2pielen += 2;
  4036. /* Length: */
  4037. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->device_name_len);
  4038. p2pielen += 2;
  4039. /* Value: */
  4040. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_name, pwdinfo->device_name_len);
  4041. p2pielen += pwdinfo->device_name_len;
  4042. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  4043. #ifdef CONFIG_WFD
  4044. wfdielen = build_invitation_req_wfd_ie(pwdinfo, pframe);
  4045. pframe += wfdielen;
  4046. pattrib->pktlen += wfdielen;
  4047. #endif
  4048. pattrib->last_txcmdsz = pattrib->pktlen;
  4049. dump_mgntframe(padapter, pmgntframe);
  4050. return;
  4051. }
  4052. void issue_p2p_invitation_response(_adapter *padapter, u8 *raddr, u8 dialogToken, u8 status_code)
  4053. {
  4054. struct p2p_channels *ch_list = &(adapter_to_rfctl(padapter)->channel_list);
  4055. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  4056. u8 action = P2P_PUB_ACTION_ACTION;
  4057. u32 p2poui = cpu_to_be32(P2POUI);
  4058. u8 oui_subtype = P2P_INVIT_RESP;
  4059. u8 p2pie[255] = { 0x00 };
  4060. u8 p2pielen = 0;
  4061. u16 len_channellist_attr = 0;
  4062. #ifdef CONFIG_WFD
  4063. u32 wfdielen = 0;
  4064. #endif
  4065. struct xmit_frame *pmgntframe;
  4066. struct pkt_attrib *pattrib;
  4067. unsigned char *pframe;
  4068. struct rtw_ieee80211_hdr *pwlanhdr;
  4069. unsigned short *fctrl;
  4070. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  4071. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  4072. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  4073. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  4074. if (pmgntframe == NULL)
  4075. return;
  4076. /* update attribute */
  4077. pattrib = &pmgntframe->attrib;
  4078. update_mgntframe_attrib(padapter, pattrib);
  4079. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  4080. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  4081. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  4082. fctrl = &(pwlanhdr->frame_ctl);
  4083. *(fctrl) = 0;
  4084. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  4085. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  4086. _rtw_memcpy(pwlanhdr->addr3, raddr, ETH_ALEN);
  4087. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  4088. pmlmeext->mgnt_seq++;
  4089. set_frame_sub_type(pframe, WIFI_ACTION);
  4090. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  4091. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  4092. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  4093. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  4094. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  4095. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  4096. pframe = rtw_set_fixed_ie(pframe, 1, &(dialogToken), &(pattrib->pktlen));
  4097. /* P2P IE Section. */
  4098. /* P2P OUI */
  4099. p2pielen = 0;
  4100. p2pie[p2pielen++] = 0x50;
  4101. p2pie[p2pielen++] = 0x6F;
  4102. p2pie[p2pielen++] = 0x9A;
  4103. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  4104. /* Commented by Albert 20101005 */
  4105. /* According to the P2P Specification, the P2P Invitation response frame should contain 5 P2P attributes */
  4106. /* 1. Status */
  4107. /* 2. Configuration Timeout */
  4108. /* 3. Operating Channel ( Only GO ) */
  4109. /* 4. P2P Group BSSID ( Only GO ) */
  4110. /* 5. Channel List */
  4111. /* P2P Status */
  4112. /* Type: */
  4113. p2pie[p2pielen++] = P2P_ATTR_STATUS;
  4114. /* Length: */
  4115. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0001);
  4116. p2pielen += 2;
  4117. /* Value: */
  4118. /* When status code is P2P_STATUS_FAIL_INFO_UNAVAILABLE. */
  4119. /* Sent the event receiving the P2P Invitation Req frame to DMP UI. */
  4120. /* DMP had to compare the MAC address to find out the profile. */
  4121. /* So, the WiFi driver will send the P2P_STATUS_FAIL_INFO_UNAVAILABLE to NB. */
  4122. /* If the UI found the corresponding profile, the WiFi driver sends the P2P Invitation Req */
  4123. /* to NB to rebuild the persistent group. */
  4124. p2pie[p2pielen++] = status_code;
  4125. /* Configuration Timeout */
  4126. /* Type: */
  4127. p2pie[p2pielen++] = P2P_ATTR_CONF_TIMEOUT;
  4128. /* Length: */
  4129. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  4130. p2pielen += 2;
  4131. /* Value: */
  4132. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P GO */
  4133. p2pie[p2pielen++] = 200; /* 2 seconds needed to be the P2P Client */
  4134. if (status_code == P2P_STATUS_SUCCESS) {
  4135. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  4136. /* The P2P Invitation request frame asks this Wi-Fi device to be the P2P GO */
  4137. /* In this case, the P2P Invitation response frame should carry the two more P2P attributes. */
  4138. /* First one is operating channel attribute. */
  4139. /* Second one is P2P Group BSSID attribute. */
  4140. /* Operating Channel */
  4141. /* Type: */
  4142. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  4143. /* Length: */
  4144. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  4145. p2pielen += 2;
  4146. /* Value: */
  4147. /* Country String */
  4148. p2pie[p2pielen++] = 'X';
  4149. p2pie[p2pielen++] = 'X';
  4150. /* The third byte should be set to 0x04. */
  4151. /* Described in the "Operating Channel Attribute" section. */
  4152. p2pie[p2pielen++] = 0x04;
  4153. /* Operating Class */
  4154. p2pie[p2pielen++] = 0x51; /* Copy from SD7 */
  4155. /* Channel Number */
  4156. p2pie[p2pielen++] = pwdinfo->operating_channel; /* operating channel number */
  4157. /* P2P Group BSSID */
  4158. /* Type: */
  4159. p2pie[p2pielen++] = P2P_ATTR_GROUP_BSSID;
  4160. /* Length: */
  4161. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(ETH_ALEN);
  4162. p2pielen += 2;
  4163. /* Value: */
  4164. /* P2P Device Address for GO */
  4165. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  4166. p2pielen += ETH_ALEN;
  4167. }
  4168. /* Channel List */
  4169. /* Type: */
  4170. p2pie[p2pielen++] = P2P_ATTR_CH_LIST;
  4171. /* Length: */
  4172. /* Country String(3) */
  4173. /* + ( Operating Class (1) + Number of Channels(1) ) * Operation Classes (?) */
  4174. /* + number of channels in all classes */
  4175. len_channellist_attr = 3
  4176. + (1 + 1) * (u16)ch_list->reg_classes
  4177. + get_reg_classes_full_count(ch_list);
  4178. #ifdef CONFIG_CONCURRENT_MODE
  4179. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0)
  4180. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(5 + 1);
  4181. else
  4182. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  4183. #else
  4184. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(len_channellist_attr);
  4185. #endif
  4186. p2pielen += 2;
  4187. /* Value: */
  4188. /* Country String */
  4189. p2pie[p2pielen++] = 'X';
  4190. p2pie[p2pielen++] = 'X';
  4191. /* The third byte should be set to 0x04. */
  4192. /* Described in the "Operating Channel Attribute" section. */
  4193. p2pie[p2pielen++] = 0x04;
  4194. /* Channel Entry List */
  4195. #ifdef CONFIG_CONCURRENT_MODE
  4196. if (rtw_mi_check_status(padapter, MI_LINKED) && padapter->registrypriv.full_ch_in_p2p_handshake == 0) {
  4197. u8 union_ch = rtw_mi_get_union_chan(padapter);
  4198. /* Operating Class */
  4199. if (union_ch > 14) {
  4200. if (union_ch >= 149)
  4201. p2pie[p2pielen++] = 0x7c;
  4202. else
  4203. p2pie[p2pielen++] = 0x73;
  4204. } else
  4205. p2pie[p2pielen++] = 0x51;
  4206. /* Number of Channels */
  4207. /* Just support 1 channel and this channel is AP's channel */
  4208. p2pie[p2pielen++] = 1;
  4209. /* Channel List */
  4210. p2pie[p2pielen++] = union_ch;
  4211. } else
  4212. #endif /* CONFIG_CONCURRENT_MODE */
  4213. {
  4214. int i, j;
  4215. for (j = 0; j < ch_list->reg_classes; j++) {
  4216. /* Operating Class */
  4217. p2pie[p2pielen++] = ch_list->reg_class[j].reg_class;
  4218. /* Number of Channels */
  4219. p2pie[p2pielen++] = ch_list->reg_class[j].channels;
  4220. /* Channel List */
  4221. for (i = 0; i < ch_list->reg_class[j].channels; i++)
  4222. p2pie[p2pielen++] = ch_list->reg_class[j].channel[i];
  4223. }
  4224. }
  4225. }
  4226. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  4227. #ifdef CONFIG_WFD
  4228. wfdielen = build_invitation_resp_wfd_ie(pwdinfo, pframe);
  4229. pframe += wfdielen;
  4230. pattrib->pktlen += wfdielen;
  4231. #endif
  4232. pattrib->last_txcmdsz = pattrib->pktlen;
  4233. dump_mgntframe(padapter, pmgntframe);
  4234. return;
  4235. }
  4236. void issue_p2p_provision_request(_adapter *padapter, u8 *pssid, u8 ussidlen, u8 *pdev_raddr)
  4237. {
  4238. unsigned char category = RTW_WLAN_CATEGORY_PUBLIC;
  4239. u8 action = P2P_PUB_ACTION_ACTION;
  4240. u8 dialogToken = 1;
  4241. u32 p2poui = cpu_to_be32(P2POUI);
  4242. u8 oui_subtype = P2P_PROVISION_DISC_REQ;
  4243. u8 wpsie[100] = { 0x00 };
  4244. u8 wpsielen = 0;
  4245. u32 p2pielen = 0;
  4246. #ifdef CONFIG_WFD
  4247. u32 wfdielen = 0;
  4248. #endif
  4249. struct xmit_frame *pmgntframe;
  4250. struct pkt_attrib *pattrib;
  4251. unsigned char *pframe;
  4252. struct rtw_ieee80211_hdr *pwlanhdr;
  4253. unsigned short *fctrl;
  4254. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  4255. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  4256. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  4257. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  4258. if (pmgntframe == NULL)
  4259. return;
  4260. RTW_INFO("[%s] In\n", __FUNCTION__);
  4261. /* update attribute */
  4262. pattrib = &pmgntframe->attrib;
  4263. update_mgntframe_attrib(padapter, pattrib);
  4264. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  4265. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  4266. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  4267. fctrl = &(pwlanhdr->frame_ctl);
  4268. *(fctrl) = 0;
  4269. _rtw_memcpy(pwlanhdr->addr1, pdev_raddr, ETH_ALEN);
  4270. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  4271. _rtw_memcpy(pwlanhdr->addr3, pdev_raddr, ETH_ALEN);
  4272. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  4273. pmlmeext->mgnt_seq++;
  4274. set_frame_sub_type(pframe, WIFI_ACTION);
  4275. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  4276. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  4277. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  4278. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  4279. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *) &(p2poui), &(pattrib->pktlen));
  4280. pframe = rtw_set_fixed_ie(pframe, 1, &(oui_subtype), &(pattrib->pktlen));
  4281. pframe = rtw_set_fixed_ie(pframe, 1, &(dialogToken), &(pattrib->pktlen));
  4282. p2pielen = build_prov_disc_request_p2p_ie(pwdinfo, pframe, pssid, ussidlen, pdev_raddr);
  4283. pframe += p2pielen;
  4284. pattrib->pktlen += p2pielen;
  4285. wpsielen = 0;
  4286. /* WPS OUI */
  4287. *(u32 *)(wpsie) = cpu_to_be32(WPSOUI);
  4288. wpsielen += 4;
  4289. /* WPS version */
  4290. /* Type: */
  4291. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_VER1);
  4292. wpsielen += 2;
  4293. /* Length: */
  4294. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4295. wpsielen += 2;
  4296. /* Value: */
  4297. wpsie[wpsielen++] = WPS_VERSION_1; /* Version 1.0 */
  4298. /* Config Method */
  4299. /* Type: */
  4300. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_CONF_METHOD);
  4301. wpsielen += 2;
  4302. /* Length: */
  4303. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  4304. wpsielen += 2;
  4305. /* Value: */
  4306. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(pwdinfo->tx_prov_disc_info.wps_config_method_request);
  4307. wpsielen += 2;
  4308. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, wpsielen, (unsigned char *) wpsie, &pattrib->pktlen);
  4309. #ifdef CONFIG_WFD
  4310. wfdielen = build_provdisc_req_wfd_ie(pwdinfo, pframe);
  4311. pframe += wfdielen;
  4312. pattrib->pktlen += wfdielen;
  4313. #endif
  4314. pattrib->last_txcmdsz = pattrib->pktlen;
  4315. dump_mgntframe(padapter, pmgntframe);
  4316. return;
  4317. }
  4318. u8 is_matched_in_profilelist(u8 *peermacaddr, struct profile_info *profileinfo)
  4319. {
  4320. u8 i, match_result = 0;
  4321. RTW_INFO("[%s] peermac = %.2X %.2X %.2X %.2X %.2X %.2X\n", __FUNCTION__,
  4322. peermacaddr[0], peermacaddr[1], peermacaddr[2], peermacaddr[3], peermacaddr[4], peermacaddr[5]);
  4323. for (i = 0; i < P2P_MAX_PERSISTENT_GROUP_NUM; i++, profileinfo++) {
  4324. RTW_INFO("[%s] profileinfo_mac = %.2X %.2X %.2X %.2X %.2X %.2X\n", __FUNCTION__,
  4325. profileinfo->peermac[0], profileinfo->peermac[1], profileinfo->peermac[2], profileinfo->peermac[3], profileinfo->peermac[4], profileinfo->peermac[5]);
  4326. if (_rtw_memcmp(peermacaddr, profileinfo->peermac, ETH_ALEN)) {
  4327. match_result = 1;
  4328. RTW_INFO("[%s] Match!\n", __FUNCTION__);
  4329. break;
  4330. }
  4331. }
  4332. return match_result ;
  4333. }
  4334. void issue_probersp_p2p(_adapter *padapter, unsigned char *da)
  4335. {
  4336. struct xmit_frame *pmgntframe;
  4337. struct pkt_attrib *pattrib;
  4338. unsigned char *pframe;
  4339. struct rtw_ieee80211_hdr *pwlanhdr;
  4340. unsigned short *fctrl;
  4341. unsigned char *mac;
  4342. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  4343. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  4344. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  4345. /* WLAN_BSSID_EX *cur_network = &(pmlmeinfo->network); */
  4346. u16 beacon_interval = 100;
  4347. u16 capInfo = 0;
  4348. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  4349. u8 wpsie[255] = { 0x00 };
  4350. u32 wpsielen = 0, p2pielen = 0;
  4351. #ifdef CONFIG_WFD
  4352. u32 wfdielen = 0;
  4353. #endif
  4354. #ifdef CONFIG_INTEL_WIDI
  4355. u8 zero_array_check[L2SDTA_SERVICE_VE_LEN] = { 0x00 };
  4356. #endif /* CONFIG_INTEL_WIDI */
  4357. /* RTW_INFO("%s\n", __FUNCTION__); */
  4358. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  4359. if (pmgntframe == NULL)
  4360. return;
  4361. /* update attribute */
  4362. pattrib = &pmgntframe->attrib;
  4363. update_mgntframe_attrib(padapter, pattrib);
  4364. if (IS_CCK_RATE(pattrib->rate)) {
  4365. /* force OFDM 6M rate */
  4366. pattrib->rate = MGN_6M;
  4367. pattrib->raid = rtw_get_mgntframe_raid(padapter, WIRELESS_11G);
  4368. }
  4369. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  4370. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  4371. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  4372. mac = adapter_mac_addr(padapter);
  4373. fctrl = &(pwlanhdr->frame_ctl);
  4374. *(fctrl) = 0;
  4375. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  4376. _rtw_memcpy(pwlanhdr->addr2, mac, ETH_ALEN);
  4377. /* Use the device address for BSSID field. */
  4378. _rtw_memcpy(pwlanhdr->addr3, mac, ETH_ALEN);
  4379. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  4380. pmlmeext->mgnt_seq++;
  4381. set_frame_sub_type(fctrl, WIFI_PROBERSP);
  4382. pattrib->hdrlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  4383. pattrib->pktlen = pattrib->hdrlen;
  4384. pframe += pattrib->hdrlen;
  4385. /* timestamp will be inserted by hardware */
  4386. pframe += 8;
  4387. pattrib->pktlen += 8;
  4388. /* beacon interval: 2 bytes */
  4389. _rtw_memcpy(pframe, (unsigned char *) &beacon_interval, 2);
  4390. pframe += 2;
  4391. pattrib->pktlen += 2;
  4392. /* capability info: 2 bytes */
  4393. /* ESS and IBSS bits must be 0 (defined in the 3.1.2.1.1 of WiFi Direct Spec) */
  4394. capInfo |= cap_ShortPremble;
  4395. capInfo |= cap_ShortSlot;
  4396. _rtw_memcpy(pframe, (unsigned char *) &capInfo, 2);
  4397. pframe += 2;
  4398. pattrib->pktlen += 2;
  4399. /* SSID */
  4400. pframe = rtw_set_ie(pframe, _SSID_IE_, 7, pwdinfo->p2p_wildcard_ssid, &pattrib->pktlen);
  4401. /* supported rates... */
  4402. /* Use the OFDM rate in the P2P probe response frame. ( 6(B), 9(B), 12, 18, 24, 36, 48, 54 ) */
  4403. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, 8, pwdinfo->support_rate, &pattrib->pktlen);
  4404. /* DS parameter set */
  4405. pframe = rtw_set_ie(pframe, _DSSET_IE_, 1, (unsigned char *)&pwdinfo->listen_channel, &pattrib->pktlen);
  4406. #ifdef CONFIG_IOCTL_CFG80211
  4407. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  4408. if (pmlmepriv->wps_probe_resp_ie != NULL && pmlmepriv->p2p_probe_resp_ie != NULL) {
  4409. /* WPS IE */
  4410. _rtw_memcpy(pframe, pmlmepriv->wps_probe_resp_ie, pmlmepriv->wps_probe_resp_ie_len);
  4411. pattrib->pktlen += pmlmepriv->wps_probe_resp_ie_len;
  4412. pframe += pmlmepriv->wps_probe_resp_ie_len;
  4413. /* P2P IE */
  4414. _rtw_memcpy(pframe, pmlmepriv->p2p_probe_resp_ie, pmlmepriv->p2p_probe_resp_ie_len);
  4415. pattrib->pktlen += pmlmepriv->p2p_probe_resp_ie_len;
  4416. pframe += pmlmepriv->p2p_probe_resp_ie_len;
  4417. }
  4418. } else
  4419. #endif /* CONFIG_IOCTL_CFG80211 */
  4420. {
  4421. /* Todo: WPS IE */
  4422. /* Noted by Albert 20100907 */
  4423. /* According to the WPS specification, all the WPS attribute is presented by Big Endian. */
  4424. wpsielen = 0;
  4425. /* WPS OUI */
  4426. *(u32 *)(wpsie) = cpu_to_be32(WPSOUI);
  4427. wpsielen += 4;
  4428. /* WPS version */
  4429. /* Type: */
  4430. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_VER1);
  4431. wpsielen += 2;
  4432. /* Length: */
  4433. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4434. wpsielen += 2;
  4435. /* Value: */
  4436. wpsie[wpsielen++] = WPS_VERSION_1; /* Version 1.0 */
  4437. #ifdef CONFIG_INTEL_WIDI
  4438. /* Commented by Kurt */
  4439. /* Appended WiDi info. only if we did issued_probereq_widi(), and then we saved ven. ext. in pmlmepriv->sa_ext. */
  4440. if (_rtw_memcmp(pmlmepriv->sa_ext, zero_array_check, L2SDTA_SERVICE_VE_LEN) == _FALSE
  4441. || pmlmepriv->num_p2p_sdt != 0) {
  4442. /* Sec dev type */
  4443. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_SEC_DEV_TYPE_LIST);
  4444. wpsielen += 2;
  4445. /* Length: */
  4446. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0008);
  4447. wpsielen += 2;
  4448. /* Value: */
  4449. /* Category ID */
  4450. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_CID_DISPLAYS);
  4451. wpsielen += 2;
  4452. /* OUI */
  4453. *(u32 *)(wpsie + wpsielen) = cpu_to_be32(INTEL_DEV_TYPE_OUI);
  4454. wpsielen += 4;
  4455. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_SCID_WIDI_CONSUMER_SINK);
  4456. wpsielen += 2;
  4457. if (_rtw_memcmp(pmlmepriv->sa_ext, zero_array_check, L2SDTA_SERVICE_VE_LEN) == _FALSE) {
  4458. /* Vendor Extension */
  4459. _rtw_memcpy(wpsie + wpsielen, pmlmepriv->sa_ext, L2SDTA_SERVICE_VE_LEN);
  4460. wpsielen += L2SDTA_SERVICE_VE_LEN;
  4461. }
  4462. }
  4463. #endif /* CONFIG_INTEL_WIDI */
  4464. /* WiFi Simple Config State */
  4465. /* Type: */
  4466. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_SIMPLE_CONF_STATE);
  4467. wpsielen += 2;
  4468. /* Length: */
  4469. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4470. wpsielen += 2;
  4471. /* Value: */
  4472. wpsie[wpsielen++] = WPS_WSC_STATE_NOT_CONFIG; /* Not Configured. */
  4473. /* Response Type */
  4474. /* Type: */
  4475. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_RESP_TYPE);
  4476. wpsielen += 2;
  4477. /* Length: */
  4478. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4479. wpsielen += 2;
  4480. /* Value: */
  4481. wpsie[wpsielen++] = WPS_RESPONSE_TYPE_8021X;
  4482. /* UUID-E */
  4483. /* Type: */
  4484. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_UUID_E);
  4485. wpsielen += 2;
  4486. /* Length: */
  4487. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0010);
  4488. wpsielen += 2;
  4489. /* Value: */
  4490. if (pwdinfo->external_uuid == 0) {
  4491. _rtw_memset(wpsie + wpsielen, 0x0, 16);
  4492. _rtw_memcpy(wpsie + wpsielen, mac, ETH_ALEN);
  4493. } else
  4494. _rtw_memcpy(wpsie + wpsielen, pwdinfo->uuid, 0x10);
  4495. wpsielen += 0x10;
  4496. /* Manufacturer */
  4497. /* Type: */
  4498. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_MANUFACTURER);
  4499. wpsielen += 2;
  4500. /* Length: */
  4501. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0007);
  4502. wpsielen += 2;
  4503. /* Value: */
  4504. _rtw_memcpy(wpsie + wpsielen, "Realtek", 7);
  4505. wpsielen += 7;
  4506. /* Model Name */
  4507. /* Type: */
  4508. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_MODEL_NAME);
  4509. wpsielen += 2;
  4510. /* Length: */
  4511. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0006);
  4512. wpsielen += 2;
  4513. /* Value: */
  4514. _rtw_memcpy(wpsie + wpsielen, "8192CU", 6);
  4515. wpsielen += 6;
  4516. /* Model Number */
  4517. /* Type: */
  4518. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_MODEL_NUMBER);
  4519. wpsielen += 2;
  4520. /* Length: */
  4521. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4522. wpsielen += 2;
  4523. /* Value: */
  4524. wpsie[wpsielen++] = 0x31; /* character 1 */
  4525. /* Serial Number */
  4526. /* Type: */
  4527. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_SERIAL_NUMBER);
  4528. wpsielen += 2;
  4529. /* Length: */
  4530. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(ETH_ALEN);
  4531. wpsielen += 2;
  4532. /* Value: */
  4533. _rtw_memcpy(wpsie + wpsielen, "123456" , ETH_ALEN);
  4534. wpsielen += ETH_ALEN;
  4535. /* Primary Device Type */
  4536. /* Type: */
  4537. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_PRIMARY_DEV_TYPE);
  4538. wpsielen += 2;
  4539. /* Length: */
  4540. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0008);
  4541. wpsielen += 2;
  4542. /* Value: */
  4543. /* Category ID */
  4544. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  4545. wpsielen += 2;
  4546. /* OUI */
  4547. *(u32 *)(wpsie + wpsielen) = cpu_to_be32(WPSOUI);
  4548. wpsielen += 4;
  4549. /* Sub Category ID */
  4550. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  4551. wpsielen += 2;
  4552. /* Device Name */
  4553. /* Type: */
  4554. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  4555. wpsielen += 2;
  4556. /* Length: */
  4557. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(pwdinfo->device_name_len);
  4558. wpsielen += 2;
  4559. /* Value: */
  4560. _rtw_memcpy(wpsie + wpsielen, pwdinfo->device_name, pwdinfo->device_name_len);
  4561. wpsielen += pwdinfo->device_name_len;
  4562. /* Config Method */
  4563. /* Type: */
  4564. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_CONF_METHOD);
  4565. wpsielen += 2;
  4566. /* Length: */
  4567. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  4568. wpsielen += 2;
  4569. /* Value: */
  4570. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(pwdinfo->supported_wps_cm);
  4571. wpsielen += 2;
  4572. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, wpsielen, (unsigned char *) wpsie, &pattrib->pktlen);
  4573. p2pielen = build_probe_resp_p2p_ie(pwdinfo, pframe);
  4574. pframe += p2pielen;
  4575. pattrib->pktlen += p2pielen;
  4576. }
  4577. #ifdef CONFIG_WFD
  4578. wfdielen = rtw_append_probe_resp_wfd_ie(padapter, pframe);
  4579. pframe += wfdielen;
  4580. pattrib->pktlen += wfdielen;
  4581. #endif
  4582. pattrib->last_txcmdsz = pattrib->pktlen;
  4583. dump_mgntframe(padapter, pmgntframe);
  4584. return;
  4585. }
  4586. int _issue_probereq_p2p(_adapter *padapter, u8 *da, int wait_ack)
  4587. {
  4588. int ret = _FAIL;
  4589. struct xmit_frame *pmgntframe;
  4590. struct pkt_attrib *pattrib;
  4591. unsigned char *pframe;
  4592. struct rtw_ieee80211_hdr *pwlanhdr;
  4593. unsigned short *fctrl;
  4594. unsigned char *mac;
  4595. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  4596. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  4597. u8 bc_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  4598. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  4599. u8 wpsie[255] = { 0x00 }, p2pie[255] = { 0x00 };
  4600. u16 wpsielen = 0, p2pielen = 0;
  4601. #ifdef CONFIG_WFD
  4602. u32 wfdielen = 0;
  4603. #endif
  4604. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  4605. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  4606. if (pmgntframe == NULL)
  4607. goto exit;
  4608. /* update attribute */
  4609. pattrib = &pmgntframe->attrib;
  4610. update_mgntframe_attrib(padapter, pattrib);
  4611. if (IS_CCK_RATE(pattrib->rate)) {
  4612. /* force OFDM 6M rate */
  4613. pattrib->rate = MGN_6M;
  4614. pattrib->raid = rtw_get_mgntframe_raid(padapter, WIRELESS_11G);
  4615. }
  4616. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  4617. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  4618. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  4619. mac = adapter_mac_addr(padapter);
  4620. fctrl = &(pwlanhdr->frame_ctl);
  4621. *(fctrl) = 0;
  4622. if (da) {
  4623. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  4624. _rtw_memcpy(pwlanhdr->addr3, da, ETH_ALEN);
  4625. } else {
  4626. if ((pwdinfo->p2p_info.scan_op_ch_only) || (pwdinfo->rx_invitereq_info.scan_op_ch_only)) {
  4627. /* This two flags will be set when this is only the P2P client mode. */
  4628. _rtw_memcpy(pwlanhdr->addr1, pwdinfo->p2p_peer_interface_addr, ETH_ALEN);
  4629. _rtw_memcpy(pwlanhdr->addr3, pwdinfo->p2p_peer_interface_addr, ETH_ALEN);
  4630. } else {
  4631. /* broadcast probe request frame */
  4632. _rtw_memcpy(pwlanhdr->addr1, bc_addr, ETH_ALEN);
  4633. _rtw_memcpy(pwlanhdr->addr3, bc_addr, ETH_ALEN);
  4634. }
  4635. }
  4636. _rtw_memcpy(pwlanhdr->addr2, mac, ETH_ALEN);
  4637. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  4638. pmlmeext->mgnt_seq++;
  4639. set_frame_sub_type(pframe, WIFI_PROBEREQ);
  4640. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  4641. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  4642. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_TX_PROVISION_DIS_REQ))
  4643. pframe = rtw_set_ie(pframe, _SSID_IE_, pwdinfo->tx_prov_disc_info.ssid.SsidLength, pwdinfo->tx_prov_disc_info.ssid.Ssid, &(pattrib->pktlen));
  4644. else
  4645. pframe = rtw_set_ie(pframe, _SSID_IE_, P2P_WILDCARD_SSID_LEN, pwdinfo->p2p_wildcard_ssid, &(pattrib->pktlen));
  4646. /* Use the OFDM rate in the P2P probe request frame. ( 6(B), 9(B), 12(B), 24(B), 36, 48, 54 ) */
  4647. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, 8, pwdinfo->support_rate, &pattrib->pktlen);
  4648. #ifdef CONFIG_IOCTL_CFG80211
  4649. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  4650. if (pmlmepriv->wps_probe_req_ie != NULL && pmlmepriv->p2p_probe_req_ie != NULL) {
  4651. /* WPS IE */
  4652. _rtw_memcpy(pframe, pmlmepriv->wps_probe_req_ie, pmlmepriv->wps_probe_req_ie_len);
  4653. pattrib->pktlen += pmlmepriv->wps_probe_req_ie_len;
  4654. pframe += pmlmepriv->wps_probe_req_ie_len;
  4655. /* P2P IE */
  4656. _rtw_memcpy(pframe, pmlmepriv->p2p_probe_req_ie, pmlmepriv->p2p_probe_req_ie_len);
  4657. pattrib->pktlen += pmlmepriv->p2p_probe_req_ie_len;
  4658. pframe += pmlmepriv->p2p_probe_req_ie_len;
  4659. }
  4660. } else
  4661. #endif /* CONFIG_IOCTL_CFG80211 */
  4662. {
  4663. /* WPS IE */
  4664. /* Noted by Albert 20110221 */
  4665. /* According to the WPS specification, all the WPS attribute is presented by Big Endian. */
  4666. wpsielen = 0;
  4667. /* WPS OUI */
  4668. *(u32 *)(wpsie) = cpu_to_be32(WPSOUI);
  4669. wpsielen += 4;
  4670. /* WPS version */
  4671. /* Type: */
  4672. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_VER1);
  4673. wpsielen += 2;
  4674. /* Length: */
  4675. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0001);
  4676. wpsielen += 2;
  4677. /* Value: */
  4678. wpsie[wpsielen++] = WPS_VERSION_1; /* Version 1.0 */
  4679. if (pmlmepriv->wps_probe_req_ie == NULL) {
  4680. /* UUID-E */
  4681. /* Type: */
  4682. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_UUID_E);
  4683. wpsielen += 2;
  4684. /* Length: */
  4685. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0010);
  4686. wpsielen += 2;
  4687. /* Value: */
  4688. if (pwdinfo->external_uuid == 0) {
  4689. _rtw_memset(wpsie + wpsielen, 0x0, 16);
  4690. _rtw_memcpy(wpsie + wpsielen, mac, ETH_ALEN);
  4691. } else
  4692. _rtw_memcpy(wpsie + wpsielen, pwdinfo->uuid, 0x10);
  4693. wpsielen += 0x10;
  4694. /* Config Method */
  4695. /* Type: */
  4696. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_CONF_METHOD);
  4697. wpsielen += 2;
  4698. /* Length: */
  4699. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  4700. wpsielen += 2;
  4701. /* Value: */
  4702. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(pwdinfo->supported_wps_cm);
  4703. wpsielen += 2;
  4704. }
  4705. /* Device Name */
  4706. /* Type: */
  4707. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  4708. wpsielen += 2;
  4709. /* Length: */
  4710. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(pwdinfo->device_name_len);
  4711. wpsielen += 2;
  4712. /* Value: */
  4713. _rtw_memcpy(wpsie + wpsielen, pwdinfo->device_name, pwdinfo->device_name_len);
  4714. wpsielen += pwdinfo->device_name_len;
  4715. /* Primary Device Type */
  4716. /* Type: */
  4717. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_PRIMARY_DEV_TYPE);
  4718. wpsielen += 2;
  4719. /* Length: */
  4720. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0008);
  4721. wpsielen += 2;
  4722. /* Value: */
  4723. /* Category ID */
  4724. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_CID_RTK_WIDI);
  4725. wpsielen += 2;
  4726. /* OUI */
  4727. *(u32 *)(wpsie + wpsielen) = cpu_to_be32(WPSOUI);
  4728. wpsielen += 4;
  4729. /* Sub Category ID */
  4730. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_PDT_SCID_RTK_DMP);
  4731. wpsielen += 2;
  4732. /* Device Password ID */
  4733. /* Type: */
  4734. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_ATTR_DEVICE_PWID);
  4735. wpsielen += 2;
  4736. /* Length: */
  4737. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(0x0002);
  4738. wpsielen += 2;
  4739. /* Value: */
  4740. *(u16 *)(wpsie + wpsielen) = cpu_to_be16(WPS_DPID_REGISTRAR_SPEC); /* Registrar-specified */
  4741. wpsielen += 2;
  4742. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, wpsielen, (unsigned char *) wpsie, &pattrib->pktlen);
  4743. /* P2P OUI */
  4744. p2pielen = 0;
  4745. p2pie[p2pielen++] = 0x50;
  4746. p2pie[p2pielen++] = 0x6F;
  4747. p2pie[p2pielen++] = 0x9A;
  4748. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  4749. /* Commented by Albert 20110221 */
  4750. /* According to the P2P Specification, the probe request frame should contain 5 P2P attributes */
  4751. /* 1. P2P Capability */
  4752. /* 2. P2P Device ID if this probe request wants to find the specific P2P device */
  4753. /* 3. Listen Channel */
  4754. /* 4. Extended Listen Timing */
  4755. /* 5. Operating Channel if this WiFi is working as the group owner now */
  4756. /* P2P Capability */
  4757. /* Type: */
  4758. p2pie[p2pielen++] = P2P_ATTR_CAPABILITY;
  4759. /* Length: */
  4760. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  4761. p2pielen += 2;
  4762. /* Value: */
  4763. /* Device Capability Bitmap, 1 byte */
  4764. p2pie[p2pielen++] = DMP_P2P_DEVCAP_SUPPORT;
  4765. /* Group Capability Bitmap, 1 byte */
  4766. if (pwdinfo->persistent_supported)
  4767. p2pie[p2pielen++] = P2P_GRPCAP_PERSISTENT_GROUP | DMP_P2P_GRPCAP_SUPPORT;
  4768. else
  4769. p2pie[p2pielen++] = DMP_P2P_GRPCAP_SUPPORT;
  4770. /* Listen Channel */
  4771. /* Type: */
  4772. p2pie[p2pielen++] = P2P_ATTR_LISTEN_CH;
  4773. /* Length: */
  4774. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  4775. p2pielen += 2;
  4776. /* Value: */
  4777. /* Country String */
  4778. p2pie[p2pielen++] = 'X';
  4779. p2pie[p2pielen++] = 'X';
  4780. /* The third byte should be set to 0x04. */
  4781. /* Described in the "Operating Channel Attribute" section. */
  4782. p2pie[p2pielen++] = 0x04;
  4783. /* Operating Class */
  4784. p2pie[p2pielen++] = 0x51; /* Copy from SD7 */
  4785. /* Channel Number */
  4786. p2pie[p2pielen++] = pwdinfo->listen_channel; /* listen channel */
  4787. /* Extended Listen Timing */
  4788. /* Type: */
  4789. p2pie[p2pielen++] = P2P_ATTR_EX_LISTEN_TIMING;
  4790. /* Length: */
  4791. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0004);
  4792. p2pielen += 2;
  4793. /* Value: */
  4794. /* Availability Period */
  4795. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  4796. p2pielen += 2;
  4797. /* Availability Interval */
  4798. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  4799. p2pielen += 2;
  4800. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  4801. /* Operating Channel (if this WiFi is working as the group owner now) */
  4802. /* Type: */
  4803. p2pie[p2pielen++] = P2P_ATTR_OPERATING_CH;
  4804. /* Length: */
  4805. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0005);
  4806. p2pielen += 2;
  4807. /* Value: */
  4808. /* Country String */
  4809. p2pie[p2pielen++] = 'X';
  4810. p2pie[p2pielen++] = 'X';
  4811. /* The third byte should be set to 0x04. */
  4812. /* Described in the "Operating Channel Attribute" section. */
  4813. p2pie[p2pielen++] = 0x04;
  4814. /* Operating Class */
  4815. p2pie[p2pielen++] = 0x51; /* Copy from SD7 */
  4816. /* Channel Number */
  4817. p2pie[p2pielen++] = pwdinfo->operating_channel; /* operating channel number */
  4818. }
  4819. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  4820. }
  4821. #ifdef CONFIG_WFD
  4822. wfdielen = rtw_append_probe_req_wfd_ie(padapter, pframe);
  4823. pframe += wfdielen;
  4824. pattrib->pktlen += wfdielen;
  4825. #endif
  4826. pattrib->last_txcmdsz = pattrib->pktlen;
  4827. if (wait_ack)
  4828. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  4829. else {
  4830. dump_mgntframe(padapter, pmgntframe);
  4831. ret = _SUCCESS;
  4832. }
  4833. exit:
  4834. return ret;
  4835. }
  4836. inline void issue_probereq_p2p(_adapter *adapter, u8 *da)
  4837. {
  4838. _issue_probereq_p2p(adapter, da, _FALSE);
  4839. }
  4840. /*
  4841. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  4842. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  4843. * try_cnt means the maximal TX count to try
  4844. */
  4845. int issue_probereq_p2p_ex(_adapter *adapter, u8 *da, int try_cnt, int wait_ms)
  4846. {
  4847. int ret;
  4848. int i = 0;
  4849. systime start = rtw_get_current_time();
  4850. do {
  4851. ret = _issue_probereq_p2p(adapter, da, wait_ms > 0 ? _TRUE : _FALSE);
  4852. i++;
  4853. if (RTW_CANNOT_RUN(adapter))
  4854. break;
  4855. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  4856. rtw_msleep_os(wait_ms);
  4857. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  4858. if (ret != _FAIL) {
  4859. ret = _SUCCESS;
  4860. #ifndef DBG_XMIT_ACK
  4861. goto exit;
  4862. #endif
  4863. }
  4864. if (try_cnt && wait_ms) {
  4865. if (da)
  4866. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", ch:%u%s, %d/%d in %u ms\n",
  4867. FUNC_ADPT_ARG(adapter), MAC_ARG(da), rtw_get_oper_ch(adapter),
  4868. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  4869. else
  4870. RTW_INFO(FUNC_ADPT_FMT", ch:%u%s, %d/%d in %u ms\n",
  4871. FUNC_ADPT_ARG(adapter), rtw_get_oper_ch(adapter),
  4872. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  4873. }
  4874. exit:
  4875. return ret;
  4876. }
  4877. #endif /* CONFIG_P2P */
  4878. s32 rtw_action_public_decache(union recv_frame *rframe, u8 token_offset)
  4879. {
  4880. _adapter *adapter = rframe->u.hdr.adapter;
  4881. struct mlme_ext_priv *mlmeext = &(adapter->mlmeextpriv);
  4882. u8 *frame = rframe->u.hdr.rx_data;
  4883. u16 seq_ctrl = ((rframe->u.hdr.attrib.seq_num & 0xffff) << 4) | (rframe->u.hdr.attrib.frag_num & 0xf);
  4884. u8 token = *(rframe->u.hdr.rx_data + sizeof(struct rtw_ieee80211_hdr_3addr) + token_offset);
  4885. if (GetRetry(frame)) {
  4886. if ((seq_ctrl == mlmeext->action_public_rxseq)
  4887. && (token == mlmeext->action_public_dialog_token)
  4888. ) {
  4889. RTW_INFO(FUNC_ADPT_FMT" seq_ctrl=0x%x, rxseq=0x%x, token:%d\n",
  4890. FUNC_ADPT_ARG(adapter), seq_ctrl, mlmeext->action_public_rxseq, token);
  4891. return _FAIL;
  4892. }
  4893. }
  4894. /* TODO: per sta seq & token */
  4895. mlmeext->action_public_rxseq = seq_ctrl;
  4896. mlmeext->action_public_dialog_token = token;
  4897. return _SUCCESS;
  4898. }
  4899. unsigned int on_action_public_p2p(union recv_frame *precv_frame)
  4900. {
  4901. _adapter *padapter = precv_frame->u.hdr.adapter;
  4902. u8 *pframe = precv_frame->u.hdr.rx_data;
  4903. uint len = precv_frame->u.hdr.len;
  4904. u8 *frame_body;
  4905. #ifdef CONFIG_P2P
  4906. u8 *p2p_ie;
  4907. u32 p2p_ielen;
  4908. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  4909. u8 result = P2P_STATUS_SUCCESS;
  4910. u8 empty_addr[ETH_ALEN] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
  4911. u8 *merged_p2pie = NULL;
  4912. u32 merged_p2p_ielen = 0;
  4913. #endif /* CONFIG_P2P */
  4914. frame_body = (unsigned char *)(pframe + sizeof(struct rtw_ieee80211_hdr_3addr));
  4915. #ifdef CONFIG_P2P
  4916. _cancel_timer_ex(&pwdinfo->reset_ch_sitesurvey);
  4917. #ifdef CONFIG_IOCTL_CFG80211
  4918. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211)
  4919. rtw_cfg80211_rx_p2p_action_public(padapter, precv_frame);
  4920. else
  4921. #endif /* CONFIG_IOCTL_CFG80211 */
  4922. {
  4923. /* Do nothing if the driver doesn't enable the P2P function. */
  4924. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE) || rtw_p2p_chk_state(pwdinfo, P2P_STATE_IDLE))
  4925. return _SUCCESS;
  4926. len -= sizeof(struct rtw_ieee80211_hdr_3addr);
  4927. switch (frame_body[6]) { /* OUI Subtype */
  4928. case P2P_GO_NEGO_REQ: {
  4929. RTW_INFO("[%s] Got GO Nego Req Frame\n", __FUNCTION__);
  4930. _rtw_memset(&pwdinfo->groupid_info, 0x00, sizeof(struct group_id_info));
  4931. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_RX_PROVISION_DIS_REQ))
  4932. rtw_p2p_set_state(pwdinfo, rtw_p2p_pre_state(pwdinfo));
  4933. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_GONEGO_FAIL)) {
  4934. /* Commented by Albert 20110526 */
  4935. /* In this case, this means the previous nego fail doesn't be reset yet. */
  4936. _cancel_timer_ex(&pwdinfo->restore_p2p_state_timer);
  4937. /* Restore the previous p2p state */
  4938. rtw_p2p_set_state(pwdinfo, rtw_p2p_pre_state(pwdinfo));
  4939. RTW_INFO("[%s] Restore the previous p2p state to %d\n", __FUNCTION__, rtw_p2p_state(pwdinfo));
  4940. }
  4941. #ifdef CONFIG_CONCURRENT_MODE
  4942. if (rtw_mi_buddy_check_fwstate(padapter, _FW_LINKED))
  4943. _cancel_timer_ex(&pwdinfo->ap_p2p_switch_timer);
  4944. #endif /* CONFIG_CONCURRENT_MODE */
  4945. /* Commented by Kurt 20110902 */
  4946. /* Add if statement to avoid receiving duplicate prov disc req. such that pre_p2p_state would be covered. */
  4947. if (!rtw_p2p_chk_state(pwdinfo, P2P_STATE_GONEGO_ING))
  4948. rtw_p2p_set_pre_state(pwdinfo, rtw_p2p_state(pwdinfo));
  4949. /* Commented by Kurt 20120113 */
  4950. /* Get peer_dev_addr here if peer doesn't issue prov_disc frame. */
  4951. if (_rtw_memcmp(pwdinfo->rx_prov_disc_info.peerDevAddr, empty_addr, ETH_ALEN))
  4952. _rtw_memcpy(pwdinfo->rx_prov_disc_info.peerDevAddr, get_addr2_ptr(pframe), ETH_ALEN);
  4953. result = process_p2p_group_negotation_req(pwdinfo, frame_body, len);
  4954. issue_p2p_GO_response(padapter, get_addr2_ptr(pframe), frame_body, len, result);
  4955. #ifdef CONFIG_INTEL_WIDI
  4956. if (padapter->mlmepriv.widi_state == INTEL_WIDI_STATE_LISTEN) {
  4957. padapter->mlmepriv.widi_state = INTEL_WIDI_STATE_WFD_CONNECTION;
  4958. _cancel_timer_ex(&(padapter->mlmepriv.listen_timer));
  4959. intel_widi_wk_cmd(padapter, INTEL_WIDI_LISTEN_STOP_WK, NULL, 0);
  4960. }
  4961. #endif /* CONFIG_INTEL_WIDI */
  4962. /* Commented by Albert 20110718 */
  4963. /* No matter negotiating or negotiation failure, the driver should set up the restore P2P state timer. */
  4964. #ifdef CONFIG_CONCURRENT_MODE
  4965. /* Commented by Albert 20120107 */
  4966. _set_timer(&pwdinfo->restore_p2p_state_timer, 3000);
  4967. #else /* CONFIG_CONCURRENT_MODE */
  4968. _set_timer(&pwdinfo->restore_p2p_state_timer, 5000);
  4969. #endif /* CONFIG_CONCURRENT_MODE */
  4970. break;
  4971. }
  4972. case P2P_GO_NEGO_RESP: {
  4973. RTW_INFO("[%s] Got GO Nego Resp Frame\n", __FUNCTION__);
  4974. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_GONEGO_ING)) {
  4975. /* Commented by Albert 20110425 */
  4976. /* The restore timer is enabled when issuing the nego request frame of rtw_p2p_connect function. */
  4977. _cancel_timer_ex(&pwdinfo->restore_p2p_state_timer);
  4978. pwdinfo->nego_req_info.benable = _FALSE;
  4979. result = process_p2p_group_negotation_resp(pwdinfo, frame_body, len);
  4980. issue_p2p_GO_confirm(pwdinfo->padapter, get_addr2_ptr(pframe), result);
  4981. if (P2P_STATUS_SUCCESS == result) {
  4982. if (rtw_p2p_role(pwdinfo) == P2P_ROLE_CLIENT) {
  4983. pwdinfo->p2p_info.operation_ch[0] = pwdinfo->peer_operating_ch;
  4984. #ifdef CONFIG_P2P_OP_CHK_SOCIAL_CH
  4985. pwdinfo->p2p_info.operation_ch[1] = 1; /* Check whether GO is operating in channel 1; */
  4986. pwdinfo->p2p_info.operation_ch[2] = 6; /* Check whether GO is operating in channel 6; */
  4987. pwdinfo->p2p_info.operation_ch[3] = 11; /* Check whether GO is operating in channel 11; */
  4988. #endif /* CONFIG_P2P_OP_CHK_SOCIAL_CH */
  4989. pwdinfo->p2p_info.scan_op_ch_only = 1;
  4990. _set_timer(&pwdinfo->reset_ch_sitesurvey2, P2P_RESET_SCAN_CH);
  4991. }
  4992. }
  4993. /* Reset the dialog token for group negotiation frames. */
  4994. pwdinfo->negotiation_dialog_token = 1;
  4995. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_GONEGO_FAIL))
  4996. _set_timer(&pwdinfo->restore_p2p_state_timer, 5000);
  4997. } else
  4998. RTW_INFO("[%s] Skipped GO Nego Resp Frame (p2p_state != P2P_STATE_GONEGO_ING)\n", __FUNCTION__);
  4999. break;
  5000. }
  5001. case P2P_GO_NEGO_CONF: {
  5002. RTW_INFO("[%s] Got GO Nego Confirm Frame\n", __FUNCTION__);
  5003. result = process_p2p_group_negotation_confirm(pwdinfo, frame_body, len);
  5004. if (P2P_STATUS_SUCCESS == result) {
  5005. if (rtw_p2p_role(pwdinfo) == P2P_ROLE_CLIENT) {
  5006. pwdinfo->p2p_info.operation_ch[0] = pwdinfo->peer_operating_ch;
  5007. #ifdef CONFIG_P2P_OP_CHK_SOCIAL_CH
  5008. pwdinfo->p2p_info.operation_ch[1] = 1; /* Check whether GO is operating in channel 1; */
  5009. pwdinfo->p2p_info.operation_ch[2] = 6; /* Check whether GO is operating in channel 6; */
  5010. pwdinfo->p2p_info.operation_ch[3] = 11; /* Check whether GO is operating in channel 11; */
  5011. #endif /* CONFIG_P2P_OP_CHK_SOCIAL_CH */
  5012. pwdinfo->p2p_info.scan_op_ch_only = 1;
  5013. _set_timer(&pwdinfo->reset_ch_sitesurvey2, P2P_RESET_SCAN_CH);
  5014. }
  5015. }
  5016. break;
  5017. }
  5018. case P2P_INVIT_REQ: {
  5019. /* Added by Albert 2010/10/05 */
  5020. /* Received the P2P Invite Request frame. */
  5021. RTW_INFO("[%s] Got invite request frame!\n", __FUNCTION__);
  5022. p2p_ie = rtw_get_p2p_ie(frame_body + _PUBLIC_ACTION_IE_OFFSET_, len - _PUBLIC_ACTION_IE_OFFSET_, NULL, &p2p_ielen);
  5023. if (p2p_ie) {
  5024. /* Parse the necessary information from the P2P Invitation Request frame. */
  5025. /* For example: The MAC address of sending this P2P Invitation Request frame. */
  5026. u32 attr_contentlen = 0;
  5027. u8 status_code = P2P_STATUS_FAIL_INFO_UNAVAILABLE;
  5028. struct group_id_info group_id;
  5029. u8 invitation_flag = 0;
  5030. merged_p2p_ielen = rtw_get_p2p_merged_ies_len(frame_body + _PUBLIC_ACTION_IE_OFFSET_, len - _PUBLIC_ACTION_IE_OFFSET_);
  5031. merged_p2pie = rtw_zmalloc(merged_p2p_ielen + 2); /* 2 is for EID and Length */
  5032. if (merged_p2pie == NULL) {
  5033. RTW_INFO("[%s] Malloc p2p ie fail\n", __FUNCTION__);
  5034. goto exit;
  5035. }
  5036. _rtw_memset(merged_p2pie, 0x00, merged_p2p_ielen);
  5037. merged_p2p_ielen = rtw_p2p_merge_ies(frame_body + _PUBLIC_ACTION_IE_OFFSET_, len - _PUBLIC_ACTION_IE_OFFSET_, merged_p2pie);
  5038. rtw_get_p2p_attr_content(merged_p2pie, merged_p2p_ielen, P2P_ATTR_INVITATION_FLAGS, &invitation_flag, &attr_contentlen);
  5039. if (attr_contentlen) {
  5040. rtw_get_p2p_attr_content(merged_p2pie, merged_p2p_ielen, P2P_ATTR_GROUP_BSSID, pwdinfo->p2p_peer_interface_addr, &attr_contentlen);
  5041. /* Commented by Albert 20120510 */
  5042. /* Copy to the pwdinfo->p2p_peer_interface_addr. */
  5043. /* So that the WFD UI ( or Sigma ) can get the peer interface address by using the following command. */
  5044. /* #> iwpriv wlan0 p2p_get peer_ifa */
  5045. /* After having the peer interface address, the sigma can find the correct conf file for wpa_supplicant. */
  5046. if (attr_contentlen) {
  5047. RTW_INFO("[%s] GO's BSSID = %.2X %.2X %.2X %.2X %.2X %.2X\n", __FUNCTION__,
  5048. pwdinfo->p2p_peer_interface_addr[0], pwdinfo->p2p_peer_interface_addr[1],
  5049. pwdinfo->p2p_peer_interface_addr[2], pwdinfo->p2p_peer_interface_addr[3],
  5050. pwdinfo->p2p_peer_interface_addr[4], pwdinfo->p2p_peer_interface_addr[5]);
  5051. }
  5052. if (invitation_flag & P2P_INVITATION_FLAGS_PERSISTENT) {
  5053. /* Re-invoke the persistent group. */
  5054. _rtw_memset(&group_id, 0x00, sizeof(struct group_id_info));
  5055. rtw_get_p2p_attr_content(merged_p2pie, merged_p2p_ielen, P2P_ATTR_GROUP_ID, (u8 *) &group_id, &attr_contentlen);
  5056. if (attr_contentlen) {
  5057. if (_rtw_memcmp(group_id.go_device_addr, adapter_mac_addr(padapter), ETH_ALEN)) {
  5058. /* The p2p device sending this p2p invitation request wants this Wi-Fi device to be the persistent GO. */
  5059. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_GO);
  5060. rtw_p2p_set_role(pwdinfo, P2P_ROLE_GO);
  5061. status_code = P2P_STATUS_SUCCESS;
  5062. } else {
  5063. /* The p2p device sending this p2p invitation request wants to be the persistent GO. */
  5064. if (is_matched_in_profilelist(pwdinfo->p2p_peer_interface_addr, &pwdinfo->profileinfo[0])) {
  5065. u8 operatingch_info[5] = { 0x00 };
  5066. if (rtw_get_p2p_attr_content(merged_p2pie, merged_p2p_ielen, P2P_ATTR_OPERATING_CH, operatingch_info,
  5067. &attr_contentlen)) {
  5068. if (rtw_chset_search_ch(adapter_to_chset(padapter), (u32)operatingch_info[4]) >= 0) {
  5069. /* The operating channel is acceptable for this device. */
  5070. pwdinfo->rx_invitereq_info.operation_ch[0] = operatingch_info[4];
  5071. #ifdef CONFIG_P2P_OP_CHK_SOCIAL_CH
  5072. pwdinfo->rx_invitereq_info.operation_ch[1] = 1; /* Check whether GO is operating in channel 1; */
  5073. pwdinfo->rx_invitereq_info.operation_ch[2] = 6; /* Check whether GO is operating in channel 6; */
  5074. pwdinfo->rx_invitereq_info.operation_ch[3] = 11; /* Check whether GO is operating in channel 11; */
  5075. #endif /* CONFIG_P2P_OP_CHK_SOCIAL_CH */
  5076. pwdinfo->rx_invitereq_info.scan_op_ch_only = 1;
  5077. _set_timer(&pwdinfo->reset_ch_sitesurvey, P2P_RESET_SCAN_CH);
  5078. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_MATCH);
  5079. rtw_p2p_set_role(pwdinfo, P2P_ROLE_CLIENT);
  5080. status_code = P2P_STATUS_SUCCESS;
  5081. } else {
  5082. /* The operating channel isn't supported by this device. */
  5083. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_DISMATCH);
  5084. rtw_p2p_set_role(pwdinfo, P2P_ROLE_DEVICE);
  5085. status_code = P2P_STATUS_FAIL_NO_COMMON_CH;
  5086. _set_timer(&pwdinfo->restore_p2p_state_timer, 3000);
  5087. }
  5088. } else {
  5089. /* Commented by Albert 20121130 */
  5090. /* Intel will use the different P2P IE to store the operating channel information */
  5091. /* Workaround for Intel WiDi 3.5 */
  5092. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_MATCH);
  5093. rtw_p2p_set_role(pwdinfo, P2P_ROLE_CLIENT);
  5094. status_code = P2P_STATUS_SUCCESS;
  5095. }
  5096. } else {
  5097. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_DISMATCH);
  5098. #ifdef CONFIG_INTEL_WIDI
  5099. _rtw_memcpy(pwdinfo->p2p_peer_device_addr, group_id.go_device_addr , ETH_ALEN);
  5100. rtw_p2p_set_role(pwdinfo, P2P_ROLE_CLIENT);
  5101. #endif /* CONFIG_INTEL_WIDI */
  5102. status_code = P2P_STATUS_FAIL_UNKNOWN_P2PGROUP;
  5103. }
  5104. }
  5105. } else {
  5106. RTW_INFO("[%s] P2P Group ID Attribute NOT FOUND!\n", __FUNCTION__);
  5107. status_code = P2P_STATUS_FAIL_INFO_UNAVAILABLE;
  5108. }
  5109. } else {
  5110. /* Received the invitation to join a P2P group. */
  5111. _rtw_memset(&group_id, 0x00, sizeof(struct group_id_info));
  5112. rtw_get_p2p_attr_content(merged_p2pie, merged_p2p_ielen, P2P_ATTR_GROUP_ID, (u8 *) &group_id, &attr_contentlen);
  5113. if (attr_contentlen) {
  5114. if (_rtw_memcmp(group_id.go_device_addr, adapter_mac_addr(padapter), ETH_ALEN)) {
  5115. /* In this case, the GO can't be myself. */
  5116. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_DISMATCH);
  5117. status_code = P2P_STATUS_FAIL_INFO_UNAVAILABLE;
  5118. } else {
  5119. /* The p2p device sending this p2p invitation request wants to join an existing P2P group */
  5120. /* Commented by Albert 2012/06/28 */
  5121. /* In this case, this Wi-Fi device should use the iwpriv command to get the peer device address. */
  5122. /* The peer device address should be the destination address for the provisioning discovery request. */
  5123. /* Then, this Wi-Fi device should use the iwpriv command to get the peer interface address. */
  5124. /* The peer interface address should be the address for WPS mac address */
  5125. _rtw_memcpy(pwdinfo->p2p_peer_device_addr, group_id.go_device_addr , ETH_ALEN);
  5126. rtw_p2p_set_role(pwdinfo, P2P_ROLE_CLIENT);
  5127. rtw_p2p_set_state(pwdinfo, P2P_STATE_RECV_INVITE_REQ_JOIN);
  5128. status_code = P2P_STATUS_SUCCESS;
  5129. }
  5130. } else {
  5131. RTW_INFO("[%s] P2P Group ID Attribute NOT FOUND!\n", __FUNCTION__);
  5132. status_code = P2P_STATUS_FAIL_INFO_UNAVAILABLE;
  5133. }
  5134. }
  5135. } else {
  5136. RTW_INFO("[%s] P2P Invitation Flags Attribute NOT FOUND!\n", __FUNCTION__);
  5137. status_code = P2P_STATUS_FAIL_INFO_UNAVAILABLE;
  5138. }
  5139. RTW_INFO("[%s] status_code = %d\n", __FUNCTION__, status_code);
  5140. pwdinfo->inviteresp_info.token = frame_body[7];
  5141. issue_p2p_invitation_response(padapter, get_addr2_ptr(pframe), pwdinfo->inviteresp_info.token, status_code);
  5142. _set_timer(&pwdinfo->restore_p2p_state_timer, 3000);
  5143. }
  5144. #ifdef CONFIG_INTEL_WIDI
  5145. if (padapter->mlmepriv.widi_state == INTEL_WIDI_STATE_LISTEN) {
  5146. padapter->mlmepriv.widi_state = INTEL_WIDI_STATE_WFD_CONNECTION;
  5147. _cancel_timer_ex(&(padapter->mlmepriv.listen_timer));
  5148. intel_widi_wk_cmd(padapter, INTEL_WIDI_LISTEN_STOP_WK, NULL, 0);
  5149. }
  5150. #endif /* CONFIG_INTEL_WIDI */
  5151. break;
  5152. }
  5153. case P2P_INVIT_RESP: {
  5154. u8 attr_content = 0x00;
  5155. u32 attr_contentlen = 0;
  5156. RTW_INFO("[%s] Got invite response frame!\n", __FUNCTION__);
  5157. _cancel_timer_ex(&pwdinfo->restore_p2p_state_timer);
  5158. p2p_ie = rtw_get_p2p_ie(frame_body + _PUBLIC_ACTION_IE_OFFSET_, len - _PUBLIC_ACTION_IE_OFFSET_, NULL, &p2p_ielen);
  5159. if (p2p_ie) {
  5160. rtw_get_p2p_attr_content(p2p_ie, p2p_ielen, P2P_ATTR_STATUS, &attr_content, &attr_contentlen);
  5161. if (attr_contentlen == 1) {
  5162. RTW_INFO("[%s] Status = %d\n", __FUNCTION__, attr_content);
  5163. pwdinfo->invitereq_info.benable = _FALSE;
  5164. if (attr_content == P2P_STATUS_SUCCESS) {
  5165. if (_rtw_memcmp(pwdinfo->invitereq_info.go_bssid, adapter_mac_addr(padapter), ETH_ALEN))
  5166. rtw_p2p_set_role(pwdinfo, P2P_ROLE_GO);
  5167. else
  5168. rtw_p2p_set_role(pwdinfo, P2P_ROLE_CLIENT);
  5169. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_INVITE_RESP_OK);
  5170. } else {
  5171. rtw_p2p_set_role(pwdinfo, P2P_ROLE_DEVICE);
  5172. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_INVITE_RESP_FAIL);
  5173. }
  5174. } else {
  5175. rtw_p2p_set_role(pwdinfo, P2P_ROLE_DEVICE);
  5176. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_INVITE_RESP_FAIL);
  5177. }
  5178. } else {
  5179. rtw_p2p_set_role(pwdinfo, P2P_ROLE_DEVICE);
  5180. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_INVITE_RESP_FAIL);
  5181. }
  5182. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_RX_INVITE_RESP_FAIL))
  5183. _set_timer(&pwdinfo->restore_p2p_state_timer, 5000);
  5184. break;
  5185. }
  5186. case P2P_DEVDISC_REQ:
  5187. process_p2p_devdisc_req(pwdinfo, pframe, len);
  5188. break;
  5189. case P2P_DEVDISC_RESP:
  5190. process_p2p_devdisc_resp(pwdinfo, pframe, len);
  5191. break;
  5192. case P2P_PROVISION_DISC_REQ:
  5193. RTW_INFO("[%s] Got Provisioning Discovery Request Frame\n", __FUNCTION__);
  5194. process_p2p_provdisc_req(pwdinfo, pframe, len);
  5195. _rtw_memcpy(pwdinfo->rx_prov_disc_info.peerDevAddr, get_addr2_ptr(pframe), ETH_ALEN);
  5196. /* 20110902 Kurt */
  5197. /* Add the following statement to avoid receiving duplicate prov disc req. such that pre_p2p_state would be covered. */
  5198. if (!rtw_p2p_chk_state(pwdinfo, P2P_STATE_RX_PROVISION_DIS_REQ))
  5199. rtw_p2p_set_pre_state(pwdinfo, rtw_p2p_state(pwdinfo));
  5200. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_PROVISION_DIS_REQ);
  5201. _set_timer(&pwdinfo->restore_p2p_state_timer, P2P_PROVISION_TIMEOUT);
  5202. #ifdef CONFIG_INTEL_WIDI
  5203. if (padapter->mlmepriv.widi_state == INTEL_WIDI_STATE_LISTEN) {
  5204. padapter->mlmepriv.widi_state = INTEL_WIDI_STATE_WFD_CONNECTION;
  5205. _cancel_timer_ex(&(padapter->mlmepriv.listen_timer));
  5206. intel_widi_wk_cmd(padapter, INTEL_WIDI_LISTEN_STOP_WK, NULL, 0);
  5207. }
  5208. #endif /* CONFIG_INTEL_WIDI */
  5209. break;
  5210. case P2P_PROVISION_DISC_RESP:
  5211. /* Commented by Albert 20110707 */
  5212. /* Should we check the pwdinfo->tx_prov_disc_info.bsent flag here?? */
  5213. RTW_INFO("[%s] Got Provisioning Discovery Response Frame\n", __FUNCTION__);
  5214. /* Commented by Albert 20110426 */
  5215. /* The restore timer is enabled when issuing the provisioing request frame in rtw_p2p_prov_disc function. */
  5216. _cancel_timer_ex(&pwdinfo->restore_p2p_state_timer);
  5217. rtw_p2p_set_state(pwdinfo, P2P_STATE_RX_PROVISION_DIS_RSP);
  5218. process_p2p_provdisc_resp(pwdinfo, pframe);
  5219. _set_timer(&pwdinfo->restore_p2p_state_timer, P2P_PROVISION_TIMEOUT);
  5220. break;
  5221. }
  5222. }
  5223. exit:
  5224. if (merged_p2pie)
  5225. rtw_mfree(merged_p2pie, merged_p2p_ielen + 2);
  5226. #endif /* CONFIG_P2P */
  5227. return _SUCCESS;
  5228. }
  5229. unsigned int on_action_public_vendor(union recv_frame *precv_frame)
  5230. {
  5231. unsigned int ret = _FAIL;
  5232. u8 *pframe = precv_frame->u.hdr.rx_data;
  5233. u8 *frame_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5234. if (_rtw_memcmp(frame_body + 2, P2P_OUI, 4) == _TRUE) {
  5235. if (rtw_action_public_decache(precv_frame, 7) == _FAIL)
  5236. goto exit;
  5237. if (!hal_chk_wl_func(precv_frame->u.hdr.adapter, WL_FUNC_MIRACAST))
  5238. rtw_rframe_del_wfd_ie(precv_frame, 8);
  5239. ret = on_action_public_p2p(precv_frame);
  5240. }
  5241. exit:
  5242. return ret;
  5243. }
  5244. unsigned int on_action_public_default(union recv_frame *precv_frame, u8 action)
  5245. {
  5246. unsigned int ret = _FAIL;
  5247. u8 *pframe = precv_frame->u.hdr.rx_data;
  5248. u8 *frame_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5249. u8 token;
  5250. _adapter *adapter = precv_frame->u.hdr.adapter;
  5251. int cnt = 0;
  5252. char msg[64];
  5253. token = frame_body[2];
  5254. if (rtw_action_public_decache(precv_frame, 2) == _FAIL)
  5255. goto exit;
  5256. #ifdef CONFIG_IOCTL_CFG80211
  5257. cnt += sprintf((msg + cnt), "%s(token:%u)", action_public_str(action), token);
  5258. rtw_cfg80211_rx_action(adapter, precv_frame, msg);
  5259. #endif
  5260. ret = _SUCCESS;
  5261. exit:
  5262. return ret;
  5263. }
  5264. unsigned int on_action_public(_adapter *padapter, union recv_frame *precv_frame)
  5265. {
  5266. unsigned int ret = _FAIL;
  5267. u8 *pframe = precv_frame->u.hdr.rx_data;
  5268. uint frame_len = precv_frame->u.hdr.len;
  5269. u8 *frame_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5270. u8 category, action;
  5271. /* check RA matches or not */
  5272. if (!_rtw_memcmp(adapter_mac_addr(padapter), GetAddr1Ptr(pframe), ETH_ALEN))
  5273. goto exit;
  5274. category = frame_body[0];
  5275. if (category != RTW_WLAN_CATEGORY_PUBLIC)
  5276. goto exit;
  5277. action = frame_body[1];
  5278. switch (action) {
  5279. case ACT_PUBLIC_BSSCOEXIST:
  5280. #ifdef CONFIG_80211N_HT
  5281. #ifdef CONFIG_AP_MODE
  5282. /*20/40 BSS Coexistence Management frame is a Public Action frame*/
  5283. if (check_fwstate(&padapter->mlmepriv, WIFI_AP_STATE) == _TRUE)
  5284. rtw_process_public_act_bsscoex(padapter, pframe, frame_len);
  5285. #endif /*CONFIG_AP_MODE*/
  5286. #endif /*CONFIG_80211N_HT*/
  5287. break;
  5288. case ACT_PUBLIC_VENDOR:
  5289. ret = on_action_public_vendor(precv_frame);
  5290. break;
  5291. default:
  5292. ret = on_action_public_default(precv_frame, action);
  5293. break;
  5294. }
  5295. exit:
  5296. return ret;
  5297. }
  5298. #if defined(CONFIG_RTW_WNM) || defined(CONFIG_RTW_80211K)
  5299. static u8 rtw_wnm_nb_elem_parsing(
  5300. u8* pdata, u32 data_len, u8 from_btm,
  5301. u32 *nb_rpt_num, u8 *nb_rpt_is_same,
  5302. struct roam_nb_info *pnb, struct wnm_btm_cant *pcandidates)
  5303. {
  5304. u8 bfound = _FALSE, ret = _SUCCESS;
  5305. u8 *ptr, *pend, *op;
  5306. u32 elem_len, subelem_len, op_len;
  5307. u32 i, nb_rpt_entries = 0;
  5308. struct nb_rpt_hdr *pie;
  5309. struct wnm_btm_cant *pcandidate;
  5310. if ((!pdata) || (!pnb))
  5311. return _FAIL;
  5312. if ((from_btm) && (!pcandidates))
  5313. return _FAIL;
  5314. ptr = pdata;
  5315. pend = ptr + data_len;
  5316. elem_len = data_len;
  5317. subelem_len = (u32)*(pdata+1);
  5318. for (i=0; i < RTW_MAX_NB_RPT_NUM; i++) {
  5319. if (((ptr + 7) > pend) || (elem_len < subelem_len))
  5320. break;
  5321. if (*ptr != 0x34) {
  5322. RTW_ERR("WNM: invalid data(0x%2x)!\n", *ptr);
  5323. ret = _FAIL;
  5324. break;
  5325. }
  5326. pie = (struct nb_rpt_hdr *)ptr;
  5327. if (from_btm) {
  5328. op = rtw_get_ie((u8 *)(ptr+15),
  5329. WNM_BTM_CAND_PREF_SUBEID,
  5330. &op_len, (subelem_len - 15));
  5331. }
  5332. ptr = (u8 *)(ptr + subelem_len + 2);
  5333. elem_len -= (subelem_len +2);
  5334. subelem_len = *(ptr+1);
  5335. if (from_btm) {
  5336. pcandidate = (pcandidates + i);
  5337. _rtw_memcpy(&pcandidate->nb_rpt, pie, sizeof(struct nb_rpt_hdr));
  5338. if (op && (op_len !=0)) {
  5339. pcandidate->preference = *(op + 2);
  5340. bfound = _TRUE;
  5341. } else
  5342. pcandidate->preference = 0;
  5343. RTW_DBG("WNM: preference check bssid("MAC_FMT
  5344. ") ,bss_info(0x%04X), reg_class(0x%02X), ch(%d),"
  5345. " phy_type(0x%02X), preference(0x%02X)\n",
  5346. MAC_ARG(pcandidate->nb_rpt.bssid), pcandidate->nb_rpt.bss_info,
  5347. pcandidate->nb_rpt.reg_class, pcandidate->nb_rpt.ch_num,
  5348. pcandidate->nb_rpt.phy_type, pcandidate->preference);
  5349. } else {
  5350. if (_rtw_memcmp(&pnb->nb_rpt[i], pie, sizeof(struct nb_rpt_hdr)) == _FALSE)
  5351. *nb_rpt_is_same = _FALSE;
  5352. _rtw_memcpy(&pnb->nb_rpt[i], pie, sizeof(struct nb_rpt_hdr));
  5353. }
  5354. nb_rpt_entries++;
  5355. }
  5356. if (from_btm)
  5357. pnb->preference_en = (bfound)?_TRUE:_FALSE;
  5358. *nb_rpt_num = nb_rpt_entries;
  5359. return ret;
  5360. }
  5361. /* selection sorting based on preference value
  5362. * IN : nb_rpt_entries - candidate num
  5363. * IN/OUT : pcandidates - candidate list
  5364. * return : TRUE - means pcandidates is updated.
  5365. */
  5366. static u8 rtw_wnm_candidates_sorting(
  5367. u32 nb_rpt_entries, struct wnm_btm_cant *pcandidates)
  5368. {
  5369. u8 updated = _FALSE;
  5370. u32 i, j, pos;
  5371. struct wnm_btm_cant swap;
  5372. struct wnm_btm_cant *pcant_1, *pcant_2;
  5373. if ((!nb_rpt_entries) || (!pcandidates))
  5374. return updated;
  5375. for (i=0; i < (nb_rpt_entries - 1); i++) {
  5376. pos = i;
  5377. for (j=(i + 1); j < nb_rpt_entries; j++) {
  5378. pcant_1 = pcandidates+pos;
  5379. pcant_2 = pcandidates+j;
  5380. if ((pcant_1->preference) < (pcant_2->preference))
  5381. pos = j;
  5382. }
  5383. if (pos != i) {
  5384. updated = _TRUE;
  5385. _rtw_memcpy(&swap, (pcandidates+i), sizeof(struct wnm_btm_cant));
  5386. _rtw_memcpy((pcandidates+i), (pcandidates+pos), sizeof(struct wnm_btm_cant));
  5387. _rtw_memcpy((pcandidates+pos), &swap, sizeof(struct wnm_btm_cant));
  5388. }
  5389. }
  5390. return updated;
  5391. }
  5392. static void rtw_wnm_nb_info_update(
  5393. u32 nb_rpt_entries, u8 from_btm,
  5394. struct roam_nb_info *pnb, struct wnm_btm_cant *pcandidates,
  5395. u8 *nb_rpt_is_same)
  5396. {
  5397. u8 is_found;
  5398. u32 i, j;
  5399. struct wnm_btm_cant *pcand;
  5400. if (!pnb)
  5401. return;
  5402. pnb->nb_rpt_ch_list_num = 0;
  5403. for (i=0; i<nb_rpt_entries; i++) {
  5404. is_found = _FALSE;
  5405. if (from_btm) {
  5406. pcand = (pcandidates+i);
  5407. if (_rtw_memcmp(&pnb->nb_rpt[i], &pcand->nb_rpt,
  5408. sizeof(struct nb_rpt_hdr)) == _FALSE)
  5409. *nb_rpt_is_same = _FALSE;
  5410. _rtw_memcpy(&pnb->nb_rpt[i], &pcand->nb_rpt, sizeof(struct nb_rpt_hdr));
  5411. }
  5412. RTW_DBG("WNM: bssid(" MAC_FMT
  5413. ") , bss_info(0x%04X), reg_class(0x%02X), ch_num(%d), phy_type(0x%02X)\n",
  5414. MAC_ARG(pnb->nb_rpt[i].bssid), pnb->nb_rpt[i].bss_info,
  5415. pnb->nb_rpt[i].reg_class, pnb->nb_rpt[i].ch_num,
  5416. pnb->nb_rpt[i].phy_type);
  5417. if (pnb->nb_rpt[i].ch_num == 0)
  5418. continue;
  5419. for (j=0; j<nb_rpt_entries; j++) {
  5420. if (pnb->nb_rpt[i].ch_num == pnb->nb_rpt_ch_list[j].hw_value) {
  5421. is_found = _TRUE;
  5422. break;
  5423. }
  5424. }
  5425. if (!is_found) {
  5426. pnb->nb_rpt_ch_list[pnb->nb_rpt_ch_list_num].hw_value = pnb->nb_rpt[i].ch_num;
  5427. pnb->nb_rpt_ch_list_num++;
  5428. }
  5429. }
  5430. }
  5431. static void rtw_wnm_btm_candidate_select(_adapter *padapter)
  5432. {
  5433. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  5434. struct roam_nb_info *pnb = &(padapter->mlmepriv.nb_info);
  5435. struct wlan_network *pnetwork;
  5436. u8 bfound = _FALSE;
  5437. u32 i;
  5438. for (i = 0; i < pnb->last_nb_rpt_entries; i++) {
  5439. pnetwork = rtw_find_network(
  5440. &(pmlmepriv->scanned_queue),
  5441. pnb->nb_rpt[i].bssid);
  5442. if (pnetwork) {
  5443. bfound = _TRUE;
  5444. break;
  5445. }
  5446. }
  5447. if (bfound) {
  5448. _rtw_memcpy(pnb->roam_target_addr, pnb->nb_rpt[i].bssid, ETH_ALEN);
  5449. RTW_INFO("WNM : select btm entry(%d) - %s("MAC_FMT", ch%u) rssi:%d\n"
  5450. , i
  5451. , pnetwork->network.Ssid.Ssid
  5452. , MAC_ARG(pnetwork->network.MacAddress)
  5453. , pnetwork->network.Configuration.DSConfig
  5454. , (int)pnetwork->network.Rssi);
  5455. } else
  5456. _rtw_memset(pnb->roam_target_addr,0, ETH_ALEN);
  5457. }
  5458. u32 rtw_wnm_btm_candidates_survey(
  5459. _adapter *padapter, u8* pframe, u32 elem_len, u8 from_btm)
  5460. {
  5461. struct roam_nb_info *pnb = &(padapter->mlmepriv.nb_info);
  5462. struct wnm_btm_cant *pcandidate_list = NULL;
  5463. u8 nb_rpt_is_same = _TRUE;
  5464. u32 ret = _FAIL;
  5465. u32 nb_rpt_entries = 0;
  5466. if (from_btm) {
  5467. u32 mlen = sizeof(struct wnm_btm_cant) * RTW_MAX_NB_RPT_NUM;
  5468. pcandidate_list = (struct wnm_btm_cant *)rtw_malloc(mlen);
  5469. if (pcandidate_list == NULL)
  5470. goto exit;
  5471. }
  5472. /*clean the status set last time*/
  5473. _rtw_memset(&pnb->nb_rpt_ch_list, 0, sizeof(pnb->nb_rpt_ch_list));
  5474. pnb->nb_rpt_valid = _FALSE;
  5475. if (!rtw_wnm_nb_elem_parsing(
  5476. pframe, elem_len, from_btm,
  5477. &nb_rpt_entries, &nb_rpt_is_same,
  5478. pnb, pcandidate_list))
  5479. goto exit;
  5480. if (nb_rpt_entries != 0) {
  5481. if ((from_btm) && (rtw_wnm_btm_preference_cap(padapter)))
  5482. rtw_wnm_candidates_sorting(nb_rpt_entries, pcandidate_list);
  5483. rtw_wnm_nb_info_update(
  5484. nb_rpt_entries, from_btm,
  5485. pnb, pcandidate_list, &nb_rpt_is_same);
  5486. }
  5487. RTW_INFO("nb_rpt_is_same = %d, nb_rpt_entries = %d, last_nb_rpt_entries = %d\n",
  5488. nb_rpt_is_same, nb_rpt_entries, pnb->last_nb_rpt_entries);
  5489. if ((nb_rpt_is_same == _TRUE) && (nb_rpt_entries == pnb->last_nb_rpt_entries))
  5490. pnb->nb_rpt_is_same = _TRUE;
  5491. else {
  5492. pnb->nb_rpt_is_same = _FALSE;
  5493. pnb->last_nb_rpt_entries = nb_rpt_entries;
  5494. }
  5495. if ((from_btm) && (nb_rpt_entries != 0))
  5496. rtw_wnm_btm_candidate_select(padapter);
  5497. pnb->nb_rpt_valid = _TRUE;
  5498. ret = _SUCCESS;
  5499. exit:
  5500. if (from_btm && pcandidate_list)
  5501. rtw_mfree((u8 *)pcandidate_list, sizeof(struct wnm_btm_cant) * RTW_MAX_NB_RPT_NUM);
  5502. return ret;
  5503. }
  5504. #endif
  5505. unsigned int OnAction_ft(_adapter *padapter, union recv_frame *precv_frame)
  5506. {
  5507. #ifdef CONFIG_RTW_80211R
  5508. u32 ret = _FAIL;
  5509. u32 frame_len = 0;
  5510. u8 action_code = 0;
  5511. u8 category = 0;
  5512. u8 *pframe = NULL;
  5513. u8 *pframe_body = NULL;
  5514. u8 sta_addr[ETH_ALEN] = {0};
  5515. u8 *pie = NULL;
  5516. u32 ft_ie_len = 0;
  5517. u32 status_code = 0;
  5518. struct mlme_ext_priv *pmlmeext = NULL;
  5519. struct mlme_ext_info *pmlmeinfo = NULL;
  5520. struct mlme_priv *pmlmepriv = NULL;
  5521. struct wlan_network *proam_target = NULL;
  5522. struct ft_roam_info *pft_roam = NULL;
  5523. _irqL irqL;
  5524. pmlmeext = &(padapter->mlmeextpriv);
  5525. pmlmeinfo = &(pmlmeext->mlmext_info);
  5526. pmlmepriv = &(padapter->mlmepriv);
  5527. pft_roam = &(pmlmepriv->ft_roam);
  5528. pframe = precv_frame->u.hdr.rx_data;
  5529. frame_len = precv_frame->u.hdr.len;
  5530. pframe_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5531. category = pframe_body[0];
  5532. if (category != RTW_WLAN_CATEGORY_FT)
  5533. goto exit;
  5534. action_code = pframe_body[1];
  5535. switch (action_code) {
  5536. case RTW_WLAN_ACTION_FT_RSP:
  5537. RTW_INFO("FT: RTW_WLAN_ACTION_FT_RSP recv.\n");
  5538. if (!_rtw_memcmp(adapter_mac_addr(padapter), &pframe_body[2], ETH_ALEN)) {
  5539. RTW_ERR("FT: Unmatched STA MAC Address "MAC_FMT"\n", MAC_ARG(&pframe_body[2]));
  5540. goto exit;
  5541. }
  5542. status_code = le16_to_cpu(*(u16 *)((SIZE_PTR)pframe + sizeof(struct rtw_ieee80211_hdr_3addr) + 14));
  5543. if (status_code != 0) {
  5544. RTW_ERR("FT: WLAN ACTION FT RESPONSE fail, status: %d\n", status_code);
  5545. goto exit;
  5546. }
  5547. if (is_zero_mac_addr(&pframe_body[8]) || is_broadcast_mac_addr(&pframe_body[8])) {
  5548. RTW_ERR("FT: Invalid Target MAC Address "MAC_FMT"\n", MAC_ARG(padapter->mlmepriv.roam_tgt_addr));
  5549. goto exit;
  5550. }
  5551. pie = rtw_get_ie(pframe_body, _MDIE_, &ft_ie_len, frame_len);
  5552. if (pie) {
  5553. if (!_rtw_memcmp(&pft_roam->mdid, pie+2, 2)) {
  5554. RTW_ERR("FT: Invalid MDID\n");
  5555. goto exit;
  5556. }
  5557. }
  5558. rtw_ft_set_status(padapter, RTW_FT_REQUESTED_STA);
  5559. _cancel_timer_ex(&pmlmeext->ft_link_timer);
  5560. /*Disconnect current AP*/
  5561. receive_disconnect(padapter, pmlmepriv->cur_network.network.MacAddress, WLAN_REASON_ACTIVE_ROAM, _FALSE);
  5562. pft_roam->ft_action_len = frame_len;
  5563. _rtw_memcpy(pft_roam->ft_action, pframe, rtw_min(frame_len, RTW_FT_MAX_IE_SZ));
  5564. ret = _SUCCESS;
  5565. break;
  5566. case RTW_WLAN_ACTION_FT_REQ:
  5567. case RTW_WLAN_ACTION_FT_CONF:
  5568. case RTW_WLAN_ACTION_FT_ACK:
  5569. default:
  5570. RTW_ERR("FT: Unsupported FT Action!\n");
  5571. break;
  5572. }
  5573. exit:
  5574. return ret;
  5575. #else
  5576. return _SUCCESS;
  5577. #endif
  5578. }
  5579. #ifdef CONFIG_RTW_WNM
  5580. u8 rtw_wmn_btm_rsp_reason_decision(_adapter *padapter, u8* req_mode)
  5581. {
  5582. struct recv_priv *precvpriv = &padapter->recvpriv;
  5583. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  5584. u8 reason = 0;
  5585. if (!rtw_wnm_btm_diff_bss(padapter)) {
  5586. /* Reject - No suitable BSS transition candidates */
  5587. reason = 7;
  5588. goto candidate_remove;
  5589. }
  5590. #ifdef CONFIG_RTW_80211R
  5591. if (rtw_ft_chk_flags(padapter, RTW_FT_BTM_ROAM)) {
  5592. /* Accept */
  5593. reason = 0;
  5594. goto under_survey;
  5595. }
  5596. #endif
  5597. if (((*req_mode) & DISASSOC_IMMINENT) == 0) {
  5598. /* Reject - Unspecified reject reason */
  5599. reason = 1;
  5600. goto candidate_remove;
  5601. }
  5602. if (precvpriv->signal_strength_data.avg_val >= pmlmepriv->roam_rssi_threshold) {
  5603. reason = 1;
  5604. goto candidate_remove;
  5605. }
  5606. under_survey:
  5607. if (check_fwstate(pmlmepriv, _FW_UNDER_SURVEY)) {
  5608. RTW_INFO("%s reject due to _FW_UNDER_SURVEY\n", __func__);
  5609. reason = 1;
  5610. }
  5611. candidate_remove:
  5612. if (reason !=0)
  5613. rtw_wnm_reset_btm_candidate(&padapter->mlmepriv.nb_info);
  5614. return reason;
  5615. }
  5616. static u32 rtw_wnm_btm_candidates_offset_get(u8* pframe)
  5617. {
  5618. u8 *pos = pframe;
  5619. u32 offset = 0;
  5620. if (!pframe)
  5621. return 0;
  5622. offset += 7;
  5623. pos += offset;
  5624. /* BSS Termination Duration check */
  5625. if (wnm_btm_bss_term_inc(pframe)) {
  5626. offset += 12;
  5627. pos += offset;
  5628. }
  5629. /* Session Information URL check*/
  5630. if (wnm_btm_ess_disassoc_im(pframe)) {
  5631. /*URL length field + URL variable length*/
  5632. offset = 1 + *(pframe + offset);
  5633. pos += offset;
  5634. }
  5635. offset = (pos - pframe);
  5636. return offset;
  5637. }
  5638. static void rtw_wnm_btm_req_hdr_parsing(u8* pframe, struct btm_req_hdr *phdr)
  5639. {
  5640. u8 *pos = pframe;
  5641. u32 offset = 0;
  5642. if (!pframe || !phdr)
  5643. return;
  5644. _rtw_memset(phdr, 0, sizeof(struct btm_req_hdr));
  5645. phdr->req_mode = wnm_btm_req_mode(pframe);
  5646. phdr->disassoc_timer = wnm_btm_disassoc_timer(pframe);
  5647. phdr->validity_interval = wnm_btm_valid_interval(pframe);
  5648. if (wnm_btm_bss_term_inc(pframe)) {
  5649. _rtw_memcpy(&phdr->term_duration,
  5650. wnm_btm_term_duration_offset(pframe),
  5651. sizeof(struct btm_term_duration));
  5652. }
  5653. RTW_DBG("WNM: req_mode(%1x), disassoc_timer(%02x), interval(%x)\n",
  5654. phdr->req_mode, phdr->disassoc_timer, phdr->validity_interval);
  5655. if (wnm_btm_bss_term_inc(pframe))
  5656. RTW_INFO("WNM: tsf(%llx), duration(%2x)\n",
  5657. phdr->term_duration.tsf, phdr->term_duration.duration);
  5658. }
  5659. void rtw_wnm_roam_scan_hdl(void *ctx)
  5660. {
  5661. _adapter *padapter = (_adapter *)ctx;
  5662. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  5663. if (rtw_is_scan_deny(padapter))
  5664. RTW_INFO("WNM: roam scan would abort by scan_deny!\n");
  5665. pmlmepriv->need_to_roam = _TRUE;
  5666. rtw_drv_scan_by_self(padapter, RTW_AUTO_SCAN_REASON_ROAM);
  5667. }
  5668. static void rtw_wnm_roam_scan(_adapter *padapter)
  5669. {
  5670. struct roam_nb_info *pnb = &(padapter->mlmepriv.nb_info);
  5671. if (rtw_is_scan_deny(padapter)) {
  5672. _cancel_timer_ex(&pnb->roam_scan_timer);
  5673. _set_timer(&pnb->roam_scan_timer, 1000);
  5674. } else
  5675. rtw_wnm_roam_scan_hdl((void *)padapter);
  5676. }
  5677. void rtw_wnm_process_btm_req(_adapter *padapter, u8* pframe, u32 frame_len)
  5678. {
  5679. struct roam_nb_info *pnb = &(padapter->mlmepriv.nb_info);
  5680. struct btm_req_hdr req_hdr;
  5681. u8 *ptr, reason;
  5682. u32 elem_len, offset;
  5683. rtw_wnm_btm_req_hdr_parsing(pframe, &req_hdr);
  5684. offset = rtw_wnm_btm_candidates_offset_get(pframe);
  5685. if ((offset == 0) || ((frame_len - offset) <= 15))
  5686. return;
  5687. ptr = (pframe + offset);
  5688. elem_len = (frame_len - offset);
  5689. rtw_wnm_btm_candidates_survey(padapter, ptr, elem_len, _TRUE);
  5690. reason = rtw_wmn_btm_rsp_reason_decision(padapter, &pframe[3]);
  5691. rtw_wnm_issue_action(padapter,
  5692. RTW_WLAN_ACTION_WNM_BTM_RSP, reason);
  5693. if (reason == 0)
  5694. rtw_wnm_roam_scan(padapter);
  5695. }
  5696. void rtw_wnm_reset_btm_candidate(struct roam_nb_info *pnb)
  5697. {
  5698. pnb->preference_en = _FALSE;
  5699. _rtw_memset(pnb->roam_target_addr, 0, ETH_ALEN);
  5700. }
  5701. void rtw_wnm_reset_btm_state(_adapter *padapter)
  5702. {
  5703. struct roam_nb_info *pnb = &(padapter->mlmepriv.nb_info);
  5704. pnb->last_nb_rpt_entries = 0;
  5705. pnb->nb_rpt_is_same = _TRUE;
  5706. pnb->nb_rpt_valid = _FALSE;
  5707. pnb->nb_rpt_ch_list_num = 0;
  5708. rtw_wnm_reset_btm_candidate(pnb);
  5709. _rtw_memset(&pnb->nb_rpt, 0, sizeof(pnb->nb_rpt));
  5710. _rtw_memset(&pnb->nb_rpt_ch_list, 0, sizeof(pnb->nb_rpt_ch_list));
  5711. }
  5712. void rtw_wnm_issue_action(_adapter *padapter, u8 action, u8 reason)
  5713. {
  5714. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  5715. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  5716. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  5717. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  5718. struct xmit_frame *pmgntframe;
  5719. struct rtw_ieee80211_hdr *pwlanhdr;
  5720. struct pkt_attrib *pattrib;
  5721. u8 category, dialog_token, termination_delay, *pframe;
  5722. u16 *fctrl;
  5723. if ((pmgntframe = alloc_mgtxmitframe(pxmitpriv)) == NULL)
  5724. return ;
  5725. pattrib = &(pmgntframe->attrib);
  5726. update_mgntframe_attrib(padapter, pattrib);
  5727. _rtw_memset(pmgntframe->buf_addr, 0, (WLANHDR_OFFSET + TXDESC_OFFSET));
  5728. pframe = (u8 *)(pmgntframe->buf_addr + TXDESC_OFFSET);
  5729. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  5730. fctrl = &(pwlanhdr->frame_ctl);
  5731. *(fctrl) = 0;
  5732. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  5733. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  5734. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  5735. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  5736. pmlmeext->mgnt_seq++;
  5737. set_frame_sub_type(pframe, WIFI_ACTION);
  5738. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  5739. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  5740. category = RTW_WLAN_CATEGORY_WNM;
  5741. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  5742. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  5743. switch (action) {
  5744. case RTW_WLAN_ACTION_WNM_BTM_QUERY:
  5745. pframe = rtw_set_fixed_ie(pframe, 1, &(dialog_token), &(pattrib->pktlen));
  5746. pframe = rtw_set_fixed_ie(pframe, 1, &(reason), &(pattrib->pktlen));
  5747. RTW_INFO("WNM: RTW_WLAN_ACTION_WNM_BTM_QUERY sent.\n");
  5748. break;
  5749. case RTW_WLAN_ACTION_WNM_BTM_RSP:
  5750. termination_delay = 0;
  5751. pframe = rtw_set_fixed_ie(pframe, 1, &(dialog_token), &(pattrib->pktlen));
  5752. pframe = rtw_set_fixed_ie(pframe, 1, &(reason), &(pattrib->pktlen));
  5753. pframe = rtw_set_fixed_ie(pframe, 1, &(termination_delay), &(pattrib->pktlen));
  5754. if (!is_zero_mac_addr(pmlmepriv->nb_info.roam_target_addr)) {
  5755. pframe = rtw_set_fixed_ie(pframe, 6,
  5756. pmlmepriv->nb_info.roam_target_addr, &(pattrib->pktlen));
  5757. }
  5758. RTW_INFO("WNM: RTW_WLAN_ACTION_WNM_BTM_RSP sent. reason = %d\n", reason);
  5759. break;
  5760. default:
  5761. goto exit;
  5762. }
  5763. pattrib->last_txcmdsz = pattrib->pktlen;
  5764. dump_mgntframe(padapter, pmgntframe);
  5765. exit:
  5766. return;
  5767. }
  5768. #endif
  5769. unsigned int OnAction_ht(_adapter *padapter, union recv_frame *precv_frame)
  5770. {
  5771. u8 *pframe = precv_frame->u.hdr.rx_data;
  5772. u8 *frame_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5773. u8 category, action;
  5774. /* check RA matches or not */
  5775. if (!_rtw_memcmp(adapter_mac_addr(padapter), GetAddr1Ptr(pframe), ETH_ALEN))
  5776. goto exit;
  5777. category = frame_body[0];
  5778. if (category != RTW_WLAN_CATEGORY_HT)
  5779. goto exit;
  5780. action = frame_body[1];
  5781. switch (action) {
  5782. case RTW_WLAN_ACTION_HT_SM_PS:
  5783. #ifdef CONFIG_80211N_HT
  5784. #ifdef CONFIG_AP_MODE
  5785. if (check_fwstate(&padapter->mlmepriv, WIFI_AP_STATE) == _TRUE)
  5786. rtw_process_ht_action_smps(padapter, get_addr2_ptr(pframe), frame_body[2]);
  5787. #endif /*CONFIG_AP_MODE*/
  5788. #endif /*CONFIG_80211N_HT*/
  5789. break;
  5790. case RTW_WLAN_ACTION_HT_COMPRESS_BEAMFORMING:
  5791. #ifdef CONFIG_BEAMFORMING
  5792. /*RTW_INFO("RTW_WLAN_ACTION_HT_COMPRESS_BEAMFORMING\n");*/
  5793. rtw_beamforming_get_report_frame(padapter, precv_frame);
  5794. #endif /*CONFIG_BEAMFORMING*/
  5795. break;
  5796. default:
  5797. break;
  5798. }
  5799. exit:
  5800. return _SUCCESS;
  5801. }
  5802. #ifdef CONFIG_IEEE80211W
  5803. unsigned int OnAction_sa_query(_adapter *padapter, union recv_frame *precv_frame)
  5804. {
  5805. u8 *pframe = precv_frame->u.hdr.rx_data;
  5806. struct rx_pkt_attrib *pattrib = &precv_frame->u.hdr.attrib;
  5807. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  5808. struct sta_info *psta;
  5809. struct sta_priv *pstapriv = &padapter->stapriv;
  5810. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  5811. u16 tid;
  5812. /* Baron */
  5813. RTW_INFO("OnAction_sa_query\n");
  5814. switch (pframe[WLAN_HDR_A3_LEN + 1]) {
  5815. case 0: /* SA Query req */
  5816. _rtw_memcpy(&tid, &pframe[WLAN_HDR_A3_LEN + 2], sizeof(u16));
  5817. RTW_INFO("OnAction_sa_query request,action=%d, tid=%04x, pframe=%02x-%02x\n"
  5818. , pframe[WLAN_HDR_A3_LEN + 1], tid, pframe[WLAN_HDR_A3_LEN + 2], pframe[WLAN_HDR_A3_LEN + 3]);
  5819. issue_action_SA_Query(padapter, get_addr2_ptr(pframe), 1, tid, IEEE80211W_RIGHT_KEY);
  5820. break;
  5821. case 1: /* SA Query rsp */
  5822. psta = rtw_get_stainfo(pstapriv, get_addr2_ptr(pframe));
  5823. if (psta != NULL)
  5824. _cancel_timer_ex(&psta->dot11w_expire_timer);
  5825. _rtw_memcpy(&tid, &pframe[WLAN_HDR_A3_LEN + 2], sizeof(u16));
  5826. RTW_INFO("OnAction_sa_query response,action=%d, tid=%04x, cancel timer\n", pframe[WLAN_HDR_A3_LEN + 1], tid);
  5827. break;
  5828. default:
  5829. break;
  5830. }
  5831. if (0) {
  5832. int pp;
  5833. printk("pattrib->pktlen = %d =>", pattrib->pkt_len);
  5834. for (pp = 0; pp < pattrib->pkt_len; pp++)
  5835. printk(" %02x ", pframe[pp]);
  5836. printk("\n");
  5837. }
  5838. return _SUCCESS;
  5839. }
  5840. #endif /* CONFIG_IEEE80211W */
  5841. unsigned int on_action_rm(_adapter *padapter, union recv_frame *precv_frame)
  5842. {
  5843. #ifdef CONFIG_RTW_80211K
  5844. return rm_on_action(padapter, precv_frame);
  5845. #else
  5846. return _SUCCESS;
  5847. #endif /* CONFIG_RTW_80211K */
  5848. }
  5849. unsigned int OnAction_wmm(_adapter *padapter, union recv_frame *precv_frame)
  5850. {
  5851. return _SUCCESS;
  5852. }
  5853. unsigned int OnAction_vht(_adapter *padapter, union recv_frame *precv_frame)
  5854. {
  5855. #ifdef CONFIG_80211AC_VHT
  5856. u8 *pframe = precv_frame->u.hdr.rx_data;
  5857. struct rtw_ieee80211_hdr_3addr *whdr = (struct rtw_ieee80211_hdr_3addr *)pframe;
  5858. u8 *frame_body = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
  5859. u8 category, action;
  5860. struct sta_info *psta = NULL;
  5861. /* check RA matches or not */
  5862. if (!_rtw_memcmp(adapter_mac_addr(padapter), GetAddr1Ptr(pframe), ETH_ALEN))
  5863. goto exit;
  5864. category = frame_body[0];
  5865. if (category != RTW_WLAN_CATEGORY_VHT)
  5866. goto exit;
  5867. action = frame_body[1];
  5868. switch (action) {
  5869. case RTW_WLAN_ACTION_VHT_COMPRESSED_BEAMFORMING:
  5870. #ifdef CONFIG_BEAMFORMING
  5871. /*RTW_INFO("RTW_WLAN_ACTION_VHT_COMPRESSED_BEAMFORMING\n");*/
  5872. rtw_beamforming_get_report_frame(padapter, precv_frame);
  5873. #endif /*CONFIG_BEAMFORMING*/
  5874. break;
  5875. case RTW_WLAN_ACTION_VHT_OPMODE_NOTIFICATION:
  5876. /* CategoryCode(1) + ActionCode(1) + OpModeNotification(1) */
  5877. /* RTW_INFO("RTW_WLAN_ACTION_VHT_OPMODE_NOTIFICATION\n"); */
  5878. psta = rtw_get_stainfo(&padapter->stapriv, whdr->addr2);
  5879. if (psta)
  5880. rtw_process_vht_op_mode_notify(padapter, &frame_body[2], psta);
  5881. break;
  5882. case RTW_WLAN_ACTION_VHT_GROUPID_MANAGEMENT:
  5883. #ifdef CONFIG_BEAMFORMING
  5884. #ifdef RTW_BEAMFORMING_VERSION_2
  5885. rtw_beamforming_get_vht_gid_mgnt_frame(padapter, precv_frame);
  5886. #endif /* RTW_BEAMFORMING_VERSION_2 */
  5887. #endif /* CONFIG_BEAMFORMING */
  5888. break;
  5889. default:
  5890. break;
  5891. }
  5892. exit:
  5893. #endif /* CONFIG_80211AC_VHT */
  5894. return _SUCCESS;
  5895. }
  5896. unsigned int OnAction_p2p(_adapter *padapter, union recv_frame *precv_frame)
  5897. {
  5898. #ifdef CONFIG_P2P
  5899. u8 *frame_body;
  5900. u8 category, OUI_Subtype, dialogToken = 0;
  5901. u8 *pframe = precv_frame->u.hdr.rx_data;
  5902. uint len = precv_frame->u.hdr.len;
  5903. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  5904. /* check RA matches or not */
  5905. if (!_rtw_memcmp(adapter_mac_addr(padapter), GetAddr1Ptr(pframe), ETH_ALEN))
  5906. return _SUCCESS;
  5907. frame_body = (unsigned char *)(pframe + sizeof(struct rtw_ieee80211_hdr_3addr));
  5908. category = frame_body[0];
  5909. if (category != RTW_WLAN_CATEGORY_P2P)
  5910. return _SUCCESS;
  5911. if (cpu_to_be32(*((u32 *)(frame_body + 1))) != P2POUI)
  5912. return _SUCCESS;
  5913. #ifdef CONFIG_IOCTL_CFG80211
  5914. if (adapter_wdev_data(padapter)->p2p_enabled
  5915. && pwdinfo->driver_interface == DRIVER_CFG80211
  5916. ) {
  5917. rtw_cfg80211_rx_action_p2p(padapter, precv_frame);
  5918. return _SUCCESS;
  5919. } else
  5920. #endif /* CONFIG_IOCTL_CFG80211 */
  5921. {
  5922. len -= sizeof(struct rtw_ieee80211_hdr_3addr);
  5923. OUI_Subtype = frame_body[5];
  5924. dialogToken = frame_body[6];
  5925. switch (OUI_Subtype) {
  5926. case P2P_NOTICE_OF_ABSENCE:
  5927. break;
  5928. case P2P_PRESENCE_REQUEST:
  5929. process_p2p_presence_req(pwdinfo, pframe, len);
  5930. break;
  5931. case P2P_PRESENCE_RESPONSE:
  5932. break;
  5933. case P2P_GO_DISC_REQUEST:
  5934. break;
  5935. default:
  5936. break;
  5937. }
  5938. }
  5939. #endif /* CONFIG_P2P */
  5940. return _SUCCESS;
  5941. }
  5942. unsigned int OnAction(_adapter *padapter, union recv_frame *precv_frame)
  5943. {
  5944. int i;
  5945. unsigned char category;
  5946. struct action_handler *ptable;
  5947. unsigned char *frame_body;
  5948. u8 *pframe = precv_frame->u.hdr.rx_data;
  5949. frame_body = (unsigned char *)(pframe + sizeof(struct rtw_ieee80211_hdr_3addr));
  5950. category = frame_body[0];
  5951. for (i = 0; i < sizeof(OnAction_tbl) / sizeof(struct action_handler); i++) {
  5952. ptable = &OnAction_tbl[i];
  5953. if (category == ptable->num)
  5954. ptable->func(padapter, precv_frame);
  5955. }
  5956. return _SUCCESS;
  5957. }
  5958. unsigned int DoReserved(_adapter *padapter, union recv_frame *precv_frame)
  5959. {
  5960. /* RTW_INFO("rcvd mgt frame(%x, %x)\n", (get_frame_sub_type(pframe) >> 4), *(unsigned int *)GetAddr1Ptr(pframe)); */
  5961. return _SUCCESS;
  5962. }
  5963. struct xmit_frame *_alloc_mgtxmitframe(struct xmit_priv *pxmitpriv, bool once)
  5964. {
  5965. struct xmit_frame *pmgntframe;
  5966. struct xmit_buf *pxmitbuf;
  5967. if (once)
  5968. pmgntframe = rtw_alloc_xmitframe_once(pxmitpriv);
  5969. else
  5970. pmgntframe = rtw_alloc_xmitframe_ext(pxmitpriv);
  5971. if (pmgntframe == NULL) {
  5972. RTW_INFO(FUNC_ADPT_FMT" alloc xmitframe fail, once:%d\n", FUNC_ADPT_ARG(pxmitpriv->adapter), once);
  5973. goto exit;
  5974. }
  5975. pxmitbuf = rtw_alloc_xmitbuf_ext(pxmitpriv);
  5976. if (pxmitbuf == NULL) {
  5977. RTW_INFO(FUNC_ADPT_FMT" alloc xmitbuf fail\n", FUNC_ADPT_ARG(pxmitpriv->adapter));
  5978. rtw_free_xmitframe(pxmitpriv, pmgntframe);
  5979. pmgntframe = NULL;
  5980. goto exit;
  5981. }
  5982. pmgntframe->frame_tag = MGNT_FRAMETAG;
  5983. pmgntframe->pxmitbuf = pxmitbuf;
  5984. pmgntframe->buf_addr = pxmitbuf->pbuf;
  5985. pxmitbuf->priv_data = pmgntframe;
  5986. exit:
  5987. return pmgntframe;
  5988. }
  5989. inline struct xmit_frame *alloc_mgtxmitframe(struct xmit_priv *pxmitpriv)
  5990. {
  5991. return _alloc_mgtxmitframe(pxmitpriv, _FALSE);
  5992. }
  5993. inline struct xmit_frame *alloc_mgtxmitframe_once(struct xmit_priv *pxmitpriv)
  5994. {
  5995. return _alloc_mgtxmitframe(pxmitpriv, _TRUE);
  5996. }
  5997. /****************************************************************************
  5998. Following are some TX fuctions for WiFi MLME
  5999. *****************************************************************************/
  6000. void update_mgnt_tx_rate(_adapter *padapter, u8 rate)
  6001. {
  6002. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6003. pmlmeext->tx_rate = rate;
  6004. /* RTW_INFO("%s(): rate = %x\n",__FUNCTION__, rate); */
  6005. }
  6006. void update_monitor_frame_attrib(_adapter *padapter, struct pkt_attrib *pattrib)
  6007. {
  6008. HAL_DATA_TYPE *pHalData = GET_HAL_DATA(padapter);
  6009. u8 wireless_mode;
  6010. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6011. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  6012. struct sta_info *psta = NULL;
  6013. struct sta_priv *pstapriv = &padapter->stapriv;
  6014. psta = rtw_get_stainfo(pstapriv, pattrib->ra);
  6015. pattrib->hdrlen = 24;
  6016. pattrib->nr_frags = 1;
  6017. pattrib->priority = 7;
  6018. pattrib->mac_id = RTW_DEFAULT_MGMT_MACID;
  6019. pattrib->qsel = QSLT_MGNT;
  6020. pattrib->pktlen = 0;
  6021. if (pmlmeext->tx_rate == IEEE80211_CCK_RATE_1MB)
  6022. wireless_mode = WIRELESS_11B;
  6023. else
  6024. wireless_mode = WIRELESS_11G;
  6025. pattrib->raid = rtw_get_mgntframe_raid(padapter, wireless_mode);
  6026. #ifdef CONFIG_80211AC_VHT
  6027. if (pHalData->rf_type == RF_1T1R)
  6028. pattrib->raid = RATEID_IDX_VHT_1SS;
  6029. else if (pHalData->rf_type == RF_2T2R || pHalData->rf_type == RF_2T4R)
  6030. pattrib->raid = RATEID_IDX_VHT_2SS;
  6031. else if (pHalData->rf_type == RF_3T3R)
  6032. pattrib->raid = RATEID_IDX_VHT_3SS;
  6033. else
  6034. pattrib->raid = RATEID_IDX_BGN_40M_1SS;
  6035. #endif
  6036. #ifdef CONFIG_80211AC_VHT
  6037. pattrib->rate = MGN_VHT1SS_MCS9;
  6038. #else
  6039. pattrib->rate = MGN_MCS7;
  6040. #endif
  6041. pattrib->encrypt = _NO_PRIVACY_;
  6042. pattrib->bswenc = _FALSE;
  6043. pattrib->qos_en = _FALSE;
  6044. pattrib->ht_en = 1;
  6045. pattrib->bwmode = CHANNEL_WIDTH_20;
  6046. pattrib->ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  6047. pattrib->sgi = _FALSE;
  6048. pattrib->seqnum = pmlmeext->mgnt_seq;
  6049. pattrib->retry_ctrl = _TRUE;
  6050. pattrib->mbssid = 0;
  6051. pattrib->hw_ssn_sel = pxmitpriv->hw_ssn_seq_no;
  6052. }
  6053. void update_mgntframe_attrib(_adapter *padapter, struct pkt_attrib *pattrib)
  6054. {
  6055. u8 wireless_mode;
  6056. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6057. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  6058. /* _rtw_memset((u8 *)(pattrib), 0, sizeof(struct pkt_attrib)); */
  6059. pattrib->hdrlen = 24;
  6060. pattrib->nr_frags = 1;
  6061. pattrib->priority = 7;
  6062. pattrib->mac_id = RTW_DEFAULT_MGMT_MACID;
  6063. pattrib->qsel = QSLT_MGNT;
  6064. #ifdef CONFIG_MCC_MODE
  6065. update_mcc_mgntframe_attrib(padapter, pattrib);
  6066. #endif
  6067. pattrib->pktlen = 0;
  6068. if (IS_CCK_RATE(pmlmeext->tx_rate))
  6069. wireless_mode = WIRELESS_11B;
  6070. else
  6071. wireless_mode = WIRELESS_11G;
  6072. pattrib->raid = rtw_get_mgntframe_raid(padapter, wireless_mode);
  6073. pattrib->rate = pmlmeext->tx_rate;
  6074. pattrib->encrypt = _NO_PRIVACY_;
  6075. pattrib->bswenc = _FALSE;
  6076. pattrib->qos_en = _FALSE;
  6077. pattrib->ht_en = _FALSE;
  6078. pattrib->bwmode = CHANNEL_WIDTH_20;
  6079. pattrib->ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  6080. pattrib->sgi = _FALSE;
  6081. pattrib->seqnum = pmlmeext->mgnt_seq;
  6082. pattrib->retry_ctrl = _TRUE;
  6083. pattrib->mbssid = 0;
  6084. pattrib->hw_ssn_sel = pxmitpriv->hw_ssn_seq_no;
  6085. }
  6086. void update_mgntframe_attrib_addr(_adapter *padapter, struct xmit_frame *pmgntframe)
  6087. {
  6088. u8 *pframe;
  6089. struct pkt_attrib *pattrib = &pmgntframe->attrib;
  6090. #if defined(CONFIG_BEAMFORMING) || defined(CONFIG_ANTENNA_DIVERSITY)
  6091. struct sta_info *sta = NULL;
  6092. #endif
  6093. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6094. _rtw_memcpy(pattrib->ra, GetAddr1Ptr(pframe), ETH_ALEN);
  6095. _rtw_memcpy(pattrib->ta, get_addr2_ptr(pframe), ETH_ALEN);
  6096. #if defined(CONFIG_BEAMFORMING) || defined(CONFIG_ANTENNA_DIVERSITY)
  6097. sta = pattrib->psta;
  6098. if (!sta) {
  6099. sta = rtw_get_stainfo(&padapter->stapriv, pattrib->ra);
  6100. pattrib->psta = sta;
  6101. }
  6102. #ifdef CONFIG_BEAMFORMING
  6103. if (sta)
  6104. update_attrib_txbf_info(padapter, pattrib, sta);
  6105. #endif
  6106. #endif /* defined(CONFIG_BEAMFORMING) || defined(CONFIG_ANTENNA_DIVERSITY) */
  6107. }
  6108. void dump_mgntframe(_adapter *padapter, struct xmit_frame *pmgntframe)
  6109. {
  6110. if (RTW_CANNOT_RUN(padapter)) {
  6111. rtw_free_xmitbuf(&padapter->xmitpriv, pmgntframe->pxmitbuf);
  6112. rtw_free_xmitframe(&padapter->xmitpriv, pmgntframe);
  6113. return;
  6114. }
  6115. rtw_hal_mgnt_xmit(padapter, pmgntframe);
  6116. }
  6117. s32 dump_mgntframe_and_wait(_adapter *padapter, struct xmit_frame *pmgntframe, int timeout_ms)
  6118. {
  6119. s32 ret = _FAIL;
  6120. _irqL irqL;
  6121. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  6122. struct xmit_buf *pxmitbuf = pmgntframe->pxmitbuf;
  6123. struct submit_ctx sctx;
  6124. if (RTW_CANNOT_RUN(padapter)) {
  6125. rtw_free_xmitbuf(&padapter->xmitpriv, pmgntframe->pxmitbuf);
  6126. rtw_free_xmitframe(&padapter->xmitpriv, pmgntframe);
  6127. return ret;
  6128. }
  6129. rtw_sctx_init(&sctx, timeout_ms);
  6130. pxmitbuf->sctx = &sctx;
  6131. ret = rtw_hal_mgnt_xmit(padapter, pmgntframe);
  6132. if (ret == _SUCCESS)
  6133. ret = rtw_sctx_wait(&sctx, __func__);
  6134. _enter_critical(&pxmitpriv->lock_sctx, &irqL);
  6135. pxmitbuf->sctx = NULL;
  6136. _exit_critical(&pxmitpriv->lock_sctx, &irqL);
  6137. return ret;
  6138. }
  6139. s32 dump_mgntframe_and_wait_ack_timeout(_adapter *padapter, struct xmit_frame *pmgntframe, int timeout_ms)
  6140. {
  6141. #ifdef CONFIG_XMIT_ACK
  6142. static u8 seq_no = 0;
  6143. s32 ret = _FAIL;
  6144. struct xmit_priv *pxmitpriv = &(GET_PRIMARY_ADAPTER(padapter))->xmitpriv;
  6145. if (RTW_CANNOT_RUN(padapter)) {
  6146. rtw_free_xmitbuf(&padapter->xmitpriv, pmgntframe->pxmitbuf);
  6147. rtw_free_xmitframe(&padapter->xmitpriv, pmgntframe);
  6148. return -1;
  6149. }
  6150. _enter_critical_mutex(&pxmitpriv->ack_tx_mutex, NULL);
  6151. pxmitpriv->ack_tx = _TRUE;
  6152. pxmitpriv->seq_no = seq_no++;
  6153. pmgntframe->ack_report = 1;
  6154. rtw_sctx_init(&(pxmitpriv->ack_tx_ops), timeout_ms);
  6155. if (rtw_hal_mgnt_xmit(padapter, pmgntframe) == _SUCCESS)
  6156. ret = rtw_sctx_wait(&(pxmitpriv->ack_tx_ops), __func__);
  6157. pxmitpriv->ack_tx = _FALSE;
  6158. _exit_critical_mutex(&pxmitpriv->ack_tx_mutex, NULL);
  6159. return ret;
  6160. #else /* !CONFIG_XMIT_ACK */
  6161. dump_mgntframe(padapter, pmgntframe);
  6162. rtw_msleep_os(50);
  6163. return _SUCCESS;
  6164. #endif /* !CONFIG_XMIT_ACK */
  6165. }
  6166. s32 dump_mgntframe_and_wait_ack(_adapter *padapter, struct xmit_frame *pmgntframe)
  6167. {
  6168. /* In this case, use 500 ms as the default wait_ack timeout */
  6169. return dump_mgntframe_and_wait_ack_timeout(padapter, pmgntframe, 500);
  6170. }
  6171. int update_hidden_ssid(u8 *ies, u32 ies_len, u8 hidden_ssid_mode)
  6172. {
  6173. u8 *ssid_ie;
  6174. sint ssid_len_ori;
  6175. int len_diff = 0;
  6176. ssid_ie = rtw_get_ie(ies, WLAN_EID_SSID, &ssid_len_ori, ies_len);
  6177. /* RTW_INFO("%s hidden_ssid_mode:%u, ssid_ie:%p, ssid_len_ori:%d\n", __FUNCTION__, hidden_ssid_mode, ssid_ie, ssid_len_ori); */
  6178. if (ssid_ie && ssid_len_ori > 0) {
  6179. switch (hidden_ssid_mode) {
  6180. case 1: {
  6181. u8 *next_ie = ssid_ie + 2 + ssid_len_ori;
  6182. u32 remain_len = 0;
  6183. remain_len = ies_len - (next_ie - ies);
  6184. ssid_ie[1] = 0;
  6185. _rtw_memcpy(ssid_ie + 2, next_ie, remain_len);
  6186. len_diff -= ssid_len_ori;
  6187. break;
  6188. }
  6189. case 2:
  6190. _rtw_memset(&ssid_ie[2], 0, ssid_len_ori);
  6191. break;
  6192. default:
  6193. break;
  6194. }
  6195. }
  6196. return len_diff;
  6197. }
  6198. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  6199. u32 rtw_build_vendor_ie(_adapter *padapter , unsigned char *pframe , u8 mgmt_frame_tyte)
  6200. {
  6201. int vendor_ie_num = 0;
  6202. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  6203. u32 len = 0;
  6204. for (vendor_ie_num = 0 ; vendor_ie_num < WLAN_MAX_VENDOR_IE_NUM ; vendor_ie_num++) {
  6205. if (pmlmepriv->vendor_ielen[vendor_ie_num] > 0 && pmlmepriv->vendor_ie_mask[vendor_ie_num] & mgmt_frame_tyte) {
  6206. _rtw_memcpy(pframe , pmlmepriv->vendor_ie[vendor_ie_num] , pmlmepriv->vendor_ielen[vendor_ie_num]);
  6207. pframe += pmlmepriv->vendor_ielen[vendor_ie_num];
  6208. len += pmlmepriv->vendor_ielen[vendor_ie_num];
  6209. }
  6210. }
  6211. return len;
  6212. }
  6213. #endif
  6214. void issue_beacon(_adapter *padapter, int timeout_ms)
  6215. {
  6216. struct xmit_frame *pmgntframe;
  6217. struct pkt_attrib *pattrib;
  6218. unsigned char *pframe;
  6219. struct rtw_ieee80211_hdr *pwlanhdr;
  6220. unsigned short *fctrl;
  6221. unsigned int rate_len;
  6222. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  6223. #if defined(CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME)
  6224. _irqL irqL;
  6225. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  6226. #endif /* #if defined (CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME) */
  6227. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6228. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  6229. WLAN_BSSID_EX *cur_network = &(pmlmeinfo->network);
  6230. u8 bc_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  6231. #ifdef CONFIG_P2P
  6232. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  6233. #endif /* CONFIG_P2P */
  6234. /* RTW_INFO("%s\n", __FUNCTION__); */
  6235. #ifdef CONFIG_BCN_ICF
  6236. pmgntframe = rtw_alloc_bcnxmitframe(pxmitpriv);
  6237. if (pmgntframe == NULL)
  6238. #else
  6239. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  6240. if (pmgntframe == NULL)
  6241. #endif
  6242. {
  6243. RTW_INFO("%s, alloc mgnt frame fail\n", __FUNCTION__);
  6244. return;
  6245. }
  6246. #if defined(CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME)
  6247. _enter_critical_bh(&pmlmepriv->bcn_update_lock, &irqL);
  6248. #endif /* #if defined (CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME) */
  6249. /* update attribute */
  6250. pattrib = &pmgntframe->attrib;
  6251. update_mgntframe_attrib(padapter, pattrib);
  6252. pattrib->qsel = QSLT_BEACON;
  6253. #if defined(CONFIG_CONCURRENT_MODE) && (!defined(CONFIG_SWTIMER_BASED_TXBCN))
  6254. if (padapter->hw_port == HW_PORT1)
  6255. pattrib->mbssid = 1;
  6256. #endif
  6257. #ifdef CONFIG_FW_HANDLE_TXBCN
  6258. if (padapter->vap_id != CONFIG_LIMITED_AP_NUM)
  6259. pattrib->mbssid = padapter->vap_id;
  6260. #endif
  6261. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  6262. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6263. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  6264. fctrl = &(pwlanhdr->frame_ctl);
  6265. *(fctrl) = 0;
  6266. _rtw_memcpy(pwlanhdr->addr1, bc_addr, ETH_ALEN);
  6267. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  6268. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(cur_network), ETH_ALEN);
  6269. SetSeqNum(pwlanhdr, 0/*pmlmeext->mgnt_seq*/);
  6270. /* pmlmeext->mgnt_seq++; */
  6271. set_frame_sub_type(pframe, WIFI_BEACON);
  6272. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  6273. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6274. if (MLME_IS_AP(padapter) || MLME_IS_MESH(padapter)) {
  6275. /* RTW_INFO("ie len=%d\n", cur_network->IELength); */
  6276. #ifdef CONFIG_P2P
  6277. /* for P2P : Primary Device Type & Device Name */
  6278. u32 wpsielen = 0, insert_len = 0;
  6279. u8 *wpsie = NULL;
  6280. wpsie = rtw_get_wps_ie(cur_network->IEs + _FIXED_IE_LENGTH_, cur_network->IELength - _FIXED_IE_LENGTH_, NULL, &wpsielen);
  6281. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO) && wpsie && wpsielen > 0) {
  6282. uint wps_offset, remainder_ielen;
  6283. u8 *premainder_ie, *pframe_wscie;
  6284. wps_offset = (uint)(wpsie - cur_network->IEs);
  6285. premainder_ie = wpsie + wpsielen;
  6286. remainder_ielen = cur_network->IELength - wps_offset - wpsielen;
  6287. #ifdef CONFIG_IOCTL_CFG80211
  6288. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  6289. if (pmlmepriv->wps_beacon_ie && pmlmepriv->wps_beacon_ie_len > 0) {
  6290. _rtw_memcpy(pframe, cur_network->IEs, wps_offset);
  6291. pframe += wps_offset;
  6292. pattrib->pktlen += wps_offset;
  6293. _rtw_memcpy(pframe, pmlmepriv->wps_beacon_ie, pmlmepriv->wps_beacon_ie_len);
  6294. pframe += pmlmepriv->wps_beacon_ie_len;
  6295. pattrib->pktlen += pmlmepriv->wps_beacon_ie_len;
  6296. /* copy remainder_ie to pframe */
  6297. _rtw_memcpy(pframe, premainder_ie, remainder_ielen);
  6298. pframe += remainder_ielen;
  6299. pattrib->pktlen += remainder_ielen;
  6300. } else {
  6301. _rtw_memcpy(pframe, cur_network->IEs, cur_network->IELength);
  6302. pframe += cur_network->IELength;
  6303. pattrib->pktlen += cur_network->IELength;
  6304. }
  6305. } else
  6306. #endif /* CONFIG_IOCTL_CFG80211 */
  6307. {
  6308. pframe_wscie = pframe + wps_offset;
  6309. _rtw_memcpy(pframe, cur_network->IEs, wps_offset + wpsielen);
  6310. pframe += (wps_offset + wpsielen);
  6311. pattrib->pktlen += (wps_offset + wpsielen);
  6312. /* now pframe is end of wsc ie, insert Primary Device Type & Device Name */
  6313. /* Primary Device Type */
  6314. /* Type: */
  6315. *(u16 *)(pframe + insert_len) = cpu_to_be16(WPS_ATTR_PRIMARY_DEV_TYPE);
  6316. insert_len += 2;
  6317. /* Length: */
  6318. *(u16 *)(pframe + insert_len) = cpu_to_be16(0x0008);
  6319. insert_len += 2;
  6320. /* Value: */
  6321. /* Category ID */
  6322. *(u16 *)(pframe + insert_len) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  6323. insert_len += 2;
  6324. /* OUI */
  6325. *(u32 *)(pframe + insert_len) = cpu_to_be32(WPSOUI);
  6326. insert_len += 4;
  6327. /* Sub Category ID */
  6328. *(u16 *)(pframe + insert_len) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  6329. insert_len += 2;
  6330. /* Device Name */
  6331. /* Type: */
  6332. *(u16 *)(pframe + insert_len) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  6333. insert_len += 2;
  6334. /* Length: */
  6335. *(u16 *)(pframe + insert_len) = cpu_to_be16(pwdinfo->device_name_len);
  6336. insert_len += 2;
  6337. /* Value: */
  6338. _rtw_memcpy(pframe + insert_len, pwdinfo->device_name, pwdinfo->device_name_len);
  6339. insert_len += pwdinfo->device_name_len;
  6340. /* update wsc ie length */
  6341. *(pframe_wscie + 1) = (wpsielen - 2) + insert_len;
  6342. /* pframe move to end */
  6343. pframe += insert_len;
  6344. pattrib->pktlen += insert_len;
  6345. /* copy remainder_ie to pframe */
  6346. _rtw_memcpy(pframe, premainder_ie, remainder_ielen);
  6347. pframe += remainder_ielen;
  6348. pattrib->pktlen += remainder_ielen;
  6349. }
  6350. } else
  6351. #endif /* CONFIG_P2P */
  6352. {
  6353. int len_diff;
  6354. _rtw_memcpy(pframe, cur_network->IEs, cur_network->IELength);
  6355. len_diff = update_hidden_ssid(
  6356. pframe + _BEACON_IE_OFFSET_
  6357. , cur_network->IELength - _BEACON_IE_OFFSET_
  6358. , pmlmeinfo->hidden_ssid_mode
  6359. );
  6360. pframe += (cur_network->IELength + len_diff);
  6361. pattrib->pktlen += (cur_network->IELength + len_diff);
  6362. }
  6363. {
  6364. u8 *wps_ie;
  6365. uint wps_ielen;
  6366. u8 sr = 0;
  6367. wps_ie = rtw_get_wps_ie(pmgntframe->buf_addr + TXDESC_OFFSET + sizeof(struct rtw_ieee80211_hdr_3addr) + _BEACON_IE_OFFSET_,
  6368. pattrib->pktlen - sizeof(struct rtw_ieee80211_hdr_3addr) - _BEACON_IE_OFFSET_, NULL, &wps_ielen);
  6369. if (wps_ie && wps_ielen > 0)
  6370. rtw_get_wps_attr_content(wps_ie, wps_ielen, WPS_ATTR_SELECTED_REGISTRAR, (u8 *)(&sr), NULL);
  6371. if (sr != 0)
  6372. set_fwstate(pmlmepriv, WIFI_UNDER_WPS);
  6373. else
  6374. _clr_fwstate_(pmlmepriv, WIFI_UNDER_WPS);
  6375. }
  6376. #ifdef CONFIG_RTW_80211K
  6377. pframe = rtw_set_ie(pframe, _EID_RRM_EN_CAP_IE_,
  6378. sizeof(padapter->rmpriv.rm_en_cap_def),
  6379. padapter->rmpriv.rm_en_cap_def, &pattrib->pktlen);
  6380. #endif
  6381. #ifdef CONFIG_P2P
  6382. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  6383. u32 len;
  6384. #ifdef CONFIG_IOCTL_CFG80211
  6385. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  6386. len = pmlmepriv->p2p_beacon_ie_len;
  6387. if (pmlmepriv->p2p_beacon_ie && len > 0)
  6388. _rtw_memcpy(pframe, pmlmepriv->p2p_beacon_ie, len);
  6389. } else
  6390. #endif /* CONFIG_IOCTL_CFG80211 */
  6391. {
  6392. len = build_beacon_p2p_ie(pwdinfo, pframe);
  6393. }
  6394. pframe += len;
  6395. pattrib->pktlen += len;
  6396. #ifdef CONFIG_MCC_MODE
  6397. pframe = rtw_hal_mcc_append_go_p2p_ie(padapter, pframe, &pattrib->pktlen);
  6398. #endif /* CONFIG_MCC_MODE*/
  6399. #ifdef CONFIG_WFD
  6400. len = rtw_append_beacon_wfd_ie(padapter, pframe);
  6401. pframe += len;
  6402. pattrib->pktlen += len;
  6403. #endif
  6404. }
  6405. #endif /* CONFIG_P2P */
  6406. #ifdef CONFIG_RTW_REPEATER_SON
  6407. rtw_rson_append_ie(padapter, pframe, &pattrib->pktlen);
  6408. #endif
  6409. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  6410. pattrib->pktlen += rtw_build_vendor_ie(padapter , pframe , WIFI_BEACON_VENDOR_IE_BIT);
  6411. #endif
  6412. goto _issue_bcn;
  6413. }
  6414. /* below for ad-hoc mode */
  6415. /* timestamp will be inserted by hardware */
  6416. pframe += 8;
  6417. pattrib->pktlen += 8;
  6418. /* beacon interval: 2 bytes */
  6419. _rtw_memcpy(pframe, (unsigned char *)(rtw_get_beacon_interval_from_ie(cur_network->IEs)), 2);
  6420. pframe += 2;
  6421. pattrib->pktlen += 2;
  6422. /* capability info: 2 bytes */
  6423. _rtw_memcpy(pframe, (unsigned char *)(rtw_get_capability_from_ie(cur_network->IEs)), 2);
  6424. pframe += 2;
  6425. pattrib->pktlen += 2;
  6426. /* SSID */
  6427. pframe = rtw_set_ie(pframe, _SSID_IE_, cur_network->Ssid.SsidLength, cur_network->Ssid.Ssid, &pattrib->pktlen);
  6428. /* supported rates... */
  6429. rate_len = rtw_get_rateset_len(cur_network->SupportedRates);
  6430. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, ((rate_len > 8) ? 8 : rate_len), cur_network->SupportedRates, &pattrib->pktlen);
  6431. /* DS parameter set */
  6432. pframe = rtw_set_ie(pframe, _DSSET_IE_, 1, (unsigned char *)&(cur_network->Configuration.DSConfig), &pattrib->pktlen);
  6433. /* if( (pmlmeinfo->state&0x03) == WIFI_FW_ADHOC_STATE) */
  6434. {
  6435. u8 erpinfo = 0;
  6436. u32 ATIMWindow;
  6437. /* IBSS Parameter Set... */
  6438. /* ATIMWindow = cur->Configuration.ATIMWindow; */
  6439. ATIMWindow = 0;
  6440. pframe = rtw_set_ie(pframe, _IBSS_PARA_IE_, 2, (unsigned char *)(&ATIMWindow), &pattrib->pktlen);
  6441. /* ERP IE */
  6442. pframe = rtw_set_ie(pframe, _ERPINFO_IE_, 1, &erpinfo, &pattrib->pktlen);
  6443. }
  6444. /* EXTERNDED SUPPORTED RATE */
  6445. if (rate_len > 8)
  6446. pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_, (rate_len - 8), (cur_network->SupportedRates + 8), &pattrib->pktlen);
  6447. /* todo:HT for adhoc */
  6448. _issue_bcn:
  6449. #if defined(CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME)
  6450. pmlmepriv->update_bcn = _FALSE;
  6451. _exit_critical_bh(&pmlmepriv->bcn_update_lock, &irqL);
  6452. #endif /* #if defined (CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME) */
  6453. if ((pattrib->pktlen + TXDESC_SIZE) > MAX_BEACON_LEN) {
  6454. RTW_ERR("beacon frame too large ,len(%d,%d)\n",
  6455. (pattrib->pktlen + TXDESC_SIZE), MAX_BEACON_LEN);
  6456. rtw_warn_on(1);
  6457. return;
  6458. }
  6459. pattrib->last_txcmdsz = pattrib->pktlen;
  6460. /* RTW_INFO("issue bcn_sz=%d\n", pattrib->last_txcmdsz); */
  6461. if (timeout_ms > 0)
  6462. dump_mgntframe_and_wait(padapter, pmgntframe, timeout_ms);
  6463. else
  6464. dump_mgntframe(padapter, pmgntframe);
  6465. }
  6466. void issue_probersp(_adapter *padapter, unsigned char *da, u8 is_valid_p2p_probereq)
  6467. {
  6468. struct xmit_frame *pmgntframe;
  6469. struct pkt_attrib *pattrib;
  6470. unsigned char *pframe;
  6471. struct rtw_ieee80211_hdr *pwlanhdr;
  6472. unsigned short *fctrl;
  6473. unsigned char *mac, *bssid;
  6474. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  6475. #if defined(CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME)
  6476. u8 *pwps_ie;
  6477. uint wps_ielen;
  6478. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  6479. #endif /* #if defined (CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME) */
  6480. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6481. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  6482. WLAN_BSSID_EX *cur_network = &(pmlmeinfo->network);
  6483. unsigned int rate_len;
  6484. #ifdef CONFIG_P2P
  6485. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  6486. #endif /* CONFIG_P2P */
  6487. /* RTW_INFO("%s\n", __FUNCTION__); */
  6488. if (da == NULL)
  6489. return;
  6490. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  6491. return;
  6492. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  6493. if (pmgntframe == NULL) {
  6494. RTW_INFO("%s, alloc mgnt frame fail\n", __FUNCTION__);
  6495. return;
  6496. }
  6497. /* update attribute */
  6498. pattrib = &pmgntframe->attrib;
  6499. update_mgntframe_attrib(padapter, pattrib);
  6500. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  6501. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6502. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  6503. mac = adapter_mac_addr(padapter);
  6504. bssid = cur_network->MacAddress;
  6505. fctrl = &(pwlanhdr->frame_ctl);
  6506. *(fctrl) = 0;
  6507. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  6508. _rtw_memcpy(pwlanhdr->addr2, mac, ETH_ALEN);
  6509. _rtw_memcpy(pwlanhdr->addr3, bssid, ETH_ALEN);
  6510. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  6511. pmlmeext->mgnt_seq++;
  6512. set_frame_sub_type(fctrl, WIFI_PROBERSP);
  6513. pattrib->hdrlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6514. pattrib->pktlen = pattrib->hdrlen;
  6515. pframe += pattrib->hdrlen;
  6516. if (cur_network->IELength > MAX_IE_SZ)
  6517. return;
  6518. #if defined(CONFIG_AP_MODE) && defined (CONFIG_NATIVEAP_MLME)
  6519. if ((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) {
  6520. pwps_ie = rtw_get_wps_ie(cur_network->IEs + _FIXED_IE_LENGTH_, cur_network->IELength - _FIXED_IE_LENGTH_, NULL, &wps_ielen);
  6521. /* inerset & update wps_probe_resp_ie */
  6522. if ((pmlmepriv->wps_probe_resp_ie != NULL) && pwps_ie && (wps_ielen > 0)) {
  6523. uint wps_offset, remainder_ielen;
  6524. u8 *premainder_ie;
  6525. wps_offset = (uint)(pwps_ie - cur_network->IEs);
  6526. premainder_ie = pwps_ie + wps_ielen;
  6527. remainder_ielen = cur_network->IELength - wps_offset - wps_ielen;
  6528. _rtw_memcpy(pframe, cur_network->IEs, wps_offset);
  6529. pframe += wps_offset;
  6530. pattrib->pktlen += wps_offset;
  6531. wps_ielen = (uint)pmlmepriv->wps_probe_resp_ie[1];/* to get ie data len */
  6532. if ((wps_offset + wps_ielen + 2) <= MAX_IE_SZ) {
  6533. _rtw_memcpy(pframe, pmlmepriv->wps_probe_resp_ie, wps_ielen + 2);
  6534. pframe += wps_ielen + 2;
  6535. pattrib->pktlen += wps_ielen + 2;
  6536. }
  6537. if ((wps_offset + wps_ielen + 2 + remainder_ielen) <= MAX_IE_SZ) {
  6538. _rtw_memcpy(pframe, premainder_ie, remainder_ielen);
  6539. pframe += remainder_ielen;
  6540. pattrib->pktlen += remainder_ielen;
  6541. }
  6542. } else {
  6543. _rtw_memcpy(pframe, cur_network->IEs, cur_network->IELength);
  6544. pframe += cur_network->IELength;
  6545. pattrib->pktlen += cur_network->IELength;
  6546. }
  6547. /* retrieve SSID IE from cur_network->Ssid */
  6548. {
  6549. u8 *ssid_ie;
  6550. sint ssid_ielen;
  6551. sint ssid_ielen_diff;
  6552. u8 buf[MAX_IE_SZ];
  6553. u8 *ies = pmgntframe->buf_addr + TXDESC_OFFSET + sizeof(struct rtw_ieee80211_hdr_3addr);
  6554. ssid_ie = rtw_get_ie(ies + _FIXED_IE_LENGTH_, _SSID_IE_, &ssid_ielen,
  6555. (pframe - ies) - _FIXED_IE_LENGTH_);
  6556. ssid_ielen_diff = cur_network->Ssid.SsidLength - ssid_ielen;
  6557. if (ssid_ie && cur_network->Ssid.SsidLength) {
  6558. uint remainder_ielen;
  6559. u8 *remainder_ie;
  6560. remainder_ie = ssid_ie + 2;
  6561. remainder_ielen = (pframe - remainder_ie);
  6562. if (remainder_ielen > MAX_IE_SZ) {
  6563. RTW_WARN(FUNC_ADPT_FMT" remainder_ielen > MAX_IE_SZ\n", FUNC_ADPT_ARG(padapter));
  6564. remainder_ielen = MAX_IE_SZ;
  6565. }
  6566. _rtw_memcpy(buf, remainder_ie, remainder_ielen);
  6567. _rtw_memcpy(remainder_ie + ssid_ielen_diff, buf, remainder_ielen);
  6568. *(ssid_ie + 1) = cur_network->Ssid.SsidLength;
  6569. _rtw_memcpy(ssid_ie + 2, cur_network->Ssid.Ssid, cur_network->Ssid.SsidLength);
  6570. pframe += ssid_ielen_diff;
  6571. pattrib->pktlen += ssid_ielen_diff;
  6572. }
  6573. }
  6574. #ifdef CONFIG_RTW_REPEATER_SON
  6575. rtw_rson_append_ie(padapter, pframe, &pattrib->pktlen);
  6576. #endif
  6577. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  6578. pattrib->pktlen += rtw_build_vendor_ie(padapter , pframe , WIFI_PROBERESP_VENDOR_IE_BIT);
  6579. #endif
  6580. } else
  6581. #endif
  6582. {
  6583. /* timestamp will be inserted by hardware */
  6584. pframe += 8;
  6585. pattrib->pktlen += 8;
  6586. /* beacon interval: 2 bytes */
  6587. _rtw_memcpy(pframe, (unsigned char *)(rtw_get_beacon_interval_from_ie(cur_network->IEs)), 2);
  6588. pframe += 2;
  6589. pattrib->pktlen += 2;
  6590. /* capability info: 2 bytes */
  6591. _rtw_memcpy(pframe, (unsigned char *)(rtw_get_capability_from_ie(cur_network->IEs)), 2);
  6592. pframe += 2;
  6593. pattrib->pktlen += 2;
  6594. /* below for ad-hoc mode */
  6595. /* SSID */
  6596. pframe = rtw_set_ie(pframe, _SSID_IE_, cur_network->Ssid.SsidLength, cur_network->Ssid.Ssid, &pattrib->pktlen);
  6597. /* supported rates... */
  6598. rate_len = rtw_get_rateset_len(cur_network->SupportedRates);
  6599. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, ((rate_len > 8) ? 8 : rate_len), cur_network->SupportedRates, &pattrib->pktlen);
  6600. /* DS parameter set */
  6601. pframe = rtw_set_ie(pframe, _DSSET_IE_, 1, (unsigned char *)&(cur_network->Configuration.DSConfig), &pattrib->pktlen);
  6602. if ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) {
  6603. u8 erpinfo = 0;
  6604. u32 ATIMWindow;
  6605. /* IBSS Parameter Set... */
  6606. /* ATIMWindow = cur->Configuration.ATIMWindow; */
  6607. ATIMWindow = 0;
  6608. pframe = rtw_set_ie(pframe, _IBSS_PARA_IE_, 2, (unsigned char *)(&ATIMWindow), &pattrib->pktlen);
  6609. /* ERP IE */
  6610. pframe = rtw_set_ie(pframe, _ERPINFO_IE_, 1, &erpinfo, &pattrib->pktlen);
  6611. }
  6612. /* EXTERNDED SUPPORTED RATE */
  6613. if (rate_len > 8)
  6614. pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_, (rate_len - 8), (cur_network->SupportedRates + 8), &pattrib->pktlen);
  6615. /* todo:HT for adhoc */
  6616. }
  6617. #ifdef CONFIG_RTW_80211K
  6618. pframe = rtw_set_ie(pframe, _EID_RRM_EN_CAP_IE_,
  6619. sizeof(padapter->rmpriv.rm_en_cap_def),
  6620. padapter->rmpriv.rm_en_cap_def, &pattrib->pktlen);
  6621. #endif
  6622. #ifdef CONFIG_P2P
  6623. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)
  6624. /* IOT issue, When wifi_spec is not set, send probe_resp with P2P IE even if probe_req has no P2P IE */
  6625. && (is_valid_p2p_probereq || !padapter->registrypriv.wifi_spec)) {
  6626. u32 len;
  6627. #ifdef CONFIG_IOCTL_CFG80211
  6628. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  6629. /* if pwdinfo->role == P2P_ROLE_DEVICE will call issue_probersp_p2p() */
  6630. len = pmlmepriv->p2p_go_probe_resp_ie_len;
  6631. if (pmlmepriv->p2p_go_probe_resp_ie && len > 0)
  6632. _rtw_memcpy(pframe, pmlmepriv->p2p_go_probe_resp_ie, len);
  6633. } else
  6634. #endif /* CONFIG_IOCTL_CFG80211 */
  6635. {
  6636. len = build_probe_resp_p2p_ie(pwdinfo, pframe);
  6637. }
  6638. pframe += len;
  6639. pattrib->pktlen += len;
  6640. #ifdef CONFIG_MCC_MODE
  6641. pframe = rtw_hal_mcc_append_go_p2p_ie(padapter, pframe, &pattrib->pktlen);
  6642. #endif /* CONFIG_MCC_MODE*/
  6643. #ifdef CONFIG_WFD
  6644. len = rtw_append_probe_resp_wfd_ie(padapter, pframe);
  6645. pframe += len;
  6646. pattrib->pktlen += len;
  6647. #endif
  6648. }
  6649. #endif /* CONFIG_P2P */
  6650. #ifdef CONFIG_AUTO_AP_MODE
  6651. {
  6652. struct sta_info *psta;
  6653. struct sta_priv *pstapriv = &padapter->stapriv;
  6654. RTW_INFO("(%s)\n", __FUNCTION__);
  6655. /* check rc station */
  6656. psta = rtw_get_stainfo(pstapriv, da);
  6657. if (psta && psta->isrc && psta->pid > 0) {
  6658. u8 RC_OUI[4] = {0x00, 0xE0, 0x4C, 0x0A};
  6659. u8 RC_INFO[14] = {0};
  6660. /* EID[1] + EID_LEN[1] + RC_OUI[4] + MAC[6] + PairingID[2] + ChannelNum[2] */
  6661. u16 cu_ch = (u16)cur_network->Configuration.DSConfig;
  6662. RTW_INFO("%s, reply rc(pid=0x%x) device "MAC_FMT" in ch=%d\n", __FUNCTION__,
  6663. psta->pid, MAC_ARG(psta->cmn.mac_addr), cu_ch);
  6664. /* append vendor specific ie */
  6665. _rtw_memcpy(RC_INFO, RC_OUI, sizeof(RC_OUI));
  6666. _rtw_memcpy(&RC_INFO[4], mac, ETH_ALEN);
  6667. _rtw_memcpy(&RC_INFO[10], (u8 *)&psta->pid, 2);
  6668. _rtw_memcpy(&RC_INFO[12], (u8 *)&cu_ch, 2);
  6669. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, sizeof(RC_INFO), RC_INFO, &pattrib->pktlen);
  6670. }
  6671. }
  6672. #endif /* CONFIG_AUTO_AP_MODE */
  6673. pattrib->last_txcmdsz = pattrib->pktlen;
  6674. dump_mgntframe(padapter, pmgntframe);
  6675. return;
  6676. }
  6677. int _issue_probereq(_adapter *padapter, const NDIS_802_11_SSID *pssid, const u8 *da, u8 ch, bool append_wps, int wait_ack)
  6678. {
  6679. int ret = _FAIL;
  6680. struct xmit_frame *pmgntframe;
  6681. struct pkt_attrib *pattrib;
  6682. unsigned char *pframe;
  6683. struct rtw_ieee80211_hdr *pwlanhdr;
  6684. unsigned short *fctrl;
  6685. unsigned char *mac;
  6686. unsigned char bssrate[NumRates];
  6687. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  6688. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  6689. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6690. int bssrate_len = 0;
  6691. u8 bc_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  6692. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  6693. struct rtw_wdev_priv *pwdev_priv = adapter_wdev_data(padapter);
  6694. #endif
  6695. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  6696. goto exit;
  6697. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  6698. if (pmgntframe == NULL)
  6699. goto exit;
  6700. /* update attribute */
  6701. pattrib = &pmgntframe->attrib;
  6702. update_mgntframe_attrib(padapter, pattrib);
  6703. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  6704. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6705. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  6706. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  6707. if ((pwdev_priv->pno_mac_addr[0] != 0xFF)
  6708. && (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) == _TRUE)
  6709. && (check_fwstate(&padapter->mlmepriv, _FW_LINKED) == _FALSE))
  6710. mac = pwdev_priv->pno_mac_addr;
  6711. else
  6712. #endif
  6713. mac = adapter_mac_addr(padapter);
  6714. fctrl = &(pwlanhdr->frame_ctl);
  6715. *(fctrl) = 0;
  6716. if (da) {
  6717. /* unicast probe request frame */
  6718. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  6719. _rtw_memcpy(pwlanhdr->addr3, da, ETH_ALEN);
  6720. } else {
  6721. /* broadcast probe request frame */
  6722. _rtw_memcpy(pwlanhdr->addr1, bc_addr, ETH_ALEN);
  6723. _rtw_memcpy(pwlanhdr->addr3, bc_addr, ETH_ALEN);
  6724. }
  6725. _rtw_memcpy(pwlanhdr->addr2, mac, ETH_ALEN);
  6726. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  6727. if ((pwdev_priv->pno_mac_addr[0] != 0xFF)
  6728. && (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) == _TRUE)
  6729. && (check_fwstate(&padapter->mlmepriv, _FW_LINKED) == _FALSE)) {
  6730. #ifdef CONFIG_RTW_DEBUG
  6731. RTW_DBG("%s pno_scan_seq_num: %d\n", __func__,
  6732. pwdev_priv->pno_scan_seq_num);
  6733. #endif
  6734. SetSeqNum(pwlanhdr, pwdev_priv->pno_scan_seq_num);
  6735. pattrib->seqnum = pwdev_priv->pno_scan_seq_num;
  6736. pattrib->qos_en = 1;
  6737. pwdev_priv->pno_scan_seq_num++;
  6738. } else
  6739. #endif
  6740. {
  6741. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  6742. pmlmeext->mgnt_seq++;
  6743. }
  6744. set_frame_sub_type(pframe, WIFI_PROBEREQ);
  6745. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  6746. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6747. if (pssid && !MLME_IS_MESH(padapter))
  6748. pframe = rtw_set_ie(pframe, _SSID_IE_, pssid->SsidLength, pssid->Ssid, &(pattrib->pktlen));
  6749. else
  6750. pframe = rtw_set_ie(pframe, _SSID_IE_, 0, NULL, &(pattrib->pktlen));
  6751. get_rate_set(padapter, bssrate, &bssrate_len);
  6752. if (bssrate_len > 8) {
  6753. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_ , 8, bssrate, &(pattrib->pktlen));
  6754. pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_ , (bssrate_len - 8), (bssrate + 8), &(pattrib->pktlen));
  6755. } else
  6756. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_ , bssrate_len , bssrate, &(pattrib->pktlen));
  6757. if (ch)
  6758. pframe = rtw_set_ie(pframe, _DSSET_IE_, 1, &ch, &pattrib->pktlen);
  6759. #ifdef CONFIG_RTW_MESH
  6760. if (MLME_IS_MESH(padapter)) {
  6761. if (pssid)
  6762. pframe = rtw_set_ie_mesh_id(pframe, &pattrib->pktlen, pssid->Ssid, pssid->SsidLength);
  6763. else
  6764. pframe = rtw_set_ie_mesh_id(pframe, &pattrib->pktlen, NULL, 0);
  6765. }
  6766. #endif
  6767. if (append_wps) {
  6768. /* add wps_ie for wps2.0 */
  6769. if (pmlmepriv->wps_probe_req_ie_len > 0 && pmlmepriv->wps_probe_req_ie) {
  6770. _rtw_memcpy(pframe, pmlmepriv->wps_probe_req_ie, pmlmepriv->wps_probe_req_ie_len);
  6771. pframe += pmlmepriv->wps_probe_req_ie_len;
  6772. pattrib->pktlen += pmlmepriv->wps_probe_req_ie_len;
  6773. /* pmlmepriv->wps_probe_req_ie_len = 0 ; */ /* reset to zero */
  6774. }
  6775. }
  6776. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  6777. pattrib->pktlen += rtw_build_vendor_ie(padapter , pframe , WIFI_PROBEREQ_VENDOR_IE_BIT);
  6778. #endif
  6779. pattrib->last_txcmdsz = pattrib->pktlen;
  6780. if (wait_ack)
  6781. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  6782. else {
  6783. dump_mgntframe(padapter, pmgntframe);
  6784. ret = _SUCCESS;
  6785. }
  6786. exit:
  6787. return ret;
  6788. }
  6789. inline void issue_probereq(_adapter *padapter, const NDIS_802_11_SSID *pssid, const u8 *da)
  6790. {
  6791. _issue_probereq(padapter, pssid, da, 0, 1, _FALSE);
  6792. }
  6793. /*
  6794. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  6795. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  6796. * try_cnt means the maximal TX count to try
  6797. */
  6798. int issue_probereq_ex(_adapter *padapter, const NDIS_802_11_SSID *pssid, const u8 *da, u8 ch, bool append_wps,
  6799. int try_cnt, int wait_ms)
  6800. {
  6801. int ret = _FAIL;
  6802. int i = 0;
  6803. systime start = rtw_get_current_time();
  6804. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  6805. goto exit;
  6806. do {
  6807. ret = _issue_probereq(padapter, pssid, da, ch, append_wps, wait_ms > 0 ? _TRUE : _FALSE);
  6808. i++;
  6809. if (RTW_CANNOT_RUN(padapter))
  6810. break;
  6811. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  6812. rtw_msleep_os(wait_ms);
  6813. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  6814. if (ret != _FAIL) {
  6815. ret = _SUCCESS;
  6816. #ifndef DBG_XMIT_ACK
  6817. goto exit;
  6818. #endif
  6819. }
  6820. if (try_cnt && wait_ms) {
  6821. if (da)
  6822. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", ch:%u%s, %d/%d in %u ms\n",
  6823. FUNC_ADPT_ARG(padapter), MAC_ARG(da), rtw_get_oper_ch(padapter),
  6824. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  6825. else
  6826. RTW_INFO(FUNC_ADPT_FMT", ch:%u%s, %d/%d in %u ms\n",
  6827. FUNC_ADPT_ARG(padapter), rtw_get_oper_ch(padapter),
  6828. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  6829. }
  6830. exit:
  6831. return ret;
  6832. }
  6833. /* if psta == NULL, indiate we are station(client) now... */
  6834. void issue_auth(_adapter *padapter, struct sta_info *psta, unsigned short status)
  6835. {
  6836. struct xmit_frame *pmgntframe;
  6837. struct pkt_attrib *pattrib;
  6838. unsigned char *pframe;
  6839. struct rtw_ieee80211_hdr *pwlanhdr;
  6840. unsigned short *fctrl;
  6841. unsigned int val32;
  6842. unsigned short val16;
  6843. int use_shared_key = 0;
  6844. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  6845. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  6846. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  6847. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  6848. return;
  6849. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  6850. if (pmgntframe == NULL)
  6851. return;
  6852. /* update attribute */
  6853. pattrib = &pmgntframe->attrib;
  6854. update_mgntframe_attrib(padapter, pattrib);
  6855. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  6856. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6857. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  6858. fctrl = &(pwlanhdr->frame_ctl);
  6859. *(fctrl) = 0;
  6860. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  6861. pmlmeext->mgnt_seq++;
  6862. set_frame_sub_type(pframe, WIFI_AUTH);
  6863. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  6864. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6865. if (psta) { /* for AP mode */
  6866. #ifdef CONFIG_NATIVEAP_MLME
  6867. _rtw_memcpy(pwlanhdr->addr1, psta->cmn.mac_addr, ETH_ALEN);
  6868. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  6869. _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(padapter), ETH_ALEN);
  6870. /* setting auth algo number */
  6871. val16 = (u16)psta->authalg;
  6872. if (status != _STATS_SUCCESSFUL_)
  6873. val16 = 0;
  6874. if (val16) {
  6875. val16 = cpu_to_le16(val16);
  6876. use_shared_key = 1;
  6877. }
  6878. pframe = rtw_set_fixed_ie(pframe, _AUTH_ALGM_NUM_, (unsigned char *)&val16, &(pattrib->pktlen));
  6879. /* setting auth seq number */
  6880. val16 = (u16)psta->auth_seq;
  6881. val16 = cpu_to_le16(val16);
  6882. pframe = rtw_set_fixed_ie(pframe, _AUTH_SEQ_NUM_, (unsigned char *)&val16, &(pattrib->pktlen));
  6883. /* setting status code... */
  6884. val16 = status;
  6885. val16 = cpu_to_le16(val16);
  6886. pframe = rtw_set_fixed_ie(pframe, _STATUS_CODE_, (unsigned char *)&val16, &(pattrib->pktlen));
  6887. /* added challenging text... */
  6888. if ((psta->auth_seq == 2) && (psta->state & WIFI_FW_AUTH_STATE) && (use_shared_key == 1))
  6889. pframe = rtw_set_ie(pframe, _CHLGETXT_IE_, 128, psta->chg_txt, &(pattrib->pktlen));
  6890. #endif
  6891. } else {
  6892. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  6893. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  6894. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  6895. #ifdef CONFIG_RTW_80211R
  6896. if (rtw_ft_roam(padapter)) {
  6897. /* 2: 802.11R FTAA */
  6898. val16 = cpu_to_le16(2);
  6899. } else
  6900. #endif
  6901. {
  6902. /* setting auth algo number */
  6903. val16 = (pmlmeinfo->auth_algo == dot11AuthAlgrthm_Shared) ? 1 : 0; /* 0:OPEN System, 1:Shared key */
  6904. if (val16) {
  6905. val16 = cpu_to_le16(val16);
  6906. use_shared_key = 1;
  6907. }
  6908. }
  6909. /* RTW_INFO("%s auth_algo= %s auth_seq=%d\n",__FUNCTION__,(pmlmeinfo->auth_algo==0)?"OPEN":"SHARED",pmlmeinfo->auth_seq); */
  6910. /* setting IV for auth seq #3 */
  6911. if ((pmlmeinfo->auth_seq == 3) && (pmlmeinfo->state & WIFI_FW_AUTH_STATE) && (use_shared_key == 1)) {
  6912. /* RTW_INFO("==> iv(%d),key_index(%d)\n",pmlmeinfo->iv,pmlmeinfo->key_index); */
  6913. val32 = ((pmlmeinfo->iv++) | (pmlmeinfo->key_index << 30));
  6914. val32 = cpu_to_le32(val32);
  6915. pframe = rtw_set_fixed_ie(pframe, 4, (unsigned char *)&val32, &(pattrib->pktlen));
  6916. pattrib->iv_len = 4;
  6917. }
  6918. pframe = rtw_set_fixed_ie(pframe, _AUTH_ALGM_NUM_, (unsigned char *)&val16, &(pattrib->pktlen));
  6919. /* setting auth seq number */
  6920. val16 = pmlmeinfo->auth_seq;
  6921. val16 = cpu_to_le16(val16);
  6922. pframe = rtw_set_fixed_ie(pframe, _AUTH_SEQ_NUM_, (unsigned char *)&val16, &(pattrib->pktlen));
  6923. /* setting status code... */
  6924. val16 = status;
  6925. val16 = cpu_to_le16(val16);
  6926. pframe = rtw_set_fixed_ie(pframe, _STATUS_CODE_, (unsigned char *)&val16, &(pattrib->pktlen));
  6927. #ifdef CONFIG_RTW_80211R
  6928. rtw_ft_build_auth_req_ies(padapter, pattrib, &pframe);
  6929. #endif
  6930. /* then checking to see if sending challenging text... */
  6931. if ((pmlmeinfo->auth_seq == 3) && (pmlmeinfo->state & WIFI_FW_AUTH_STATE) && (use_shared_key == 1)) {
  6932. pframe = rtw_set_ie(pframe, _CHLGETXT_IE_, 128, pmlmeinfo->chg_txt, &(pattrib->pktlen));
  6933. SetPrivacy(fctrl);
  6934. pattrib->hdrlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6935. pattrib->encrypt = _WEP40_;
  6936. pattrib->icv_len = 4;
  6937. pattrib->pktlen += pattrib->icv_len;
  6938. }
  6939. }
  6940. pattrib->last_txcmdsz = pattrib->pktlen;
  6941. rtw_wep_encrypt(padapter, (u8 *)pmgntframe);
  6942. RTW_INFO("%s\n", __FUNCTION__);
  6943. dump_mgntframe(padapter, pmgntframe);
  6944. return;
  6945. }
  6946. void issue_asocrsp(_adapter *padapter, unsigned short status, struct sta_info *pstat, int pkt_type)
  6947. {
  6948. #ifdef CONFIG_AP_MODE
  6949. struct xmit_frame *pmgntframe;
  6950. struct rtw_ieee80211_hdr *pwlanhdr;
  6951. struct pkt_attrib *pattrib;
  6952. unsigned char *pbuf, *pframe;
  6953. unsigned short val, ie_status;
  6954. unsigned short *fctrl;
  6955. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  6956. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  6957. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  6958. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  6959. WLAN_BSSID_EX *pnetwork = &(pmlmeinfo->network);
  6960. u8 *ie = pnetwork->IEs;
  6961. #ifdef CONFIG_P2P
  6962. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  6963. #ifdef CONFIG_WFD
  6964. u32 wfdielen = 0;
  6965. #endif
  6966. #endif /* CONFIG_P2P */
  6967. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  6968. return;
  6969. RTW_INFO("%s\n", __FUNCTION__);
  6970. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  6971. if (pmgntframe == NULL)
  6972. return;
  6973. /* update attribute */
  6974. pattrib = &pmgntframe->attrib;
  6975. update_mgntframe_attrib(padapter, pattrib);
  6976. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  6977. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  6978. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  6979. fctrl = &(pwlanhdr->frame_ctl);
  6980. *(fctrl) = 0;
  6981. _rtw_memcpy((void *)GetAddr1Ptr(pwlanhdr), pstat->cmn.mac_addr, ETH_ALEN);
  6982. _rtw_memcpy((void *)get_addr2_ptr(pwlanhdr), adapter_mac_addr(padapter), ETH_ALEN);
  6983. _rtw_memcpy((void *)GetAddr3Ptr(pwlanhdr), get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  6984. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  6985. pmlmeext->mgnt_seq++;
  6986. if ((pkt_type == WIFI_ASSOCRSP) || (pkt_type == WIFI_REASSOCRSP))
  6987. set_frame_sub_type(pwlanhdr, pkt_type);
  6988. else
  6989. return;
  6990. pattrib->hdrlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  6991. pattrib->pktlen += pattrib->hdrlen;
  6992. pframe += pattrib->hdrlen;
  6993. /* capability */
  6994. val = *(unsigned short *)rtw_get_capability_from_ie(ie);
  6995. pframe = rtw_set_fixed_ie(pframe, _CAPABILITY_ , (unsigned char *)&val, &(pattrib->pktlen));
  6996. ie_status = cpu_to_le16(status);
  6997. pframe = rtw_set_fixed_ie(pframe , _STATUS_CODE_ , (unsigned char *)&ie_status, &(pattrib->pktlen));
  6998. val = cpu_to_le16(pstat->cmn.aid | BIT(14) | BIT(15));
  6999. pframe = rtw_set_fixed_ie(pframe, _ASOC_ID_ , (unsigned char *)&val, &(pattrib->pktlen));
  7000. if (pstat->bssratelen <= 8)
  7001. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, pstat->bssratelen, pstat->bssrateset, &(pattrib->pktlen));
  7002. else {
  7003. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, 8, pstat->bssrateset, &(pattrib->pktlen));
  7004. pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_, (pstat->bssratelen - 8), pstat->bssrateset + 8, &(pattrib->pktlen));
  7005. }
  7006. #ifdef CONFIG_IEEE80211W
  7007. if (status == _STATS_REFUSED_TEMPORARILY_) {
  7008. u8 timeout_itvl[5];
  7009. u32 timeout_interval = 3000;
  7010. /* Association Comeback time */
  7011. timeout_itvl[0] = 0x03;
  7012. timeout_interval = cpu_to_le32(timeout_interval);
  7013. _rtw_memcpy(timeout_itvl + 1, &timeout_interval, 4);
  7014. pframe = rtw_set_ie(pframe, _TIMEOUT_ITVL_IE_, 5, timeout_itvl, &(pattrib->pktlen));
  7015. }
  7016. #endif /* CONFIG_IEEE80211W */
  7017. #ifdef CONFIG_80211N_HT
  7018. if ((pstat->flags & WLAN_STA_HT) && (pmlmepriv->htpriv.ht_option)) {
  7019. uint ie_len = 0;
  7020. /* FILL HT CAP INFO IE */
  7021. /* p = hostapd_eid_ht_capabilities_info(hapd, p); */
  7022. pbuf = rtw_get_ie(ie + _BEACON_IE_OFFSET_, _HT_CAPABILITY_IE_, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_));
  7023. if (pbuf && ie_len > 0) {
  7024. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7025. pframe += (ie_len + 2);
  7026. pattrib->pktlen += (ie_len + 2);
  7027. }
  7028. /* FILL HT ADD INFO IE */
  7029. /* p = hostapd_eid_ht_operation(hapd, p); */
  7030. pbuf = rtw_get_ie(ie + _BEACON_IE_OFFSET_, _HT_ADD_INFO_IE_, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_));
  7031. if (pbuf && ie_len > 0) {
  7032. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7033. pframe += (ie_len + 2);
  7034. pattrib->pktlen += (ie_len + 2);
  7035. }
  7036. }
  7037. #endif
  7038. /*adding EXT_CAPAB_IE */
  7039. if (pmlmepriv->ext_capab_ie_len > 0) {
  7040. uint ie_len = 0;
  7041. pbuf = rtw_get_ie(ie + _BEACON_IE_OFFSET_, _EXT_CAP_IE_, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_));
  7042. if (pbuf && ie_len > 0) {
  7043. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7044. pframe += (ie_len + 2);
  7045. pattrib->pktlen += (ie_len + 2);
  7046. }
  7047. }
  7048. #ifdef CONFIG_80211AC_VHT
  7049. if ((pstat->flags & WLAN_STA_VHT) && (pmlmepriv->vhtpriv.vht_option)
  7050. && (pstat->wpa_pairwise_cipher != WPA_CIPHER_TKIP)
  7051. && (pstat->wpa2_pairwise_cipher != WPA_CIPHER_TKIP)) {
  7052. u32 ie_len = 0;
  7053. /* FILL VHT CAP IE */
  7054. pbuf = rtw_get_ie(ie + _BEACON_IE_OFFSET_, EID_VHTCapability, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_));
  7055. if (pbuf && ie_len > 0) {
  7056. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7057. pframe += (ie_len + 2);
  7058. pattrib->pktlen += (ie_len + 2);
  7059. }
  7060. /* FILL VHT OPERATION IE */
  7061. pbuf = rtw_get_ie(ie + _BEACON_IE_OFFSET_, EID_VHTOperation, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_));
  7062. if (pbuf && ie_len > 0) {
  7063. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7064. pframe += (ie_len + 2);
  7065. pattrib->pktlen += (ie_len + 2);
  7066. }
  7067. }
  7068. #endif /* CONFIG_80211AC_VHT */
  7069. /* FILL WMM IE */
  7070. if ((pstat->flags & WLAN_STA_WME) && (pmlmepriv->qospriv.qos_option)) {
  7071. uint ie_len = 0;
  7072. unsigned char WMM_PARA_IE[] = {0x00, 0x50, 0xf2, 0x02, 0x01, 0x01};
  7073. for (pbuf = ie + _BEACON_IE_OFFSET_; ; pbuf += (ie_len + 2)) {
  7074. pbuf = rtw_get_ie(pbuf, _VENDOR_SPECIFIC_IE_, &ie_len, (pnetwork->IELength - _BEACON_IE_OFFSET_ - (ie_len + 2)));
  7075. if (pbuf && _rtw_memcmp(pbuf + 2, WMM_PARA_IE, 6)) {
  7076. _rtw_memcpy(pframe, pbuf, ie_len + 2);
  7077. pframe += (ie_len + 2);
  7078. pattrib->pktlen += (ie_len + 2);
  7079. break;
  7080. }
  7081. if ((pbuf == NULL) || (ie_len == 0))
  7082. break;
  7083. }
  7084. }
  7085. if (pmlmeinfo->assoc_AP_vendor == HT_IOT_PEER_REALTEK)
  7086. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, 6 , REALTEK_96B_IE, &(pattrib->pktlen));
  7087. /* add WPS IE ie for wps 2.0 */
  7088. if (pmlmepriv->wps_assoc_resp_ie && pmlmepriv->wps_assoc_resp_ie_len > 0) {
  7089. _rtw_memcpy(pframe, pmlmepriv->wps_assoc_resp_ie, pmlmepriv->wps_assoc_resp_ie_len);
  7090. pframe += pmlmepriv->wps_assoc_resp_ie_len;
  7091. pattrib->pktlen += pmlmepriv->wps_assoc_resp_ie_len;
  7092. }
  7093. #ifdef CONFIG_P2P
  7094. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO) && (pstat->is_p2p_device == _TRUE)) {
  7095. u32 len;
  7096. if (padapter->wdinfo.driver_interface == DRIVER_CFG80211) {
  7097. len = 0;
  7098. if (pmlmepriv->p2p_assoc_resp_ie && pmlmepriv->p2p_assoc_resp_ie_len > 0) {
  7099. len = pmlmepriv->p2p_assoc_resp_ie_len;
  7100. _rtw_memcpy(pframe, pmlmepriv->p2p_assoc_resp_ie, len);
  7101. }
  7102. } else
  7103. len = build_assoc_resp_p2p_ie(pwdinfo, pframe, pstat->p2p_status_code);
  7104. pframe += len;
  7105. pattrib->pktlen += len;
  7106. }
  7107. #ifdef CONFIG_WFD
  7108. if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
  7109. wfdielen = rtw_append_assoc_resp_wfd_ie(padapter, pframe);
  7110. pframe += wfdielen;
  7111. pattrib->pktlen += wfdielen;
  7112. }
  7113. #endif
  7114. #endif /* CONFIG_P2P */
  7115. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  7116. pattrib->pktlen += rtw_build_vendor_ie(padapter , pframe , WIFI_ASSOCRESP_VENDOR_IE_BIT);
  7117. #endif
  7118. pattrib->last_txcmdsz = pattrib->pktlen;
  7119. dump_mgntframe(padapter, pmgntframe);
  7120. #endif
  7121. }
  7122. void _issue_assocreq(_adapter *padapter, u8 is_reassoc)
  7123. {
  7124. int ret = _FAIL;
  7125. struct xmit_frame *pmgntframe;
  7126. struct pkt_attrib *pattrib;
  7127. unsigned char *pframe;
  7128. struct rtw_ieee80211_hdr *pwlanhdr;
  7129. unsigned short *fctrl;
  7130. unsigned short val16;
  7131. unsigned int i, j, index = 0;
  7132. unsigned char bssrate[NumRates], sta_bssrate[NumRates];
  7133. PNDIS_802_11_VARIABLE_IEs pIE;
  7134. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7135. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  7136. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7137. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7138. int bssrate_len = 0, sta_bssrate_len = 0;
  7139. u8 vs_ie_length = 0;
  7140. #ifdef CONFIG_P2P
  7141. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  7142. u8 p2pie[255] = { 0x00 };
  7143. u16 p2pielen = 0;
  7144. #ifdef CONFIG_WFD
  7145. u32 wfdielen = 0;
  7146. #endif
  7147. #endif /* CONFIG_P2P */
  7148. #ifdef CONFIG_DFS
  7149. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  7150. u16 cap;
  7151. /* Dot H */
  7152. u8 pow_cap_ele[2] = { 0x00 };
  7153. u8 sup_ch[30 * 2] = {0x00 }, sup_ch_idx = 0, idx_5g = 2; /* For supported channel */
  7154. #endif /* CONFIG_DFS */
  7155. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7156. goto exit;
  7157. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7158. if (pmgntframe == NULL)
  7159. goto exit;
  7160. /* update attribute */
  7161. pattrib = &pmgntframe->attrib;
  7162. update_mgntframe_attrib(padapter, pattrib);
  7163. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7164. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7165. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7166. fctrl = &(pwlanhdr->frame_ctl);
  7167. *(fctrl) = 0;
  7168. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7169. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7170. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7171. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7172. pmlmeext->mgnt_seq++;
  7173. if (is_reassoc == _TRUE)
  7174. set_frame_sub_type(pframe, WIFI_REASSOCREQ);
  7175. else
  7176. set_frame_sub_type(pframe, WIFI_ASSOCREQ);
  7177. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  7178. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  7179. /* caps */
  7180. #ifdef CONFIG_DFS
  7181. _rtw_memcpy(&cap, rtw_get_capability_from_ie(pmlmeinfo->network.IEs), 2);
  7182. cap |= cap_SpecMgmt;
  7183. _rtw_memcpy(pframe, &cap, 2);
  7184. #else
  7185. _rtw_memcpy(pframe, rtw_get_capability_from_ie(pmlmeinfo->network.IEs), 2);
  7186. #endif /* CONFIG_DFS */
  7187. pframe += 2;
  7188. pattrib->pktlen += 2;
  7189. /* listen interval */
  7190. /* todo: listen interval for power saving */
  7191. val16 = cpu_to_le16(3);
  7192. _rtw_memcpy(pframe , (unsigned char *)&val16, 2);
  7193. pframe += 2;
  7194. pattrib->pktlen += 2;
  7195. /*Construct Current AP Field for Reassoc-Req only*/
  7196. if (is_reassoc == _TRUE) {
  7197. _rtw_memcpy(pframe, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7198. pframe += ETH_ALEN;
  7199. pattrib->pktlen += ETH_ALEN;
  7200. }
  7201. /* SSID */
  7202. pframe = rtw_set_ie(pframe, _SSID_IE_, pmlmeinfo->network.Ssid.SsidLength, pmlmeinfo->network.Ssid.Ssid, &(pattrib->pktlen));
  7203. #ifdef CONFIG_DFS
  7204. /* Dot H */
  7205. if (pmlmeext->cur_channel > 14) {
  7206. pow_cap_ele[0] = 13; /* Minimum transmit power capability */
  7207. pow_cap_ele[1] = 21; /* Maximum transmit power capability */
  7208. pframe = rtw_set_ie(pframe, EID_PowerCap, 2, pow_cap_ele, &(pattrib->pktlen));
  7209. /* supported channels */
  7210. while (sup_ch_idx < rfctl->max_chan_nums && rfctl->channel_set[sup_ch_idx].ChannelNum != 0) {
  7211. if (rfctl->channel_set[sup_ch_idx].ChannelNum <= 14) {
  7212. /* TODO: fix 2.4G supported channel when channel doesn't start from 1 and continuous */
  7213. sup_ch[0] = 1; /* First channel number */
  7214. sup_ch[1] = rfctl->channel_set[sup_ch_idx].ChannelNum; /* Number of channel */
  7215. } else {
  7216. sup_ch[idx_5g++] = rfctl->channel_set[sup_ch_idx].ChannelNum;
  7217. sup_ch[idx_5g++] = 1;
  7218. }
  7219. sup_ch_idx++;
  7220. }
  7221. pframe = rtw_set_ie(pframe, EID_SupportedChannels, idx_5g, sup_ch, &(pattrib->pktlen));
  7222. }
  7223. #endif /* CONFIG_DFS */
  7224. /* supported rate & extended supported rate */
  7225. #if 1 /* Check if the AP's supported rates are also supported by STA. */
  7226. get_rate_set(padapter, sta_bssrate, &sta_bssrate_len);
  7227. /* RTW_INFO("sta_bssrate_len=%d\n", sta_bssrate_len); */
  7228. if (pmlmeext->cur_channel == 14) /* for JAPAN, channel 14 can only uses B Mode(CCK) */
  7229. sta_bssrate_len = 4;
  7230. /* for (i = 0; i < sta_bssrate_len; i++) { */
  7231. /* RTW_INFO("sta_bssrate[%d]=%02X\n", i, sta_bssrate[i]); */
  7232. /* } */
  7233. for (i = 0; i < NDIS_802_11_LENGTH_RATES_EX; i++) {
  7234. if (pmlmeinfo->network.SupportedRates[i] == 0)
  7235. break;
  7236. RTW_INFO("network.SupportedRates[%d]=%02X\n", i, pmlmeinfo->network.SupportedRates[i]);
  7237. }
  7238. for (i = 0; i < NDIS_802_11_LENGTH_RATES_EX; i++) {
  7239. if (pmlmeinfo->network.SupportedRates[i] == 0)
  7240. break;
  7241. /* Check if the AP's supported rates are also supported by STA. */
  7242. for (j = 0; j < sta_bssrate_len; j++) {
  7243. /* Avoid the proprietary data rate (22Mbps) of Handlink WSG-4000 AP */
  7244. if ((pmlmeinfo->network.SupportedRates[i] | IEEE80211_BASIC_RATE_MASK)
  7245. == (sta_bssrate[j] | IEEE80211_BASIC_RATE_MASK)) {
  7246. /* RTW_INFO("match i = %d, j=%d\n", i, j); */
  7247. break;
  7248. } else {
  7249. /* RTW_INFO("not match: %02X != %02X\n", (pmlmeinfo->network.SupportedRates[i]|IEEE80211_BASIC_RATE_MASK), (sta_bssrate[j]|IEEE80211_BASIC_RATE_MASK)); */
  7250. }
  7251. }
  7252. if (j == sta_bssrate_len) {
  7253. /* the rate is not supported by STA */
  7254. RTW_INFO("%s(): the rate[%d]=%02X is not supported by STA!\n", __FUNCTION__, i, pmlmeinfo->network.SupportedRates[i]);
  7255. } else {
  7256. /* the rate is supported by STA */
  7257. bssrate[index++] = pmlmeinfo->network.SupportedRates[i];
  7258. }
  7259. }
  7260. bssrate_len = index;
  7261. RTW_INFO("bssrate_len = %d\n", bssrate_len);
  7262. #else /* Check if the AP's supported rates are also supported by STA. */
  7263. #if 0
  7264. get_rate_set(padapter, bssrate, &bssrate_len);
  7265. #else
  7266. for (bssrate_len = 0; bssrate_len < NumRates; bssrate_len++) {
  7267. if (pmlmeinfo->network.SupportedRates[bssrate_len] == 0)
  7268. break;
  7269. if (pmlmeinfo->network.SupportedRates[bssrate_len] == 0x2C) /* Avoid the proprietary data rate (22Mbps) of Handlink WSG-4000 AP */
  7270. break;
  7271. bssrate[bssrate_len] = pmlmeinfo->network.SupportedRates[bssrate_len];
  7272. }
  7273. #endif
  7274. #endif /* Check if the AP's supported rates are also supported by STA. */
  7275. if ((bssrate_len == 0) && (pmlmeinfo->network.SupportedRates[0] != 0)) {
  7276. rtw_free_xmitbuf(pxmitpriv, pmgntframe->pxmitbuf);
  7277. rtw_free_xmitframe(pxmitpriv, pmgntframe);
  7278. goto exit; /* don't connect to AP if no joint supported rate */
  7279. }
  7280. if (bssrate_len > 8) {
  7281. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_ , 8, bssrate, &(pattrib->pktlen));
  7282. pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_ , (bssrate_len - 8), (bssrate + 8), &(pattrib->pktlen));
  7283. } else if (bssrate_len > 0)
  7284. pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_ , bssrate_len , bssrate, &(pattrib->pktlen));
  7285. else
  7286. RTW_INFO("%s: Connect to AP without 11b and 11g data rate!\n", __FUNCTION__);
  7287. #ifdef CONFIG_RTW_80211K
  7288. if (pmlmeinfo->network.PhyInfo.rm_en_cap[0] /* RM Enabled Capabilities */
  7289. | pmlmeinfo->network.PhyInfo.rm_en_cap[1]
  7290. | pmlmeinfo->network.PhyInfo.rm_en_cap[2]
  7291. | pmlmeinfo->network.PhyInfo.rm_en_cap[3]
  7292. | pmlmeinfo->network.PhyInfo.rm_en_cap[4])
  7293. pframe = rtw_set_ie(pframe, _EID_RRM_EN_CAP_IE_, 5,
  7294. (u8 *)padapter->rmpriv.rm_en_cap_def, &(pattrib->pktlen));
  7295. #endif /* CONFIG_RTW_80211K */
  7296. /* vendor specific IE, such as WPA, WMM, WPS */
  7297. for (i = sizeof(NDIS_802_11_FIXED_IEs); i < pmlmeinfo->network.IELength;) {
  7298. pIE = (PNDIS_802_11_VARIABLE_IEs)(pmlmeinfo->network.IEs + i);
  7299. switch (pIE->ElementID) {
  7300. case _VENDOR_SPECIFIC_IE_:
  7301. if ((_rtw_memcmp(pIE->data, RTW_WPA_OUI, 4)) ||
  7302. (_rtw_memcmp(pIE->data, WMM_OUI, 4)) ||
  7303. (_rtw_memcmp(pIE->data, WPS_OUI, 4))) {
  7304. vs_ie_length = pIE->Length;
  7305. if ((!padapter->registrypriv.wifi_spec) && (_rtw_memcmp(pIE->data, WPS_OUI, 4))) {
  7306. /* Commented by Kurt 20110629 */
  7307. /* In some older APs, WPS handshake */
  7308. /* would be fail if we append vender extensions informations to AP */
  7309. vs_ie_length = 14;
  7310. }
  7311. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, vs_ie_length, pIE->data, &(pattrib->pktlen));
  7312. }
  7313. break;
  7314. case EID_WPA2:
  7315. #ifdef CONFIG_RTW_80211R
  7316. if ((is_reassoc) && (rtw_ft_roam(padapter))) {
  7317. rtw_ft_update_rsnie(padapter, _TRUE, pattrib, &pframe);
  7318. } else
  7319. #endif
  7320. pframe = rtw_set_ie(pframe, EID_WPA2, pIE->Length, pIE->data, &(pattrib->pktlen));
  7321. break;
  7322. #ifdef CONFIG_80211N_HT
  7323. case EID_HTCapability:
  7324. if (padapter->mlmepriv.htpriv.ht_option == _TRUE) {
  7325. if (!(is_ap_in_tkip(padapter))) {
  7326. _rtw_memcpy(&(pmlmeinfo->HT_caps), pIE->data, sizeof(struct HT_caps_element));
  7327. pmlmeinfo->HT_caps.u.HT_cap_element.HT_caps_info = cpu_to_le16(pmlmeinfo->HT_caps.u.HT_cap_element.HT_caps_info);
  7328. pframe = rtw_set_ie(pframe, EID_HTCapability, pIE->Length , (u8 *)(&(pmlmeinfo->HT_caps)), &(pattrib->pktlen));
  7329. }
  7330. }
  7331. break;
  7332. case EID_EXTCapability:
  7333. if (padapter->mlmepriv.htpriv.ht_option == _TRUE)
  7334. pframe = rtw_set_ie(pframe, EID_EXTCapability, pIE->Length, pIE->data, &(pattrib->pktlen));
  7335. break;
  7336. #endif /* CONFIG_80211N_HT */
  7337. #ifdef CONFIG_80211AC_VHT
  7338. case EID_VHTCapability:
  7339. if (padapter->mlmepriv.vhtpriv.vht_option == _TRUE)
  7340. pframe = rtw_set_ie(pframe, EID_VHTCapability, pIE->Length, pIE->data, &(pattrib->pktlen));
  7341. break;
  7342. case EID_OpModeNotification:
  7343. if (padapter->mlmepriv.vhtpriv.vht_option == _TRUE)
  7344. pframe = rtw_set_ie(pframe, EID_OpModeNotification, pIE->Length, pIE->data, &(pattrib->pktlen));
  7345. break;
  7346. #endif /* CONFIG_80211AC_VHT */
  7347. default:
  7348. break;
  7349. }
  7350. i += (pIE->Length + 2);
  7351. }
  7352. if (pmlmeinfo->assoc_AP_vendor == HT_IOT_PEER_REALTEK)
  7353. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, 6 , REALTEK_96B_IE, &(pattrib->pktlen));
  7354. #ifdef CONFIG_WAPI_SUPPORT
  7355. rtw_build_assoc_req_wapi_ie(padapter, pframe, pattrib);
  7356. #endif
  7357. #ifdef CONFIG_P2P
  7358. #ifdef CONFIG_IOCTL_CFG80211
  7359. if (adapter_wdev_data(padapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
  7360. if (pmlmepriv->p2p_assoc_req_ie && pmlmepriv->p2p_assoc_req_ie_len > 0) {
  7361. _rtw_memcpy(pframe, pmlmepriv->p2p_assoc_req_ie, pmlmepriv->p2p_assoc_req_ie_len);
  7362. pframe += pmlmepriv->p2p_assoc_req_ie_len;
  7363. pattrib->pktlen += pmlmepriv->p2p_assoc_req_ie_len;
  7364. }
  7365. } else
  7366. #endif /* CONFIG_IOCTL_CFG80211 */
  7367. {
  7368. if (!rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE) && !rtw_p2p_chk_state(pwdinfo, P2P_STATE_IDLE)) {
  7369. /* Should add the P2P IE in the association request frame. */
  7370. /* P2P OUI */
  7371. p2pielen = 0;
  7372. p2pie[p2pielen++] = 0x50;
  7373. p2pie[p2pielen++] = 0x6F;
  7374. p2pie[p2pielen++] = 0x9A;
  7375. p2pie[p2pielen++] = 0x09; /* WFA P2P v1.0 */
  7376. /* Commented by Albert 20101109 */
  7377. /* According to the P2P Specification, the association request frame should contain 3 P2P attributes */
  7378. /* 1. P2P Capability */
  7379. /* 2. Extended Listen Timing */
  7380. /* 3. Device Info */
  7381. /* Commented by Albert 20110516 */
  7382. /* 4. P2P Interface */
  7383. /* P2P Capability */
  7384. /* Type: */
  7385. p2pie[p2pielen++] = P2P_ATTR_CAPABILITY;
  7386. /* Length: */
  7387. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0002);
  7388. p2pielen += 2;
  7389. /* Value: */
  7390. /* Device Capability Bitmap, 1 byte */
  7391. p2pie[p2pielen++] = DMP_P2P_DEVCAP_SUPPORT;
  7392. /* Group Capability Bitmap, 1 byte */
  7393. if (pwdinfo->persistent_supported)
  7394. p2pie[p2pielen++] = P2P_GRPCAP_PERSISTENT_GROUP | DMP_P2P_GRPCAP_SUPPORT;
  7395. else
  7396. p2pie[p2pielen++] = DMP_P2P_GRPCAP_SUPPORT;
  7397. /* Extended Listen Timing */
  7398. /* Type: */
  7399. p2pie[p2pielen++] = P2P_ATTR_EX_LISTEN_TIMING;
  7400. /* Length: */
  7401. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x0004);
  7402. p2pielen += 2;
  7403. /* Value: */
  7404. /* Availability Period */
  7405. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  7406. p2pielen += 2;
  7407. /* Availability Interval */
  7408. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0xFFFF);
  7409. p2pielen += 2;
  7410. /* Device Info */
  7411. /* Type: */
  7412. p2pie[p2pielen++] = P2P_ATTR_DEVICE_INFO;
  7413. /* Length: */
  7414. /* 21->P2P Device Address (6bytes) + Config Methods (2bytes) + Primary Device Type (8bytes) */
  7415. /* + NumofSecondDevType (1byte) + WPS Device Name ID field (2bytes) + WPS Device Name Len field (2bytes) */
  7416. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(21 + pwdinfo->device_name_len);
  7417. p2pielen += 2;
  7418. /* Value: */
  7419. /* P2P Device Address */
  7420. _rtw_memcpy(p2pie + p2pielen, adapter_mac_addr(padapter), ETH_ALEN);
  7421. p2pielen += ETH_ALEN;
  7422. /* Config Method */
  7423. /* This field should be big endian. Noted by P2P specification. */
  7424. if ((pwdinfo->ui_got_wps_info == P2P_GOT_WPSINFO_PEER_DISPLAY_PIN) ||
  7425. (pwdinfo->ui_got_wps_info == P2P_GOT_WPSINFO_SELF_DISPLAY_PIN))
  7426. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_CONFIG_METHOD_DISPLAY);
  7427. else
  7428. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_CONFIG_METHOD_PBC);
  7429. p2pielen += 2;
  7430. /* Primary Device Type */
  7431. /* Category ID */
  7432. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_CID_MULIT_MEDIA);
  7433. p2pielen += 2;
  7434. /* OUI */
  7435. *(u32 *)(p2pie + p2pielen) = cpu_to_be32(WPSOUI);
  7436. p2pielen += 4;
  7437. /* Sub Category ID */
  7438. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_PDT_SCID_MEDIA_SERVER);
  7439. p2pielen += 2;
  7440. /* Number of Secondary Device Types */
  7441. p2pie[p2pielen++] = 0x00; /* No Secondary Device Type List */
  7442. /* Device Name */
  7443. /* Type: */
  7444. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(WPS_ATTR_DEVICE_NAME);
  7445. p2pielen += 2;
  7446. /* Length: */
  7447. *(u16 *)(p2pie + p2pielen) = cpu_to_be16(pwdinfo->device_name_len);
  7448. p2pielen += 2;
  7449. /* Value: */
  7450. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_name, pwdinfo->device_name_len);
  7451. p2pielen += pwdinfo->device_name_len;
  7452. /* P2P Interface */
  7453. /* Type: */
  7454. p2pie[p2pielen++] = P2P_ATTR_INTERFACE;
  7455. /* Length: */
  7456. *(u16 *)(p2pie + p2pielen) = cpu_to_le16(0x000D);
  7457. p2pielen += 2;
  7458. /* Value: */
  7459. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_addr, ETH_ALEN); /* P2P Device Address */
  7460. p2pielen += ETH_ALEN;
  7461. p2pie[p2pielen++] = 1; /* P2P Interface Address Count */
  7462. _rtw_memcpy(p2pie + p2pielen, pwdinfo->device_addr, ETH_ALEN); /* P2P Interface Address List */
  7463. p2pielen += ETH_ALEN;
  7464. pframe = rtw_set_ie(pframe, _VENDOR_SPECIFIC_IE_, p2pielen, (unsigned char *) p2pie, &pattrib->pktlen);
  7465. }
  7466. }
  7467. #endif /* CONFIG_P2P */
  7468. #ifdef CONFIG_WFD
  7469. wfdielen = rtw_append_assoc_req_wfd_ie(padapter, pframe);
  7470. pframe += wfdielen;
  7471. pattrib->pktlen += wfdielen;
  7472. #endif
  7473. #ifdef CONFIG_RTW_REPEATER_SON
  7474. rtw_rson_append_ie(padapter, pframe, &pattrib->pktlen);
  7475. #endif
  7476. #ifdef CONFIG_APPEND_VENDOR_IE_ENABLE
  7477. pattrib->pktlen += rtw_build_vendor_ie(padapter , pframe , WIFI_ASSOCREQ_VENDOR_IE_BIT);
  7478. #endif
  7479. #ifdef CONFIG_RTW_80211R
  7480. rtw_ft_build_assoc_req_ies(padapter, is_reassoc, pattrib, &pframe);
  7481. #endif
  7482. pattrib->last_txcmdsz = pattrib->pktlen;
  7483. dump_mgntframe(padapter, pmgntframe);
  7484. ret = _SUCCESS;
  7485. exit:
  7486. if (ret == _SUCCESS)
  7487. rtw_buf_update(&pmlmepriv->assoc_req, &pmlmepriv->assoc_req_len, (u8 *)pwlanhdr, pattrib->pktlen);
  7488. else
  7489. rtw_buf_free(&pmlmepriv->assoc_req, &pmlmepriv->assoc_req_len);
  7490. return;
  7491. }
  7492. void issue_assocreq(_adapter *padapter)
  7493. {
  7494. _issue_assocreq(padapter, _FALSE);
  7495. }
  7496. void issue_reassocreq(_adapter *padapter)
  7497. {
  7498. _issue_assocreq(padapter, _TRUE);
  7499. }
  7500. /* when wait_ack is ture, this function shoule be called at process context */
  7501. static int _issue_nulldata(_adapter *padapter, unsigned char *da, unsigned int power_mode, int wait_ack)
  7502. {
  7503. int ret = _FAIL;
  7504. struct xmit_frame *pmgntframe;
  7505. struct pkt_attrib *pattrib;
  7506. unsigned char *pframe;
  7507. struct rtw_ieee80211_hdr *pwlanhdr;
  7508. unsigned short *fctrl;
  7509. struct xmit_priv *pxmitpriv;
  7510. struct mlme_ext_priv *pmlmeext;
  7511. struct mlme_ext_info *pmlmeinfo;
  7512. u8 a4_shift;
  7513. /* RTW_INFO("%s:%d\n", __FUNCTION__, power_mode); */
  7514. if (!padapter)
  7515. goto exit;
  7516. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7517. goto exit;
  7518. pxmitpriv = &(padapter->xmitpriv);
  7519. pmlmeext = &(padapter->mlmeextpriv);
  7520. pmlmeinfo = &(pmlmeext->mlmext_info);
  7521. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7522. if (pmgntframe == NULL)
  7523. goto exit;
  7524. /* update attribute */
  7525. pattrib = &pmgntframe->attrib;
  7526. update_mgntframe_attrib(padapter, pattrib);
  7527. pattrib->retry_ctrl = _FALSE;
  7528. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7529. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7530. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7531. fctrl = &(pwlanhdr->frame_ctl);
  7532. *(fctrl) = 0;
  7533. if (MLME_IS_AP(padapter))
  7534. SetFrDs(fctrl);
  7535. else if (MLME_IS_STA(padapter))
  7536. SetToDs(fctrl);
  7537. else if (MLME_IS_MESH(padapter)) {
  7538. SetToDs(fctrl);
  7539. SetFrDs(fctrl);
  7540. }
  7541. if (power_mode)
  7542. SetPwrMgt(fctrl);
  7543. if (get_tofr_ds(fctrl) == 3) {
  7544. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  7545. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7546. _rtw_memcpy(pwlanhdr->addr3, da, ETH_ALEN);
  7547. _rtw_memcpy(pwlanhdr->addr4, adapter_mac_addr(padapter), ETH_ALEN);
  7548. a4_shift = ETH_ALEN;
  7549. pattrib->hdrlen += ETH_ALEN;
  7550. } else {
  7551. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  7552. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7553. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7554. a4_shift = 0;
  7555. }
  7556. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7557. pmlmeext->mgnt_seq++;
  7558. set_frame_sub_type(pframe, WIFI_DATA_NULL);
  7559. pframe += sizeof(struct rtw_ieee80211_hdr_3addr) + a4_shift;
  7560. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr) + a4_shift;
  7561. pattrib->last_txcmdsz = pattrib->pktlen;
  7562. if (wait_ack)
  7563. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  7564. else {
  7565. dump_mgntframe(padapter, pmgntframe);
  7566. ret = _SUCCESS;
  7567. }
  7568. exit:
  7569. return ret;
  7570. }
  7571. /*
  7572. * When wait_ms > 0, this function should be called at process context
  7573. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  7574. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  7575. * try_cnt means the maximal TX count to try
  7576. * da == NULL for station mode
  7577. */
  7578. int issue_nulldata(_adapter *padapter, unsigned char *da, unsigned int power_mode, int try_cnt, int wait_ms)
  7579. {
  7580. int ret = _FAIL;
  7581. int i = 0;
  7582. systime start = rtw_get_current_time();
  7583. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7584. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7585. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7586. goto exit;
  7587. /* da == NULL, assum it's null data for sta to ap */
  7588. if (da == NULL)
  7589. da = get_my_bssid(&(pmlmeinfo->network));
  7590. do {
  7591. ret = _issue_nulldata(padapter, da, power_mode, wait_ms > 0 ? _TRUE : _FALSE);
  7592. i++;
  7593. if (RTW_CANNOT_RUN(padapter))
  7594. break;
  7595. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  7596. rtw_msleep_os(wait_ms);
  7597. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  7598. if (ret != _FAIL) {
  7599. ret = _SUCCESS;
  7600. #ifndef DBG_XMIT_ACK
  7601. goto exit;
  7602. #endif
  7603. }
  7604. if (try_cnt && wait_ms) {
  7605. if (da)
  7606. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", ch:%u%s, %d/%d in %u ms\n",
  7607. FUNC_ADPT_ARG(padapter), MAC_ARG(da), rtw_get_oper_ch(padapter),
  7608. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7609. else
  7610. RTW_INFO(FUNC_ADPT_FMT", ch:%u%s, %d/%d in %u ms\n",
  7611. FUNC_ADPT_ARG(padapter), rtw_get_oper_ch(padapter),
  7612. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7613. }
  7614. exit:
  7615. return ret;
  7616. }
  7617. /* when wait_ack is ture, this function shoule be called at process context */
  7618. static int _issue_qos_nulldata(_adapter *padapter, unsigned char *da, u16 tid, u8 ps, int wait_ack)
  7619. {
  7620. int ret = _FAIL;
  7621. struct xmit_frame *pmgntframe;
  7622. struct pkt_attrib *pattrib;
  7623. unsigned char *pframe;
  7624. struct rtw_ieee80211_hdr *pwlanhdr;
  7625. unsigned short *fctrl, *qc;
  7626. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7627. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7628. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7629. u8 a4_shift;
  7630. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7631. goto exit;
  7632. /* RTW_INFO("%s\n", __FUNCTION__); */
  7633. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7634. if (pmgntframe == NULL)
  7635. goto exit;
  7636. /* update attribute */
  7637. pattrib = &pmgntframe->attrib;
  7638. update_mgntframe_attrib(padapter, pattrib);
  7639. pattrib->hdrlen += 2;
  7640. pattrib->qos_en = _TRUE;
  7641. pattrib->eosp = 1;
  7642. pattrib->ack_policy = 0;
  7643. pattrib->mdata = 0;
  7644. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7645. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7646. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7647. fctrl = &(pwlanhdr->frame_ctl);
  7648. *(fctrl) = 0;
  7649. if (MLME_IS_AP(padapter))
  7650. SetFrDs(fctrl);
  7651. else if (MLME_IS_STA(padapter))
  7652. SetToDs(fctrl);
  7653. else if (MLME_IS_MESH(padapter)) {
  7654. SetToDs(fctrl);
  7655. SetFrDs(fctrl);
  7656. }
  7657. if (ps)
  7658. SetPwrMgt(fctrl);
  7659. if (pattrib->mdata)
  7660. SetMData(fctrl);
  7661. if (get_tofr_ds(fctrl) == 3) {
  7662. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  7663. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7664. _rtw_memcpy(pwlanhdr->addr3, da, ETH_ALEN);
  7665. _rtw_memcpy(pwlanhdr->addr4, adapter_mac_addr(padapter), ETH_ALEN);
  7666. a4_shift = ETH_ALEN;
  7667. pattrib->hdrlen += ETH_ALEN;
  7668. } else {
  7669. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  7670. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7671. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7672. a4_shift = 0;
  7673. }
  7674. qc = (unsigned short *)(pframe + pattrib->hdrlen - 2);
  7675. SetPriority(qc, tid);
  7676. SetEOSP(qc, pattrib->eosp);
  7677. SetAckpolicy(qc, pattrib->ack_policy);
  7678. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7679. pmlmeext->mgnt_seq++;
  7680. set_frame_sub_type(pframe, WIFI_QOS_DATA_NULL);
  7681. pframe += sizeof(struct rtw_ieee80211_hdr_3addr_qos) + a4_shift;
  7682. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr_qos) + a4_shift;
  7683. pattrib->last_txcmdsz = pattrib->pktlen;
  7684. if (wait_ack)
  7685. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  7686. else {
  7687. dump_mgntframe(padapter, pmgntframe);
  7688. ret = _SUCCESS;
  7689. }
  7690. exit:
  7691. return ret;
  7692. }
  7693. /*
  7694. * when wait_ms >0 , this function should be called at process context
  7695. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  7696. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  7697. * try_cnt means the maximal TX count to try
  7698. * da == NULL for station mode
  7699. */
  7700. int issue_qos_nulldata(_adapter *padapter, unsigned char *da, u16 tid, u8 ps, int try_cnt, int wait_ms)
  7701. {
  7702. int ret = _FAIL;
  7703. int i = 0;
  7704. systime start = rtw_get_current_time();
  7705. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7706. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7707. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7708. goto exit;
  7709. /* da == NULL, assum it's null data for sta to ap*/
  7710. if (da == NULL)
  7711. da = get_my_bssid(&(pmlmeinfo->network));
  7712. do {
  7713. ret = _issue_qos_nulldata(padapter, da, tid, ps, wait_ms > 0 ? _TRUE : _FALSE);
  7714. i++;
  7715. if (RTW_CANNOT_RUN(padapter))
  7716. break;
  7717. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  7718. rtw_msleep_os(wait_ms);
  7719. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  7720. if (ret != _FAIL) {
  7721. ret = _SUCCESS;
  7722. #ifndef DBG_XMIT_ACK
  7723. goto exit;
  7724. #endif
  7725. }
  7726. if (try_cnt && wait_ms) {
  7727. if (da)
  7728. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", ch:%u%s, %d/%d in %u ms\n",
  7729. FUNC_ADPT_ARG(padapter), MAC_ARG(da), rtw_get_oper_ch(padapter),
  7730. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7731. else
  7732. RTW_INFO(FUNC_ADPT_FMT", ch:%u%s, %d/%d in %u ms\n",
  7733. FUNC_ADPT_ARG(padapter), rtw_get_oper_ch(padapter),
  7734. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7735. }
  7736. exit:
  7737. return ret;
  7738. }
  7739. static int _issue_deauth(_adapter *padapter, unsigned char *da, unsigned short reason, u8 wait_ack, u8 key_type)
  7740. {
  7741. struct xmit_frame *pmgntframe;
  7742. struct pkt_attrib *pattrib;
  7743. unsigned char *pframe;
  7744. struct rtw_ieee80211_hdr *pwlanhdr;
  7745. unsigned short *fctrl;
  7746. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7747. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7748. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7749. int ret = _FAIL;
  7750. #ifdef CONFIG_P2P
  7751. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  7752. #endif /* CONFIG_P2P */
  7753. /* RTW_INFO("%s to "MAC_FMT"\n", __func__, MAC_ARG(da)); */
  7754. #ifdef CONFIG_P2P
  7755. if (!(rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE)) && (pwdinfo->rx_invitereq_info.scan_op_ch_only)) {
  7756. _cancel_timer_ex(&pwdinfo->reset_ch_sitesurvey);
  7757. _set_timer(&pwdinfo->reset_ch_sitesurvey, 10);
  7758. }
  7759. #endif /* CONFIG_P2P */
  7760. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7761. goto exit;
  7762. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7763. if (pmgntframe == NULL)
  7764. goto exit;
  7765. /* update attribute */
  7766. pattrib = &pmgntframe->attrib;
  7767. update_mgntframe_attrib(padapter, pattrib);
  7768. pattrib->retry_ctrl = _FALSE;
  7769. pattrib->key_type = key_type;
  7770. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7771. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7772. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7773. fctrl = &(pwlanhdr->frame_ctl);
  7774. *(fctrl) = 0;
  7775. _rtw_memcpy(pwlanhdr->addr1, da, ETH_ALEN);
  7776. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7777. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7778. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7779. pmlmeext->mgnt_seq++;
  7780. set_frame_sub_type(pframe, WIFI_DEAUTH);
  7781. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  7782. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  7783. reason = cpu_to_le16(reason);
  7784. pframe = rtw_set_fixed_ie(pframe, _RSON_CODE_ , (unsigned char *)&reason, &(pattrib->pktlen));
  7785. pattrib->last_txcmdsz = pattrib->pktlen;
  7786. if (wait_ack)
  7787. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  7788. else {
  7789. dump_mgntframe(padapter, pmgntframe);
  7790. ret = _SUCCESS;
  7791. }
  7792. exit:
  7793. return ret;
  7794. }
  7795. int issue_deauth(_adapter *padapter, unsigned char *da, unsigned short reason)
  7796. {
  7797. RTW_INFO("%s to "MAC_FMT"\n", __func__, MAC_ARG(da));
  7798. return _issue_deauth(padapter, da, reason, _FALSE, IEEE80211W_RIGHT_KEY);
  7799. }
  7800. #ifdef CONFIG_IEEE80211W
  7801. int issue_deauth_11w(_adapter *padapter, unsigned char *da, unsigned short reason, u8 key_type)
  7802. {
  7803. RTW_INFO("%s to "MAC_FMT"\n", __func__, MAC_ARG(da));
  7804. return _issue_deauth(padapter, da, reason, _FALSE, key_type);
  7805. }
  7806. #endif /* CONFIG_IEEE80211W */
  7807. /*
  7808. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  7809. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  7810. * try_cnt means the maximal TX count to try
  7811. */
  7812. int issue_deauth_ex(_adapter *padapter, u8 *da, unsigned short reason, int try_cnt,
  7813. int wait_ms)
  7814. {
  7815. int ret = _FAIL;
  7816. int i = 0;
  7817. systime start = rtw_get_current_time();
  7818. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7819. goto exit;
  7820. do {
  7821. ret = _issue_deauth(padapter, da, reason, wait_ms > 0 ? _TRUE : _FALSE, IEEE80211W_RIGHT_KEY);
  7822. i++;
  7823. if (RTW_CANNOT_RUN(padapter))
  7824. break;
  7825. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  7826. rtw_msleep_os(wait_ms);
  7827. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  7828. if (ret != _FAIL) {
  7829. ret = _SUCCESS;
  7830. #ifndef DBG_XMIT_ACK
  7831. goto exit;
  7832. #endif
  7833. }
  7834. if (try_cnt && wait_ms) {
  7835. if (da)
  7836. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", ch:%u%s, %d/%d in %u ms\n",
  7837. FUNC_ADPT_ARG(padapter), MAC_ARG(da), rtw_get_oper_ch(padapter),
  7838. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7839. else
  7840. RTW_INFO(FUNC_ADPT_FMT", ch:%u%s, %d/%d in %u ms\n",
  7841. FUNC_ADPT_ARG(padapter), rtw_get_oper_ch(padapter),
  7842. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  7843. }
  7844. exit:
  7845. return ret;
  7846. }
  7847. void issue_action_spct_ch_switch(_adapter *padapter, u8 *ra, u8 new_ch, u8 ch_offset)
  7848. {
  7849. struct xmit_frame *pmgntframe;
  7850. struct pkt_attrib *pattrib;
  7851. unsigned char *pframe;
  7852. struct rtw_ieee80211_hdr *pwlanhdr;
  7853. unsigned short *fctrl;
  7854. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7855. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7856. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7857. return;
  7858. RTW_INFO(FUNC_NDEV_FMT" ra="MAC_FMT", ch:%u, offset:%u\n",
  7859. FUNC_NDEV_ARG(padapter->pnetdev), MAC_ARG(ra), new_ch, ch_offset);
  7860. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7861. if (pmgntframe == NULL)
  7862. return;
  7863. /* update attribute */
  7864. pattrib = &pmgntframe->attrib;
  7865. update_mgntframe_attrib(padapter, pattrib);
  7866. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7867. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7868. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7869. fctrl = &(pwlanhdr->frame_ctl);
  7870. *(fctrl) = 0;
  7871. _rtw_memcpy(pwlanhdr->addr1, ra, ETH_ALEN); /* RA */
  7872. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN); /* TA */
  7873. _rtw_memcpy(pwlanhdr->addr3, ra, ETH_ALEN); /* DA = RA */
  7874. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7875. pmlmeext->mgnt_seq++;
  7876. set_frame_sub_type(pframe, WIFI_ACTION);
  7877. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  7878. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  7879. /* category, action */
  7880. {
  7881. u8 category, action;
  7882. category = RTW_WLAN_CATEGORY_SPECTRUM_MGMT;
  7883. action = RTW_WLAN_ACTION_SPCT_CHL_SWITCH;
  7884. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  7885. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  7886. }
  7887. pframe = rtw_set_ie_ch_switch(pframe, &(pattrib->pktlen), 0, new_ch, 0);
  7888. pframe = rtw_set_ie_secondary_ch_offset(pframe, &(pattrib->pktlen),
  7889. hal_ch_offset_to_secondary_ch_offset(ch_offset));
  7890. pattrib->last_txcmdsz = pattrib->pktlen;
  7891. dump_mgntframe(padapter, pmgntframe);
  7892. }
  7893. #ifdef CONFIG_IEEE80211W
  7894. void issue_action_SA_Query(_adapter *padapter, unsigned char *raddr, unsigned char action, unsigned short tid, u8 key_type)
  7895. {
  7896. u8 category = RTW_WLAN_CATEGORY_SA_QUERY;
  7897. u16 reason_code;
  7898. struct xmit_frame *pmgntframe;
  7899. struct pkt_attrib *pattrib;
  7900. u8 *pframe;
  7901. struct rtw_ieee80211_hdr *pwlanhdr;
  7902. u16 *fctrl;
  7903. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7904. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7905. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7906. struct sta_info *psta;
  7907. struct sta_priv *pstapriv = &padapter->stapriv;
  7908. struct registry_priv *pregpriv = &padapter->registrypriv;
  7909. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  7910. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  7911. return;
  7912. RTW_INFO("%s, %04x\n", __FUNCTION__, tid);
  7913. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  7914. if (pmgntframe == NULL) {
  7915. RTW_INFO("%s: alloc_mgtxmitframe fail\n", __FUNCTION__);
  7916. return;
  7917. }
  7918. /* update attribute */
  7919. pattrib = &pmgntframe->attrib;
  7920. update_mgntframe_attrib(padapter, pattrib);
  7921. pattrib->key_type = key_type;
  7922. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  7923. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  7924. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  7925. fctrl = &(pwlanhdr->frame_ctl);
  7926. *(fctrl) = 0;
  7927. if (raddr)
  7928. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  7929. else
  7930. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7931. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  7932. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  7933. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  7934. pmlmeext->mgnt_seq++;
  7935. set_frame_sub_type(pframe, WIFI_ACTION);
  7936. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  7937. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  7938. pframe = rtw_set_fixed_ie(pframe, 1, &category, &pattrib->pktlen);
  7939. pframe = rtw_set_fixed_ie(pframe, 1, &action, &pattrib->pktlen);
  7940. switch (action) {
  7941. case 0: /* SA Query req */
  7942. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)&pmlmeext->sa_query_seq, &pattrib->pktlen);
  7943. pmlmeext->sa_query_seq++;
  7944. /* send sa query request to AP, AP should reply sa query response in 1 second */
  7945. if (pattrib->key_type == IEEE80211W_RIGHT_KEY) {
  7946. psta = rtw_get_stainfo(pstapriv, pwlanhdr->addr1);
  7947. if (psta != NULL) {
  7948. /* RTW_INFO("%s, %d, set dot11w_expire_timer\n", __func__, __LINE__); */
  7949. _set_timer(&psta->dot11w_expire_timer, 1000);
  7950. }
  7951. }
  7952. break;
  7953. case 1: /* SA Query rsp */
  7954. tid = cpu_to_le16(tid);
  7955. /* RTW_INFO("rtw_set_fixed_ie, %04x\n", tid); */
  7956. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)&tid, &pattrib->pktlen);
  7957. break;
  7958. default:
  7959. break;
  7960. }
  7961. pattrib->last_txcmdsz = pattrib->pktlen;
  7962. dump_mgntframe(padapter, pmgntframe);
  7963. }
  7964. #endif /* CONFIG_IEEE80211W */
  7965. /**
  7966. * issue_action_ba - internal function to TX Block Ack action frame
  7967. * @padapter: the adapter to TX
  7968. * @raddr: receiver address
  7969. * @action: Block Ack Action
  7970. * @tid: tid
  7971. * @size: the announced AMPDU buffer size. used by ADDBA_RESP
  7972. * @status: status/reason code. used by ADDBA_RESP, DELBA
  7973. * @initiator: if we are the initiator of AMPDU association. used by DELBA
  7974. * @wait_ack: used xmit ack
  7975. *
  7976. * Returns:
  7977. * _SUCCESS: No xmit ack is used or acked
  7978. * _FAIL: not acked when using xmit ack
  7979. */
  7980. static int issue_action_ba(_adapter *padapter, unsigned char *raddr, unsigned char action
  7981. , u8 tid, u8 size, u16 status, u8 initiator, int wait_ack)
  7982. {
  7983. int ret = _FAIL;
  7984. u8 category = RTW_WLAN_CATEGORY_BACK;
  7985. u16 start_seq;
  7986. u16 BA_para_set;
  7987. u16 BA_timeout_value;
  7988. u16 BA_starting_seqctrl;
  7989. struct xmit_frame *pmgntframe;
  7990. struct pkt_attrib *pattrib;
  7991. u8 *pframe;
  7992. struct rtw_ieee80211_hdr *pwlanhdr;
  7993. u16 *fctrl;
  7994. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  7995. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  7996. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  7997. struct sta_info *psta;
  7998. struct sta_priv *pstapriv = &padapter->stapriv;
  7999. struct registry_priv *pregpriv = &padapter->registrypriv;
  8000. #ifdef CONFIG_80211N_HT
  8001. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  8002. goto exit;
  8003. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  8004. if (pmgntframe == NULL)
  8005. goto exit;
  8006. /* update attribute */
  8007. pattrib = &pmgntframe->attrib;
  8008. update_mgntframe_attrib(padapter, pattrib);
  8009. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  8010. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  8011. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  8012. fctrl = &(pwlanhdr->frame_ctl);
  8013. *(fctrl) = 0;
  8014. /* _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN); */
  8015. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN);
  8016. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  8017. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  8018. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  8019. pmlmeext->mgnt_seq++;
  8020. set_frame_sub_type(pframe, WIFI_ACTION);
  8021. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  8022. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  8023. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  8024. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  8025. if (category == 3) {
  8026. switch (action) {
  8027. case RTW_WLAN_ACTION_ADDBA_REQ:
  8028. do {
  8029. pmlmeinfo->dialogToken++;
  8030. } while (pmlmeinfo->dialogToken == 0);
  8031. pframe = rtw_set_fixed_ie(pframe, 1, &(pmlmeinfo->dialogToken), &(pattrib->pktlen));
  8032. #if defined(CONFIG_RTL8188E) && defined(CONFIG_SDIO_HCI)
  8033. BA_para_set = (0x0802 | ((tid & 0xf) << 2)); /* immediate ack & 16 buffer size */
  8034. #else
  8035. BA_para_set = (0x1002 | ((tid & 0xf) << 2)); /* immediate ack & 64 buffer size */
  8036. #endif
  8037. #ifdef CONFIG_TX_AMSDU
  8038. if (padapter->tx_amsdu >= 1) /* TX AMSDU enabled */
  8039. BA_para_set |= BIT(0);
  8040. else /* TX AMSDU disabled */
  8041. BA_para_set &= ~BIT(0);
  8042. #endif
  8043. BA_para_set = cpu_to_le16(BA_para_set);
  8044. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(BA_para_set)), &(pattrib->pktlen));
  8045. /* BA_timeout_value = 0xffff; */ /* max: 65535 TUs(~ 65 ms) */
  8046. BA_timeout_value = 5000;/* ~ 5ms */
  8047. BA_timeout_value = cpu_to_le16(BA_timeout_value);
  8048. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(BA_timeout_value)), &(pattrib->pktlen));
  8049. /* if ((psta = rtw_get_stainfo(pstapriv, pmlmeinfo->network.MacAddress)) != NULL) */
  8050. psta = rtw_get_stainfo(pstapriv, raddr);
  8051. if (psta != NULL) {
  8052. start_seq = (psta->sta_xmitpriv.txseq_tid[tid & 0x07] & 0xfff) + 1;
  8053. RTW_INFO("BA_starting_seqctrl = %d for TID=%d\n", start_seq, tid & 0x07);
  8054. psta->BA_starting_seqctrl[tid & 0x07] = start_seq;
  8055. BA_starting_seqctrl = start_seq << 4;
  8056. }
  8057. BA_starting_seqctrl = cpu_to_le16(BA_starting_seqctrl);
  8058. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(BA_starting_seqctrl)), &(pattrib->pktlen));
  8059. break;
  8060. case RTW_WLAN_ACTION_ADDBA_RESP:
  8061. pframe = rtw_set_fixed_ie(pframe, 1, &(pmlmeinfo->ADDBA_req.dialog_token), &(pattrib->pktlen));
  8062. status = cpu_to_le16(status);
  8063. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&status), &(pattrib->pktlen));
  8064. BA_para_set = le16_to_cpu(pmlmeinfo->ADDBA_req.BA_para_set);
  8065. BA_para_set &= ~IEEE80211_ADDBA_PARAM_TID_MASK;
  8066. BA_para_set |= (tid << 2) & IEEE80211_ADDBA_PARAM_TID_MASK;
  8067. BA_para_set &= ~RTW_IEEE80211_ADDBA_PARAM_BUF_SIZE_MASK;
  8068. BA_para_set |= (size << 6) & RTW_IEEE80211_ADDBA_PARAM_BUF_SIZE_MASK;
  8069. if (!padapter->registrypriv.wifi_spec) {
  8070. if (pregpriv->rx_ampdu_amsdu == 0) /* disabled */
  8071. BA_para_set &= ~BIT(0);
  8072. else if (pregpriv->rx_ampdu_amsdu == 1) /* enabled */
  8073. BA_para_set |= BIT(0);
  8074. }
  8075. BA_para_set = cpu_to_le16(BA_para_set);
  8076. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(BA_para_set)), &(pattrib->pktlen));
  8077. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(pmlmeinfo->ADDBA_req.BA_timeout_value)), &(pattrib->pktlen));
  8078. break;
  8079. case RTW_WLAN_ACTION_DELBA:
  8080. BA_para_set = 0;
  8081. BA_para_set |= (tid << 12) & IEEE80211_DELBA_PARAM_TID_MASK;
  8082. BA_para_set |= (initiator << 11) & IEEE80211_DELBA_PARAM_INITIATOR_MASK;
  8083. BA_para_set = cpu_to_le16(BA_para_set);
  8084. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(BA_para_set)), &(pattrib->pktlen));
  8085. status = cpu_to_le16(status);
  8086. pframe = rtw_set_fixed_ie(pframe, 2, (unsigned char *)(&(status)), &(pattrib->pktlen));
  8087. break;
  8088. default:
  8089. break;
  8090. }
  8091. }
  8092. pattrib->last_txcmdsz = pattrib->pktlen;
  8093. if (wait_ack)
  8094. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  8095. else {
  8096. dump_mgntframe(padapter, pmgntframe);
  8097. ret = _SUCCESS;
  8098. }
  8099. exit:
  8100. #endif /* CONFIG_80211N_HT */
  8101. return ret;
  8102. }
  8103. /**
  8104. * issue_addba_req - TX ADDBA_REQ
  8105. * @adapter: the adapter to TX
  8106. * @ra: receiver address
  8107. * @tid: tid
  8108. */
  8109. inline void issue_addba_req(_adapter *adapter, unsigned char *ra, u8 tid)
  8110. {
  8111. issue_action_ba(adapter, ra, RTW_WLAN_ACTION_ADDBA_REQ
  8112. , tid
  8113. , 0 /* unused */
  8114. , 0 /* unused */
  8115. , 0 /* unused */
  8116. , _FALSE
  8117. );
  8118. RTW_INFO(FUNC_ADPT_FMT" ra="MAC_FMT" tid=%u\n"
  8119. , FUNC_ADPT_ARG(adapter), MAC_ARG(ra), tid);
  8120. }
  8121. /**
  8122. * issue_addba_rsp - TX ADDBA_RESP
  8123. * @adapter: the adapter to TX
  8124. * @ra: receiver address
  8125. * @tid: tid
  8126. * @status: status code
  8127. * @size: the announced AMPDU buffer size
  8128. */
  8129. inline void issue_addba_rsp(_adapter *adapter, unsigned char *ra, u8 tid, u16 status, u8 size)
  8130. {
  8131. issue_action_ba(adapter, ra, RTW_WLAN_ACTION_ADDBA_RESP
  8132. , tid
  8133. , size
  8134. , status
  8135. , 0 /* unused */
  8136. , _FALSE
  8137. );
  8138. RTW_INFO(FUNC_ADPT_FMT" ra="MAC_FMT" status=%u, tid=%u, size=%u\n"
  8139. , FUNC_ADPT_ARG(adapter), MAC_ARG(ra), status, tid, size);
  8140. }
  8141. /**
  8142. * issue_addba_rsp_wait_ack - TX ADDBA_RESP and wait ack
  8143. * @adapter: the adapter to TX
  8144. * @ra: receiver address
  8145. * @tid: tid
  8146. * @status: status code
  8147. * @size: the announced AMPDU buffer size
  8148. * @try_cnt: the maximal TX count to try
  8149. * @wait_ms: == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  8150. * > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  8151. */
  8152. inline u8 issue_addba_rsp_wait_ack(_adapter *adapter, unsigned char *ra, u8 tid, u16 status, u8 size, int try_cnt, int wait_ms)
  8153. {
  8154. int ret = _FAIL;
  8155. int i = 0;
  8156. systime start = rtw_get_current_time();
  8157. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(adapter)))
  8158. goto exit;
  8159. do {
  8160. ret = issue_action_ba(adapter, ra, RTW_WLAN_ACTION_ADDBA_RESP
  8161. , tid
  8162. , size
  8163. , status
  8164. , 0 /* unused */
  8165. , _TRUE
  8166. );
  8167. i++;
  8168. if (RTW_CANNOT_RUN(adapter))
  8169. break;
  8170. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  8171. rtw_msleep_os(wait_ms);
  8172. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  8173. if (ret != _FAIL) {
  8174. ret = _SUCCESS;
  8175. #ifndef DBG_XMIT_ACK
  8176. /* goto exit; */
  8177. #endif
  8178. }
  8179. if (try_cnt && wait_ms) {
  8180. RTW_INFO(FUNC_ADPT_FMT" ra="MAC_FMT" status:=%u tid=%u size:%u%s, %d/%d in %u ms\n"
  8181. , FUNC_ADPT_ARG(adapter), MAC_ARG(ra), status, tid, size
  8182. , ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  8183. }
  8184. exit:
  8185. return ret;
  8186. }
  8187. /**
  8188. * issue_del_ba - TX DELBA
  8189. * @adapter: the adapter to TX
  8190. * @ra: receiver address
  8191. * @tid: tid
  8192. * @reason: reason code
  8193. * @initiator: if we are the initiator of AMPDU association. used by DELBA
  8194. */
  8195. inline void issue_del_ba(_adapter *adapter, unsigned char *ra, u8 tid, u16 reason, u8 initiator)
  8196. {
  8197. issue_action_ba(adapter, ra, RTW_WLAN_ACTION_DELBA
  8198. , tid
  8199. , 0 /* unused */
  8200. , reason
  8201. , initiator
  8202. , _FALSE
  8203. );
  8204. RTW_INFO(FUNC_ADPT_FMT" ra="MAC_FMT" reason=%u, tid=%u, initiator=%u\n"
  8205. , FUNC_ADPT_ARG(adapter), MAC_ARG(ra), reason, tid, initiator);
  8206. }
  8207. /**
  8208. * issue_del_ba_ex - TX DELBA with xmit ack options
  8209. * @adapter: the adapter to TX
  8210. * @ra: receiver address
  8211. * @tid: tid
  8212. * @reason: reason code
  8213. * @initiator: if we are the initiator of AMPDU association. used by DELBA
  8214. * @try_cnt: the maximal TX count to try
  8215. * @wait_ms: == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  8216. * > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  8217. */
  8218. int issue_del_ba_ex(_adapter *adapter, unsigned char *ra, u8 tid, u16 reason, u8 initiator
  8219. , int try_cnt, int wait_ms)
  8220. {
  8221. int ret = _FAIL;
  8222. int i = 0;
  8223. systime start = rtw_get_current_time();
  8224. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(adapter)))
  8225. goto exit;
  8226. do {
  8227. ret = issue_action_ba(adapter, ra, RTW_WLAN_ACTION_DELBA
  8228. , tid
  8229. , 0 /* unused */
  8230. , reason
  8231. , initiator
  8232. , wait_ms > 0 ? _TRUE : _FALSE
  8233. );
  8234. i++;
  8235. if (RTW_CANNOT_RUN(adapter))
  8236. break;
  8237. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  8238. rtw_msleep_os(wait_ms);
  8239. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  8240. if (ret != _FAIL) {
  8241. ret = _SUCCESS;
  8242. #ifndef DBG_XMIT_ACK
  8243. /* goto exit; */
  8244. #endif
  8245. }
  8246. if (try_cnt && wait_ms) {
  8247. RTW_INFO(FUNC_ADPT_FMT" ra="MAC_FMT" reason=%u, tid=%u, initiator=%u%s, %d/%d in %u ms\n"
  8248. , FUNC_ADPT_ARG(adapter), MAC_ARG(ra), reason, tid, initiator
  8249. , ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  8250. }
  8251. exit:
  8252. return ret;
  8253. }
  8254. void issue_action_BSSCoexistPacket(_adapter *padapter)
  8255. {
  8256. _irqL irqL;
  8257. _list *plist, *phead;
  8258. unsigned char category, action;
  8259. struct xmit_frame *pmgntframe;
  8260. struct pkt_attrib *pattrib;
  8261. unsigned char *pframe;
  8262. struct rtw_ieee80211_hdr *pwlanhdr;
  8263. unsigned short *fctrl;
  8264. struct wlan_network *pnetwork = NULL;
  8265. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  8266. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  8267. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  8268. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8269. _queue *queue = &(pmlmepriv->scanned_queue);
  8270. u8 InfoContent[16] = {0};
  8271. u8 ICS[8][15];
  8272. #ifdef CONFIG_80211N_HT
  8273. if ((pmlmepriv->num_FortyMHzIntolerant == 0) || (pmlmepriv->num_sta_no_ht == 0))
  8274. return;
  8275. if (_TRUE == pmlmeinfo->bwmode_updated)
  8276. return;
  8277. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  8278. return;
  8279. RTW_INFO("%s\n", __FUNCTION__);
  8280. category = RTW_WLAN_CATEGORY_PUBLIC;
  8281. action = ACT_PUBLIC_BSSCOEXIST;
  8282. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  8283. if (pmgntframe == NULL)
  8284. return;
  8285. /* update attribute */
  8286. pattrib = &pmgntframe->attrib;
  8287. update_mgntframe_attrib(padapter, pattrib);
  8288. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  8289. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  8290. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  8291. fctrl = &(pwlanhdr->frame_ctl);
  8292. *(fctrl) = 0;
  8293. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  8294. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  8295. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
  8296. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  8297. pmlmeext->mgnt_seq++;
  8298. set_frame_sub_type(pframe, WIFI_ACTION);
  8299. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  8300. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  8301. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  8302. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  8303. /* */
  8304. if (pmlmepriv->num_FortyMHzIntolerant > 0) {
  8305. u8 iedata = 0;
  8306. iedata |= BIT(2);/* 20 MHz BSS Width Request */
  8307. pframe = rtw_set_ie(pframe, EID_BSSCoexistence, 1, &iedata, &(pattrib->pktlen));
  8308. }
  8309. /* */
  8310. _rtw_memset(ICS, 0, sizeof(ICS));
  8311. if (pmlmepriv->num_sta_no_ht > 0) {
  8312. int i;
  8313. _enter_critical_bh(&(pmlmepriv->scanned_queue.lock), &irqL);
  8314. phead = get_list_head(queue);
  8315. plist = get_next(phead);
  8316. while (1) {
  8317. int len;
  8318. u8 *p;
  8319. WLAN_BSSID_EX *pbss_network;
  8320. if (rtw_end_of_queue_search(phead, plist) == _TRUE)
  8321. break;
  8322. pnetwork = LIST_CONTAINOR(plist, struct wlan_network, list);
  8323. plist = get_next(plist);
  8324. pbss_network = (WLAN_BSSID_EX *)&pnetwork->network;
  8325. p = rtw_get_ie(pbss_network->IEs + _FIXED_IE_LENGTH_, _HT_CAPABILITY_IE_, &len, pbss_network->IELength - _FIXED_IE_LENGTH_);
  8326. if ((p == NULL) || (len == 0)) { /* non-HT */
  8327. if ((pbss_network->Configuration.DSConfig <= 0) || (pbss_network->Configuration.DSConfig > 14))
  8328. continue;
  8329. ICS[0][pbss_network->Configuration.DSConfig] = 1;
  8330. if (ICS[0][0] == 0)
  8331. ICS[0][0] = 1;
  8332. }
  8333. }
  8334. _exit_critical_bh(&(pmlmepriv->scanned_queue.lock), &irqL);
  8335. for (i = 0; i < 8; i++) {
  8336. if (ICS[i][0] == 1) {
  8337. int j, k = 0;
  8338. InfoContent[k] = i;
  8339. /* SET_BSS_INTOLERANT_ELE_REG_CLASS(InfoContent,i); */
  8340. k++;
  8341. for (j = 1; j <= 14; j++) {
  8342. if (ICS[i][j] == 1) {
  8343. if (k < 16) {
  8344. InfoContent[k] = j; /* channel number */
  8345. /* SET_BSS_INTOLERANT_ELE_CHANNEL(InfoContent+k, j); */
  8346. k++;
  8347. }
  8348. }
  8349. }
  8350. pframe = rtw_set_ie(pframe, EID_BSSIntolerantChlReport, k, InfoContent, &(pattrib->pktlen));
  8351. }
  8352. }
  8353. }
  8354. pattrib->last_txcmdsz = pattrib->pktlen;
  8355. dump_mgntframe(padapter, pmgntframe);
  8356. #endif /* CONFIG_80211N_HT */
  8357. }
  8358. /* Spatial Multiplexing Powersave (SMPS) action frame */
  8359. int _issue_action_SM_PS(_adapter *padapter , unsigned char *raddr , u8 NewMimoPsMode , u8 wait_ack)
  8360. {
  8361. int ret = _FAIL;
  8362. unsigned char category = RTW_WLAN_CATEGORY_HT;
  8363. u8 action = RTW_WLAN_ACTION_HT_SM_PS;
  8364. u8 sm_power_control = 0;
  8365. struct xmit_frame *pmgntframe;
  8366. struct pkt_attrib *pattrib;
  8367. unsigned char *pframe;
  8368. struct rtw_ieee80211_hdr *pwlanhdr;
  8369. unsigned short *fctrl;
  8370. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  8371. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  8372. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8373. if (NewMimoPsMode == WLAN_HT_CAP_SM_PS_DISABLED) {
  8374. sm_power_control = sm_power_control & ~(BIT(0)); /* SM Power Save Enable = 0 SM Power Save Disable */
  8375. } else if (NewMimoPsMode == WLAN_HT_CAP_SM_PS_STATIC) {
  8376. sm_power_control = sm_power_control | BIT(0); /* SM Power Save Enable = 1 SM Power Save Enable */
  8377. sm_power_control = sm_power_control & ~(BIT(1)); /* SM Mode = 0 Static Mode */
  8378. } else if (NewMimoPsMode == WLAN_HT_CAP_SM_PS_DYNAMIC) {
  8379. sm_power_control = sm_power_control | BIT(0); /* SM Power Save Enable = 1 SM Power Save Enable */
  8380. sm_power_control = sm_power_control | BIT(1); /* SM Mode = 1 Dynamic Mode */
  8381. } else
  8382. return ret;
  8383. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  8384. return ret;
  8385. RTW_INFO("%s, sm_power_control=%u, NewMimoPsMode=%u\n", __FUNCTION__ , sm_power_control , NewMimoPsMode);
  8386. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  8387. if (pmgntframe == NULL)
  8388. return ret;
  8389. /* update attribute */
  8390. pattrib = &pmgntframe->attrib;
  8391. update_mgntframe_attrib(padapter, pattrib);
  8392. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  8393. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  8394. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  8395. fctrl = &(pwlanhdr->frame_ctl);
  8396. *(fctrl) = 0;
  8397. _rtw_memcpy(pwlanhdr->addr1, raddr, ETH_ALEN); /* RA */
  8398. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN); /* TA */
  8399. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN); /* DA = RA */
  8400. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  8401. pmlmeext->mgnt_seq++;
  8402. set_frame_sub_type(pframe, WIFI_ACTION);
  8403. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  8404. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  8405. /* category, action */
  8406. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  8407. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  8408. pframe = rtw_set_fixed_ie(pframe, 1, &(sm_power_control), &(pattrib->pktlen));
  8409. pattrib->last_txcmdsz = pattrib->pktlen;
  8410. if (wait_ack)
  8411. ret = dump_mgntframe_and_wait_ack(padapter, pmgntframe);
  8412. else {
  8413. dump_mgntframe(padapter, pmgntframe);
  8414. ret = _SUCCESS;
  8415. }
  8416. if (ret != _SUCCESS)
  8417. RTW_INFO("%s, ack to\n", __func__);
  8418. return ret;
  8419. }
  8420. /*
  8421. * wait_ms == 0 means that there is no need to wait ack through C2H_CCX_TX_RPT
  8422. * wait_ms > 0 means you want to wait ack through C2H_CCX_TX_RPT, and the value of wait_ms means the interval between each TX
  8423. * try_cnt means the maximal TX count to try
  8424. */
  8425. int issue_action_SM_PS_wait_ack(_adapter *padapter, unsigned char *raddr, u8 NewMimoPsMode, int try_cnt, int wait_ms)
  8426. {
  8427. int ret = _FAIL;
  8428. int i = 0;
  8429. systime start = rtw_get_current_time();
  8430. if (rtw_rfctl_is_tx_blocked_by_ch_waiting(adapter_to_rfctl(padapter)))
  8431. goto exit;
  8432. do {
  8433. ret = _issue_action_SM_PS(padapter, raddr, NewMimoPsMode , wait_ms > 0 ? _TRUE : _FALSE);
  8434. i++;
  8435. if (RTW_CANNOT_RUN(padapter))
  8436. break;
  8437. if (i < try_cnt && wait_ms > 0 && ret == _FAIL)
  8438. rtw_msleep_os(wait_ms);
  8439. } while ((i < try_cnt) && ((ret == _FAIL) || (wait_ms == 0)));
  8440. if (ret != _FAIL) {
  8441. ret = _SUCCESS;
  8442. #ifndef DBG_XMIT_ACK
  8443. goto exit;
  8444. #endif
  8445. }
  8446. if (try_cnt && wait_ms) {
  8447. if (raddr)
  8448. RTW_INFO(FUNC_ADPT_FMT" to "MAC_FMT", %s , %d/%d in %u ms\n",
  8449. FUNC_ADPT_ARG(padapter), MAC_ARG(raddr),
  8450. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  8451. else
  8452. RTW_INFO(FUNC_ADPT_FMT", %s , %d/%d in %u ms\n",
  8453. FUNC_ADPT_ARG(padapter),
  8454. ret == _SUCCESS ? ", acked" : "", i, try_cnt, rtw_get_passing_time_ms(start));
  8455. }
  8456. exit:
  8457. return ret;
  8458. }
  8459. int issue_action_SM_PS(_adapter *padapter , unsigned char *raddr , u8 NewMimoPsMode)
  8460. {
  8461. RTW_INFO("%s to "MAC_FMT"\n", __func__, MAC_ARG(raddr));
  8462. return _issue_action_SM_PS(padapter, raddr, NewMimoPsMode , _FALSE);
  8463. }
  8464. /**
  8465. * _send_delba_sta_tid - Cancel the AMPDU association for the specific @sta, @tid
  8466. * @adapter: the adapter to which @sta belongs
  8467. * @initiator: if we are the initiator of AMPDU association
  8468. * @sta: the sta to be checked
  8469. * @tid: the tid to be checked
  8470. * @force: cancel and send DELBA even when no AMPDU association is setup
  8471. * @wait_ack: send delba with xmit ack (valid when initiator == 0)
  8472. *
  8473. * Returns:
  8474. * _FAIL if sta is NULL
  8475. * when initiator is 1, always _SUCCESS
  8476. * when initiator is 0, _SUCCESS if DELBA is acked
  8477. */
  8478. static unsigned int _send_delba_sta_tid(_adapter *adapter, u8 initiator, struct sta_info *sta, u8 tid
  8479. , u8 force, int wait_ack)
  8480. {
  8481. int ret = _SUCCESS;
  8482. if (sta == NULL) {
  8483. ret = _FAIL;
  8484. goto exit;
  8485. }
  8486. if (initiator == 0) {
  8487. /* recipient */
  8488. if (force || sta->recvreorder_ctrl[tid].enable == _TRUE) {
  8489. u8 ampdu_size_bak = sta->recvreorder_ctrl[tid].ampdu_size;
  8490. sta->recvreorder_ctrl[tid].enable = _FALSE;
  8491. sta->recvreorder_ctrl[tid].ampdu_size = RX_AMPDU_SIZE_INVALID;
  8492. if (rtw_del_rx_ampdu_test_trigger_no_tx_fail())
  8493. ret = _FAIL;
  8494. else if (wait_ack)
  8495. ret = issue_del_ba_ex(adapter, sta->cmn.mac_addr, tid, 37, initiator, 3, 1);
  8496. else
  8497. issue_del_ba(adapter, sta->cmn.mac_addr, tid, 37, initiator);
  8498. if (ret == _FAIL && sta->recvreorder_ctrl[tid].enable == _FALSE)
  8499. sta->recvreorder_ctrl[tid].ampdu_size = ampdu_size_bak;
  8500. }
  8501. } else if (initiator == 1) {
  8502. /* originator */
  8503. #ifdef CONFIG_80211N_HT
  8504. if (force || sta->htpriv.agg_enable_bitmap & BIT(tid)) {
  8505. sta->htpriv.agg_enable_bitmap &= ~BIT(tid);
  8506. sta->htpriv.candidate_tid_bitmap &= ~BIT(tid);
  8507. issue_del_ba(adapter, sta->cmn.mac_addr, tid, 37, initiator);
  8508. }
  8509. #endif
  8510. }
  8511. exit:
  8512. return ret;
  8513. }
  8514. inline unsigned int send_delba_sta_tid(_adapter *adapter, u8 initiator, struct sta_info *sta, u8 tid
  8515. , u8 force)
  8516. {
  8517. return _send_delba_sta_tid(adapter, initiator, sta, tid, force, 0);
  8518. }
  8519. inline unsigned int send_delba_sta_tid_wait_ack(_adapter *adapter, u8 initiator, struct sta_info *sta, u8 tid
  8520. , u8 force)
  8521. {
  8522. return _send_delba_sta_tid(adapter, initiator, sta, tid, force, 1);
  8523. }
  8524. unsigned int send_delba(_adapter *padapter, u8 initiator, u8 *addr)
  8525. {
  8526. struct sta_priv *pstapriv = &padapter->stapriv;
  8527. struct sta_info *psta = NULL;
  8528. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  8529. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8530. u16 tid;
  8531. if ((pmlmeinfo->state & 0x03) != WIFI_FW_AP_STATE)
  8532. if (!(pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS))
  8533. return _SUCCESS;
  8534. psta = rtw_get_stainfo(pstapriv, addr);
  8535. if (psta == NULL)
  8536. return _SUCCESS;
  8537. #if 0
  8538. RTW_INFO("%s:%s\n", __func__, (initiator == 0) ? "RX_DIR" : "TX_DIR");
  8539. if (initiator == 1) /* originator */
  8540. RTW_INFO("tx agg_enable_bitmap(0x%08x)\n", psta->htpriv.agg_enable_bitmap);
  8541. #endif
  8542. for (tid = 0; tid < TID_NUM; tid++)
  8543. send_delba_sta_tid(padapter, initiator, psta, tid, 0);
  8544. return _SUCCESS;
  8545. }
  8546. unsigned int send_beacon(_adapter *padapter)
  8547. {
  8548. #ifdef CONFIG_PCI_HCI
  8549. #ifdef CONFIG_FW_HANDLE_TXBCN
  8550. u8 vap_id = padapter->vap_id;
  8551. /* bypass TX BCN because vap_id is invalid*/
  8552. if (vap_id == CONFIG_LIMITED_AP_NUM)
  8553. return _SUCCESS;
  8554. #endif
  8555. /* bypass TX BCN queue because op ch is switching/waiting */
  8556. if (check_fwstate(&padapter->mlmepriv, WIFI_OP_CH_SWITCHING)
  8557. || IS_CH_WAITING(adapter_to_rfctl(padapter))
  8558. )
  8559. return _SUCCESS;
  8560. /* RTW_INFO("%s\n", __FUNCTION__); */
  8561. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_VALID, NULL);
  8562. /* 8192EE Port select for Beacon DL */
  8563. rtw_hal_set_hwreg(padapter, HW_VAR_DL_BCN_SEL, NULL);
  8564. #ifdef CONFIG_FW_HANDLE_TXBCN
  8565. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_HEAD_SEL, &vap_id);
  8566. #endif
  8567. issue_beacon(padapter, 0);
  8568. #ifdef CONFIG_FW_HANDLE_TXBCN
  8569. vap_id = 0xFF;
  8570. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_HEAD_SEL, &vap_id);
  8571. #endif
  8572. #ifdef RTL8814AE_SW_BCN
  8573. if (GET_HAL_DATA(padapter)->bCorrectBCN != 0)
  8574. RTW_INFO("%s, line%d, Warnning, pHalData->bCorrectBCN != 0\n", __func__, __LINE__);
  8575. GET_HAL_DATA(padapter)->bCorrectBCN = 1;
  8576. #endif
  8577. return _SUCCESS;
  8578. #endif /*CONFIG_PCI_HCI*/
  8579. #if defined(CONFIG_USB_HCI) || defined(CONFIG_SDIO_HCI) || defined(CONFIG_GSPI_HCI)
  8580. u8 bxmitok = _FALSE;
  8581. int issue = 0;
  8582. int poll = 0;
  8583. systime start = rtw_get_current_time();
  8584. #ifdef CONFIG_FW_HANDLE_TXBCN
  8585. u8 vap_id = padapter->vap_id;
  8586. /* bypass TX BCN because vap_id is invalid*/
  8587. if (vap_id == CONFIG_LIMITED_AP_NUM)
  8588. return _SUCCESS;
  8589. #endif
  8590. /* bypass TX BCN queue because op ch is switching/waiting */
  8591. if (check_fwstate(&padapter->mlmepriv, WIFI_OP_CH_SWITCHING)
  8592. || IS_CH_WAITING(adapter_to_rfctl(padapter))
  8593. )
  8594. return _SUCCESS;
  8595. #if defined(CONFIG_USB_HCI)
  8596. #if defined(CONFIG_RTL8812A)
  8597. if (IS_FULL_SPEED_USB(padapter)) {
  8598. issue_beacon(padapter, 300);
  8599. bxmitok = _TRUE;
  8600. } else
  8601. #endif
  8602. #endif
  8603. {
  8604. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_VALID, NULL);
  8605. rtw_hal_set_hwreg(padapter, HW_VAR_DL_BCN_SEL, NULL);
  8606. #ifdef CONFIG_FW_HANDLE_TXBCN
  8607. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_HEAD_SEL, &vap_id);
  8608. #endif
  8609. do {
  8610. issue_beacon(padapter, 100);
  8611. issue++;
  8612. do {
  8613. rtw_yield_os();
  8614. rtw_hal_get_hwreg(padapter, HW_VAR_BCN_VALID, (u8 *)(&bxmitok));
  8615. poll++;
  8616. } while ((poll % 10) != 0 && _FALSE == bxmitok && !RTW_CANNOT_RUN(padapter));
  8617. } while (bxmitok == _FALSE && (issue < 100) && !RTW_CANNOT_RUN(padapter));
  8618. #ifdef CONFIG_FW_HANDLE_TXBCN
  8619. vap_id = 0xFF;
  8620. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_HEAD_SEL, &vap_id);
  8621. #endif
  8622. }
  8623. if (RTW_CANNOT_RUN(padapter))
  8624. return _FAIL;
  8625. if (_FALSE == bxmitok) {
  8626. RTW_INFO("%s fail! %u ms\n", __FUNCTION__, rtw_get_passing_time_ms(start));
  8627. #ifdef CONFIG_BCN_RECOVERY
  8628. GET_HAL_DATA(padapter)->issue_bcn_fail++;
  8629. #endif /*CONFIG_BCN_RECOVERY*/
  8630. return _FAIL;
  8631. } else {
  8632. u32 passing_time = rtw_get_passing_time_ms(start);
  8633. if (passing_time > 100 || issue > 3)
  8634. RTW_INFO("%s success, issue:%d, poll:%d, %u ms\n", __FUNCTION__, issue, poll, rtw_get_passing_time_ms(start));
  8635. else if (0)
  8636. RTW_INFO("%s success, issue:%d, poll:%d, %u ms\n", __FUNCTION__, issue, poll, rtw_get_passing_time_ms(start));
  8637. #ifdef CONFIG_FW_CORRECT_BCN
  8638. rtw_hal_fw_correct_bcn(padapter);
  8639. #endif
  8640. return _SUCCESS;
  8641. }
  8642. #endif /*defined(CONFIG_USB_HCI) || defined(CONFIG_SDIO_HCI) || defined(CONFIG_GSPI_HCI)*/
  8643. }
  8644. /****************************************************************************
  8645. Following are some utitity fuctions for WiFi MLME
  8646. *****************************************************************************/
  8647. BOOLEAN IsLegal5GChannel(
  8648. IN PADAPTER Adapter,
  8649. IN u8 channel)
  8650. {
  8651. int i = 0;
  8652. u8 Channel_5G[45] = {36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58,
  8653. 60, 62, 64, 100, 102, 104, 106, 108, 110, 112, 114, 116, 118, 120, 122,
  8654. 124, 126, 128, 130, 132, 134, 136, 138, 140, 149, 151, 153, 155, 157, 159,
  8655. 161, 163, 165
  8656. };
  8657. for (i = 0; i < sizeof(Channel_5G); i++)
  8658. if (channel == Channel_5G[i])
  8659. return _TRUE;
  8660. return _FALSE;
  8661. }
  8662. /* collect bss info from Beacon and Probe request/response frames. */
  8663. u8 collect_bss_info(_adapter *padapter, union recv_frame *precv_frame, WLAN_BSSID_EX *bssid)
  8664. {
  8665. int i;
  8666. sint len;
  8667. u8 *p;
  8668. u8 rf_path;
  8669. u16 val16, subtype;
  8670. u8 *pframe = precv_frame->u.hdr.rx_data;
  8671. u32 packet_len = precv_frame->u.hdr.len;
  8672. u8 ie_offset;
  8673. HAL_DATA_TYPE *pHalData = GET_HAL_DATA(padapter);
  8674. struct registry_priv *pregistrypriv = &padapter->registrypriv;
  8675. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  8676. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8677. len = packet_len - sizeof(struct rtw_ieee80211_hdr_3addr);
  8678. if (len > MAX_IE_SZ) {
  8679. /* RTW_INFO("IE too long for survey event\n"); */
  8680. return _FAIL;
  8681. }
  8682. _rtw_memset(bssid, 0, sizeof(WLAN_BSSID_EX));
  8683. subtype = get_frame_sub_type(pframe);
  8684. if (subtype == WIFI_BEACON) {
  8685. bssid->Reserved[0] = BSS_TYPE_BCN;
  8686. ie_offset = _BEACON_IE_OFFSET_;
  8687. } else {
  8688. /* FIXME : more type */
  8689. if (subtype == WIFI_PROBERSP) {
  8690. ie_offset = _PROBERSP_IE_OFFSET_;
  8691. bssid->Reserved[0] = BSS_TYPE_PROB_RSP;
  8692. } else if (subtype == WIFI_PROBEREQ) {
  8693. ie_offset = _PROBEREQ_IE_OFFSET_;
  8694. bssid->Reserved[0] = BSS_TYPE_PROB_REQ;
  8695. } else {
  8696. bssid->Reserved[0] = BSS_TYPE_UNDEF;
  8697. ie_offset = _FIXED_IE_LENGTH_;
  8698. }
  8699. }
  8700. bssid->Length = sizeof(WLAN_BSSID_EX) - MAX_IE_SZ + len;
  8701. /* below is to copy the information element */
  8702. bssid->IELength = len;
  8703. _rtw_memcpy(bssid->IEs, (pframe + sizeof(struct rtw_ieee80211_hdr_3addr)), bssid->IELength);
  8704. /* get the signal strength */
  8705. /* bssid->Rssi = precv_frame->u.hdr.attrib.SignalStrength; */ /* 0-100 index. */
  8706. bssid->Rssi = precv_frame->u.hdr.attrib.phy_info.recv_signal_power; /* in dBM.raw data */
  8707. bssid->PhyInfo.SignalQuality = precv_frame->u.hdr.attrib.phy_info.signal_quality;/* in percentage */
  8708. bssid->PhyInfo.SignalStrength = precv_frame->u.hdr.attrib.phy_info.signal_strength;/* in percentage */
  8709. /* get rx_snr */
  8710. if (precv_frame->u.hdr.attrib.data_rate >= DESC_RATE11M) {
  8711. bssid->PhyInfo.is_cck_rate = 0;
  8712. for (rf_path = 0; rf_path < pHalData->NumTotalRFPath; rf_path++)
  8713. bssid->PhyInfo.rx_snr[rf_path] =
  8714. precv_frame->u.hdr.attrib.phy_info.rx_snr[rf_path];
  8715. } else
  8716. bssid->PhyInfo.is_cck_rate = 1;
  8717. #ifdef CONFIG_ANTENNA_DIVERSITY
  8718. rtw_hal_get_odm_var(padapter, HAL_ODM_ANTDIV_SELECT, &(bssid->PhyInfo.Optimum_antenna), NULL);
  8719. #endif
  8720. /* checking SSID */
  8721. p = rtw_get_ie(bssid->IEs + ie_offset, _SSID_IE_, &len, bssid->IELength - ie_offset);
  8722. if (p == NULL) {
  8723. RTW_INFO("marc: cannot find SSID for survey event\n");
  8724. return _FAIL;
  8725. }
  8726. if (*(p + 1)) {
  8727. if (len > NDIS_802_11_LENGTH_SSID) {
  8728. RTW_INFO("%s()-%d: IE too long (%d) for survey event\n", __FUNCTION__, __LINE__, len);
  8729. return _FAIL;
  8730. }
  8731. _rtw_memcpy(bssid->Ssid.Ssid, (p + 2), *(p + 1));
  8732. bssid->Ssid.SsidLength = *(p + 1);
  8733. } else
  8734. bssid->Ssid.SsidLength = 0;
  8735. _rtw_memset(bssid->SupportedRates, 0, NDIS_802_11_LENGTH_RATES_EX);
  8736. /* checking rate info... */
  8737. i = 0;
  8738. p = rtw_get_ie(bssid->IEs + ie_offset, _SUPPORTEDRATES_IE_, &len, bssid->IELength - ie_offset);
  8739. if (p != NULL) {
  8740. if (len > NDIS_802_11_LENGTH_RATES_EX) {
  8741. RTW_INFO("%s()-%d: IE too long (%d) for survey event\n", __FUNCTION__, __LINE__, len);
  8742. return _FAIL;
  8743. }
  8744. if (rtw_validate_value(_SUPPORTEDRATES_IE_, p+2, len) == _FALSE) {
  8745. rtw_absorb_ssid_ifneed(padapter, bssid, pframe);
  8746. RTW_DBG_DUMP("Invalidated Support Rate IE --", p, len+2);
  8747. return _FAIL;
  8748. }
  8749. _rtw_memcpy(bssid->SupportedRates, (p + 2), len);
  8750. i = len;
  8751. }
  8752. p = rtw_get_ie(bssid->IEs + ie_offset, _EXT_SUPPORTEDRATES_IE_, &len, bssid->IELength - ie_offset);
  8753. if (p != NULL) {
  8754. if (len > (NDIS_802_11_LENGTH_RATES_EX - i)) {
  8755. RTW_INFO("%s()-%d: IE too long (%d) for survey event\n", __FUNCTION__, __LINE__, len);
  8756. return _FAIL;
  8757. }
  8758. if (rtw_validate_value(_EXT_SUPPORTEDRATES_IE_, p+2, len) == _FALSE) {
  8759. rtw_absorb_ssid_ifneed(padapter, bssid, pframe);
  8760. RTW_DBG_DUMP("Invalidated EXT Support Rate IE --", p, len+2);
  8761. return _FAIL;
  8762. }
  8763. _rtw_memcpy(bssid->SupportedRates + i, (p + 2), len);
  8764. }
  8765. /* todo: */
  8766. #if 0
  8767. if (judge_network_type(bssid->SupportedRates, (len + i)) == WIRELESS_11B)
  8768. bssid->NetworkTypeInUse = Ndis802_11DS;
  8769. else
  8770. #endif
  8771. {
  8772. bssid->NetworkTypeInUse = Ndis802_11OFDM24;
  8773. }
  8774. #ifdef CONFIG_P2P
  8775. if (subtype == WIFI_PROBEREQ) {
  8776. u8 *p2p_ie;
  8777. u32 p2p_ielen;
  8778. /* Set Listion Channel */
  8779. p2p_ie = rtw_get_p2p_ie(bssid->IEs, bssid->IELength, NULL, &p2p_ielen);
  8780. if (p2p_ie) {
  8781. u32 attr_contentlen = 0;
  8782. u8 listen_ch[5] = { 0x00 };
  8783. rtw_get_p2p_attr_content(p2p_ie, p2p_ielen, P2P_ATTR_LISTEN_CH, listen_ch, &attr_contentlen);
  8784. bssid->Configuration.DSConfig = listen_ch[4];
  8785. } else {
  8786. /* use current channel */
  8787. bssid->Configuration.DSConfig = padapter->mlmeextpriv.cur_channel;
  8788. RTW_INFO("%s()-%d: Cannot get p2p_ie. set DSconfig to op_ch(%d)\n", __FUNCTION__, __LINE__, bssid->Configuration.DSConfig);
  8789. }
  8790. /* FIXME */
  8791. bssid->InfrastructureMode = Ndis802_11Infrastructure;
  8792. _rtw_memcpy(bssid->MacAddress, get_addr2_ptr(pframe), ETH_ALEN);
  8793. bssid->Privacy = 1;
  8794. return _SUCCESS;
  8795. }
  8796. #endif /* CONFIG_P2P */
  8797. if (bssid->IELength < 12)
  8798. return _FAIL;
  8799. /* Checking for DSConfig */
  8800. p = rtw_get_ie(bssid->IEs + ie_offset, _DSSET_IE_, &len, bssid->IELength - ie_offset);
  8801. bssid->Configuration.DSConfig = 0;
  8802. bssid->Configuration.Length = 0;
  8803. if (p)
  8804. bssid->Configuration.DSConfig = *(p + 2);
  8805. else {
  8806. /* In 5G, some ap do not have DSSET IE */
  8807. /* checking HT info for channel */
  8808. p = rtw_get_ie(bssid->IEs + ie_offset, _HT_ADD_INFO_IE_, &len, bssid->IELength - ie_offset);
  8809. if (p) {
  8810. struct HT_info_element *HT_info = (struct HT_info_element *)(p + 2);
  8811. bssid->Configuration.DSConfig = HT_info->primary_channel;
  8812. } else {
  8813. /* use current channel */
  8814. bssid->Configuration.DSConfig = rtw_get_oper_ch(padapter);
  8815. }
  8816. }
  8817. _rtw_memcpy(&bssid->Configuration.BeaconPeriod, rtw_get_beacon_interval_from_ie(bssid->IEs), 2);
  8818. bssid->Configuration.BeaconPeriod = le32_to_cpu(bssid->Configuration.BeaconPeriod);
  8819. val16 = rtw_get_capability((WLAN_BSSID_EX *)bssid);
  8820. if ((val16 & 0x03) == cap_ESS) {
  8821. bssid->InfrastructureMode = Ndis802_11Infrastructure;
  8822. _rtw_memcpy(bssid->MacAddress, get_addr2_ptr(pframe), ETH_ALEN);
  8823. } else if ((val16 & 0x03) == cap_IBSS){
  8824. bssid->InfrastructureMode = Ndis802_11IBSS;
  8825. _rtw_memcpy(bssid->MacAddress, GetAddr3Ptr(pframe), ETH_ALEN);
  8826. } else if ((val16 & 0x03) == 0x00){
  8827. u8 *mesh_id_ie, *mesh_conf_ie;
  8828. sint mesh_id_ie_len, mesh_conf_ie_len;
  8829. mesh_id_ie = rtw_get_ie(bssid->IEs + ie_offset, WLAN_EID_MESH_ID, &mesh_id_ie_len, bssid->IELength - ie_offset);
  8830. mesh_conf_ie = rtw_get_ie(bssid->IEs + ie_offset, WLAN_EID_MESH_CONFIG, &mesh_conf_ie_len, bssid->IELength - ie_offset);
  8831. if (mesh_id_ie || mesh_conf_ie) {
  8832. if (!mesh_id_ie) {
  8833. RTW_INFO("cannot find Mesh ID for survey event\n");
  8834. return _FAIL;
  8835. }
  8836. if (mesh_id_ie_len) {
  8837. if (mesh_id_ie_len > NDIS_802_11_LENGTH_SSID) {
  8838. RTW_INFO("Mesh ID too long (%d) for survey event\n", mesh_id_ie_len);
  8839. return _FAIL;
  8840. }
  8841. _rtw_memcpy(bssid->mesh_id.Ssid, (mesh_id_ie + 2), mesh_id_ie_len);
  8842. bssid->mesh_id.SsidLength = mesh_id_ie_len;
  8843. } else
  8844. bssid->mesh_id.SsidLength = 0;
  8845. if (!mesh_conf_ie) {
  8846. RTW_INFO("cannot find Mesh config for survey event\n");
  8847. return _FAIL;
  8848. }
  8849. if (mesh_conf_ie_len != 7) {
  8850. RTW_INFO("invalid Mesh conf IE len (%d) for survey event\n", mesh_conf_ie_len);
  8851. return _FAIL;
  8852. }
  8853. bssid->InfrastructureMode = Ndis802_11_mesh;
  8854. _rtw_memcpy(bssid->MacAddress, GetAddr3Ptr(pframe), ETH_ALEN);
  8855. } else {
  8856. /* default cases */
  8857. bssid->InfrastructureMode = Ndis802_11IBSS;
  8858. _rtw_memcpy(bssid->MacAddress, GetAddr3Ptr(pframe), ETH_ALEN);
  8859. }
  8860. }
  8861. if (val16 & BIT(4))
  8862. bssid->Privacy = 1;
  8863. else
  8864. bssid->Privacy = 0;
  8865. bssid->Configuration.ATIMWindow = 0;
  8866. /* 20/40 BSS Coexistence check */
  8867. if ((pregistrypriv->wifi_spec == 1) && (_FALSE == pmlmeinfo->bwmode_updated)) {
  8868. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  8869. #ifdef CONFIG_80211N_HT
  8870. p = rtw_get_ie(bssid->IEs + ie_offset, _HT_CAPABILITY_IE_, &len, bssid->IELength - ie_offset);
  8871. if (p && len > 0) {
  8872. struct HT_caps_element *pHT_caps;
  8873. pHT_caps = (struct HT_caps_element *)(p + 2);
  8874. if (pHT_caps->u.HT_cap_element.HT_caps_info & BIT(14))
  8875. pmlmepriv->num_FortyMHzIntolerant++;
  8876. } else
  8877. pmlmepriv->num_sta_no_ht++;
  8878. #endif /* CONFIG_80211N_HT */
  8879. }
  8880. #ifdef CONFIG_INTEL_WIDI
  8881. /* process_intel_widi_query_or_tigger(padapter, bssid); */
  8882. if (process_intel_widi_query_or_tigger(padapter, bssid))
  8883. return _FAIL;
  8884. #endif /* CONFIG_INTEL_WIDI */
  8885. #if defined(DBG_RX_SIGNAL_DISPLAY_SSID_MONITORED) & 1
  8886. if (strcmp(bssid->Ssid.Ssid, DBG_RX_SIGNAL_DISPLAY_SSID_MONITORED) == 0) {
  8887. RTW_INFO("Receiving %s("MAC_FMT", DSConfig:%u) from ch%u with ss:%3u, sq:%3u, RawRSSI:%3ld\n"
  8888. , bssid->Ssid.Ssid, MAC_ARG(bssid->MacAddress), bssid->Configuration.DSConfig
  8889. , rtw_get_oper_ch(padapter)
  8890. , bssid->PhyInfo.SignalStrength, bssid->PhyInfo.SignalQuality, bssid->Rssi
  8891. );
  8892. }
  8893. #endif
  8894. /* mark bss info receving from nearby channel as SignalQuality 101 */
  8895. if (bssid->Configuration.DSConfig != rtw_get_oper_ch(padapter))
  8896. bssid->PhyInfo.SignalQuality = 101;
  8897. #ifdef CONFIG_RTW_80211K
  8898. p = rtw_get_ie(bssid->IEs + ie_offset, _EID_RRM_EN_CAP_IE_, &len, bssid->IELength - ie_offset);
  8899. if (p)
  8900. _rtw_memcpy(bssid->PhyInfo.rm_en_cap, (p + 2), *(p + 1));
  8901. /* save freerun counter */
  8902. bssid->PhyInfo.free_cnt = precv_frame->u.hdr.attrib.free_cnt;
  8903. #endif
  8904. return _SUCCESS;
  8905. }
  8906. void start_create_ibss(_adapter *padapter)
  8907. {
  8908. unsigned short caps;
  8909. u8 val8;
  8910. u8 join_type;
  8911. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  8912. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8913. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  8914. u8 doiqk = _FALSE;
  8915. pmlmeext->cur_channel = (u8)pnetwork->Configuration.DSConfig;
  8916. pmlmeinfo->bcn_interval = get_beacon_interval(pnetwork);
  8917. /* update wireless mode */
  8918. update_wireless_mode(padapter);
  8919. /* udpate capability */
  8920. caps = rtw_get_capability((WLAN_BSSID_EX *)pnetwork);
  8921. update_capinfo(padapter, caps);
  8922. if (caps & cap_IBSS) { /* adhoc master */
  8923. /* set_opmode_cmd(padapter, adhoc); */ /* removed */
  8924. val8 = 0xcf;
  8925. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_CFG, (u8 *)(&val8));
  8926. doiqk = _TRUE;
  8927. rtw_hal_set_hwreg(padapter , HW_VAR_DO_IQK , &doiqk);
  8928. /* switch channel */
  8929. set_channel_bwmode(padapter, pmlmeext->cur_channel, HAL_PRIME_CHNL_OFFSET_DONT_CARE, CHANNEL_WIDTH_20);
  8930. doiqk = _FALSE;
  8931. rtw_hal_set_hwreg(padapter , HW_VAR_DO_IQK , &doiqk);
  8932. beacon_timing_control(padapter);
  8933. /* set msr to WIFI_FW_ADHOC_STATE */
  8934. pmlmeinfo->state = WIFI_FW_ADHOC_STATE;
  8935. Set_MSR(padapter, (pmlmeinfo->state & 0x3));
  8936. /* issue beacon */
  8937. if (send_beacon(padapter) == _FAIL) {
  8938. report_join_res(padapter, -1, WLAN_STATUS_UNSPECIFIED_FAILURE);
  8939. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  8940. } else {
  8941. rtw_hal_set_hwreg(padapter, HW_VAR_BSSID, padapter->registrypriv.dev_network.MacAddress);
  8942. rtw_hal_rcr_set_chk_bssid(padapter, MLME_ADHOC_STARTED);
  8943. join_type = 0;
  8944. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_JOIN, (u8 *)(&join_type));
  8945. report_join_res(padapter, 1, WLAN_STATUS_SUCCESS);
  8946. pmlmeinfo->state |= WIFI_FW_ASSOC_SUCCESS;
  8947. rtw_indicate_connect(padapter);
  8948. }
  8949. } else {
  8950. RTW_INFO("start_create_ibss, invalid cap:%x\n", caps);
  8951. return;
  8952. }
  8953. /* update bc/mc sta_info */
  8954. update_bmc_sta(padapter);
  8955. }
  8956. void start_clnt_join(_adapter *padapter)
  8957. {
  8958. unsigned short caps;
  8959. u8 val8;
  8960. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  8961. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  8962. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  8963. int beacon_timeout;
  8964. u8 ASIX_ID[] = {0x00, 0x0E, 0xC6};
  8965. /* update wireless mode */
  8966. update_wireless_mode(padapter);
  8967. /* udpate capability */
  8968. caps = rtw_get_capability((WLAN_BSSID_EX *)pnetwork);
  8969. update_capinfo(padapter, caps);
  8970. /* check if sta is ASIX peer and fix IOT issue if it is. */
  8971. if (_rtw_memcmp(get_my_bssid(&pmlmeinfo->network) , ASIX_ID , 3)) {
  8972. u8 iot_flag = _TRUE;
  8973. rtw_hal_set_hwreg(padapter, HW_VAR_ASIX_IOT, (u8 *)(&iot_flag));
  8974. }
  8975. if (caps & cap_ESS) {
  8976. Set_MSR(padapter, WIFI_FW_STATION_STATE);
  8977. val8 = (pmlmeinfo->auth_algo == dot11AuthAlgrthm_8021X) ? 0xcc : 0xcf;
  8978. #ifdef CONFIG_WAPI_SUPPORT
  8979. if (padapter->wapiInfo.bWapiEnable && pmlmeinfo->auth_algo == dot11AuthAlgrthm_WAPI) {
  8980. /* Disable TxUseDefaultKey, RxUseDefaultKey, RxBroadcastUseDefaultKey. */
  8981. val8 = 0x4c;
  8982. }
  8983. #endif
  8984. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_CFG, (u8 *)(&val8));
  8985. #ifdef CONFIG_DEAUTH_BEFORE_CONNECT
  8986. /* Because of AP's not receiving deauth before */
  8987. /* AP may: 1)not response auth or 2)deauth us after link is complete */
  8988. /* issue deauth before issuing auth to deal with the situation */
  8989. /* Commented by Albert 2012/07/21 */
  8990. /* For the Win8 P2P connection, it will be hard to have a successful connection if this Wi-Fi doesn't connect to it. */
  8991. {
  8992. #ifdef CONFIG_P2P
  8993. _queue *queue = &(padapter->mlmepriv.scanned_queue);
  8994. _list *head = get_list_head(queue);
  8995. _list *pos = get_next(head);
  8996. struct wlan_network *scanned = NULL;
  8997. u8 ie_offset = 0;
  8998. _irqL irqL;
  8999. bool has_p2p_ie = _FALSE;
  9000. _enter_critical_bh(&(padapter->mlmepriv.scanned_queue.lock), &irqL);
  9001. for (pos = get_next(head); !rtw_end_of_queue_search(head, pos); pos = get_next(pos)) {
  9002. scanned = LIST_CONTAINOR(pos, struct wlan_network, list);
  9003. if (_rtw_memcmp(&(scanned->network.Ssid), &(pnetwork->Ssid), sizeof(NDIS_802_11_SSID)) == _TRUE
  9004. && _rtw_memcmp(scanned->network.MacAddress, pnetwork->MacAddress, sizeof(NDIS_802_11_MAC_ADDRESS)) == _TRUE
  9005. ) {
  9006. ie_offset = (scanned->network.Reserved[0] == BSS_TYPE_PROB_REQ ? 0 : 12);
  9007. if (rtw_get_p2p_ie(scanned->network.IEs + ie_offset, scanned->network.IELength - ie_offset, NULL, NULL))
  9008. has_p2p_ie = _TRUE;
  9009. break;
  9010. }
  9011. }
  9012. _exit_critical_bh(&(padapter->mlmepriv.scanned_queue.lock), &irqL);
  9013. if (scanned == NULL || rtw_end_of_queue_search(head, pos) || has_p2p_ie == _FALSE)
  9014. #endif /* CONFIG_P2P */
  9015. /* To avoid connecting to AP fail during resume process, change retry count from 5 to 1 */
  9016. issue_deauth_ex(padapter, pnetwork->MacAddress, WLAN_REASON_DEAUTH_LEAVING, 1, 100);
  9017. }
  9018. #endif /* CONFIG_DEAUTH_BEFORE_CONNECT */
  9019. /* here wait for receiving the beacon to start auth */
  9020. /* and enable a timer */
  9021. beacon_timeout = decide_wait_for_beacon_timeout(pmlmeinfo->bcn_interval);
  9022. set_link_timer(pmlmeext, beacon_timeout);
  9023. _set_timer(&padapter->mlmepriv.assoc_timer,
  9024. (REAUTH_TO * REAUTH_LIMIT) + (REASSOC_TO * REASSOC_LIMIT) + beacon_timeout);
  9025. #ifdef CONFIG_RTW_80211R
  9026. if (rtw_ft_roam(padapter)) {
  9027. rtw_ft_start_clnt_join(padapter);
  9028. } else
  9029. #endif
  9030. {
  9031. rtw_sta_linking_test_set_start();
  9032. pmlmeinfo->state = WIFI_FW_AUTH_NULL | WIFI_FW_STATION_STATE;
  9033. }
  9034. } else if (caps & cap_IBSS) { /* adhoc client */
  9035. Set_MSR(padapter, WIFI_FW_ADHOC_STATE);
  9036. val8 = 0xcf;
  9037. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_CFG, (u8 *)(&val8));
  9038. beacon_timing_control(padapter);
  9039. pmlmeinfo->state = WIFI_FW_ADHOC_STATE;
  9040. report_join_res(padapter, 1, WLAN_STATUS_SUCCESS);
  9041. } else {
  9042. /* RTW_INFO("marc: invalid cap:%x\n", caps); */
  9043. return;
  9044. }
  9045. }
  9046. void start_clnt_auth(_adapter *padapter)
  9047. {
  9048. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9049. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9050. _cancel_timer_ex(&pmlmeext->link_timer);
  9051. pmlmeinfo->state &= (~WIFI_FW_AUTH_NULL);
  9052. pmlmeinfo->state |= WIFI_FW_AUTH_STATE;
  9053. pmlmeinfo->auth_seq = 1;
  9054. pmlmeinfo->reauth_count = 0;
  9055. pmlmeinfo->reassoc_count = 0;
  9056. pmlmeinfo->link_count = 0;
  9057. pmlmeext->retry = 0;
  9058. #ifdef CONFIG_RTW_80211R
  9059. if (rtw_ft_roam(padapter)) {
  9060. rtw_ft_set_status(padapter, RTW_FT_AUTHENTICATING_STA);
  9061. RTW_PRINT("start ft auth\n");
  9062. } else
  9063. #endif
  9064. RTW_PRINT("start auth\n");
  9065. issue_auth(padapter, NULL, 0);
  9066. set_link_timer(pmlmeext, REAUTH_TO);
  9067. }
  9068. void start_clnt_assoc(_adapter *padapter)
  9069. {
  9070. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9071. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9072. _cancel_timer_ex(&pmlmeext->link_timer);
  9073. pmlmeinfo->state &= (~(WIFI_FW_AUTH_NULL | WIFI_FW_AUTH_STATE));
  9074. pmlmeinfo->state |= (WIFI_FW_AUTH_SUCCESS | WIFI_FW_ASSOC_STATE);
  9075. #ifdef CONFIG_RTW_80211R
  9076. if (rtw_ft_roam(padapter))
  9077. issue_reassocreq(padapter);
  9078. else
  9079. #endif
  9080. issue_assocreq(padapter);
  9081. set_link_timer(pmlmeext, REASSOC_TO);
  9082. }
  9083. unsigned int receive_disconnect(_adapter *padapter, unsigned char *MacAddr, unsigned short reason, u8 locally_generated)
  9084. {
  9085. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9086. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9087. if (!(_rtw_memcmp(MacAddr, get_my_bssid(&pmlmeinfo->network), ETH_ALEN)))
  9088. return _SUCCESS;
  9089. RTW_INFO("%s\n", __FUNCTION__);
  9090. #ifdef CONFIG_RTW_REPEATER_SON
  9091. rtw_rson_do_disconnect(padapter);
  9092. #endif
  9093. if ((pmlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE) {
  9094. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS) {
  9095. if (report_del_sta_event(padapter, MacAddr, reason, _TRUE, locally_generated) != _FAIL)
  9096. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  9097. } else if (pmlmeinfo->state & WIFI_FW_LINKING_STATE) {
  9098. if (report_join_res(padapter, -2, reason) != _FAIL)
  9099. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  9100. } else
  9101. RTW_INFO(FUNC_ADPT_FMT" - End to Disconnect\n", FUNC_ADPT_ARG(padapter));
  9102. #ifdef CONFIG_RTW_80211R
  9103. rtw_ft_roam_status_reset(padapter);
  9104. #endif
  9105. #ifdef CONFIG_RTW_WNM
  9106. rtw_wnm_reset_btm_state(padapter);
  9107. #endif
  9108. }
  9109. return _SUCCESS;
  9110. }
  9111. #ifdef CONFIG_80211D
  9112. static void process_80211d(PADAPTER padapter, WLAN_BSSID_EX *bssid)
  9113. {
  9114. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  9115. struct registry_priv *pregistrypriv;
  9116. struct mlme_ext_priv *pmlmeext;
  9117. RT_CHANNEL_INFO *chplan_new;
  9118. u8 channel;
  9119. u8 i;
  9120. pregistrypriv = &padapter->registrypriv;
  9121. pmlmeext = &padapter->mlmeextpriv;
  9122. /* Adjust channel plan by AP Country IE */
  9123. if (pregistrypriv->enable80211d
  9124. && (!pmlmeext->update_channel_plan_by_ap_done)) {
  9125. u8 *ie, *p;
  9126. u32 len;
  9127. RT_CHANNEL_PLAN chplan_ap;
  9128. RT_CHANNEL_INFO *chplan_sta = NULL;
  9129. u8 country[4];
  9130. u8 fcn; /* first channel number */
  9131. u8 noc; /* number of channel */
  9132. u8 j, k;
  9133. ie = rtw_get_ie(bssid->IEs + _FIXED_IE_LENGTH_, _COUNTRY_IE_, &len, bssid->IELength - _FIXED_IE_LENGTH_);
  9134. if (!ie)
  9135. return;
  9136. if (len < 6)
  9137. return;
  9138. ie += 2;
  9139. p = ie;
  9140. ie += len;
  9141. _rtw_memset(country, 0, 4);
  9142. _rtw_memcpy(country, p, 3);
  9143. p += 3;
  9144. RTW_INFO("%s: 802.11d country=%s\n", __FUNCTION__, country);
  9145. i = 0;
  9146. while ((ie - p) >= 3) {
  9147. fcn = *(p++);
  9148. noc = *(p++);
  9149. p++;
  9150. for (j = 0; j < noc; j++) {
  9151. if (fcn <= 14)
  9152. channel = fcn + j; /* 2.4 GHz */
  9153. else
  9154. channel = fcn + j * 4; /* 5 GHz */
  9155. chplan_ap.Channel[i++] = channel;
  9156. }
  9157. }
  9158. chplan_ap.Len = i;
  9159. #ifdef CONFIG_RTW_DEBUG
  9160. i = 0;
  9161. RTW_INFO("%s: AP[%s] channel plan {", __FUNCTION__, bssid->Ssid.Ssid);
  9162. while ((i < chplan_ap.Len) && (chplan_ap.Channel[i] != 0)) {
  9163. _RTW_INFO("%02d,", chplan_ap.Channel[i]);
  9164. i++;
  9165. }
  9166. _RTW_INFO("}\n");
  9167. #endif
  9168. chplan_sta = rtw_malloc(sizeof(RT_CHANNEL_INFO) * MAX_CHANNEL_NUM);
  9169. if (!chplan_sta)
  9170. goto done_update_chplan_from_ap;
  9171. _rtw_memcpy(chplan_sta, rfctl->channel_set, sizeof(RT_CHANNEL_INFO) * MAX_CHANNEL_NUM);
  9172. #ifdef CONFIG_RTW_DEBUG
  9173. i = 0;
  9174. RTW_INFO("%s: STA channel plan {", __FUNCTION__);
  9175. while ((i < MAX_CHANNEL_NUM) && (chplan_sta[i].ChannelNum != 0)) {
  9176. _RTW_INFO("%02d(%c),", chplan_sta[i].ChannelNum, chplan_sta[i].ScanType == SCAN_PASSIVE ? 'p' : 'a');
  9177. i++;
  9178. }
  9179. _RTW_INFO("}\n");
  9180. #endif
  9181. _rtw_memset(rfctl->channel_set, 0, sizeof(rfctl->channel_set));
  9182. chplan_new = rfctl->channel_set;
  9183. i = j = k = 0;
  9184. if (pregistrypriv->wireless_mode & WIRELESS_11G) {
  9185. do {
  9186. if ((i == MAX_CHANNEL_NUM)
  9187. || (chplan_sta[i].ChannelNum == 0)
  9188. || (chplan_sta[i].ChannelNum > 14))
  9189. break;
  9190. if ((j == chplan_ap.Len) || (chplan_ap.Channel[j] > 14))
  9191. break;
  9192. if (chplan_sta[i].ChannelNum == chplan_ap.Channel[j]) {
  9193. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9194. chplan_new[k].ScanType = SCAN_ACTIVE;
  9195. i++;
  9196. j++;
  9197. k++;
  9198. } else if (chplan_sta[i].ChannelNum < chplan_ap.Channel[j]) {
  9199. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9200. #if 0
  9201. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9202. #else
  9203. chplan_new[k].ScanType = SCAN_PASSIVE;
  9204. #endif
  9205. i++;
  9206. k++;
  9207. } else if (chplan_sta[i].ChannelNum > chplan_ap.Channel[j]) {
  9208. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9209. chplan_new[k].ScanType = SCAN_ACTIVE;
  9210. j++;
  9211. k++;
  9212. }
  9213. } while (1);
  9214. /* change AP not support channel to Passive scan */
  9215. while ((i < MAX_CHANNEL_NUM)
  9216. && (chplan_sta[i].ChannelNum != 0)
  9217. && (chplan_sta[i].ChannelNum <= 14)) {
  9218. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9219. #if 0
  9220. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9221. #else
  9222. chplan_new[k].ScanType = SCAN_PASSIVE;
  9223. #endif
  9224. i++;
  9225. k++;
  9226. }
  9227. /* add channel AP supported */
  9228. while ((j < chplan_ap.Len) && (chplan_ap.Channel[j] <= 14)) {
  9229. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9230. chplan_new[k].ScanType = SCAN_ACTIVE;
  9231. j++;
  9232. k++;
  9233. }
  9234. } else {
  9235. /* keep original STA 2.4G channel plan */
  9236. while ((i < MAX_CHANNEL_NUM)
  9237. && (chplan_sta[i].ChannelNum != 0)
  9238. && (chplan_sta[i].ChannelNum <= 14)) {
  9239. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9240. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9241. i++;
  9242. k++;
  9243. }
  9244. /* skip AP 2.4G channel plan */
  9245. while ((j < chplan_ap.Len) && (chplan_ap.Channel[j] <= 14))
  9246. j++;
  9247. }
  9248. if (pregistrypriv->wireless_mode & WIRELESS_11A) {
  9249. do {
  9250. if ((i >= MAX_CHANNEL_NUM)
  9251. || (chplan_sta[i].ChannelNum == 0))
  9252. break;
  9253. if ((j == chplan_ap.Len) || (chplan_ap.Channel[j] == 0))
  9254. break;
  9255. if (chplan_sta[i].ChannelNum == chplan_ap.Channel[j]) {
  9256. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9257. chplan_new[k].ScanType = SCAN_ACTIVE;
  9258. i++;
  9259. j++;
  9260. k++;
  9261. } else if (chplan_sta[i].ChannelNum < chplan_ap.Channel[j]) {
  9262. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9263. #if 0
  9264. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9265. #else
  9266. chplan_new[k].ScanType = SCAN_PASSIVE;
  9267. #endif
  9268. i++;
  9269. k++;
  9270. } else if (chplan_sta[i].ChannelNum > chplan_ap.Channel[j]) {
  9271. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9272. chplan_new[k].ScanType = SCAN_ACTIVE;
  9273. j++;
  9274. k++;
  9275. }
  9276. } while (1);
  9277. /* change AP not support channel to Passive scan */
  9278. while ((i < MAX_CHANNEL_NUM) && (chplan_sta[i].ChannelNum != 0)) {
  9279. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9280. #if 0
  9281. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9282. #else
  9283. chplan_new[k].ScanType = SCAN_PASSIVE;
  9284. #endif
  9285. i++;
  9286. k++;
  9287. }
  9288. /* add channel AP supported */
  9289. while ((j < chplan_ap.Len) && (chplan_ap.Channel[j] != 0)) {
  9290. chplan_new[k].ChannelNum = chplan_ap.Channel[j];
  9291. chplan_new[k].ScanType = SCAN_ACTIVE;
  9292. j++;
  9293. k++;
  9294. }
  9295. } else {
  9296. /* keep original STA 5G channel plan */
  9297. while ((i < MAX_CHANNEL_NUM) && (chplan_sta[i].ChannelNum != 0)) {
  9298. chplan_new[k].ChannelNum = chplan_sta[i].ChannelNum;
  9299. chplan_new[k].ScanType = chplan_sta[i].ScanType;
  9300. i++;
  9301. k++;
  9302. }
  9303. }
  9304. pmlmeext->update_channel_plan_by_ap_done = 1;
  9305. #ifdef CONFIG_RTW_DEBUG
  9306. k = 0;
  9307. RTW_INFO("%s: new STA channel plan {", __FUNCTION__);
  9308. while ((k < MAX_CHANNEL_NUM) && (chplan_new[k].ChannelNum != 0)) {
  9309. _RTW_INFO("%02d(%c),", chplan_new[k].ChannelNum, chplan_new[k].ScanType == SCAN_PASSIVE ? 'p' : 'c');
  9310. k++;
  9311. }
  9312. _RTW_INFO("}\n");
  9313. #endif
  9314. #if 0
  9315. /* recover the right channel index */
  9316. channel = chplan_sta[pmlmeext->sitesurvey_res.channel_idx].ChannelNum;
  9317. k = 0;
  9318. while ((k < MAX_CHANNEL_NUM) && (chplan_new[k].ChannelNum != 0)) {
  9319. if (chplan_new[k].ChannelNum == channel) {
  9320. RTW_INFO("%s: change mlme_ext sitesurvey channel index from %d to %d\n",
  9321. __FUNCTION__, pmlmeext->sitesurvey_res.channel_idx, k);
  9322. pmlmeext->sitesurvey_res.channel_idx = k;
  9323. break;
  9324. }
  9325. k++;
  9326. }
  9327. #endif
  9328. done_update_chplan_from_ap:
  9329. if (chplan_sta)
  9330. rtw_mfree(chplan_sta, sizeof(RT_CHANNEL_INFO) * MAX_CHANNEL_NUM);
  9331. }
  9332. }
  9333. #endif
  9334. /****************************************************************************
  9335. Following are the functions to report events
  9336. *****************************************************************************/
  9337. void report_survey_event(_adapter *padapter, union recv_frame *precv_frame)
  9338. {
  9339. struct cmd_obj *pcmd_obj;
  9340. u8 *pevtcmd;
  9341. u32 cmdsz;
  9342. struct survey_event *psurvey_evt;
  9343. struct C2HEvent_Header *pc2h_evt_hdr;
  9344. struct mlme_ext_priv *pmlmeext;
  9345. struct cmd_priv *pcmdpriv;
  9346. /* u8 *pframe = precv_frame->u.hdr.rx_data; */
  9347. /* uint len = precv_frame->u.hdr.len; */
  9348. RT_CHANNEL_INFO *chset = adapter_to_chset(padapter);
  9349. int ch_set_idx = -1;
  9350. if (!padapter)
  9351. return;
  9352. pmlmeext = &padapter->mlmeextpriv;
  9353. pcmdpriv = &padapter->cmdpriv;
  9354. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9355. if (pcmd_obj == NULL)
  9356. return;
  9357. cmdsz = (sizeof(struct survey_event) + sizeof(struct C2HEvent_Header));
  9358. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9359. if (pevtcmd == NULL) {
  9360. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9361. return;
  9362. }
  9363. _rtw_init_listhead(&pcmd_obj->list);
  9364. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9365. pcmd_obj->cmdsz = cmdsz;
  9366. pcmd_obj->parmbuf = pevtcmd;
  9367. pcmd_obj->rsp = NULL;
  9368. pcmd_obj->rspsz = 0;
  9369. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9370. pc2h_evt_hdr->len = sizeof(struct survey_event);
  9371. pc2h_evt_hdr->ID = GEN_EVT_CODE(_Survey);
  9372. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9373. psurvey_evt = (struct survey_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9374. if (collect_bss_info(padapter, precv_frame, (WLAN_BSSID_EX *)&psurvey_evt->bss) == _FAIL) {
  9375. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9376. rtw_mfree((u8 *)pevtcmd, cmdsz);
  9377. return;
  9378. }
  9379. #ifdef CONFIG_80211D
  9380. process_80211d(padapter, &psurvey_evt->bss);
  9381. #endif
  9382. ch_set_idx = rtw_chset_search_ch(chset, psurvey_evt->bss.Configuration.DSConfig);
  9383. if (ch_set_idx >= 0) {
  9384. if (psurvey_evt->bss.InfrastructureMode == Ndis802_11Infrastructure) {
  9385. if (chset[ch_set_idx].ScanType == SCAN_PASSIVE
  9386. && !rtw_is_dfs_ch(psurvey_evt->bss.Configuration.DSConfig)
  9387. ) {
  9388. RTW_INFO("%s: change ch:%d to active\n", __func__, psurvey_evt->bss.Configuration.DSConfig);
  9389. chset[ch_set_idx].ScanType = SCAN_ACTIVE;
  9390. }
  9391. #ifdef CONFIG_DFS
  9392. if (hidden_ssid_ap(&psurvey_evt->bss))
  9393. chset[ch_set_idx].hidden_bss_cnt++;
  9394. #endif
  9395. }
  9396. }
  9397. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9398. pmlmeext->sitesurvey_res.bss_cnt++;
  9399. return;
  9400. }
  9401. void report_surveydone_event(_adapter *padapter)
  9402. {
  9403. struct cmd_obj *pcmd_obj;
  9404. u8 *pevtcmd;
  9405. u32 cmdsz;
  9406. struct surveydone_event *psurveydone_evt;
  9407. struct C2HEvent_Header *pc2h_evt_hdr;
  9408. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9409. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  9410. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9411. if (pcmd_obj == NULL)
  9412. return;
  9413. cmdsz = (sizeof(struct surveydone_event) + sizeof(struct C2HEvent_Header));
  9414. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9415. if (pevtcmd == NULL) {
  9416. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9417. return;
  9418. }
  9419. _rtw_init_listhead(&pcmd_obj->list);
  9420. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9421. pcmd_obj->cmdsz = cmdsz;
  9422. pcmd_obj->parmbuf = pevtcmd;
  9423. pcmd_obj->rsp = NULL;
  9424. pcmd_obj->rspsz = 0;
  9425. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9426. pc2h_evt_hdr->len = sizeof(struct surveydone_event);
  9427. pc2h_evt_hdr->ID = GEN_EVT_CODE(_SurveyDone);
  9428. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9429. psurveydone_evt = (struct surveydone_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9430. psurveydone_evt->bss_cnt = pmlmeext->sitesurvey_res.bss_cnt;
  9431. RTW_INFO("survey done event(%x) band:%d for "ADPT_FMT"\n", psurveydone_evt->bss_cnt, padapter->setband, ADPT_ARG(padapter));
  9432. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9433. return;
  9434. }
  9435. u32 report_join_res(_adapter *padapter, int aid_res, u16 status)
  9436. {
  9437. struct cmd_obj *pcmd_obj;
  9438. u8 *pevtcmd;
  9439. u32 cmdsz;
  9440. struct joinbss_event *pjoinbss_evt;
  9441. struct C2HEvent_Header *pc2h_evt_hdr;
  9442. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9443. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9444. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  9445. u32 ret = _FAIL;
  9446. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9447. if (pcmd_obj == NULL)
  9448. goto exit;
  9449. cmdsz = (sizeof(struct joinbss_event) + sizeof(struct C2HEvent_Header));
  9450. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9451. if (pevtcmd == NULL) {
  9452. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9453. goto exit;
  9454. }
  9455. _rtw_init_listhead(&pcmd_obj->list);
  9456. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9457. pcmd_obj->cmdsz = cmdsz;
  9458. pcmd_obj->parmbuf = pevtcmd;
  9459. pcmd_obj->rsp = NULL;
  9460. pcmd_obj->rspsz = 0;
  9461. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9462. pc2h_evt_hdr->len = sizeof(struct joinbss_event);
  9463. pc2h_evt_hdr->ID = GEN_EVT_CODE(_JoinBss);
  9464. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9465. pjoinbss_evt = (struct joinbss_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9466. _rtw_memcpy((unsigned char *)(&(pjoinbss_evt->network.network)), &(pmlmeinfo->network), sizeof(WLAN_BSSID_EX));
  9467. pjoinbss_evt->network.join_res = pjoinbss_evt->network.aid = aid_res;
  9468. RTW_INFO("report_join_res(%d, %u)\n", aid_res, status);
  9469. rtw_joinbss_event_prehandle(padapter, (u8 *)&pjoinbss_evt->network, status);
  9470. ret = rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9471. exit:
  9472. return ret;
  9473. }
  9474. void report_wmm_edca_update(_adapter *padapter)
  9475. {
  9476. struct cmd_obj *pcmd_obj;
  9477. u8 *pevtcmd;
  9478. u32 cmdsz;
  9479. struct wmm_event *pwmm_event;
  9480. struct C2HEvent_Header *pc2h_evt_hdr;
  9481. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9482. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  9483. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9484. if (pcmd_obj == NULL)
  9485. return;
  9486. cmdsz = (sizeof(struct wmm_event) + sizeof(struct C2HEvent_Header));
  9487. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9488. if (pevtcmd == NULL) {
  9489. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9490. return;
  9491. }
  9492. _rtw_init_listhead(&pcmd_obj->list);
  9493. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9494. pcmd_obj->cmdsz = cmdsz;
  9495. pcmd_obj->parmbuf = pevtcmd;
  9496. pcmd_obj->rsp = NULL;
  9497. pcmd_obj->rspsz = 0;
  9498. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9499. pc2h_evt_hdr->len = sizeof(struct wmm_event);
  9500. pc2h_evt_hdr->ID = GEN_EVT_CODE(_WMM);
  9501. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9502. pwmm_event = (struct wmm_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9503. pwmm_event->wmm = 0;
  9504. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9505. return;
  9506. }
  9507. u32 report_del_sta_event(_adapter *padapter, unsigned char *MacAddr, unsigned short reason, bool enqueue, u8 locally_generated)
  9508. {
  9509. struct cmd_obj *pcmd_obj;
  9510. u8 *pevtcmd;
  9511. u32 cmdsz;
  9512. struct sta_info *psta;
  9513. int mac_id = -1;
  9514. struct stadel_event *pdel_sta_evt;
  9515. struct C2HEvent_Header *pc2h_evt_hdr;
  9516. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9517. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  9518. u8 res = _SUCCESS;
  9519. /* prepare cmd parameter */
  9520. cmdsz = (sizeof(struct stadel_event) + sizeof(struct C2HEvent_Header));
  9521. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9522. if (pevtcmd == NULL) {
  9523. res = _FAIL;
  9524. goto exit;
  9525. }
  9526. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9527. pc2h_evt_hdr->len = sizeof(struct stadel_event);
  9528. pc2h_evt_hdr->ID = GEN_EVT_CODE(_DelSTA);
  9529. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9530. pdel_sta_evt = (struct stadel_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9531. _rtw_memcpy((unsigned char *)(&(pdel_sta_evt->macaddr)), MacAddr, ETH_ALEN);
  9532. _rtw_memcpy((unsigned char *)(pdel_sta_evt->rsvd), (unsigned char *)(&reason), 2);
  9533. psta = rtw_get_stainfo(&padapter->stapriv, MacAddr);
  9534. if (psta)
  9535. mac_id = (int)psta->cmn.mac_id;
  9536. else
  9537. mac_id = (-1);
  9538. pdel_sta_evt->mac_id = mac_id;
  9539. pdel_sta_evt->locally_generated = locally_generated;
  9540. if (!enqueue) {
  9541. /* do directly */
  9542. rtw_stadel_event_callback(padapter, (u8 *)pdel_sta_evt);
  9543. rtw_mfree(pevtcmd, cmdsz);
  9544. } else {
  9545. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9546. if (pcmd_obj == NULL) {
  9547. rtw_mfree(pevtcmd, cmdsz);
  9548. res = _FAIL;
  9549. goto exit;
  9550. }
  9551. _rtw_init_listhead(&pcmd_obj->list);
  9552. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9553. pcmd_obj->cmdsz = cmdsz;
  9554. pcmd_obj->parmbuf = pevtcmd;
  9555. pcmd_obj->rsp = NULL;
  9556. pcmd_obj->rspsz = 0;
  9557. res = rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9558. }
  9559. exit:
  9560. RTW_INFO(FUNC_ADPT_FMT" "MAC_FMT" mac_id=%d, enqueue:%d, res:%u\n"
  9561. , FUNC_ADPT_ARG(padapter), MAC_ARG(MacAddr), mac_id, enqueue, res);
  9562. return res;
  9563. }
  9564. void report_add_sta_event(_adapter *padapter, unsigned char *MacAddr)
  9565. {
  9566. struct cmd_obj *pcmd_obj;
  9567. u8 *pevtcmd;
  9568. u32 cmdsz;
  9569. struct stassoc_event *padd_sta_evt;
  9570. struct C2HEvent_Header *pc2h_evt_hdr;
  9571. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9572. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  9573. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  9574. if (pcmd_obj == NULL)
  9575. return;
  9576. cmdsz = (sizeof(struct stassoc_event) + sizeof(struct C2HEvent_Header));
  9577. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  9578. if (pevtcmd == NULL) {
  9579. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  9580. return;
  9581. }
  9582. _rtw_init_listhead(&pcmd_obj->list);
  9583. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  9584. pcmd_obj->cmdsz = cmdsz;
  9585. pcmd_obj->parmbuf = pevtcmd;
  9586. pcmd_obj->rsp = NULL;
  9587. pcmd_obj->rspsz = 0;
  9588. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  9589. pc2h_evt_hdr->len = sizeof(struct stassoc_event);
  9590. pc2h_evt_hdr->ID = GEN_EVT_CODE(_AddSTA);
  9591. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  9592. padd_sta_evt = (struct stassoc_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  9593. _rtw_memcpy((unsigned char *)(&(padd_sta_evt->macaddr)), MacAddr, ETH_ALEN);
  9594. RTW_INFO("report_add_sta_event: add STA\n");
  9595. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  9596. return;
  9597. }
  9598. bool rtw_port_switch_chk(_adapter *adapter)
  9599. {
  9600. bool switch_needed = _FALSE;
  9601. #ifdef CONFIG_CONCURRENT_MODE
  9602. #ifdef CONFIG_RUNTIME_PORT_SWITCH
  9603. struct dvobj_priv *dvobj = adapter_to_dvobj(adapter);
  9604. struct pwrctrl_priv *pwrctl = dvobj_to_pwrctl(dvobj);
  9605. _adapter *if_port0 = NULL;
  9606. _adapter *if_port1 = NULL;
  9607. struct mlme_ext_info *if_port0_mlmeinfo = NULL;
  9608. struct mlme_ext_info *if_port1_mlmeinfo = NULL;
  9609. int i;
  9610. for (i = 0; i < dvobj->iface_nums; i++) {
  9611. if (get_hw_port(dvobj->padapters[i]) == HW_PORT0) {
  9612. if_port0 = dvobj->padapters[i];
  9613. if_port0_mlmeinfo = &(if_port0->mlmeextpriv.mlmext_info);
  9614. } else if (get_hw_port(dvobj->padapters[i]) == HW_PORT1) {
  9615. if_port1 = dvobj->padapters[i];
  9616. if_port1_mlmeinfo = &(if_port1->mlmeextpriv.mlmext_info);
  9617. }
  9618. }
  9619. if (if_port0 == NULL) {
  9620. rtw_warn_on(1);
  9621. goto exit;
  9622. }
  9623. if (if_port1 == NULL) {
  9624. rtw_warn_on(1);
  9625. goto exit;
  9626. }
  9627. #ifdef DBG_RUNTIME_PORT_SWITCH
  9628. RTW_INFO(FUNC_ADPT_FMT" wowlan_mode:%u\n"
  9629. ADPT_FMT", port0, mlmeinfo->state:0x%08x, p2p_state:%d, %d\n"
  9630. ADPT_FMT", port1, mlmeinfo->state:0x%08x, p2p_state:%d, %d\n",
  9631. FUNC_ADPT_ARG(adapter), pwrctl->wowlan_mode,
  9632. ADPT_ARG(if_port0), if_port0_mlmeinfo->state, rtw_p2p_state(&if_port0->wdinfo), rtw_p2p_chk_state(&if_port0->wdinfo, P2P_STATE_NONE),
  9633. ADPT_ARG(if_port1), if_port1_mlmeinfo->state, rtw_p2p_state(&if_port1->wdinfo), rtw_p2p_chk_state(&if_port1->wdinfo, P2P_STATE_NONE));
  9634. #endif /* DBG_RUNTIME_PORT_SWITCH */
  9635. #ifdef CONFIG_WOWLAN
  9636. /* WOWLAN interface(primary, for now) should be port0 */
  9637. if (pwrctl->wowlan_mode == _TRUE) {
  9638. if (!is_primary_adapter(if_port0)) {
  9639. RTW_INFO("%s "ADPT_FMT" enable WOWLAN\n", __func__, ADPT_ARG(if_port1));
  9640. switch_needed = _TRUE;
  9641. }
  9642. goto exit;
  9643. }
  9644. #endif /* CONFIG_WOWLAN */
  9645. /* AP/Mesh should use port0 for ctl frame's ack */
  9646. if ((if_port1_mlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) {
  9647. RTW_INFO("%s "ADPT_FMT" is AP/GO/Mesh\n", __func__, ADPT_ARG(if_port1));
  9648. switch_needed = _TRUE;
  9649. goto exit;
  9650. }
  9651. /* GC should use port0 for p2p ps */
  9652. if (((if_port1_mlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE)
  9653. && (if_port1_mlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)
  9654. #ifdef CONFIG_P2P
  9655. && !rtw_p2p_chk_state(&if_port1->wdinfo, P2P_STATE_NONE)
  9656. #endif
  9657. && !check_fwstate(&if_port1->mlmepriv, WIFI_UNDER_WPS)
  9658. ) {
  9659. RTW_INFO("%s "ADPT_FMT" is GC\n", __func__, ADPT_ARG(if_port1));
  9660. switch_needed = _TRUE;
  9661. goto exit;
  9662. }
  9663. /* port1 linked, but port0 not linked */
  9664. if ((if_port1_mlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)
  9665. && !(if_port0_mlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)
  9666. && ((if_port0_mlmeinfo->state & 0x03) != WIFI_FW_AP_STATE)
  9667. ) {
  9668. RTW_INFO("%s "ADPT_FMT" is SINGLE_LINK\n", __func__, ADPT_ARG(if_port1));
  9669. switch_needed = _TRUE;
  9670. goto exit;
  9671. }
  9672. exit:
  9673. #ifdef DBG_RUNTIME_PORT_SWITCH
  9674. RTW_INFO(FUNC_ADPT_FMT" ret:%d\n", FUNC_ADPT_ARG(adapter), switch_needed);
  9675. #endif /* DBG_RUNTIME_PORT_SWITCH */
  9676. #endif /* CONFIG_RUNTIME_PORT_SWITCH */
  9677. #endif /* CONFIG_CONCURRENT_MODE */
  9678. return switch_needed;
  9679. }
  9680. /****************************************************************************
  9681. Following are the event callback functions
  9682. *****************************************************************************/
  9683. /* for sta/adhoc mode */
  9684. void update_sta_info(_adapter *padapter, struct sta_info *psta)
  9685. {
  9686. _irqL irqL;
  9687. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  9688. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9689. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9690. /* ERP */
  9691. VCS_update(padapter, psta);
  9692. #ifdef CONFIG_80211N_HT
  9693. /* HT */
  9694. if (pmlmepriv->htpriv.ht_option) {
  9695. psta->htpriv.ht_option = _TRUE;
  9696. psta->htpriv.ampdu_enable = pmlmepriv->htpriv.ampdu_enable;
  9697. psta->htpriv.rx_ampdu_min_spacing = (pmlmeinfo->HT_caps.u.HT_cap_element.AMPDU_para & IEEE80211_HT_CAP_AMPDU_DENSITY) >> 2;
  9698. if (support_short_GI(padapter, &(pmlmeinfo->HT_caps), CHANNEL_WIDTH_20))
  9699. psta->htpriv.sgi_20m = _TRUE;
  9700. if (support_short_GI(padapter, &(pmlmeinfo->HT_caps), CHANNEL_WIDTH_40))
  9701. psta->htpriv.sgi_40m = _TRUE;
  9702. psta->qos_option = _TRUE;
  9703. psta->htpriv.ldpc_cap = pmlmepriv->htpriv.ldpc_cap;
  9704. psta->htpriv.stbc_cap = pmlmepriv->htpriv.stbc_cap;
  9705. psta->htpriv.beamform_cap = pmlmepriv->htpriv.beamform_cap;
  9706. _rtw_memcpy(&psta->htpriv.ht_cap, &pmlmeinfo->HT_caps, sizeof(struct rtw_ieee80211_ht_cap));
  9707. #ifdef CONFIG_BEAMFORMING
  9708. psta->htpriv.beamform_cap = pmlmepriv->htpriv.beamform_cap;
  9709. psta->cmn.bf_info.ht_beamform_cap = pmlmepriv->htpriv.beamform_cap;
  9710. #endif
  9711. } else
  9712. #endif /* CONFIG_80211N_HT */
  9713. {
  9714. #ifdef CONFIG_80211N_HT
  9715. psta->htpriv.ht_option = _FALSE;
  9716. psta->htpriv.ampdu_enable = _FALSE;
  9717. psta->htpriv.tx_amsdu_enable = _FALSE;
  9718. psta->htpriv.sgi_20m = _FALSE;
  9719. psta->htpriv.sgi_40m = _FALSE;
  9720. #endif /* CONFIG_80211N_HT */
  9721. psta->qos_option = _FALSE;
  9722. }
  9723. #ifdef CONFIG_80211N_HT
  9724. psta->htpriv.ch_offset = pmlmeext->cur_ch_offset;
  9725. psta->htpriv.agg_enable_bitmap = 0x0;/* reset */
  9726. psta->htpriv.candidate_tid_bitmap = 0x0;/* reset */
  9727. #endif /* CONFIG_80211N_HT */
  9728. psta->cmn.bw_mode = pmlmeext->cur_bwmode;
  9729. /* QoS */
  9730. if (pmlmepriv->qospriv.qos_option)
  9731. psta->qos_option = _TRUE;
  9732. #ifdef CONFIG_80211AC_VHT
  9733. _rtw_memcpy(&psta->vhtpriv, &pmlmepriv->vhtpriv, sizeof(struct vht_priv));
  9734. if (psta->vhtpriv.vht_option) {
  9735. psta->cmn.ra_info.is_vht_enable = _TRUE;
  9736. #ifdef CONFIG_BEAMFORMING
  9737. psta->vhtpriv.beamform_cap = pmlmepriv->vhtpriv.beamform_cap;
  9738. psta->cmn.bf_info.vht_beamform_cap = pmlmepriv->vhtpriv.beamform_cap;
  9739. #endif /*CONFIG_BEAMFORMING*/
  9740. }
  9741. #endif /* CONFIG_80211AC_VHT */
  9742. psta->cmn.ra_info.is_support_sgi = query_ra_short_GI(psta, rtw_get_tx_bw_mode(padapter, psta));
  9743. update_ldpc_stbc_cap(psta);
  9744. _enter_critical_bh(&psta->lock, &irqL);
  9745. psta->state = _FW_LINKED;
  9746. _exit_critical_bh(&psta->lock, &irqL);
  9747. }
  9748. static void rtw_mlmeext_disconnect(_adapter *padapter)
  9749. {
  9750. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  9751. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9752. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9753. u8 self_action = MLME_ACTION_UNKNOWN;
  9754. u8 state_backup = (pmlmeinfo->state & 0x03);
  9755. u8 ASIX_ID[] = {0x00, 0x0E, 0xC6};
  9756. if (MLME_IS_AP(padapter))
  9757. self_action = MLME_AP_STOPPED;
  9758. else if (MLME_IS_MESH(padapter))
  9759. self_action = MLME_MESH_STOPPED;
  9760. else if (MLME_IS_STA(padapter))
  9761. self_action = MLME_STA_DISCONNECTED;
  9762. else if (MLME_IS_ADHOC(padapter) || MLME_IS_ADHOC_MASTER(padapter))
  9763. self_action = MLME_ADHOC_STOPPED;
  9764. else {
  9765. RTW_INFO("state:0x%x\n", MLME_STATE(padapter));
  9766. rtw_warn_on(1);
  9767. }
  9768. /* set_opmode_cmd(padapter, infra_client_with_mlme); */
  9769. #ifdef CONFIG_HW_P0_TSF_SYNC
  9770. if (self_action == MLME_STA_DISCONNECTED)
  9771. correct_TSF(padapter, self_action);
  9772. #endif
  9773. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_DISCONNECT, 0);
  9774. rtw_hal_set_hwreg(padapter, HW_VAR_BSSID, null_addr);
  9775. if (self_action == MLME_STA_DISCONNECTED)
  9776. rtw_hal_rcr_set_chk_bssid(padapter, self_action);
  9777. /* set MSR to no link state->infra. mode */
  9778. Set_MSR(padapter, _HW_STATE_STATION_);
  9779. /* check if sta is ASIX peer and fix IOT issue if it is. */
  9780. if (_rtw_memcmp(get_my_bssid(&pmlmeinfo->network) , ASIX_ID , 3)) {
  9781. u8 iot_flag = _FALSE;
  9782. rtw_hal_set_hwreg(padapter, HW_VAR_ASIX_IOT, (u8 *)(&iot_flag));
  9783. }
  9784. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  9785. #ifdef CONFIG_MCC_MODE
  9786. /* mcc disconnect setting before download LPS rsvd page */
  9787. rtw_hal_set_mcc_setting_disconnect(padapter);
  9788. #endif /* CONFIG_MCC_MODE */
  9789. if (state_backup == WIFI_FW_STATION_STATE) {
  9790. if (rtw_port_switch_chk(padapter) == _TRUE) {
  9791. rtw_hal_set_hwreg(padapter, HW_VAR_PORT_SWITCH, NULL);
  9792. #ifdef CONFIG_LPS
  9793. {
  9794. _adapter *port0_iface = dvobj_get_port0_adapter(adapter_to_dvobj(padapter));
  9795. if (port0_iface)
  9796. rtw_lps_ctrl_wk_cmd(port0_iface, LPS_CTRL_CONNECT, 0);
  9797. }
  9798. #endif
  9799. }
  9800. }
  9801. /* switch to the 20M Hz mode after disconnect */
  9802. pmlmeext->cur_bwmode = CHANNEL_WIDTH_20;
  9803. pmlmeext->cur_ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  9804. #ifdef CONFIG_FCS_MODE
  9805. if (EN_FCS(padapter))
  9806. rtw_hal_set_hwreg(padapter, HW_VAR_STOP_FCS_MODE, NULL);
  9807. #endif
  9808. if (!(MLME_IS_STA(padapter) && MLME_IS_OPCH_SW(padapter))) {
  9809. /* DFS and channel status no need to check here for STA under OPCH_SW */
  9810. u8 ch, bw, offset;
  9811. #ifdef CONFIG_DFS_MASTER
  9812. rtw_dfs_rd_en_decision(padapter, self_action, 0);
  9813. #endif
  9814. if (rtw_mi_get_ch_setting_union_no_self(padapter, &ch, &bw, &offset) != 0) {
  9815. set_channel_bwmode(padapter, ch, offset, bw);
  9816. rtw_mi_update_union_chan_inf(padapter, ch, offset, bw);
  9817. }
  9818. }
  9819. flush_all_cam_entry(padapter);
  9820. _cancel_timer_ex(&pmlmeext->link_timer);
  9821. /* pmlmepriv->LinkDetectInfo.TrafficBusyState = _FALSE; */
  9822. pmlmepriv->LinkDetectInfo.TrafficTransitionCount = 0;
  9823. pmlmepriv->LinkDetectInfo.LowPowerTransitionCount = 0;
  9824. #ifdef CONFIG_TDLS
  9825. padapter->tdlsinfo.ap_prohibited = _FALSE;
  9826. /* For TDLS channel switch, currently we only allow it to work in wifi logo test mode */
  9827. if (padapter->registrypriv.wifi_spec == 1)
  9828. padapter->tdlsinfo.ch_switch_prohibited = _FALSE;
  9829. #endif /* CONFIG_TDLS */
  9830. #ifdef CONFIG_WMMPS_STA
  9831. if (check_fwstate(pmlmepriv, WIFI_STATION_STATE)) {
  9832. /* reset currently related uapsd setting when the connection has broken */
  9833. pmlmepriv->qospriv.uapsd_max_sp_len = 0;
  9834. pmlmepriv->qospriv.uapsd_tid = 0;
  9835. pmlmepriv->qospriv.uapsd_tid_delivery_enabled = 0;
  9836. pmlmepriv->qospriv.uapsd_tid_trigger_enabled = 0;
  9837. pmlmepriv->qospriv.uapsd_ap_supported = 0;
  9838. }
  9839. #endif /* CONFIG_WMMPS_STA */
  9840. }
  9841. void mlmeext_joinbss_event_callback(_adapter *padapter, int join_res)
  9842. {
  9843. struct sta_info *psta;
  9844. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9845. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9846. WLAN_BSSID_EX *cur_network = &(pmlmeinfo->network);
  9847. struct sta_priv *pstapriv = &padapter->stapriv;
  9848. u8 join_type;
  9849. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  9850. #ifndef CONFIG_IOCTL_CFG80211
  9851. struct security_priv *psecuritypriv = &padapter->securitypriv;
  9852. #endif
  9853. if (pmlmepriv->wpa_phase == _TRUE)
  9854. pmlmepriv->wpa_phase = _FALSE;
  9855. if (join_res < 0) {
  9856. join_type = 1;
  9857. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_JOIN, (u8 *)(&join_type));
  9858. rtw_hal_set_hwreg(padapter, HW_VAR_BSSID, null_addr);
  9859. if ((pmlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE)
  9860. rtw_hal_rcr_set_chk_bssid(padapter, MLME_STA_DISCONNECTED);
  9861. goto exit_mlmeext_joinbss_event_callback;
  9862. }
  9863. #ifdef CONFIG_ARP_KEEP_ALIVE
  9864. pmlmepriv->bGetGateway = 1;
  9865. pmlmepriv->GetGatewayTryCnt = 0;
  9866. #endif
  9867. if ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) {
  9868. /* update bc/mc sta_info */
  9869. update_bmc_sta(padapter);
  9870. }
  9871. /* turn on dynamic functions */
  9872. /* Switch_DM_Func(padapter, DYNAMIC_ALL_FUNC_ENABLE, _TRUE); */
  9873. /* update IOT-releated issue */
  9874. update_IOT_info(padapter);
  9875. rtw_hal_set_hwreg(padapter, HW_VAR_BASIC_RATE, cur_network->SupportedRates);
  9876. /* BCN interval */
  9877. rtw_hal_set_hwreg(padapter, HW_VAR_BEACON_INTERVAL, (u8 *)(&pmlmeinfo->bcn_interval));
  9878. /* udpate capability */
  9879. update_capinfo(padapter, pmlmeinfo->capability);
  9880. /* WMM, Update EDCA param */
  9881. WMMOnAssocRsp(padapter);
  9882. #ifdef CONFIG_80211N_HT
  9883. /* HT */
  9884. HTOnAssocRsp(padapter);
  9885. #endif /* CONFIG_80211N_HT */
  9886. #ifdef CONFIG_80211AC_VHT
  9887. /* VHT */
  9888. VHTOnAssocRsp(padapter);
  9889. #endif
  9890. psta = rtw_get_stainfo(pstapriv, cur_network->MacAddress);
  9891. if (psta) { /* only for infra. mode */
  9892. psta->wireless_mode = pmlmeext->cur_wireless_mode;
  9893. /* set per sta rate after updating HT cap. */
  9894. set_sta_rate(padapter, psta);
  9895. rtw_sta_media_status_rpt(padapter, psta, 1);
  9896. /* wakeup macid after join bss successfully to ensure
  9897. the subsequent data frames can be sent out normally */
  9898. rtw_hal_macid_wakeup(padapter, psta->cmn.mac_id);
  9899. }
  9900. #ifndef CONFIG_IOCTL_CFG80211
  9901. if (is_wep_enc(psecuritypriv->dot11PrivacyAlgrthm))
  9902. rtw_sec_restore_wep_key(padapter);
  9903. #endif /* CONFIG_IOCTL_CFG80211 */
  9904. if (rtw_port_switch_chk(padapter) == _TRUE)
  9905. rtw_hal_set_hwreg(padapter, HW_VAR_PORT_SWITCH, NULL);
  9906. join_type = 2;
  9907. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_JOIN, (u8 *)(&join_type));
  9908. if ((pmlmeinfo->state & 0x03) == WIFI_FW_STATION_STATE) {
  9909. rtw_hal_rcr_set_chk_bssid(padapter, MLME_STA_CONNECTED);
  9910. /* correcting TSF */
  9911. correct_TSF(padapter, MLME_STA_CONNECTED);
  9912. /* set_link_timer(pmlmeext, DISCONNECT_TO); */
  9913. }
  9914. #ifdef CONFIG_LPS
  9915. #ifndef CONFIG_FW_MULTI_PORT_SUPPORT
  9916. if (get_hw_port(padapter) == HW_PORT0)
  9917. #endif
  9918. rtw_lps_ctrl_wk_cmd(padapter, LPS_CTRL_CONNECT, 0);
  9919. #endif
  9920. #ifdef CONFIG_BEAMFORMING
  9921. if (psta)
  9922. beamforming_wk_cmd(padapter, BEAMFORMING_CTRL_ENTER, (u8 *)psta, sizeof(struct sta_info), 0);
  9923. #endif/*CONFIG_BEAMFORMING*/
  9924. exit_mlmeext_joinbss_event_callback:
  9925. rtw_join_done_chk_ch(padapter, join_res);
  9926. #ifdef CONFIG_RTW_REPEATER_SON
  9927. rtw_rson_join_done(padapter);
  9928. #endif
  9929. RTW_INFO("=>%s - End to Connection without 4-way\n", __FUNCTION__);
  9930. }
  9931. /* currently only adhoc mode will go here */
  9932. void mlmeext_sta_add_event_callback(_adapter *padapter, struct sta_info *psta)
  9933. {
  9934. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  9935. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9936. u8 join_type;
  9937. RTW_INFO("%s\n", __FUNCTION__);
  9938. if ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) {
  9939. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS) { /* adhoc master or sta_count>1 */
  9940. /* nothing to do */
  9941. } else { /* adhoc client */
  9942. /* update TSF Value */
  9943. /* update_TSF(pmlmeext, pframe, len); */
  9944. /* correcting TSF */
  9945. correct_TSF(padapter, MLME_ADHOC_STARTED);
  9946. /* start beacon */
  9947. if (send_beacon(padapter) == _FAIL)
  9948. rtw_warn_on(1);
  9949. pmlmeinfo->state |= WIFI_FW_ASSOC_SUCCESS;
  9950. }
  9951. join_type = 2;
  9952. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_JOIN, (u8 *)(&join_type));
  9953. }
  9954. /* update adhoc sta_info */
  9955. update_sta_info(padapter, psta);
  9956. rtw_hal_update_sta_ra_info(padapter, psta);
  9957. /* ToDo: HT for Ad-hoc */
  9958. psta->wireless_mode = rtw_check_network_type(psta->bssrateset, psta->bssratelen, pmlmeext->cur_channel);
  9959. rtw_hal_set_odm_var(padapter, HAL_ODM_STA_INFO, psta, _TRUE);
  9960. /* rate radaptive */
  9961. Update_RA_Entry(padapter, psta);
  9962. }
  9963. void mlmeext_sta_del_event_callback(_adapter *padapter)
  9964. {
  9965. if (is_client_associated_to_ap(padapter) || is_IBSS_empty(padapter))
  9966. rtw_mlmeext_disconnect(padapter);
  9967. }
  9968. /****************************************************************************
  9969. Following are the functions for the timer handlers
  9970. *****************************************************************************/
  9971. void _linked_info_dump(_adapter *padapter)
  9972. {
  9973. if (padapter->bLinkInfoDump) {
  9974. rtw_hal_get_def_var(padapter, HW_DEF_RA_INFO_DUMP, RTW_DBGDUMP);
  9975. rtw_hal_set_odm_var(padapter, HAL_ODM_RX_INFO_DUMP, RTW_DBGDUMP, _FALSE);
  9976. }
  9977. }
  9978. /********************************************************************
  9979. When station does not receive any packet in MAX_CONTINUAL_NORXPACKET_COUNT*2 seconds,
  9980. recipient station will teardown the block ack by issuing DELBA frame.
  9981. *********************************************************************/
  9982. void rtw_delba_check(_adapter *padapter, struct sta_info *psta, u8 from_timer)
  9983. {
  9984. int i = 0;
  9985. int ret = _SUCCESS;
  9986. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  9987. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  9988. /*
  9989. IOT issue,occur Broadcom ap(Buffalo WZR-D1800H,Netgear R6300).
  9990. AP is originator.AP does not transmit unicast packets when STA response its BAR.
  9991. This case probably occur ap issue BAR after AP builds BA.
  9992. Follow 802.11 spec, STA shall maintain an inactivity timer for every negotiated Block Ack setup.
  9993. The inactivity timer is not reset when MPDUs corresponding to other TIDs are received.
  9994. */
  9995. if (pmlmeinfo->assoc_AP_vendor == HT_IOT_PEER_BROADCOM) {
  9996. for (i = 0; i < TID_NUM ; i++) {
  9997. if ((psta->recvreorder_ctrl[i].enable) &&
  9998. (sta_rx_data_qos_pkts(psta, i) == sta_last_rx_data_qos_pkts(psta, i)) ) {
  9999. if (_TRUE == rtw_inc_and_chk_continual_no_rx_packet(psta, i)) {
  10000. /* send a DELBA frame to the peer STA with the Reason Code field set to TIMEOUT */
  10001. if (!from_timer)
  10002. ret = issue_del_ba_ex(padapter, psta->cmn.mac_addr, i, 39, 0, 3, 1);
  10003. else
  10004. issue_del_ba(padapter, psta->cmn.mac_addr, i, 39, 0);
  10005. psta->recvreorder_ctrl[i].enable = _FALSE;
  10006. if (ret != _FAIL)
  10007. psta->recvreorder_ctrl[i].ampdu_size = RX_AMPDU_SIZE_INVALID;
  10008. rtw_reset_continual_no_rx_packet(psta, i);
  10009. }
  10010. } else {
  10011. /* The inactivity timer is reset when MPDUs to the TID is received. */
  10012. rtw_reset_continual_no_rx_packet(psta, i);
  10013. }
  10014. }
  10015. }
  10016. }
  10017. u8 chk_ap_is_alive(_adapter *padapter, struct sta_info *psta)
  10018. {
  10019. u8 ret = _FALSE;
  10020. #ifdef DBG_EXPIRATION_CHK
  10021. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10022. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10023. RTW_INFO(FUNC_ADPT_FMT" rx:"STA_PKTS_FMT", beacon:%llu, probersp_to_self:%llu"
  10024. /*", probersp_bm:%llu, probersp_uo:%llu, probereq:%llu, BI:%u"*/
  10025. ", retry:%u\n"
  10026. , FUNC_ADPT_ARG(padapter)
  10027. , STA_RX_PKTS_DIFF_ARG(psta)
  10028. , psta->sta_stats.rx_beacon_pkts - psta->sta_stats.last_rx_beacon_pkts
  10029. , psta->sta_stats.rx_probersp_pkts - psta->sta_stats.last_rx_probersp_pkts
  10030. /*, psta->sta_stats.rx_probersp_bm_pkts - psta->sta_stats.last_rx_probersp_bm_pkts
  10031. , psta->sta_stats.rx_probersp_uo_pkts - psta->sta_stats.last_rx_probersp_uo_pkts
  10032. , psta->sta_stats.rx_probereq_pkts - psta->sta_stats.last_rx_probereq_pkts
  10033. , pmlmeinfo->bcn_interval*/
  10034. , pmlmeext->retry
  10035. );
  10036. RTW_INFO(FUNC_ADPT_FMT" tx_pkts:%llu, link_count:%u\n", FUNC_ADPT_ARG(padapter)
  10037. , sta_tx_pkts(psta)
  10038. , pmlmeinfo->link_count
  10039. );
  10040. #endif
  10041. if ((sta_rx_data_pkts(psta) == sta_last_rx_data_pkts(psta))
  10042. && sta_rx_beacon_pkts(psta) == sta_last_rx_beacon_pkts(psta)
  10043. && sta_rx_probersp_pkts(psta) == sta_last_rx_probersp_pkts(psta)
  10044. )
  10045. ret = _FALSE;
  10046. else
  10047. ret = _TRUE;
  10048. sta_update_last_rx_pkts(psta);
  10049. return ret;
  10050. }
  10051. u8 chk_adhoc_peer_is_alive(struct sta_info *psta)
  10052. {
  10053. u8 ret = _TRUE;
  10054. #ifdef DBG_EXPIRATION_CHK
  10055. RTW_INFO("sta:"MAC_FMT", rssi:%d, rx:"STA_PKTS_FMT", beacon:%llu, probersp_to_self:%llu"
  10056. /*", probersp_bm:%llu, probersp_uo:%llu, probereq:%llu, BI:%u"*/
  10057. ", expire_to:%u\n"
  10058. , MAC_ARG(psta->cmn.mac_addr)
  10059. , psta->cmn.rssi_stat.rssi
  10060. , STA_RX_PKTS_DIFF_ARG(psta)
  10061. , psta->sta_stats.rx_beacon_pkts - psta->sta_stats.last_rx_beacon_pkts
  10062. , psta->sta_stats.rx_probersp_pkts - psta->sta_stats.last_rx_probersp_pkts
  10063. /*, psta->sta_stats.rx_probersp_bm_pkts - psta->sta_stats.last_rx_probersp_bm_pkts
  10064. , psta->sta_stats.rx_probersp_uo_pkts - psta->sta_stats.last_rx_probersp_uo_pkts
  10065. , psta->sta_stats.rx_probereq_pkts - psta->sta_stats.last_rx_probereq_pkts
  10066. , pmlmeinfo->bcn_interval*/
  10067. , psta->expire_to
  10068. );
  10069. #endif
  10070. if (sta_rx_data_pkts(psta) == sta_last_rx_data_pkts(psta)
  10071. && sta_rx_beacon_pkts(psta) == sta_last_rx_beacon_pkts(psta)
  10072. && sta_rx_probersp_pkts(psta) == sta_last_rx_probersp_pkts(psta))
  10073. ret = _FALSE;
  10074. sta_update_last_rx_pkts(psta);
  10075. return ret;
  10076. }
  10077. #ifdef CONFIG_TDLS
  10078. u8 chk_tdls_peer_sta_is_alive(_adapter *padapter, struct sta_info *psta)
  10079. {
  10080. if ((psta->sta_stats.rx_data_pkts == psta->sta_stats.last_rx_data_pkts)
  10081. && (psta->sta_stats.rx_tdls_disc_rsp_pkts == psta->sta_stats.last_rx_tdls_disc_rsp_pkts))
  10082. return _FALSE;
  10083. return _TRUE;
  10084. }
  10085. void linked_status_chk_tdls(_adapter *padapter)
  10086. {
  10087. struct candidate_pool {
  10088. struct sta_info *psta;
  10089. u8 addr[ETH_ALEN];
  10090. };
  10091. struct sta_priv *pstapriv = &padapter->stapriv;
  10092. _irqL irqL;
  10093. u8 ack_chk;
  10094. struct sta_info *psta;
  10095. int i, num_teardown = 0, num_checkalive = 0;
  10096. _list *plist, *phead;
  10097. struct tdls_txmgmt txmgmt;
  10098. struct candidate_pool checkalive[MAX_ALLOWED_TDLS_STA_NUM];
  10099. struct candidate_pool teardown[MAX_ALLOWED_TDLS_STA_NUM];
  10100. u8 tdls_sta_max = _FALSE;
  10101. #define ALIVE_MIN 2
  10102. #define ALIVE_MAX 5
  10103. _rtw_memset(&txmgmt, 0x00, sizeof(struct tdls_txmgmt));
  10104. _rtw_memset(checkalive, 0x00, sizeof(checkalive));
  10105. _rtw_memset(teardown, 0x00, sizeof(teardown));
  10106. if ((padapter->tdlsinfo.link_established == _TRUE)) {
  10107. _enter_critical_bh(&pstapriv->sta_hash_lock, &irqL);
  10108. for (i = 0; i < NUM_STA; i++) {
  10109. phead = &(pstapriv->sta_hash[i]);
  10110. plist = get_next(phead);
  10111. while ((rtw_end_of_queue_search(phead, plist)) == _FALSE) {
  10112. psta = LIST_CONTAINOR(plist, struct sta_info, hash_list);
  10113. plist = get_next(plist);
  10114. if (psta->tdls_sta_state & TDLS_LINKED_STATE) {
  10115. psta->alive_count++;
  10116. if (psta->alive_count >= ALIVE_MIN) {
  10117. if (chk_tdls_peer_sta_is_alive(padapter, psta) == _FALSE) {
  10118. if (psta->alive_count < ALIVE_MAX) {
  10119. _rtw_memcpy(checkalive[num_checkalive].addr, psta->cmn.mac_addr, ETH_ALEN);
  10120. checkalive[num_checkalive].psta = psta;
  10121. num_checkalive++;
  10122. } else {
  10123. _rtw_memcpy(teardown[num_teardown].addr, psta->cmn.mac_addr, ETH_ALEN);
  10124. teardown[num_teardown].psta = psta;
  10125. num_teardown++;
  10126. }
  10127. } else
  10128. psta->alive_count = 0;
  10129. }
  10130. psta->sta_stats.last_rx_data_pkts = psta->sta_stats.rx_data_pkts;
  10131. psta->sta_stats.last_rx_tdls_disc_rsp_pkts = psta->sta_stats.rx_tdls_disc_rsp_pkts;
  10132. if ((num_checkalive >= MAX_ALLOWED_TDLS_STA_NUM) || (num_teardown >= MAX_ALLOWED_TDLS_STA_NUM)) {
  10133. tdls_sta_max = _TRUE;
  10134. break;
  10135. }
  10136. }
  10137. }
  10138. if (tdls_sta_max == _TRUE)
  10139. break;
  10140. }
  10141. _exit_critical_bh(&pstapriv->sta_hash_lock, &irqL);
  10142. if (num_checkalive > 0) {
  10143. for (i = 0; i < num_checkalive; i++) {
  10144. _rtw_memcpy(txmgmt.peer, checkalive[i].addr, ETH_ALEN);
  10145. issue_tdls_dis_req(padapter, &txmgmt);
  10146. issue_tdls_dis_req(padapter, &txmgmt);
  10147. issue_tdls_dis_req(padapter, &txmgmt);
  10148. }
  10149. }
  10150. if (num_teardown > 0) {
  10151. for (i = 0; i < num_teardown; i++) {
  10152. RTW_INFO("[%s %d] Send teardown to "MAC_FMT"\n", __FUNCTION__, __LINE__, MAC_ARG(teardown[i].addr));
  10153. txmgmt.status_code = _RSON_TDLS_TEAR_TOOFAR_;
  10154. _rtw_memcpy(txmgmt.peer, teardown[i].addr, ETH_ALEN);
  10155. issue_tdls_teardown(padapter, &txmgmt, _FALSE);
  10156. }
  10157. }
  10158. }
  10159. }
  10160. #endif /* CONFIG_TDLS */
  10161. /* from_timer == 1 means driver is in LPS */
  10162. void linked_status_chk(_adapter *padapter, u8 from_timer)
  10163. {
  10164. u32 i;
  10165. struct sta_info *psta;
  10166. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10167. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10168. struct sta_priv *pstapriv = &padapter->stapriv;
  10169. #if defined(CONFIG_ARP_KEEP_ALIVE) || defined(CONFIG_LAYER2_ROAMING)
  10170. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  10171. #endif
  10172. #ifdef CONFIG_LAYER2_ROAMING
  10173. struct recv_priv *precvpriv = &padapter->recvpriv;
  10174. #endif
  10175. if (padapter->registrypriv.mp_mode == _TRUE)
  10176. return;
  10177. if (is_client_associated_to_ap(padapter)) {
  10178. /* linked infrastructure client mode */
  10179. int tx_chk = _SUCCESS, rx_chk = _SUCCESS;
  10180. int rx_chk_limit;
  10181. int link_count_limit;
  10182. #if defined(CONFIG_RTW_REPEATER_SON)
  10183. rtw_rson_scan_wk_cmd(padapter, RSON_SCAN_PROCESS);
  10184. #elif defined(CONFIG_LAYER2_ROAMING)
  10185. if (rtw_chk_roam_flags(padapter, RTW_ROAM_ACTIVE)) {
  10186. RTW_INFO("signal_strength_data.avg_val = %d\n", precvpriv->signal_strength_data.avg_val);
  10187. if ((precvpriv->signal_strength_data.avg_val < pmlmepriv->roam_rssi_threshold)
  10188. && (rtw_get_passing_time_ms(pmlmepriv->last_roaming) >= pmlmepriv->roam_scan_int*2000)) {
  10189. #ifdef CONFIG_RTW_80211K
  10190. rtw_roam_nb_discover(padapter, _FALSE);
  10191. #endif
  10192. pmlmepriv->need_to_roam = _TRUE;
  10193. rtw_drv_scan_by_self(padapter, RTW_AUTO_SCAN_REASON_ROAM);
  10194. pmlmepriv->last_roaming = rtw_get_current_time();
  10195. } else
  10196. pmlmepriv->need_to_roam = _FALSE;
  10197. }
  10198. #endif
  10199. #ifdef CONFIG_MCC_MODE
  10200. /*
  10201. * due to tx ps null date to ao, so ap doest not tx pkt to driver
  10202. * we may check chk_ap_is_alive fail, and may issue_probereq to wrong channel under sitesurvey
  10203. * don't keep alive check under MCC
  10204. */
  10205. if (rtw_hal_mcc_link_status_chk(padapter, __func__) == _FALSE)
  10206. return;
  10207. #endif
  10208. #if defined(DBG_ROAMING_TEST) || defined(CONFIG_RTW_REPEATER_SON)
  10209. rx_chk_limit = 1;
  10210. #elif defined(CONFIG_ACTIVE_KEEP_ALIVE_CHECK) && !defined(CONFIG_LPS_LCLK_WD_TIMER)
  10211. rx_chk_limit = 4;
  10212. #else
  10213. rx_chk_limit = 8;
  10214. #endif
  10215. #ifdef CONFIG_ARP_KEEP_ALIVE
  10216. if (!from_timer && pmlmepriv->bGetGateway == 1 && pmlmepriv->GetGatewayTryCnt < 3) {
  10217. RTW_INFO("do rtw_gw_addr_query() : %d\n", pmlmepriv->GetGatewayTryCnt);
  10218. pmlmepriv->GetGatewayTryCnt++;
  10219. if (rtw_gw_addr_query(padapter) == 0)
  10220. pmlmepriv->bGetGateway = 0;
  10221. else {
  10222. _rtw_memset(pmlmepriv->gw_ip, 0, 4);
  10223. _rtw_memset(pmlmepriv->gw_mac_addr, 0, ETH_ALEN);
  10224. }
  10225. }
  10226. #endif
  10227. #ifdef CONFIG_P2P
  10228. if (!rtw_p2p_chk_state(&padapter->wdinfo, P2P_STATE_NONE)) {
  10229. if (!from_timer)
  10230. link_count_limit = 3; /* 8 sec */
  10231. else
  10232. link_count_limit = 15; /* 32 sec */
  10233. } else
  10234. #endif /* CONFIG_P2P */
  10235. {
  10236. if (!from_timer)
  10237. link_count_limit = 7; /* 16 sec */
  10238. else
  10239. link_count_limit = 29; /* 60 sec */
  10240. }
  10241. #ifdef CONFIG_TDLS
  10242. #ifdef CONFIG_TDLS_CH_SW
  10243. if (ATOMIC_READ(&padapter->tdlsinfo.chsw_info.chsw_on) == _TRUE)
  10244. return;
  10245. #endif /* CONFIG_TDLS_CH_SW */
  10246. #ifdef CONFIG_TDLS_AUTOCHECKALIVE
  10247. linked_status_chk_tdls(padapter);
  10248. #endif /* CONFIG_TDLS_AUTOCHECKALIVE */
  10249. #endif /* CONFIG_TDLS */
  10250. psta = rtw_get_stainfo(pstapriv, pmlmeinfo->network.MacAddress);
  10251. if (psta != NULL) {
  10252. bool is_p2p_enable = _FALSE;
  10253. #ifdef CONFIG_P2P
  10254. is_p2p_enable = !rtw_p2p_chk_state(&padapter->wdinfo, P2P_STATE_NONE);
  10255. #endif
  10256. #ifdef CONFIG_ISSUE_DELBA_WHEN_NO_TRAFFIC
  10257. /*issue delba when ap does not tx data packet that is Broadcom ap */
  10258. rtw_delba_check(padapter, psta, from_timer);
  10259. #endif
  10260. if (chk_ap_is_alive(padapter, psta) == _FALSE)
  10261. rx_chk = _FAIL;
  10262. if (sta_last_tx_pkts(psta) == sta_tx_pkts(psta))
  10263. tx_chk = _FAIL;
  10264. #ifdef CONFIG_ACTIVE_KEEP_ALIVE_CHECK
  10265. if (!from_timer && pmlmeext->active_keep_alive_check && (rx_chk == _FAIL || tx_chk == _FAIL)
  10266. ) {
  10267. u8 backup_ch = 0, backup_bw = 0, backup_offset = 0;
  10268. u8 union_ch = 0, union_bw = 0, union_offset = 0;
  10269. u8 switch_channel_by_drv = _TRUE;
  10270. #ifdef CONFIG_MCC_MODE
  10271. if (MCC_EN(padapter)) {
  10272. /* driver doesn't switch channel under MCC */
  10273. if (rtw_hal_check_mcc_status(padapter, MCC_STATUS_DOING_MCC))
  10274. switch_channel_by_drv = _FALSE;
  10275. }
  10276. #endif
  10277. if (switch_channel_by_drv) {
  10278. if (!rtw_mi_get_ch_setting_union(padapter, &union_ch, &union_bw, &union_offset)
  10279. || pmlmeext->cur_channel != union_ch)
  10280. goto bypass_active_keep_alive;
  10281. /* switch to correct channel of current network before issue keep-alive frames */
  10282. if (rtw_get_oper_ch(padapter) != pmlmeext->cur_channel) {
  10283. backup_ch = rtw_get_oper_ch(padapter);
  10284. backup_bw = rtw_get_oper_bw(padapter);
  10285. backup_offset = rtw_get_oper_choffset(padapter);
  10286. set_channel_bwmode(padapter, union_ch, union_offset, union_bw);
  10287. }
  10288. }
  10289. if (rx_chk != _SUCCESS)
  10290. issue_probereq_ex(padapter, &pmlmeinfo->network.Ssid, psta->cmn.mac_addr, 0, 0, 3, 1);
  10291. if ((tx_chk != _SUCCESS && pmlmeinfo->link_count++ == link_count_limit) || rx_chk != _SUCCESS) {
  10292. if (rtw_mi_check_fwstate(padapter, _FW_UNDER_SURVEY))
  10293. tx_chk = issue_nulldata(padapter, psta->cmn.mac_addr, 1, 3, 1);
  10294. else
  10295. tx_chk = issue_nulldata(padapter, psta->cmn.mac_addr, 0, 3, 1);
  10296. /* if tx acked and p2p disabled, set rx_chk _SUCCESS to reset retry count */
  10297. if (tx_chk == _SUCCESS && !is_p2p_enable)
  10298. rx_chk = _SUCCESS;
  10299. }
  10300. /* back to the original operation channel */
  10301. if (backup_ch > 0 && switch_channel_by_drv)
  10302. set_channel_bwmode(padapter, backup_ch, backup_offset, backup_bw);
  10303. bypass_active_keep_alive:
  10304. ;
  10305. } else
  10306. #endif /* CONFIG_ACTIVE_KEEP_ALIVE_CHECK */
  10307. {
  10308. if (rx_chk != _SUCCESS) {
  10309. if (pmlmeext->retry == 0) {
  10310. #ifdef DBG_EXPIRATION_CHK
  10311. RTW_INFO("issue_probereq to trigger probersp, retry=%d\n", pmlmeext->retry);
  10312. #endif
  10313. issue_probereq_ex(padapter, &pmlmeinfo->network.Ssid, pmlmeinfo->network.MacAddress, 0, 0, 0, (from_timer ? 0 : 1));
  10314. issue_probereq_ex(padapter, &pmlmeinfo->network.Ssid, pmlmeinfo->network.MacAddress, 0, 0, 0, (from_timer ? 0 : 1));
  10315. issue_probereq_ex(padapter, &pmlmeinfo->network.Ssid, pmlmeinfo->network.MacAddress, 0, 0, 0, (from_timer ? 0 : 1));
  10316. }
  10317. }
  10318. if (tx_chk != _SUCCESS && pmlmeinfo->link_count++ == link_count_limit
  10319. #ifdef CONFIG_MCC_MODE
  10320. /* FW tx nulldata under MCC mode, we just check ap is alive */
  10321. && (!rtw_hal_check_mcc_status(padapter, MCC_STATUS_NEED_MCC))
  10322. #endif /* CONFIG_MCC_MODE */
  10323. ) {
  10324. #ifdef DBG_EXPIRATION_CHK
  10325. RTW_INFO("%s issue_nulldata(%d)\n", __FUNCTION__, from_timer ? 1 : 0);
  10326. #endif
  10327. if (from_timer || rtw_mi_check_fwstate(padapter, _FW_UNDER_SURVEY))
  10328. tx_chk = issue_nulldata(padapter, NULL, 1, 0, 0);
  10329. else
  10330. tx_chk = issue_nulldata(padapter, NULL, 0, 1, 1);
  10331. }
  10332. }
  10333. if (rx_chk == _FAIL) {
  10334. pmlmeext->retry++;
  10335. if (pmlmeext->retry > rx_chk_limit) {
  10336. RTW_PRINT(FUNC_ADPT_FMT" disconnect or roaming\n",
  10337. FUNC_ADPT_ARG(padapter));
  10338. receive_disconnect(padapter, pmlmeinfo->network.MacAddress
  10339. , WLAN_REASON_EXPIRATION_CHK, _FALSE);
  10340. return;
  10341. }
  10342. } else
  10343. pmlmeext->retry = 0;
  10344. if (tx_chk == _FAIL)
  10345. pmlmeinfo->link_count %= (link_count_limit + 1);
  10346. else {
  10347. psta->sta_stats.last_tx_pkts = psta->sta_stats.tx_pkts;
  10348. pmlmeinfo->link_count = 0;
  10349. }
  10350. } /* end of if ((psta = rtw_get_stainfo(pstapriv, passoc_res->network.MacAddress)) != NULL) */
  10351. } else if (is_client_associated_to_ibss(padapter)) {
  10352. _irqL irqL;
  10353. _list *phead, *plist, dlist;
  10354. _rtw_init_listhead(&dlist);
  10355. _enter_critical_bh(&pstapriv->sta_hash_lock, &irqL);
  10356. for (i = 0; i < NUM_STA; i++) {
  10357. phead = &(pstapriv->sta_hash[i]);
  10358. plist = get_next(phead);
  10359. while ((rtw_end_of_queue_search(phead, plist)) == _FALSE) {
  10360. psta = LIST_CONTAINOR(plist, struct sta_info, hash_list);
  10361. plist = get_next(plist);
  10362. if (is_broadcast_mac_addr(psta->cmn.mac_addr))
  10363. continue;
  10364. if (chk_adhoc_peer_is_alive(psta) || !psta->expire_to)
  10365. psta->expire_to = pstapriv->adhoc_expire_to;
  10366. else
  10367. psta->expire_to--;
  10368. if (psta->expire_to <= 0) {
  10369. rtw_list_delete(&psta->list);
  10370. rtw_list_insert_tail(&psta->list, &dlist);
  10371. }
  10372. }
  10373. }
  10374. _exit_critical_bh(&pstapriv->sta_hash_lock, &irqL);
  10375. plist = get_next(&dlist);
  10376. while (rtw_end_of_queue_search(&dlist, plist) == _FALSE) {
  10377. psta = LIST_CONTAINOR(plist, struct sta_info, list);
  10378. plist = get_next(plist);
  10379. rtw_list_delete(&psta->list);
  10380. RTW_INFO(FUNC_ADPT_FMT" ibss expire "MAC_FMT"\n"
  10381. , FUNC_ADPT_ARG(padapter), MAC_ARG(psta->cmn.mac_addr));
  10382. report_del_sta_event(padapter, psta->cmn.mac_addr, WLAN_REASON_EXPIRATION_CHK, from_timer ? _TRUE : _FALSE, _FALSE);
  10383. }
  10384. }
  10385. }
  10386. void survey_timer_hdl(void *ctx)
  10387. {
  10388. _adapter *padapter = (_adapter *)ctx;
  10389. struct cmd_obj *cmd;
  10390. struct sitesurvey_parm *psurveyPara;
  10391. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  10392. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10393. if (mlmeext_scan_state(pmlmeext) > SCAN_DISABLE) {
  10394. cmd = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  10395. if (cmd == NULL) {
  10396. rtw_warn_on(1);
  10397. goto exit;
  10398. }
  10399. psurveyPara = (struct sitesurvey_parm *)rtw_zmalloc(sizeof(struct sitesurvey_parm));
  10400. if (psurveyPara == NULL) {
  10401. rtw_warn_on(1);
  10402. rtw_mfree((unsigned char *)cmd, sizeof(struct cmd_obj));
  10403. goto exit;
  10404. }
  10405. init_h2fwcmd_w_parm_no_rsp(cmd, psurveyPara, GEN_CMD_CODE(_SiteSurvey));
  10406. rtw_enqueue_cmd(pcmdpriv, cmd);
  10407. }
  10408. exit:
  10409. return;
  10410. }
  10411. #ifdef CONFIG_RTW_REPEATER_SON
  10412. /* 100ms pass, stop rson_scan */
  10413. void rson_timer_hdl(void *ctx)
  10414. {
  10415. _adapter *padapter = (_adapter *)ctx;
  10416. rtw_rson_scan_wk_cmd(padapter, RSON_SCAN_DISABLE);
  10417. }
  10418. #endif
  10419. void link_timer_hdl(void *ctx)
  10420. {
  10421. _adapter *padapter = (_adapter *)ctx;
  10422. /* static unsigned int rx_pkt = 0; */
  10423. /* static u64 tx_cnt = 0; */
  10424. /* struct xmit_priv *pxmitpriv = &(padapter->xmitpriv); */
  10425. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10426. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10427. /* struct sta_priv *pstapriv = &padapter->stapriv; */
  10428. #ifdef CONFIG_RTW_80211R
  10429. struct sta_priv *pstapriv = &padapter->stapriv;
  10430. struct sta_info *psta = NULL;
  10431. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  10432. #endif
  10433. if (rtw_sta_linking_test_force_fail())
  10434. RTW_INFO("rtw_sta_linking_test_force_fail\n");
  10435. if (pmlmeext->join_abort && pmlmeinfo->state != WIFI_FW_NULL_STATE) {
  10436. RTW_INFO(FUNC_ADPT_FMT" join abort\n", FUNC_ADPT_ARG(padapter));
  10437. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  10438. report_join_res(padapter, -4, WLAN_STATUS_UNSPECIFIED_FAILURE);
  10439. goto exit;
  10440. }
  10441. if (pmlmeinfo->state & WIFI_FW_AUTH_NULL) {
  10442. RTW_INFO("link_timer_hdl:no beacon while connecting\n");
  10443. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  10444. report_join_res(padapter, -3, WLAN_STATUS_UNSPECIFIED_FAILURE);
  10445. } else if (pmlmeinfo->state & WIFI_FW_AUTH_STATE) {
  10446. /* re-auth timer */
  10447. if (++pmlmeinfo->reauth_count > REAUTH_LIMIT) {
  10448. /* if (pmlmeinfo->auth_algo != dot11AuthAlgrthm_Auto) */
  10449. /* { */
  10450. pmlmeinfo->state = 0;
  10451. if (pmlmeinfo->auth_status) {
  10452. report_join_res(padapter, -1, pmlmeinfo->auth_status);
  10453. pmlmeinfo->auth_status = 0; /* reset */
  10454. } else
  10455. report_join_res(padapter, -1, WLAN_STATUS_UNSPECIFIED_FAILURE);
  10456. return;
  10457. /* } */
  10458. /* else */
  10459. /* { */
  10460. /* pmlmeinfo->auth_algo = dot11AuthAlgrthm_Shared; */
  10461. /* pmlmeinfo->reauth_count = 0; */
  10462. /* } */
  10463. }
  10464. RTW_INFO("link_timer_hdl: auth timeout and try again\n");
  10465. pmlmeinfo->auth_seq = 1;
  10466. issue_auth(padapter, NULL, 0);
  10467. set_link_timer(pmlmeext, REAUTH_TO);
  10468. } else if (pmlmeinfo->state & WIFI_FW_ASSOC_STATE) {
  10469. /* re-assoc timer */
  10470. if (++pmlmeinfo->reassoc_count > REASSOC_LIMIT) {
  10471. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  10472. #ifdef CONFIG_RTW_80211R
  10473. if (rtw_ft_roam(padapter)) {
  10474. psta = rtw_get_stainfo(pstapriv, pmlmeinfo->network.MacAddress);
  10475. if (psta)
  10476. rtw_free_stainfo(padapter, psta);
  10477. }
  10478. #endif
  10479. report_join_res(padapter, -2, WLAN_STATUS_UNSPECIFIED_FAILURE);
  10480. return;
  10481. }
  10482. #ifdef CONFIG_RTW_80211R
  10483. if (rtw_ft_roam(padapter)) {
  10484. RTW_INFO("link_timer_hdl: reassoc timeout and try again\n");
  10485. issue_reassocreq(padapter);
  10486. } else
  10487. #endif
  10488. {
  10489. RTW_INFO("link_timer_hdl: assoc timeout and try again\n");
  10490. issue_assocreq(padapter);
  10491. }
  10492. set_link_timer(pmlmeext, REASSOC_TO);
  10493. }
  10494. exit:
  10495. return;
  10496. }
  10497. void addba_timer_hdl(void *ctx)
  10498. {
  10499. struct sta_info *psta = (struct sta_info *)ctx;
  10500. #ifdef CONFIG_80211N_HT
  10501. struct ht_priv *phtpriv;
  10502. if (!psta)
  10503. return;
  10504. phtpriv = &psta->htpriv;
  10505. if ((phtpriv->ht_option == _TRUE) && (phtpriv->ampdu_enable == _TRUE)) {
  10506. if (phtpriv->candidate_tid_bitmap)
  10507. phtpriv->candidate_tid_bitmap = 0x0;
  10508. }
  10509. #endif /* CONFIG_80211N_HT */
  10510. }
  10511. #ifdef CONFIG_IEEE80211W
  10512. void report_sta_timeout_event(_adapter *padapter, u8 *MacAddr, unsigned short reason)
  10513. {
  10514. struct cmd_obj *pcmd_obj;
  10515. u8 *pevtcmd;
  10516. u32 cmdsz;
  10517. struct sta_info *psta;
  10518. int mac_id;
  10519. struct stadel_event *pdel_sta_evt;
  10520. struct C2HEvent_Header *pc2h_evt_hdr;
  10521. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10522. struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
  10523. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  10524. if (pcmd_obj == NULL)
  10525. return;
  10526. cmdsz = (sizeof(struct stadel_event) + sizeof(struct C2HEvent_Header));
  10527. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  10528. if (pevtcmd == NULL) {
  10529. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  10530. return;
  10531. }
  10532. _rtw_init_listhead(&pcmd_obj->list);
  10533. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  10534. pcmd_obj->cmdsz = cmdsz;
  10535. pcmd_obj->parmbuf = pevtcmd;
  10536. pcmd_obj->rsp = NULL;
  10537. pcmd_obj->rspsz = 0;
  10538. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  10539. pc2h_evt_hdr->len = sizeof(struct stadel_event);
  10540. pc2h_evt_hdr->ID = GEN_EVT_CODE(_TimeoutSTA);
  10541. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  10542. pdel_sta_evt = (struct stadel_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  10543. _rtw_memcpy((unsigned char *)(&(pdel_sta_evt->macaddr)), MacAddr, ETH_ALEN);
  10544. _rtw_memcpy((unsigned char *)(pdel_sta_evt->rsvd), (unsigned char *)(&reason), 2);
  10545. psta = rtw_get_stainfo(&padapter->stapriv, MacAddr);
  10546. if (psta)
  10547. mac_id = (int)psta->cmn.mac_id;
  10548. else
  10549. mac_id = (-1);
  10550. pdel_sta_evt->mac_id = mac_id;
  10551. RTW_INFO("report_del_sta_event: delete STA, mac_id=%d\n", mac_id);
  10552. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  10553. return;
  10554. }
  10555. void clnt_sa_query_timeout(_adapter *padapter)
  10556. {
  10557. struct mlme_ext_priv *mlmeext = &(padapter->mlmeextpriv);
  10558. struct mlme_ext_info *mlmeinfo = &(mlmeext->mlmext_info);
  10559. RTW_INFO(FUNC_ADPT_FMT"\n", FUNC_ADPT_ARG(padapter));
  10560. receive_disconnect(padapter, get_my_bssid(&(mlmeinfo->network)), WLAN_REASON_SA_QUERY_TIMEOUT, _FALSE);
  10561. }
  10562. void sa_query_timer_hdl(void *ctx)
  10563. {
  10564. struct sta_info *psta = (struct sta_info *)ctx;
  10565. _adapter *padapter = psta->padapter;
  10566. _irqL irqL;
  10567. struct sta_priv *pstapriv = &padapter->stapriv;
  10568. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  10569. if (check_fwstate(pmlmepriv, WIFI_STATION_STATE) == _TRUE &&
  10570. check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE)
  10571. clnt_sa_query_timeout(padapter);
  10572. else if (check_fwstate(pmlmepriv, WIFI_AP_STATE) == _TRUE)
  10573. report_sta_timeout_event(padapter, psta->cmn.mac_addr, WLAN_REASON_PREV_AUTH_NOT_VALID);
  10574. }
  10575. #endif /* CONFIG_IEEE80211W */
  10576. #ifdef CONFIG_RTW_80211R
  10577. void rtw_ft_update_bcn(_adapter *padapter, union recv_frame *precv_frame)
  10578. {
  10579. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10580. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10581. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  10582. u8 *pframe = precv_frame->u.hdr.rx_data;
  10583. uint len = precv_frame->u.hdr.len;
  10584. WLAN_BSSID_EX *pbss;
  10585. if (rtw_ft_chk_status(padapter,RTW_FT_ASSOCIATED_STA)
  10586. && (pmlmepriv->ft_roam.ft_updated_bcn == _FALSE)) {
  10587. pbss = (WLAN_BSSID_EX*)rtw_malloc(sizeof(WLAN_BSSID_EX));
  10588. if (pbss) {
  10589. if (collect_bss_info(padapter, precv_frame, pbss) == _SUCCESS) {
  10590. struct beacon_keys recv_beacon;
  10591. update_network(&(pmlmepriv->cur_network.network), pbss, padapter, _TRUE);
  10592. rtw_get_bcn_info(&(pmlmepriv->cur_network));
  10593. /* update bcn keys */
  10594. if (rtw_get_bcn_keys(padapter, pframe, len, &recv_beacon) == _TRUE) {
  10595. RTW_INFO("%s: beacon keys ready\n", __func__);
  10596. _rtw_memcpy(&pmlmepriv->cur_beacon_keys,
  10597. &recv_beacon, sizeof(recv_beacon));
  10598. } else {
  10599. RTW_ERR("%s: get beacon keys failed\n", __func__);
  10600. _rtw_memset(&pmlmepriv->cur_beacon_keys, 0, sizeof(recv_beacon));
  10601. }
  10602. #ifdef CONFIG_BCN_CNT_CONFIRM_HDL
  10603. pmlmepriv->new_beacon_cnts = 0;
  10604. #endif
  10605. }
  10606. rtw_mfree((u8*)pbss, sizeof(WLAN_BSSID_EX));
  10607. }
  10608. /* check the vendor of the assoc AP */
  10609. pmlmeinfo->assoc_AP_vendor =
  10610. check_assoc_AP(pframe+sizeof(struct rtw_ieee80211_hdr_3addr),
  10611. (len - sizeof(struct rtw_ieee80211_hdr_3addr)));
  10612. /* update TSF Value */
  10613. update_TSF(pmlmeext, pframe, len);
  10614. pmlmeext->bcn_cnt = 0;
  10615. pmlmeext->last_bcn_cnt = 0;
  10616. pmlmepriv->ft_roam.ft_updated_bcn = _TRUE;
  10617. }
  10618. }
  10619. void rtw_ft_start_clnt_join(_adapter *padapter)
  10620. {
  10621. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  10622. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10623. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10624. struct ft_roam_info *pft_roam = &(pmlmepriv->ft_roam);
  10625. if (rtw_ft_otd_roam(padapter)) {
  10626. pmlmeinfo->state = WIFI_FW_AUTH_SUCCESS | WIFI_FW_STATION_STATE;
  10627. pft_roam->ft_event.ies =
  10628. (pft_roam->ft_action + sizeof(struct rtw_ieee80211_hdr_3addr) + 16);
  10629. pft_roam->ft_event.ies_len =
  10630. (pft_roam->ft_action_len - sizeof(struct rtw_ieee80211_hdr_3addr));
  10631. /*Not support RIC*/
  10632. pft_roam->ft_event.ric_ies = NULL;
  10633. pft_roam->ft_event.ric_ies_len = 0;
  10634. rtw_ft_report_evt(padapter);
  10635. return;
  10636. }
  10637. pmlmeinfo->state = WIFI_FW_AUTH_NULL | WIFI_FW_STATION_STATE;
  10638. start_clnt_auth(padapter);
  10639. }
  10640. u8 rtw_ft_update_rsnie(
  10641. _adapter *padapter, u8 bwrite,
  10642. struct pkt_attrib *pattrib, u8 **pframe)
  10643. {
  10644. struct ft_roam_info *pft_roam = &(padapter->mlmepriv.ft_roam);
  10645. u8 *pie;
  10646. u32 len;
  10647. pie = rtw_get_ie(pft_roam->updated_ft_ies, EID_WPA2, &len,
  10648. pft_roam->updated_ft_ies_len);
  10649. if (!bwrite)
  10650. return (pie)?_SUCCESS:_FAIL;
  10651. if (pie) {
  10652. *pframe = rtw_set_ie(((u8 *)*pframe), EID_WPA2, len,
  10653. pie+2, &(pattrib->pktlen));
  10654. } else
  10655. return _FAIL;
  10656. return _SUCCESS;
  10657. }
  10658. static u8 rtw_ft_update_mdie(
  10659. _adapter *padapter, struct pkt_attrib *pattrib, u8 **pframe)
  10660. {
  10661. struct ft_roam_info *pft_roam = &(padapter->mlmepriv.ft_roam);
  10662. u8 *pie, mdie[3];
  10663. u32 len = 3;
  10664. if (rtw_ft_roam(padapter)) {
  10665. if ((pie = rtw_get_ie(pft_roam->updated_ft_ies, _MDIE_,
  10666. &len, pft_roam->updated_ft_ies_len))) {
  10667. pie = (pie + 2); /* ignore md-id & length */
  10668. } else
  10669. return _FAIL;
  10670. } else {
  10671. *((u16 *)&mdie[0]) = pft_roam->mdid;
  10672. mdie[2] = pft_roam->ft_cap;
  10673. pie = &mdie[0];
  10674. }
  10675. *pframe = rtw_set_ie(((u8 *)*pframe), _MDIE_, len , pie, &(pattrib->pktlen));
  10676. return _SUCCESS;
  10677. }
  10678. static u8 rtw_ft_update_ftie(
  10679. _adapter *padapter, struct pkt_attrib *pattrib, u8 **pframe)
  10680. {
  10681. struct ft_roam_info *pft_roam = &(padapter->mlmepriv.ft_roam);
  10682. u8 *pie;
  10683. u32 len;
  10684. if ((pie = rtw_get_ie(pft_roam->updated_ft_ies, _FTIE_, &len,
  10685. pft_roam->updated_ft_ies_len)) != NULL) {
  10686. *pframe = rtw_set_ie(*pframe, _FTIE_, len ,
  10687. (pie+2), &(pattrib->pktlen));
  10688. } else
  10689. return _FAIL;
  10690. return _SUCCESS;
  10691. }
  10692. void rtw_ft_build_auth_req_ies(_adapter *padapter,
  10693. struct pkt_attrib *pattrib, u8 **pframe)
  10694. {
  10695. u8 ftie_append = _TRUE;
  10696. if (!pattrib || !(*pframe))
  10697. return;
  10698. if (!rtw_ft_roam(padapter))
  10699. return;
  10700. ftie_append = rtw_ft_update_rsnie(padapter, _TRUE, pattrib, pframe);
  10701. rtw_ft_update_mdie(padapter, pattrib, pframe);
  10702. if (ftie_append)
  10703. rtw_ft_update_ftie(padapter, pattrib, pframe);
  10704. }
  10705. void rtw_ft_build_assoc_req_ies(_adapter *padapter,
  10706. u8 is_reassoc, struct pkt_attrib *pattrib, u8 **pframe)
  10707. {
  10708. if (!pattrib || !(*pframe))
  10709. return;
  10710. if (rtw_ft_chk_flags(padapter, RTW_FT_PEER_EN))
  10711. rtw_ft_update_mdie(padapter, pattrib, pframe);
  10712. if ((!is_reassoc) || (!rtw_ft_roam(padapter)))
  10713. return;
  10714. if (rtw_ft_update_rsnie(padapter, _FALSE, pattrib, pframe))
  10715. rtw_ft_update_ftie(padapter, pattrib, pframe);
  10716. }
  10717. u8 rtw_ft_update_auth_rsp_ies(_adapter *padapter, u8 *pframe, u32 len)
  10718. {
  10719. u8 ret = _SUCCESS;
  10720. u8 target_ap_addr[ETH_ALEN] = {0};
  10721. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10722. struct ft_roam_info *pft_roam = &(pmlmepriv->ft_roam);
  10723. if (!rtw_ft_roam(padapter))
  10724. return _FAIL;
  10725. /*rtw_ft_report_reassoc_evt already,
  10726. * and waiting for cfg80211_rtw_update_ft_ies */
  10727. if (rtw_ft_authed_sta(padapter))
  10728. return ret;
  10729. if (!pframe || !len)
  10730. return _FAIL;
  10731. rtw_buf_update(&pmlmepriv->auth_rsp,
  10732. &pmlmepriv->auth_rsp_len, pframe, len);
  10733. pft_roam->ft_event.ies =
  10734. (pmlmepriv->auth_rsp + sizeof(struct rtw_ieee80211_hdr_3addr) + 6);
  10735. pft_roam->ft_event.ies_len =
  10736. (pmlmepriv->auth_rsp_len - sizeof(struct rtw_ieee80211_hdr_3addr) - 6);
  10737. /*Not support RIC*/
  10738. pft_roam->ft_event.ric_ies = NULL;
  10739. pft_roam->ft_event.ric_ies_len = 0;
  10740. _rtw_memcpy(target_ap_addr, pmlmepriv->assoc_bssid, ETH_ALEN);
  10741. rtw_ft_report_reassoc_evt(padapter, target_ap_addr);
  10742. return ret;
  10743. }
  10744. static void rtw_ft_start_clnt_action(_adapter *padapter, u8 *pTargetAddr)
  10745. {
  10746. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10747. rtw_ft_set_status(padapter, RTW_FT_REQUESTING_STA);
  10748. rtw_ft_issue_action_req(padapter, pTargetAddr);
  10749. _set_timer(&pmlmeext->ft_link_timer, REASSOC_TO);
  10750. }
  10751. void rtw_ft_start_roam(_adapter *padapter, u8 *pTargetAddr)
  10752. {
  10753. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  10754. if (rtw_ft_otd_roam(padapter)) {
  10755. rtw_ft_start_clnt_action(padapter, pTargetAddr);
  10756. } else {
  10757. /*wait a little time to retrieve packets buffered in the current ap while scan*/
  10758. _set_timer(&pmlmeext->ft_roam_timer, 30);
  10759. }
  10760. }
  10761. void rtw_ft_issue_action_req(_adapter *padapter, u8 *pTargetAddr)
  10762. {
  10763. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10764. struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
  10765. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  10766. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10767. struct xmit_frame *pmgntframe;
  10768. struct rtw_ieee80211_hdr *pwlanhdr;
  10769. struct pkt_attrib *pattrib;
  10770. u8 *pframe;
  10771. u8 category = RTW_WLAN_CATEGORY_FT;
  10772. u8 action = RTW_WLAN_ACTION_FT_REQ;
  10773. pmgntframe = alloc_mgtxmitframe(pxmitpriv);
  10774. if (pmgntframe == NULL)
  10775. return;
  10776. pattrib = &pmgntframe->attrib;
  10777. update_mgntframe_attrib(padapter, pattrib);
  10778. _rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
  10779. pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
  10780. pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
  10781. pwlanhdr->frame_ctl = 0;
  10782. _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  10783. _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(padapter), ETH_ALEN);
  10784. _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&pmlmeinfo->network), ETH_ALEN);
  10785. SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
  10786. pmlmeext->mgnt_seq++;
  10787. set_frame_sub_type(pframe, WIFI_ACTION);
  10788. pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
  10789. pattrib->pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
  10790. pframe = rtw_set_fixed_ie(pframe, 1, &(category), &(pattrib->pktlen));
  10791. pframe = rtw_set_fixed_ie(pframe, 1, &(action), &(pattrib->pktlen));
  10792. _rtw_memcpy(pframe, adapter_mac_addr(padapter), ETH_ALEN);
  10793. pframe += ETH_ALEN;
  10794. pattrib->pktlen += ETH_ALEN;
  10795. _rtw_memcpy(pframe, pTargetAddr, ETH_ALEN);
  10796. pframe += ETH_ALEN;
  10797. pattrib->pktlen += ETH_ALEN;
  10798. rtw_ft_update_mdie(padapter, pattrib, &pframe);
  10799. if (rtw_ft_update_rsnie(padapter, _TRUE, pattrib, &pframe))
  10800. rtw_ft_update_ftie(padapter, pattrib, &pframe);
  10801. pattrib->last_txcmdsz = pattrib->pktlen;
  10802. dump_mgntframe(padapter, pmgntframe);
  10803. }
  10804. void rtw_ft_report_evt(_adapter *padapter)
  10805. {
  10806. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10807. struct ft_roam_info *pft_roam = &(pmlmepriv->ft_roam);
  10808. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  10809. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10810. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)&(pmlmeinfo->network);
  10811. struct cfg80211_ft_event_params ft_evt_parms;
  10812. _irqL irqL;
  10813. _rtw_memset(&ft_evt_parms, 0, sizeof(ft_evt_parms));
  10814. rtw_ft_update_stainfo(padapter, pnetwork);
  10815. if (!pnetwork)
  10816. goto err_2;
  10817. ft_evt_parms.ies_len = pft_roam->ft_event.ies_len;
  10818. ft_evt_parms.ies = rtw_zmalloc(ft_evt_parms.ies_len);
  10819. if (ft_evt_parms.ies)
  10820. _rtw_memcpy((void *)ft_evt_parms.ies, pft_roam->ft_event.ies, ft_evt_parms.ies_len);
  10821. else
  10822. goto err_2;
  10823. ft_evt_parms.target_ap = rtw_zmalloc(ETH_ALEN);
  10824. if (ft_evt_parms.target_ap)
  10825. _rtw_memcpy((void *)ft_evt_parms.target_ap, pnetwork->MacAddress, ETH_ALEN);
  10826. else
  10827. goto err_1;
  10828. ft_evt_parms.ric_ies = pft_roam->ft_event.ric_ies;
  10829. ft_evt_parms.ric_ies_len = pft_roam->ft_event.ric_ies_len;
  10830. rtw_ft_lock_set_status(padapter, RTW_FT_AUTHENTICATED_STA, &irqL);
  10831. rtw_cfg80211_ft_event(padapter, &ft_evt_parms);
  10832. RTW_INFO("FT: rtw_ft_report_evt\n");
  10833. rtw_mfree((u8 *)pft_roam->ft_event.target_ap, ETH_ALEN);
  10834. err_1:
  10835. rtw_mfree((u8 *)ft_evt_parms.ies, ft_evt_parms.ies_len);
  10836. err_2:
  10837. return;
  10838. }
  10839. void rtw_ft_report_reassoc_evt(_adapter *padapter, u8 *pMacAddr)
  10840. {
  10841. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  10842. struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
  10843. struct cmd_obj *pcmd_obj = NULL;
  10844. struct stassoc_event *passoc_sta_evt = NULL;
  10845. struct C2HEvent_Header *pc2h_evt_hdr = NULL;
  10846. u8 *pevtcmd = NULL;
  10847. u32 cmdsz = 0;
  10848. pcmd_obj = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  10849. if (pcmd_obj == NULL)
  10850. return;
  10851. cmdsz = (sizeof(struct stassoc_event) + sizeof(struct C2HEvent_Header));
  10852. pevtcmd = (u8 *)rtw_zmalloc(cmdsz);
  10853. if (pevtcmd == NULL) {
  10854. rtw_mfree((u8 *)pcmd_obj, sizeof(struct cmd_obj));
  10855. return;
  10856. }
  10857. _rtw_init_listhead(&pcmd_obj->list);
  10858. pcmd_obj->cmdcode = GEN_CMD_CODE(_Set_MLME_EVT);
  10859. pcmd_obj->cmdsz = cmdsz;
  10860. pcmd_obj->parmbuf = pevtcmd;
  10861. pcmd_obj->rsp = NULL;
  10862. pcmd_obj->rspsz = 0;
  10863. pc2h_evt_hdr = (struct C2HEvent_Header *)(pevtcmd);
  10864. pc2h_evt_hdr->len = sizeof(struct stassoc_event);
  10865. pc2h_evt_hdr->ID = GEN_EVT_CODE(_FT_REASSOC);
  10866. pc2h_evt_hdr->seq = ATOMIC_INC_RETURN(&pmlmeext->event_seq);
  10867. passoc_sta_evt = (struct stassoc_event *)(pevtcmd + sizeof(struct C2HEvent_Header));
  10868. _rtw_memcpy((unsigned char *)(&(passoc_sta_evt->macaddr)), pMacAddr, ETH_ALEN);
  10869. rtw_enqueue_cmd(pcmdpriv, pcmd_obj);
  10870. }
  10871. void rtw_ft_link_timer_hdl(void *ctx)
  10872. {
  10873. _adapter *padapter = (_adapter *)ctx;
  10874. struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
  10875. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  10876. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10877. struct ft_roam_info *pft_roam = &(pmlmepriv->ft_roam);
  10878. if (rtw_ft_chk_status(padapter, RTW_FT_REQUESTING_STA)) {
  10879. if (pft_roam->ft_req_retry_cnt < RTW_FT_ACTION_REQ_LMT) {
  10880. pft_roam->ft_req_retry_cnt++;
  10881. rtw_ft_issue_action_req(padapter, (u8 *)pmlmepriv->roam_network->network.MacAddress);
  10882. _set_timer(&pmlmeext->ft_link_timer, REASSOC_TO);
  10883. } else {
  10884. pft_roam->ft_req_retry_cnt = 0;
  10885. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS)
  10886. rtw_ft_set_status(padapter, RTW_FT_ASSOCIATED_STA);
  10887. else
  10888. rtw_ft_reset_status(padapter);
  10889. }
  10890. }
  10891. }
  10892. void rtw_ft_roam_timer_hdl(void *ctx)
  10893. {
  10894. _adapter *padapter = (_adapter *)ctx;
  10895. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  10896. receive_disconnect(padapter, pmlmepriv->cur_network.network.MacAddress
  10897. , WLAN_REASON_ACTIVE_ROAM, _FALSE);
  10898. }
  10899. void rtw_ft_roam_status_reset(_adapter *padapter)
  10900. {
  10901. struct ft_roam_info *pft_roam = &(padapter->mlmepriv.ft_roam);
  10902. if ((rtw_to_roam(padapter) > 0) &&
  10903. (!rtw_ft_chk_status(padapter, RTW_FT_REQUESTED_STA))) {
  10904. rtw_ft_reset_status(padapter);
  10905. }
  10906. padapter->mlmepriv.ft_roam.ft_updated_bcn = _FALSE;
  10907. }
  10908. #endif
  10909. u8 NULL_hdl(_adapter *padapter, u8 *pbuf)
  10910. {
  10911. return H2C_SUCCESS;
  10912. }
  10913. #ifdef CONFIG_AUTO_AP_MODE
  10914. void rtw_auto_ap_rx_msg_dump(_adapter *padapter, union recv_frame *precv_frame, u8 *ehdr_pos)
  10915. {
  10916. struct rx_pkt_attrib *pattrib = &precv_frame->u.hdr.attrib;
  10917. struct sta_info *psta = precv_frame->u.hdr.psta;
  10918. struct ethhdr *ehdr = (struct ethhdr *)ehdr_pos;
  10919. RTW_INFO("eth rx: got eth_type=0x%x\n", ntohs(ehdr->h_proto));
  10920. if (psta && psta->isrc && psta->pid > 0) {
  10921. u16 rx_pid;
  10922. rx_pid = *(u16 *)(ehdr_pos + ETH_HLEN);
  10923. RTW_INFO("eth rx(pid=0x%x): sta("MAC_FMT") pid=0x%x\n",
  10924. rx_pid, MAC_ARG(psta->cmn.mac_addr), psta->pid);
  10925. if (rx_pid == psta->pid) {
  10926. int i;
  10927. u16 len = *(u16 *)(ehdr_pos + ETH_HLEN + 2);
  10928. /* u16 ctrl_type = *(u16 *)(ehdr_pos + ETH_HLEN + 4); */
  10929. /* RTW_INFO("eth, RC: len=0x%x, ctrl_type=0x%x\n", len, ctrl_type); */
  10930. RTW_INFO("eth, RC: len=0x%x\n", len);
  10931. for (i = 0; i < len; i++)
  10932. RTW_INFO("0x%x\n", *(ehdr_pos + ETH_HLEN + 4 + i));
  10933. /* RTW_INFO("0x%x\n", *(ehdr_pos + ETH_HLEN + 6 + i)); */
  10934. RTW_INFO("eth, RC-end\n");
  10935. }
  10936. }
  10937. }
  10938. void rtw_start_auto_ap(_adapter *adapter)
  10939. {
  10940. RTW_INFO("%s\n", __FUNCTION__);
  10941. rtw_set_802_11_infrastructure_mode(adapter, Ndis802_11APMode);
  10942. rtw_setopmode_cmd(adapter, Ndis802_11APMode, RTW_CMDF_WAIT_ACK);
  10943. }
  10944. static int rtw_auto_ap_start_beacon(_adapter *adapter)
  10945. {
  10946. int ret = 0;
  10947. u8 *pbuf = NULL;
  10948. uint len;
  10949. u8 supportRate[16];
  10950. int sz = 0, rateLen;
  10951. u8 *ie;
  10952. u8 wireless_mode, oper_channel;
  10953. u8 ssid[3] = {0}; /* hidden ssid */
  10954. u32 ssid_len = sizeof(ssid);
  10955. struct mlme_priv *pmlmepriv = &(adapter->mlmepriv);
  10956. if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != _TRUE)
  10957. return -EINVAL;
  10958. len = 128;
  10959. pbuf = rtw_zmalloc(len);
  10960. if (!pbuf)
  10961. return -ENOMEM;
  10962. /* generate beacon */
  10963. ie = pbuf;
  10964. /* timestamp will be inserted by hardware */
  10965. sz += 8;
  10966. ie += sz;
  10967. /* beacon interval : 2bytes */
  10968. *(u16 *)ie = cpu_to_le16((u16)100); /* BCN_INTERVAL=100; */
  10969. sz += 2;
  10970. ie += 2;
  10971. /* capability info */
  10972. *(u16 *)ie = 0;
  10973. *(u16 *)ie |= cpu_to_le16(cap_ESS);
  10974. *(u16 *)ie |= cpu_to_le16(cap_ShortPremble);
  10975. /* *(u16*)ie |= cpu_to_le16(cap_Privacy); */
  10976. sz += 2;
  10977. ie += 2;
  10978. /* SSID */
  10979. ie = rtw_set_ie(ie, _SSID_IE_, ssid_len, ssid, &sz);
  10980. /* supported rates */
  10981. wireless_mode = WIRELESS_11BG_24N;
  10982. rtw_set_supported_rate(supportRate, wireless_mode) ;
  10983. rateLen = rtw_get_rateset_len(supportRate);
  10984. if (rateLen > 8)
  10985. ie = rtw_set_ie(ie, _SUPPORTEDRATES_IE_, 8, supportRate, &sz);
  10986. else
  10987. ie = rtw_set_ie(ie, _SUPPORTEDRATES_IE_, rateLen, supportRate, &sz);
  10988. /* DS parameter set */
  10989. if (rtw_mi_check_status(adapter, MI_LINKED))
  10990. oper_channel = rtw_mi_get_union_chan(adapter);
  10991. else
  10992. oper_channel = adapter_to_dvobj(adapter)->oper_channel;
  10993. ie = rtw_set_ie(ie, _DSSET_IE_, 1, &oper_channel, &sz);
  10994. /* ext supported rates */
  10995. if (rateLen > 8)
  10996. ie = rtw_set_ie(ie, _EXT_SUPPORTEDRATES_IE_, (rateLen - 8), (supportRate + 8), &sz);
  10997. RTW_INFO("%s, start auto ap beacon sz=%d\n", __FUNCTION__, sz);
  10998. /* lunch ap mode & start to issue beacon */
  10999. if (rtw_check_beacon_data(adapter, pbuf, sz) == _SUCCESS) {
  11000. } else
  11001. ret = -EINVAL;
  11002. rtw_mfree(pbuf, len);
  11003. return ret;
  11004. }
  11005. #endif/* CONFIG_AUTO_AP_MODE */
  11006. u8 setopmode_hdl(_adapter *padapter, u8 *pbuf)
  11007. {
  11008. u8 type;
  11009. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11010. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  11011. struct setopmode_parm *psetop = (struct setopmode_parm *)pbuf;
  11012. if (psetop->mode == Ndis802_11APMode
  11013. || psetop->mode == Ndis802_11_mesh
  11014. ) {
  11015. pmlmeinfo->state = WIFI_FW_AP_STATE;
  11016. type = _HW_STATE_AP_;
  11017. } else if (psetop->mode == Ndis802_11Infrastructure) {
  11018. pmlmeinfo->state &= ~(BIT(0) | BIT(1)); /* clear state */
  11019. pmlmeinfo->state |= WIFI_FW_STATION_STATE;/* set to STATION_STATE */
  11020. type = _HW_STATE_STATION_;
  11021. } else if (psetop->mode == Ndis802_11IBSS)
  11022. type = _HW_STATE_ADHOC_;
  11023. else if (psetop->mode == Ndis802_11Monitor)
  11024. type = _HW_STATE_MONITOR_;
  11025. else
  11026. type = _HW_STATE_NOLINK_;
  11027. #ifdef CONFIG_AP_PORT_SWAP
  11028. rtw_hal_set_hwreg(padapter, HW_VAR_PORT_SWITCH, (u8 *)(&type));
  11029. #endif
  11030. rtw_hal_set_hwreg(padapter, HW_VAR_SET_OPMODE, (u8 *)(&type));
  11031. #ifdef CONFIG_AUTO_AP_MODE
  11032. if (psetop->mode == Ndis802_11APMode)
  11033. rtw_auto_ap_start_beacon(padapter);
  11034. #endif
  11035. if (rtw_port_switch_chk(padapter) == _TRUE) {
  11036. rtw_hal_set_hwreg(padapter, HW_VAR_PORT_SWITCH, NULL);
  11037. if (psetop->mode == Ndis802_11APMode)
  11038. adapter_to_pwrctl(padapter)->fw_psmode_iface_id = 0xff; /* ap mode won't dowload rsvd pages */
  11039. else if (psetop->mode == Ndis802_11Infrastructure) {
  11040. #ifdef CONFIG_LPS
  11041. _adapter *port0_iface = dvobj_get_port0_adapter(adapter_to_dvobj(padapter));
  11042. if (port0_iface)
  11043. rtw_lps_ctrl_wk_cmd(port0_iface, LPS_CTRL_CONNECT, 0);
  11044. #endif
  11045. }
  11046. }
  11047. #ifdef CONFIG_BT_COEXIST
  11048. if (psetop->mode == Ndis802_11APMode
  11049. || psetop->mode == Ndis802_11_mesh
  11050. || psetop->mode == Ndis802_11Monitor
  11051. ) {
  11052. /* Do this after port switch to */
  11053. /* prevent from downloading rsvd page to wrong port */
  11054. rtw_btcoex_MediaStatusNotify(padapter, 1); /* connect */
  11055. }
  11056. #endif /* CONFIG_BT_COEXIST */
  11057. return H2C_SUCCESS;
  11058. }
  11059. u8 createbss_hdl(_adapter *padapter, u8 *pbuf)
  11060. {
  11061. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11062. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  11063. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  11064. WLAN_BSSID_EX *pdev_network = &padapter->registrypriv.dev_network;
  11065. struct createbss_parm *parm = (struct createbss_parm *)pbuf;
  11066. u8 ret = H2C_SUCCESS;
  11067. /* u8 initialgain; */
  11068. #ifdef CONFIG_AP_MODE
  11069. if ((parm->req_ch == 0 && pmlmeinfo->state == WIFI_FW_AP_STATE)
  11070. || parm->req_ch != 0
  11071. ) {
  11072. start_bss_network(padapter, parm);
  11073. goto exit;
  11074. }
  11075. #endif
  11076. /* below is for ad-hoc master */
  11077. if (parm->adhoc) {
  11078. rtw_warn_on(pdev_network->InfrastructureMode != Ndis802_11IBSS);
  11079. rtw_joinbss_reset(padapter);
  11080. pmlmeext->cur_bwmode = CHANNEL_WIDTH_20;
  11081. pmlmeext->cur_ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  11082. pmlmeinfo->ERP_enable = 0;
  11083. pmlmeinfo->WMM_enable = 0;
  11084. pmlmeinfo->HT_enable = 0;
  11085. pmlmeinfo->HT_caps_enable = 0;
  11086. pmlmeinfo->HT_info_enable = 0;
  11087. pmlmeinfo->agg_enable_bitmap = 0;
  11088. pmlmeinfo->candidate_tid_bitmap = 0;
  11089. /* cancel link timer */
  11090. _cancel_timer_ex(&pmlmeext->link_timer);
  11091. /* clear CAM */
  11092. flush_all_cam_entry(padapter);
  11093. pdev_network->Length = get_WLAN_BSSID_EX_sz(pdev_network);
  11094. _rtw_memcpy(pnetwork, pdev_network, FIELD_OFFSET(WLAN_BSSID_EX, IELength));
  11095. pnetwork->IELength = pdev_network->IELength;
  11096. if (pnetwork->IELength > MAX_IE_SZ) {
  11097. ret = H2C_PARAMETERS_ERROR;
  11098. goto ibss_post_hdl;
  11099. }
  11100. _rtw_memcpy(pnetwork->IEs, pdev_network->IEs, pnetwork->IELength);
  11101. start_create_ibss(padapter);
  11102. } else {
  11103. rtw_warn_on(1);
  11104. ret = H2C_PARAMETERS_ERROR;
  11105. }
  11106. ibss_post_hdl:
  11107. rtw_create_ibss_post_hdl(padapter, ret);
  11108. exit:
  11109. return ret;
  11110. }
  11111. u8 join_cmd_hdl(_adapter *padapter, u8 *pbuf)
  11112. {
  11113. u8 join_type;
  11114. PNDIS_802_11_VARIABLE_IEs pIE;
  11115. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11116. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  11117. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  11118. #ifdef CONFIG_ANTENNA_DIVERSITY
  11119. struct joinbss_parm *pparm = (struct joinbss_parm *)pbuf;
  11120. #endif /* CONFIG_ANTENNA_DIVERSITY */
  11121. u32 i;
  11122. /* u8 initialgain; */
  11123. /* u32 acparm; */
  11124. u8 u_ch, u_bw, u_offset;
  11125. u8 doiqk = _FALSE;
  11126. /* check already connecting to AP or not */
  11127. if (pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS) {
  11128. if (pmlmeinfo->state & WIFI_FW_STATION_STATE)
  11129. issue_deauth_ex(padapter, pnetwork->MacAddress, WLAN_REASON_DEAUTH_LEAVING, 1, 100);
  11130. pmlmeinfo->state = WIFI_FW_NULL_STATE;
  11131. /* clear CAM */
  11132. flush_all_cam_entry(padapter);
  11133. _cancel_timer_ex(&pmlmeext->link_timer);
  11134. /* set MSR to nolink->infra. mode */
  11135. /* Set_MSR(padapter, _HW_STATE_NOLINK_); */
  11136. Set_MSR(padapter, _HW_STATE_STATION_);
  11137. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_DISCONNECT, 0);
  11138. if (pmlmeinfo->state & WIFI_FW_STATION_STATE)
  11139. rtw_hal_rcr_set_chk_bssid(padapter, MLME_STA_DISCONNECTED);
  11140. }
  11141. #ifdef CONFIG_ANTENNA_DIVERSITY
  11142. rtw_antenna_select_cmd(padapter, pparm->network.PhyInfo.Optimum_antenna, _FALSE);
  11143. #endif
  11144. #ifdef CONFIG_WAPI_SUPPORT
  11145. rtw_wapi_clear_all_cam_entry(padapter);
  11146. #endif
  11147. rtw_joinbss_reset(padapter);
  11148. pmlmeinfo->ERP_enable = 0;
  11149. pmlmeinfo->WMM_enable = 0;
  11150. pmlmeinfo->HT_enable = 0;
  11151. pmlmeinfo->HT_caps_enable = 0;
  11152. pmlmeinfo->HT_info_enable = 0;
  11153. pmlmeinfo->agg_enable_bitmap = 0;
  11154. pmlmeinfo->candidate_tid_bitmap = 0;
  11155. pmlmeinfo->bwmode_updated = _FALSE;
  11156. /* pmlmeinfo->assoc_AP_vendor = HT_IOT_PEER_MAX; */
  11157. pmlmeinfo->VHT_enable = 0;
  11158. #ifdef ROKU_PRIVATE
  11159. pmlmeinfo->ht_vht_received = 0;
  11160. _rtw_memset(pmlmeinfo->SupportedRates_infra_ap, 0, NDIS_802_11_LENGTH_RATES_EX);
  11161. #endif /* ROKU_PRIVATE */
  11162. _rtw_memcpy(pnetwork, pbuf, FIELD_OFFSET(WLAN_BSSID_EX, IELength));
  11163. pnetwork->IELength = ((WLAN_BSSID_EX *)pbuf)->IELength;
  11164. if (pnetwork->IELength > MAX_IE_SZ) /* Check pbuf->IELength */
  11165. return H2C_PARAMETERS_ERROR;
  11166. if (pnetwork->IELength < 2) {
  11167. report_join_res(padapter, (-4), WLAN_STATUS_UNSPECIFIED_FAILURE);
  11168. return H2C_SUCCESS;
  11169. }
  11170. _rtw_memcpy(pnetwork->IEs, ((WLAN_BSSID_EX *)pbuf)->IEs, pnetwork->IELength);
  11171. pmlmeinfo->bcn_interval = get_beacon_interval(pnetwork);
  11172. /* Check AP vendor to move rtw_joinbss_cmd() */
  11173. /* pmlmeinfo->assoc_AP_vendor = check_assoc_AP(pnetwork->IEs, pnetwork->IELength); */
  11174. /* sizeof(NDIS_802_11_FIXED_IEs) */
  11175. for (i = _FIXED_IE_LENGTH_ ; i < pnetwork->IELength - 2 ;) {
  11176. pIE = (PNDIS_802_11_VARIABLE_IEs)(pnetwork->IEs + i);
  11177. switch (pIE->ElementID) {
  11178. case _VENDOR_SPECIFIC_IE_: /* Get WMM IE. */
  11179. if (_rtw_memcmp(pIE->data, WMM_OUI, 4))
  11180. WMM_param_handler(padapter, pIE);
  11181. break;
  11182. #ifdef CONFIG_80211N_HT
  11183. case _HT_CAPABILITY_IE_: /* Get HT Cap IE. */
  11184. pmlmeinfo->HT_caps_enable = 1;
  11185. break;
  11186. case _HT_EXTRA_INFO_IE_: /* Get HT Info IE. */
  11187. pmlmeinfo->HT_info_enable = 1;
  11188. break;
  11189. #endif /* CONFIG_80211N_HT */
  11190. #ifdef CONFIG_80211AC_VHT
  11191. case EID_VHTCapability: /* Get VHT Cap IE. */
  11192. pmlmeinfo->VHT_enable = 1;
  11193. break;
  11194. case EID_VHTOperation: /* Get VHT Operation IE. */
  11195. break;
  11196. #endif /* CONFIG_80211AC_VHT */
  11197. default:
  11198. break;
  11199. }
  11200. i += (pIE->Length + 2);
  11201. }
  11202. rtw_bss_get_chbw(pnetwork
  11203. , &pmlmeext->cur_channel, &pmlmeext->cur_bwmode, &pmlmeext->cur_ch_offset, 1, 1);
  11204. rtw_adjust_chbw(padapter, pmlmeext->cur_channel, &pmlmeext->cur_bwmode, &pmlmeext->cur_ch_offset);
  11205. #if 0
  11206. if (padapter->registrypriv.wifi_spec) {
  11207. /* for WiFi test, follow WMM test plan spec */
  11208. acparm = 0x002F431C; /* VO */
  11209. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_VO, (u8 *)(&acparm));
  11210. acparm = 0x005E541C; /* VI */
  11211. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_VI, (u8 *)(&acparm));
  11212. acparm = 0x0000A525; /* BE */
  11213. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_BE, (u8 *)(&acparm));
  11214. acparm = 0x0000A549; /* BK */
  11215. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_BK, (u8 *)(&acparm));
  11216. /* for WiFi test, mixed mode with intel STA under bg mode throughput issue */
  11217. if (padapter->mlmepriv.htpriv.ht_option == _FALSE) {
  11218. acparm = 0x00004320;
  11219. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_BE, (u8 *)(&acparm));
  11220. }
  11221. } else {
  11222. acparm = 0x002F3217; /* VO */
  11223. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_VO, (u8 *)(&acparm));
  11224. acparm = 0x005E4317; /* VI */
  11225. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_VI, (u8 *)(&acparm));
  11226. acparm = 0x00105320; /* BE */
  11227. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_BE, (u8 *)(&acparm));
  11228. acparm = 0x0000A444; /* BK */
  11229. rtw_hal_set_hwreg(padapter, HW_VAR_AC_PARAM_BK, (u8 *)(&acparm));
  11230. }
  11231. #endif
  11232. /* check channel, bandwidth, offset and switch */
  11233. if (rtw_chk_start_clnt_join(padapter, &u_ch, &u_bw, &u_offset) == _FAIL) {
  11234. report_join_res(padapter, (-4), WLAN_STATUS_UNSPECIFIED_FAILURE);
  11235. return H2C_SUCCESS;
  11236. }
  11237. /* disable dynamic functions, such as high power, DIG */
  11238. /*rtw_phydm_func_disable_all(padapter);*/
  11239. /* config the initial gain under linking, need to write the BB registers */
  11240. /* initialgain = 0x1E; */
  11241. /*rtw_hal_set_odm_var(padapter, HAL_ODM_INITIAL_GAIN, &initialgain, _FALSE);*/
  11242. rtw_hal_set_hwreg(padapter, HW_VAR_BSSID, pmlmeinfo->network.MacAddress);
  11243. if (MLME_IS_STA(padapter))
  11244. rtw_hal_rcr_set_chk_bssid(padapter, MLME_STA_CONNECTING);
  11245. else
  11246. rtw_hal_rcr_set_chk_bssid(padapter, MLME_ADHOC_STARTED);
  11247. join_type = 0;
  11248. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_JOIN, (u8 *)(&join_type));
  11249. doiqk = _TRUE;
  11250. rtw_hal_set_hwreg(padapter , HW_VAR_DO_IQK , &doiqk);
  11251. set_channel_bwmode(padapter, u_ch, u_offset, u_bw);
  11252. rtw_mi_update_union_chan_inf(padapter, u_ch, u_offset, u_bw);
  11253. doiqk = _FALSE;
  11254. rtw_hal_set_hwreg(padapter , HW_VAR_DO_IQK , &doiqk);
  11255. /* cancel link timer */
  11256. _cancel_timer_ex(&pmlmeext->link_timer);
  11257. start_clnt_join(padapter);
  11258. return H2C_SUCCESS;
  11259. }
  11260. u8 disconnect_hdl(_adapter *padapter, unsigned char *pbuf)
  11261. {
  11262. #ifdef CONFIG_DFS
  11263. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  11264. #endif
  11265. struct disconnect_parm *param = (struct disconnect_parm *)pbuf;
  11266. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11267. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  11268. WLAN_BSSID_EX *pnetwork = (WLAN_BSSID_EX *)(&(pmlmeinfo->network));
  11269. u8 val8;
  11270. if (is_client_associated_to_ap(padapter)
  11271. #ifdef CONFIG_DFS
  11272. && !IS_RADAR_DETECTED(rfctl) && !rfctl->csa_ch
  11273. #endif
  11274. ) {
  11275. #ifdef CONFIG_PLATFORM_ROCKCHIPS
  11276. /* To avoid connecting to AP fail during resume process, change retry count from 5 to 1 */
  11277. issue_deauth_ex(padapter, pnetwork->MacAddress, WLAN_REASON_DEAUTH_LEAVING, 1, 100);
  11278. #else
  11279. issue_deauth_ex(padapter, pnetwork->MacAddress, WLAN_REASON_DEAUTH_LEAVING, param->deauth_timeout_ms / 100, 100);
  11280. #endif /* CONFIG_PLATFORM_ROCKCHIPS */
  11281. }
  11282. #ifndef CONFIG_SUPPORT_MULTI_BCN
  11283. if (((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) || ((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE)) {
  11284. /* Stop BCN */
  11285. val8 = 0;
  11286. rtw_hal_set_hwreg(padapter, HW_VAR_BCN_FUNC, (u8 *)(&val8));
  11287. }
  11288. #endif
  11289. rtw_mlmeext_disconnect(padapter);
  11290. rtw_free_uc_swdec_pending_queue(padapter);
  11291. rtw_sta_mstatus_report(padapter);
  11292. return H2C_SUCCESS;
  11293. }
  11294. static const char *const _scan_state_str[] = {
  11295. "SCAN_DISABLE",
  11296. "SCAN_START",
  11297. "SCAN_PS_ANNC_WAIT",
  11298. "SCAN_ENTER",
  11299. "SCAN_PROCESS",
  11300. "SCAN_BACKING_OP",
  11301. "SCAN_BACK_OP",
  11302. "SCAN_LEAVING_OP",
  11303. "SCAN_LEAVE_OP",
  11304. "SCAN_SW_ANTDIV_BL",
  11305. "SCAN_TO_P2P_LISTEN",
  11306. "SCAN_P2P_LISTEN",
  11307. "SCAN_COMPLETE",
  11308. "SCAN_STATE_MAX",
  11309. };
  11310. const char *scan_state_str(u8 state)
  11311. {
  11312. state = (state >= SCAN_STATE_MAX) ? SCAN_STATE_MAX : state;
  11313. return _scan_state_str[state];
  11314. }
  11315. static bool scan_abort_hdl(_adapter *adapter)
  11316. {
  11317. struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
  11318. struct ss_res *ss = &pmlmeext->sitesurvey_res;
  11319. #ifdef CONFIG_P2P
  11320. struct wifidirect_info *pwdinfo = &adapter->wdinfo;
  11321. #endif
  11322. bool ret = _FALSE;
  11323. if (pmlmeext->scan_abort == _TRUE) {
  11324. #ifdef CONFIG_P2P
  11325. if (!rtw_p2p_chk_state(&adapter->wdinfo, P2P_STATE_NONE)) {
  11326. rtw_p2p_findphase_ex_set(pwdinfo, P2P_FINDPHASE_EX_MAX);
  11327. ss->channel_idx = 3;
  11328. RTW_INFO("%s idx:%d, cnt:%u\n", __FUNCTION__
  11329. , ss->channel_idx
  11330. , pwdinfo->find_phase_state_exchange_cnt
  11331. );
  11332. } else
  11333. #endif
  11334. {
  11335. ss->channel_idx = ss->ch_num;
  11336. RTW_INFO("%s idx:%d\n", __FUNCTION__
  11337. , ss->channel_idx
  11338. );
  11339. }
  11340. pmlmeext->scan_abort = _FALSE;
  11341. ret = _TRUE;
  11342. }
  11343. return ret;
  11344. }
  11345. u8 rtw_scan_sparse(_adapter *adapter, struct rtw_ieee80211_channel *ch, u8 ch_num)
  11346. {
  11347. /* interval larger than this is treated as backgroud scan */
  11348. #ifndef RTW_SCAN_SPARSE_BG_INTERVAL_MS
  11349. #define RTW_SCAN_SPARSE_BG_INTERVAL_MS 12000
  11350. #endif
  11351. #ifndef RTW_SCAN_SPARSE_CH_NUM_MIRACAST
  11352. #define RTW_SCAN_SPARSE_CH_NUM_MIRACAST 1
  11353. #endif
  11354. #ifndef RTW_SCAN_SPARSE_CH_NUM_BG
  11355. #define RTW_SCAN_SPARSE_CH_NUM_BG 4
  11356. #endif
  11357. #ifdef CONFIG_LAYER2_ROAMING
  11358. #ifndef RTW_SCAN_SPARSE_CH_NUM_ROAMING_ACTIVE
  11359. #define RTW_SCAN_SPARSE_CH_NUM_ROAMING_ACTIVE 1
  11360. #endif
  11361. #endif
  11362. #define SCAN_SPARSE_CH_NUM_INVALID 255
  11363. static u8 token = 255;
  11364. u32 interval;
  11365. bool busy_traffic = _FALSE;
  11366. bool miracast_enabled = _FALSE;
  11367. bool bg_scan = _FALSE;
  11368. u8 max_allow_ch = SCAN_SPARSE_CH_NUM_INVALID;
  11369. u8 scan_division_num;
  11370. u8 ret_num = ch_num;
  11371. struct registry_priv *regsty = dvobj_to_regsty(adapter_to_dvobj(adapter));
  11372. struct mlme_ext_priv *mlmeext = &adapter->mlmeextpriv;
  11373. if (regsty->wifi_spec)
  11374. goto exit;
  11375. /* assume ch_num > 6 is normal scan */
  11376. if (ch_num <= 6)
  11377. goto exit;
  11378. if (mlmeext->last_scan_time == 0)
  11379. mlmeext->last_scan_time = rtw_get_current_time();
  11380. interval = rtw_get_passing_time_ms(mlmeext->last_scan_time);
  11381. if (rtw_mi_busy_traffic_check(adapter, _FALSE))
  11382. busy_traffic = _TRUE;
  11383. if (rtw_mi_check_miracast_enabled(adapter))
  11384. miracast_enabled = _TRUE;
  11385. if (interval > RTW_SCAN_SPARSE_BG_INTERVAL_MS)
  11386. bg_scan = _TRUE;
  11387. /* max_allow_ch by conditions*/
  11388. #if RTW_SCAN_SPARSE_MIRACAST
  11389. if (miracast_enabled == _TRUE && busy_traffic == _TRUE)
  11390. max_allow_ch = rtw_min(max_allow_ch, RTW_SCAN_SPARSE_CH_NUM_MIRACAST);
  11391. #endif
  11392. #if RTW_SCAN_SPARSE_BG
  11393. if (bg_scan == _TRUE)
  11394. max_allow_ch = rtw_min(max_allow_ch, RTW_SCAN_SPARSE_CH_NUM_BG);
  11395. #endif
  11396. #if defined(CONFIG_LAYER2_ROAMING) && defined(RTW_SCAN_SPARSE_ROAMING_ACTIVE)
  11397. if (rtw_chk_roam_flags(adapter, RTW_ROAM_ACTIVE)) {
  11398. if (busy_traffic == _TRUE && adapter->mlmepriv.need_to_roam == _TRUE)
  11399. max_allow_ch = rtw_min(max_allow_ch, RTW_SCAN_SPARSE_CH_NUM_ROAMING_ACTIVE);
  11400. }
  11401. #endif
  11402. if (max_allow_ch != SCAN_SPARSE_CH_NUM_INVALID) {
  11403. int i;
  11404. int k = 0;
  11405. scan_division_num = (ch_num / max_allow_ch) + ((ch_num % max_allow_ch) ? 1 : 0);
  11406. token = (token + 1) % scan_division_num;
  11407. if (0)
  11408. RTW_INFO("scan_division_num:%u, token:%u\n", scan_division_num, token);
  11409. for (i = 0; i < ch_num; i++) {
  11410. if (ch[i].hw_value && (i % scan_division_num) == token
  11411. ) {
  11412. if (i != k)
  11413. _rtw_memcpy(&ch[k], &ch[i], sizeof(struct rtw_ieee80211_channel));
  11414. k++;
  11415. }
  11416. }
  11417. _rtw_memset(&ch[k], 0, sizeof(struct rtw_ieee80211_channel));
  11418. ret_num = k;
  11419. mlmeext->last_scan_time = rtw_get_current_time();
  11420. }
  11421. exit:
  11422. return ret_num;
  11423. }
  11424. #ifdef CONFIG_SCAN_BACKOP
  11425. u8 rtw_scan_backop_decision(_adapter *adapter)
  11426. {
  11427. struct mlme_ext_priv *mlmeext = &adapter->mlmeextpriv;
  11428. struct mi_state mstate;
  11429. u8 backop_flags = 0;
  11430. rtw_mi_status(adapter, &mstate);
  11431. if ((MSTATE_STA_LD_NUM(&mstate) && mlmeext_chk_scan_backop_flags_sta(mlmeext, SS_BACKOP_EN))
  11432. || (MSTATE_STA_NUM(&mstate) && mlmeext_chk_scan_backop_flags_sta(mlmeext, SS_BACKOP_EN_NL)))
  11433. backop_flags |= mlmeext_scan_backop_flags_sta(mlmeext);
  11434. #ifdef CONFIG_AP_MODE
  11435. if ((MSTATE_AP_LD_NUM(&mstate) && mlmeext_chk_scan_backop_flags_ap(mlmeext, SS_BACKOP_EN))
  11436. || (MSTATE_AP_NUM(&mstate) && mlmeext_chk_scan_backop_flags_ap(mlmeext, SS_BACKOP_EN_NL)))
  11437. backop_flags |= mlmeext_scan_backop_flags_ap(mlmeext);
  11438. #endif
  11439. #ifdef CONFIG_RTW_MESH
  11440. if ((MSTATE_MESH_LD_NUM(&mstate) && mlmeext_chk_scan_backop_flags_mesh(mlmeext, SS_BACKOP_EN))
  11441. || (MSTATE_MESH_NUM(&mstate) && mlmeext_chk_scan_backop_flags_mesh(mlmeext, SS_BACKOP_EN_NL)))
  11442. backop_flags |= mlmeext_scan_backop_flags_mesh(mlmeext);
  11443. #endif
  11444. return backop_flags;
  11445. }
  11446. #endif
  11447. #define SCANNING_TIMEOUT_EX 2000
  11448. u32 rtw_scan_timeout_decision(_adapter *padapter)
  11449. {
  11450. u32 back_op_times= 0;
  11451. u8 max_chan_num;
  11452. u16 scan_ms;
  11453. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11454. struct ss_res *ss = &pmlmeext->sitesurvey_res;
  11455. if (is_supported_5g(padapter->registrypriv.wireless_mode)
  11456. && IsSupported24G(padapter->registrypriv.wireless_mode))
  11457. max_chan_num = MAX_CHANNEL_NUM;/* dual band */
  11458. else
  11459. max_chan_num = MAX_CHANNEL_NUM_2G;/*single band*/
  11460. #ifdef CONFIG_SCAN_BACKOP
  11461. if (rtw_scan_backop_decision(padapter))
  11462. back_op_times = (max_chan_num / ss->scan_cnt_max) * ss->backop_ms;
  11463. #endif
  11464. if (ss->duration)
  11465. scan_ms = ss->duration;
  11466. else
  11467. #if defined(CONFIG_RTW_ACS) && defined(CONFIG_RTW_ACS_DBG)
  11468. if (IS_ACS_ENABLE(padapter) && rtw_is_acs_st_valid(padapter))
  11469. scan_ms = rtw_acs_get_adv_st(padapter);
  11470. else
  11471. #endif /*CONFIG_RTW_ACS*/
  11472. scan_ms = ss->scan_ch_ms;
  11473. ss->scan_timeout_ms = (scan_ms * max_chan_num) + back_op_times + SCANNING_TIMEOUT_EX;
  11474. #ifdef DBG_SITESURVEY
  11475. RTW_INFO("%s , scan_timeout_ms = %d (ms)\n", __func__, ss->scan_timeout_ms);
  11476. #endif /*DBG_SITESURVEY*/
  11477. return ss->scan_timeout_ms;
  11478. }
  11479. static int rtw_scan_ch_decision(_adapter *padapter, struct rtw_ieee80211_channel *out,
  11480. u32 out_num, struct rtw_ieee80211_channel *in, u32 in_num)
  11481. {
  11482. int i, j;
  11483. int set_idx;
  11484. u8 chan;
  11485. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  11486. /* clear first */
  11487. _rtw_memset(out, 0, sizeof(struct rtw_ieee80211_channel) * out_num);
  11488. /* acquire channels from in */
  11489. j = 0;
  11490. for (i = 0; i < in_num; i++) {
  11491. if (0)
  11492. RTW_INFO(FUNC_ADPT_FMT" "CHAN_FMT"\n", FUNC_ADPT_ARG(padapter), CHAN_ARG(&in[i]));
  11493. if (!in[i].hw_value || (in[i].flags & RTW_IEEE80211_CHAN_DISABLED))
  11494. continue;
  11495. if (rtw_mlme_band_check(padapter, in[i].hw_value) == _FALSE)
  11496. continue;
  11497. set_idx = rtw_chset_search_ch(rfctl->channel_set, in[i].hw_value);
  11498. if (set_idx >= 0) {
  11499. if (j >= out_num) {
  11500. RTW_PRINT(FUNC_ADPT_FMT" out_num:%u not enough\n",
  11501. FUNC_ADPT_ARG(padapter), out_num);
  11502. break;
  11503. }
  11504. _rtw_memcpy(&out[j], &in[i], sizeof(struct rtw_ieee80211_channel));
  11505. if (rfctl->channel_set[set_idx].ScanType == SCAN_PASSIVE)
  11506. out[j].flags |= RTW_IEEE80211_CHAN_PASSIVE_SCAN;
  11507. j++;
  11508. }
  11509. if (j >= out_num)
  11510. break;
  11511. }
  11512. /* if out is empty, use channel_set as default */
  11513. if (j == 0) {
  11514. for (i = 0; i < rfctl->max_chan_nums; i++) {
  11515. chan = rfctl->channel_set[i].ChannelNum;
  11516. if (rtw_mlme_band_check(padapter, chan) == _TRUE) {
  11517. if (rtw_mlme_ignore_chan(padapter, chan) == _TRUE)
  11518. continue;
  11519. if (0)
  11520. RTW_INFO(FUNC_ADPT_FMT" ch:%u\n", FUNC_ADPT_ARG(padapter), chan);
  11521. if (j >= out_num) {
  11522. RTW_PRINT(FUNC_ADPT_FMT" out_num:%u not enough\n",
  11523. FUNC_ADPT_ARG(padapter), out_num);
  11524. break;
  11525. }
  11526. out[j].hw_value = chan;
  11527. if (rfctl->channel_set[i].ScanType == SCAN_PASSIVE)
  11528. out[j].flags |= RTW_IEEE80211_CHAN_PASSIVE_SCAN;
  11529. j++;
  11530. }
  11531. }
  11532. }
  11533. /* scan_sparse */
  11534. j = rtw_scan_sparse(padapter, out, j);
  11535. return j;
  11536. }
  11537. static void sitesurvey_res_reset(_adapter *adapter, struct sitesurvey_parm *parm)
  11538. {
  11539. struct ss_res *ss = &adapter->mlmeextpriv.sitesurvey_res;
  11540. RT_CHANNEL_INFO *chset = adapter_to_chset(adapter);
  11541. int i;
  11542. ss->bss_cnt = 0;
  11543. ss->channel_idx = 0;
  11544. #ifdef CONFIG_DFS
  11545. ss->dfs_ch_ssid_scan = 0;
  11546. #endif
  11547. ss->igi_scan = 0;
  11548. ss->igi_before_scan = 0;
  11549. #ifdef CONFIG_SCAN_BACKOP
  11550. ss->scan_cnt = 0;
  11551. #endif
  11552. #if defined(CONFIG_ANTENNA_DIVERSITY) || defined(DBG_SCAN_SW_ANTDIV_BL)
  11553. ss->is_sw_antdiv_bl_scan = 0;
  11554. #endif
  11555. ss->ssid_num = 0;
  11556. for (i = 0; i < RTW_SSID_SCAN_AMOUNT; i++) {
  11557. if (parm->ssid[i].SsidLength) {
  11558. _rtw_memcpy(ss->ssid[i].Ssid, parm->ssid[i].Ssid, IW_ESSID_MAX_SIZE);
  11559. ss->ssid[i].SsidLength = parm->ssid[i].SsidLength;
  11560. ss->ssid_num++;
  11561. } else
  11562. ss->ssid[i].SsidLength = 0;
  11563. }
  11564. ss->ch_num = rtw_scan_ch_decision(adapter
  11565. , ss->ch, RTW_CHANNEL_SCAN_AMOUNT
  11566. , parm->ch, parm->ch_num
  11567. );
  11568. #ifdef CONFIG_DFS
  11569. for (i = 0; i < MAX_CHANNEL_NUM; i++)
  11570. chset[i].hidden_bss_cnt = 0;
  11571. #endif
  11572. ss->bw = parm->bw;
  11573. ss->igi = parm->igi;
  11574. ss->token = parm->token;
  11575. ss->duration = parm->duration;
  11576. ss->scan_mode = parm->scan_mode;
  11577. ss->token = parm->token;
  11578. }
  11579. static u8 sitesurvey_pick_ch_behavior(_adapter *padapter, u8 *ch, RT_SCAN_TYPE *type)
  11580. {
  11581. u8 next_state;
  11582. u8 scan_ch = 0;
  11583. RT_SCAN_TYPE scan_type = SCAN_PASSIVE;
  11584. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11585. struct ss_res *ss = &pmlmeext->sitesurvey_res;
  11586. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  11587. int ch_set_idx;
  11588. #ifdef CONFIG_P2P
  11589. struct wifidirect_info *pwdinfo = &padapter->wdinfo;
  11590. #endif
  11591. #ifdef CONFIG_SCAN_BACKOP
  11592. u8 backop_flags = 0;
  11593. #endif
  11594. /* handle scan abort request */
  11595. scan_abort_hdl(padapter);
  11596. #ifdef CONFIG_P2P
  11597. if (pwdinfo->rx_invitereq_info.scan_op_ch_only || pwdinfo->p2p_info.scan_op_ch_only) {
  11598. if (pwdinfo->rx_invitereq_info.scan_op_ch_only)
  11599. scan_ch = pwdinfo->rx_invitereq_info.operation_ch[ss->channel_idx];
  11600. else
  11601. scan_ch = pwdinfo->p2p_info.operation_ch[ss->channel_idx];
  11602. scan_type = SCAN_ACTIVE;
  11603. } else if (rtw_p2p_findphase_ex_is_social(pwdinfo)) {
  11604. /*
  11605. * Commented by Albert 2011/06/03
  11606. * The driver is in the find phase, it should go through the social channel.
  11607. */
  11608. scan_ch = pwdinfo->social_chan[ss->channel_idx];
  11609. ch_set_idx = rtw_chset_search_ch(rfctl->channel_set, scan_ch);
  11610. if (ch_set_idx >= 0)
  11611. scan_type = rfctl->channel_set[ch_set_idx].ScanType;
  11612. else
  11613. scan_type = SCAN_ACTIVE;
  11614. } else
  11615. #endif /* CONFIG_P2P */
  11616. {
  11617. struct rtw_ieee80211_channel *ch;
  11618. #ifdef CONFIG_SCAN_BACKOP
  11619. backop_flags = rtw_scan_backop_decision(padapter);
  11620. #endif
  11621. #ifdef CONFIG_DFS
  11622. #ifdef CONFIG_SCAN_BACKOP
  11623. if (!(backop_flags && ss->scan_cnt >= ss->scan_cnt_max))
  11624. #endif
  11625. {
  11626. #ifdef CONFIG_RTW_WIFI_HAL
  11627. if (adapter_to_dvobj(padapter)->nodfs) {
  11628. while ( ss->channel_idx < ss->ch_num && rtw_is_dfs_ch(ss->ch[ss->channel_idx].hw_value))
  11629. ss->channel_idx++;
  11630. } else
  11631. #endif
  11632. if (ss->channel_idx != 0 && ss->dfs_ch_ssid_scan == 0
  11633. && pmlmeext->sitesurvey_res.ssid_num
  11634. && rtw_is_dfs_ch(ss->ch[ss->channel_idx - 1].hw_value)
  11635. ) {
  11636. ch_set_idx = rtw_chset_search_ch(rfctl->channel_set, ss->ch[ss->channel_idx - 1].hw_value);
  11637. if (ch_set_idx != -1 && rfctl->channel_set[ch_set_idx].hidden_bss_cnt
  11638. && (!IS_DFS_SLAVE_WITH_RD(rfctl)
  11639. || rtw_odm_dfs_domain_unknown(rfctl_to_dvobj(rfctl))
  11640. || !CH_IS_NON_OCP(&rfctl->channel_set[ch_set_idx]))
  11641. ) {
  11642. ss->channel_idx--;
  11643. ss->dfs_ch_ssid_scan = 1;
  11644. }
  11645. } else
  11646. ss->dfs_ch_ssid_scan = 0;
  11647. }
  11648. #endif /* CONFIG_DFS */
  11649. if (ss->channel_idx < ss->ch_num) {
  11650. ch = &ss->ch[ss->channel_idx];
  11651. scan_ch = ch->hw_value;
  11652. #if defined(CONFIG_RTW_ACS) && defined(CONFIG_RTW_ACS_DBG)
  11653. if (IS_ACS_ENABLE(padapter) && rtw_is_acs_passiv_scan(padapter))
  11654. scan_type = SCAN_PASSIVE;
  11655. else
  11656. #endif /*CONFIG_RTW_ACS*/
  11657. scan_type = (ch->flags & RTW_IEEE80211_CHAN_PASSIVE_SCAN) ? SCAN_PASSIVE : SCAN_ACTIVE;
  11658. }
  11659. }
  11660. if (scan_ch != 0) {
  11661. next_state = SCAN_PROCESS;
  11662. #ifdef CONFIG_SCAN_BACKOP
  11663. if (backop_flags) {
  11664. if (ss->scan_cnt < ss->scan_cnt_max)
  11665. ss->scan_cnt++;
  11666. else {
  11667. mlmeext_assign_scan_backop_flags(pmlmeext, backop_flags);
  11668. next_state = SCAN_BACKING_OP;
  11669. }
  11670. }
  11671. #endif
  11672. } else if (rtw_p2p_findphase_ex_is_needed(pwdinfo)) {
  11673. /* go p2p listen */
  11674. next_state = SCAN_TO_P2P_LISTEN;
  11675. #ifdef CONFIG_ANTENNA_DIVERSITY
  11676. } else if (rtw_hal_antdiv_before_linked(padapter)) {
  11677. /* go sw antdiv before link */
  11678. next_state = SCAN_SW_ANTDIV_BL;
  11679. #endif
  11680. } else {
  11681. next_state = SCAN_COMPLETE;
  11682. #if defined(DBG_SCAN_SW_ANTDIV_BL)
  11683. {
  11684. /* for SCAN_SW_ANTDIV_BL state testing */
  11685. struct dvobj_priv *dvobj = adapter_to_dvobj(padapter);
  11686. int i;
  11687. bool is_linked = _FALSE;
  11688. for (i = 0; i < dvobj->iface_nums; i++) {
  11689. if (rtw_linked_check(dvobj->padapters[i]))
  11690. is_linked = _TRUE;
  11691. }
  11692. if (!is_linked) {
  11693. static bool fake_sw_antdiv_bl_state = 0;
  11694. if (fake_sw_antdiv_bl_state == 0) {
  11695. next_state = SCAN_SW_ANTDIV_BL;
  11696. fake_sw_antdiv_bl_state = 1;
  11697. } else
  11698. fake_sw_antdiv_bl_state = 0;
  11699. }
  11700. }
  11701. #endif /* defined(DBG_SCAN_SW_ANTDIV_BL) */
  11702. }
  11703. #ifdef CONFIG_SCAN_BACKOP
  11704. if (next_state != SCAN_PROCESS)
  11705. ss->scan_cnt = 0;
  11706. #endif
  11707. #ifdef DBG_FIXED_CHAN
  11708. if (pmlmeext->fixed_chan != 0xff && next_state == SCAN_PROCESS)
  11709. scan_ch = pmlmeext->fixed_chan;
  11710. #endif
  11711. if (ch)
  11712. *ch = scan_ch;
  11713. if (type)
  11714. *type = scan_type;
  11715. return next_state;
  11716. }
  11717. void site_survey(_adapter *padapter, u8 survey_channel, RT_SCAN_TYPE ScanType)
  11718. {
  11719. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11720. struct ss_res *ss = &pmlmeext->sitesurvey_res;
  11721. u8 ssid_scan = 0;
  11722. #ifdef CONFIG_P2P
  11723. #ifndef CONFIG_IOCTL_CFG80211
  11724. struct wifidirect_info *pwdinfo = &(padapter->wdinfo);
  11725. #endif
  11726. #endif
  11727. if (survey_channel != 0) {
  11728. set_channel_bwmode(padapter, survey_channel, HAL_PRIME_CHNL_OFFSET_DONT_CARE, CHANNEL_WIDTH_20);
  11729. #ifdef CONFIG_DFS
  11730. if (ScanType == SCAN_PASSIVE && ss->dfs_ch_ssid_scan)
  11731. ssid_scan = 1;
  11732. else
  11733. #endif
  11734. if (ScanType == SCAN_ACTIVE) {
  11735. #ifdef CONFIG_P2P
  11736. #ifdef CONFIG_IOCTL_CFG80211
  11737. if (rtw_cfg80211_is_p2p_scan(padapter))
  11738. #else
  11739. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_SCAN)
  11740. || rtw_p2p_chk_state(pwdinfo, P2P_STATE_FIND_PHASE_SEARCH))
  11741. #endif
  11742. {
  11743. issue_probereq_p2p(padapter, NULL);
  11744. issue_probereq_p2p(padapter, NULL);
  11745. issue_probereq_p2p(padapter, NULL);
  11746. } else
  11747. #endif /* CONFIG_P2P */
  11748. {
  11749. if (pmlmeext->sitesurvey_res.scan_mode == SCAN_ACTIVE) {
  11750. /* IOT issue, When wifi_spec is not set, send one probe req without WPS IE. */
  11751. if (padapter->registrypriv.wifi_spec)
  11752. issue_probereq(padapter, NULL, NULL);
  11753. else
  11754. issue_probereq_ex(padapter, NULL, NULL, 0, 0, 0, 0);
  11755. issue_probereq(padapter, NULL, NULL);
  11756. }
  11757. ssid_scan = 1;
  11758. }
  11759. }
  11760. if (ssid_scan) {
  11761. int i;
  11762. for (i = 0; i < RTW_SSID_SCAN_AMOUNT; i++) {
  11763. if (pmlmeext->sitesurvey_res.ssid[i].SsidLength) {
  11764. /* IOT issue, When wifi_spec is not set, send one probe req without WPS IE. */
  11765. if (padapter->registrypriv.wifi_spec)
  11766. issue_probereq(padapter, &(pmlmeext->sitesurvey_res.ssid[i]), NULL);
  11767. else
  11768. issue_probereq_ex(padapter, &(pmlmeext->sitesurvey_res.ssid[i]), NULL, 0, 0, 0, 0);
  11769. issue_probereq(padapter, &(pmlmeext->sitesurvey_res.ssid[i]), NULL);
  11770. }
  11771. }
  11772. }
  11773. } else {
  11774. /* channel number is 0 or this channel is not valid. */
  11775. rtw_warn_on(1);
  11776. }
  11777. return;
  11778. }
  11779. void survey_done_set_ch_bw(_adapter *padapter)
  11780. {
  11781. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  11782. u8 cur_channel = 0;
  11783. u8 cur_bwmode;
  11784. u8 cur_ch_offset;
  11785. #ifdef CONFIG_MCC_MODE
  11786. if (!rtw_hal_mcc_change_scan_flag(padapter, &cur_channel, &cur_bwmode, &cur_ch_offset)) {
  11787. if (0)
  11788. RTW_INFO(FUNC_ADPT_FMT" back to AP channel - ch:%u, bw:%u, offset:%u\n",
  11789. FUNC_ADPT_ARG(padapter), cur_channel, cur_bwmode, cur_ch_offset);
  11790. goto exit;
  11791. }
  11792. #endif
  11793. if (rtw_mi_get_ch_setting_union(padapter, &cur_channel, &cur_bwmode, &cur_ch_offset) != 0) {
  11794. if (0)
  11795. RTW_INFO(FUNC_ADPT_FMT" back to linked/linking union - ch:%u, bw:%u, offset:%u\n",
  11796. FUNC_ADPT_ARG(padapter), cur_channel, cur_bwmode, cur_ch_offset);
  11797. } else {
  11798. #ifdef CONFIG_P2P
  11799. struct dvobj_priv *dvobj = adapter_to_dvobj(padapter);
  11800. _adapter *iface;
  11801. int i;
  11802. for (i = 0; i < dvobj->iface_nums; i++) {
  11803. iface = dvobj->padapters[i];
  11804. if (!iface)
  11805. continue;
  11806. #ifdef CONFIG_IOCTL_CFG80211
  11807. if (iface->wdinfo.driver_interface == DRIVER_CFG80211 && !adapter_wdev_data(iface)->p2p_enabled)
  11808. continue;
  11809. #endif
  11810. if (rtw_p2p_chk_state(&iface->wdinfo, P2P_STATE_LISTEN)) {
  11811. cur_channel = iface->wdinfo.listen_channel;
  11812. cur_bwmode = CHANNEL_WIDTH_20;
  11813. cur_ch_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
  11814. if (0)
  11815. RTW_INFO(FUNC_ADPT_FMT" back to "ADPT_FMT"'s listen ch - ch:%u, bw:%u, offset:%u\n",
  11816. FUNC_ADPT_ARG(padapter), ADPT_ARG(iface), cur_channel, cur_bwmode, cur_ch_offset);
  11817. break;
  11818. }
  11819. }
  11820. #endif /* CONFIG_P2P */
  11821. if (cur_channel == 0) {
  11822. cur_channel = pmlmeext->cur_channel;
  11823. cur_bwmode = pmlmeext->cur_bwmode;
  11824. cur_ch_offset = pmlmeext->cur_ch_offset;
  11825. if (0)
  11826. RTW_INFO(FUNC_ADPT_FMT" back to ch:%u, bw:%u, offset:%u\n",
  11827. FUNC_ADPT_ARG(padapter), cur_channel, cur_bwmode, cur_ch_offset);
  11828. }
  11829. }
  11830. #ifdef CONFIG_MCC_MODE
  11831. exit:
  11832. #endif
  11833. set_channel_bwmode(padapter, cur_channel, cur_ch_offset, cur_bwmode);
  11834. }
  11835. /**
  11836. * rtw_ps_annc - check and doing ps announcement for all the adapters
  11837. * @adapter: the requesting adapter
  11838. * @ps: power saving or not
  11839. *
  11840. * Returns: 0: no ps announcement is doing. 1: ps announcement is doing
  11841. */
  11842. u8 rtw_ps_annc(_adapter *adapter, bool ps)
  11843. {
  11844. struct dvobj_priv *dvobj = adapter_to_dvobj(adapter);
  11845. _adapter *iface;
  11846. int i;
  11847. u8 ps_anc = 0;
  11848. for (i = 0; i < dvobj->iface_nums; i++) {
  11849. iface = dvobj->padapters[i];
  11850. if (!iface)
  11851. continue;
  11852. if (MLME_IS_STA(iface)) {
  11853. if (is_client_associated_to_ap(iface) == _TRUE) {
  11854. /* TODO: TDLS peers */
  11855. #ifdef CONFIG_MCC_MODE
  11856. /* for two station case */
  11857. if (MCC_EN(adapter) && rtw_hal_check_mcc_status(adapter, MCC_STATUS_NEED_MCC)) {
  11858. u8 ch = iface->mlmeextpriv.cur_channel;
  11859. u8 offset = iface->mlmeextpriv.cur_ch_offset;
  11860. u8 bw = iface->mlmeextpriv.cur_bwmode;
  11861. set_channel_bwmode(iface, ch, offset, bw);
  11862. }
  11863. #endif /* CONFIG_MCC_MODE */
  11864. issue_nulldata(iface, NULL, ps, 3, 500);
  11865. ps_anc = 1;
  11866. }
  11867. #ifdef CONFIG_RTW_MESH
  11868. } else if (MLME_IS_MESH(iface)) {
  11869. if (rtw_mesh_ps_annc(iface, ps))
  11870. ps_anc = 1;
  11871. #endif
  11872. }
  11873. }
  11874. return ps_anc;
  11875. }
  11876. void rtw_leave_opch(_adapter *adapter)
  11877. {
  11878. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  11879. #ifdef CONFIG_MCC_MODE
  11880. if (MCC_EN(adapter) && rtw_hal_check_mcc_status(adapter, MCC_STATUS_DOING_MCC))
  11881. return;
  11882. #endif
  11883. _enter_critical_mutex(&rfctl->offch_mutex, NULL);
  11884. if (rfctl->offch_state == OFFCHS_NONE) {
  11885. /* prepare to leave operating channel */
  11886. rfctl->offch_state = OFFCHS_LEAVING_OP;
  11887. /* clear HW TX queue */
  11888. rtw_hal_set_hwreg(adapter, HW_VAR_CHECK_TXBUF, 0);
  11889. rtw_hal_macid_sleep_all_used(adapter);
  11890. rtw_ps_annc(adapter, 1);
  11891. rfctl->offch_state = OFFCHS_LEAVE_OP;
  11892. }
  11893. _exit_critical_mutex(&rfctl->offch_mutex, NULL);
  11894. }
  11895. void rtw_back_opch(_adapter *adapter)
  11896. {
  11897. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  11898. #ifdef CONFIG_MCC_MODE
  11899. if (MCC_EN(adapter) && rtw_hal_check_mcc_status(adapter, MCC_STATUS_DOING_MCC))
  11900. return;
  11901. #endif
  11902. _enter_critical_mutex(&rfctl->offch_mutex, NULL);
  11903. if (rfctl->offch_state != OFFCHS_NONE) {
  11904. rfctl->offch_state = OFFCHS_BACKING_OP;
  11905. rtw_hal_macid_wakeup_all_used(adapter);
  11906. rtw_ps_annc(adapter, 0);
  11907. rfctl->offch_state = OFFCHS_NONE;
  11908. rtw_mi_os_xmit_schedule(adapter);
  11909. }
  11910. _exit_critical_mutex(&rfctl->offch_mutex, NULL);
  11911. }
  11912. void sitesurvey_set_igi(_adapter *adapter)
  11913. {
  11914. struct mlme_ext_priv *mlmeext = &adapter->mlmeextpriv;
  11915. struct ss_res *ss = &mlmeext->sitesurvey_res;
  11916. u8 igi;
  11917. #ifdef CONFIG_P2P
  11918. struct wifidirect_info *pwdinfo = &adapter->wdinfo;
  11919. #endif
  11920. switch (mlmeext_scan_state(mlmeext)) {
  11921. case SCAN_ENTER:
  11922. #ifdef CONFIG_P2P
  11923. #ifdef CONFIG_IOCTL_CFG80211
  11924. if (pwdinfo->driver_interface == DRIVER_CFG80211 && rtw_cfg80211_is_p2p_scan(adapter))
  11925. igi = 0x30;
  11926. else
  11927. #endif /* CONFIG_IOCTL_CFG80211 */
  11928. if (!rtw_p2p_chk_state(pwdinfo, P2P_STATE_NONE))
  11929. igi = 0x28;
  11930. else
  11931. #endif /* CONFIG_P2P */
  11932. if (ss->igi)
  11933. igi = ss->igi;
  11934. else
  11935. #if defined(CONFIG_RTW_ACS) && defined(CONFIG_RTW_ACS_DBG)
  11936. if (IS_ACS_ENABLE(adapter) && rtw_is_acs_igi_valid(adapter))
  11937. igi = rtw_acs_get_adv_igi(adapter);
  11938. else
  11939. #endif /*CONFIG_RTW_ACS*/
  11940. igi = 0x1e;
  11941. /* record IGI status */
  11942. ss->igi_scan = igi;
  11943. rtw_hal_get_odm_var(adapter, HAL_ODM_INITIAL_GAIN, &ss->igi_before_scan, NULL);
  11944. /* disable DIG and set IGI for scan */
  11945. rtw_hal_set_odm_var(adapter, HAL_ODM_INITIAL_GAIN, &igi, _FALSE);
  11946. break;
  11947. case SCAN_COMPLETE:
  11948. case SCAN_TO_P2P_LISTEN:
  11949. /* enable DIG and restore IGI */
  11950. igi = 0xff;
  11951. rtw_hal_set_odm_var(adapter, HAL_ODM_INITIAL_GAIN, &igi, _FALSE);
  11952. break;
  11953. #ifdef CONFIG_SCAN_BACKOP
  11954. case SCAN_BACKING_OP:
  11955. /* write IGI for op channel when DIG is not enabled */
  11956. odm_write_dig(adapter_to_phydm(adapter), ss->igi_before_scan);
  11957. break;
  11958. case SCAN_LEAVE_OP:
  11959. /* write IGI for scan when DIG is not enabled */
  11960. odm_write_dig(adapter_to_phydm(adapter), ss->igi_scan);
  11961. break;
  11962. #endif /* CONFIG_SCAN_BACKOP */
  11963. default:
  11964. rtw_warn_on(1);
  11965. break;
  11966. }
  11967. }
  11968. void sitesurvey_set_msr(_adapter *adapter, bool enter)
  11969. {
  11970. u8 network_type;
  11971. struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
  11972. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  11973. if (enter) {
  11974. #ifdef CONFIG_MI_WITH_MBSSID_CAM
  11975. rtw_hal_get_hwreg(adapter, HW_VAR_MEDIA_STATUS, (u8 *)(&pmlmeinfo->hw_media_state));
  11976. #endif
  11977. /* set MSR to no link state */
  11978. network_type = _HW_STATE_NOLINK_;
  11979. } else {
  11980. #ifdef CONFIG_MI_WITH_MBSSID_CAM
  11981. network_type = pmlmeinfo->hw_media_state;
  11982. #else
  11983. network_type = pmlmeinfo->state & 0x3;
  11984. #endif
  11985. }
  11986. Set_MSR(adapter, network_type);
  11987. }
  11988. void sitesurvey_set_offch_state(_adapter *adapter, u8 scan_state)
  11989. {
  11990. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  11991. _enter_critical_mutex(&rfctl->offch_mutex, NULL);
  11992. switch (scan_state) {
  11993. case SCAN_DISABLE:
  11994. case SCAN_BACK_OP:
  11995. rfctl->offch_state = OFFCHS_NONE;
  11996. break;
  11997. case SCAN_START:
  11998. case SCAN_LEAVING_OP:
  11999. rfctl->offch_state = OFFCHS_LEAVING_OP;
  12000. break;
  12001. case SCAN_ENTER:
  12002. case SCAN_LEAVE_OP:
  12003. rfctl->offch_state = OFFCHS_LEAVE_OP;
  12004. break;
  12005. case SCAN_COMPLETE:
  12006. case SCAN_BACKING_OP:
  12007. rfctl->offch_state = OFFCHS_BACKING_OP;
  12008. break;
  12009. default:
  12010. break;
  12011. }
  12012. _exit_critical_mutex(&rfctl->offch_mutex, NULL);
  12013. }
  12014. u8 sitesurvey_cmd_hdl(_adapter *padapter, u8 *pbuf)
  12015. {
  12016. struct sitesurvey_parm *pparm = (struct sitesurvey_parm *)pbuf;
  12017. #ifdef DBG_CHECK_FW_PS_STATE
  12018. struct dvobj_priv *dvobj = padapter->dvobj;
  12019. struct debug_priv *pdbgpriv = &dvobj->drv_dbg;
  12020. #endif
  12021. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12022. struct ss_res *ss = &pmlmeext->sitesurvey_res;
  12023. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  12024. struct rtw_wdev_priv *pwdev_priv = adapter_wdev_data(padapter);
  12025. #endif
  12026. u8 val8;
  12027. #ifdef CONFIG_P2P
  12028. struct wifidirect_info *pwdinfo = &padapter->wdinfo;
  12029. #endif
  12030. #ifdef DBG_CHECK_FW_PS_STATE
  12031. if (rtw_fw_ps_state(padapter) == _FAIL) {
  12032. RTW_INFO("scan without leave 32k\n");
  12033. pdbgpriv->dbg_scan_pwr_state_cnt++;
  12034. }
  12035. #endif /* DBG_CHECK_FW_PS_STATE */
  12036. /* increase channel idx */
  12037. if (mlmeext_chk_scan_state(pmlmeext, SCAN_PROCESS))
  12038. ss->channel_idx++;
  12039. /* update scan state to next state (assigned by previous cmd hdl) */
  12040. if (mlmeext_scan_state(pmlmeext) != mlmeext_scan_next_state(pmlmeext))
  12041. mlmeext_set_scan_state(pmlmeext, mlmeext_scan_next_state(pmlmeext));
  12042. operation_by_state:
  12043. switch (mlmeext_scan_state(pmlmeext)) {
  12044. case SCAN_DISABLE:
  12045. /*
  12046. * SW parameter initialization
  12047. */
  12048. sitesurvey_res_reset(padapter, pparm);
  12049. mlmeext_set_scan_state(pmlmeext, SCAN_START);
  12050. goto operation_by_state;
  12051. case SCAN_START:
  12052. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  12053. if ((pwdev_priv->pno_mac_addr[0] != 0xFF)
  12054. && (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) == _TRUE)
  12055. && (check_fwstate(&padapter->mlmepriv, _FW_LINKED) == _FALSE)) {
  12056. u16 seq_num;
  12057. rtw_hal_pno_random_gen_mac_addr(padapter);
  12058. rtw_hal_set_hw_mac_addr(padapter, pwdev_priv->pno_mac_addr);
  12059. get_random_bytes(&seq_num, 2);
  12060. pwdev_priv->pno_scan_seq_num = seq_num & 0xFFF;
  12061. RTW_INFO("%s pno_scan_seq_num %d\n", __func__,
  12062. pwdev_priv->pno_scan_seq_num);
  12063. }
  12064. #endif
  12065. /*
  12066. * prepare to leave operating channel
  12067. */
  12068. #ifdef CONFIG_MCC_MODE
  12069. rtw_hal_set_mcc_setting_scan_start(padapter);
  12070. #endif /* CONFIG_MCC_MODE */
  12071. /* apply rx ampdu setting */
  12072. if (ss->rx_ampdu_accept != RX_AMPDU_ACCEPT_INVALID
  12073. || ss->rx_ampdu_size != RX_AMPDU_SIZE_INVALID)
  12074. rtw_rx_ampdu_apply(padapter);
  12075. /* clear HW TX queue before scan */
  12076. rtw_hal_set_hwreg(padapter, HW_VAR_CHECK_TXBUF, 0);
  12077. rtw_hal_macid_sleep_all_used(padapter);
  12078. /* power save state announcement */
  12079. if (rtw_ps_annc(padapter, 1)) {
  12080. mlmeext_set_scan_state(pmlmeext, SCAN_PS_ANNC_WAIT);
  12081. mlmeext_set_scan_next_state(pmlmeext, SCAN_ENTER);
  12082. set_survey_timer(pmlmeext, 50); /* delay 50ms to protect nulldata(1) */
  12083. } else {
  12084. mlmeext_set_scan_state(pmlmeext, SCAN_ENTER);
  12085. goto operation_by_state;
  12086. }
  12087. break;
  12088. case SCAN_ENTER:
  12089. /*
  12090. * HW register and DM setting for enter scan
  12091. */
  12092. rtw_phydm_ability_backup(padapter);
  12093. sitesurvey_set_igi(padapter);
  12094. /* config dynamic functions for off channel */
  12095. rtw_phydm_func_for_offchannel(padapter);
  12096. /* set MSR to no link state */
  12097. sitesurvey_set_msr(padapter, _TRUE);
  12098. val8 = 1; /* under site survey */
  12099. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_SITESURVEY, (u8 *)(&val8));
  12100. mlmeext_set_scan_state(pmlmeext, SCAN_PROCESS);
  12101. goto operation_by_state;
  12102. case SCAN_PROCESS: {
  12103. u8 scan_ch;
  12104. RT_SCAN_TYPE scan_type;
  12105. u8 next_state;
  12106. u32 scan_ms;
  12107. #ifdef CONFIG_RTW_ACS
  12108. if (IS_ACS_ENABLE(padapter))
  12109. rtw_acs_get_rst(padapter);
  12110. #endif
  12111. next_state = sitesurvey_pick_ch_behavior(padapter, &scan_ch, &scan_type);
  12112. if (next_state != SCAN_PROCESS) {
  12113. mlmeext_set_scan_state(pmlmeext, next_state);
  12114. goto operation_by_state;
  12115. }
  12116. /* still SCAN_PROCESS state */
  12117. #ifdef DBG_SITESURVEY
  12118. #ifdef CONFIG_P2P
  12119. RTW_INFO(FUNC_ADPT_FMT" %s ch:%u (cnt:%u,idx:%d) at %dms, %c%c%c%c\n"
  12120. , FUNC_ADPT_ARG(padapter)
  12121. , mlmeext_scan_state_str(pmlmeext)
  12122. , scan_ch
  12123. , pwdinfo->find_phase_state_exchange_cnt, ss->channel_idx
  12124. , rtw_get_passing_time_ms(padapter->mlmepriv.scan_start_time)
  12125. , scan_type ? 'A' : 'P', ss->scan_mode ? 'A' : 'P'
  12126. , ss->ssid[0].SsidLength ? 'S' : ' '
  12127. , ss->dfs_ch_ssid_scan ? 'D' : ' '
  12128. );
  12129. #else
  12130. RTW_INFO(FUNC_ADPT_FMT" %s ch:%u (idx:%d) at %dms, %c%c%c%c\n"
  12131. , FUNC_ADPT_ARG(padapter)
  12132. , mlmeext_scan_state_str(pmlmeext)
  12133. , scan_ch
  12134. , ss->channel_idx
  12135. , rtw_get_passing_time_ms(padapter->mlmepriv.scan_start_time)
  12136. , scan_type ? 'A' : 'P', ss->scan_mode ? 'A' : 'P'
  12137. , ss->ssid[0].SsidLength ? 'S' : ' '
  12138. , ss->dfs_ch_ssid_scan ? 'D' : ' '
  12139. );
  12140. #endif /* CONFIG_P2P */
  12141. #endif /*DBG_SITESURVEY*/
  12142. #ifdef DBG_FIXED_CHAN
  12143. if (pmlmeext->fixed_chan != 0xff)
  12144. RTW_INFO(FUNC_ADPT_FMT" fixed_chan:%u\n", pmlmeext->fixed_chan);
  12145. #endif
  12146. site_survey(padapter, scan_ch, scan_type);
  12147. #if defined(CONFIG_ATMEL_RC_PATCH)
  12148. if (check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE)
  12149. scan_ms = 20;
  12150. else
  12151. scan_ms = 40;
  12152. #else
  12153. #if defined(CONFIG_RTW_ACS) && defined(CONFIG_RTW_ACS_DBG)
  12154. if (IS_ACS_ENABLE(padapter) && rtw_is_acs_st_valid(padapter))
  12155. scan_ms = rtw_acs_get_adv_st(padapter);
  12156. else
  12157. #endif /*CONFIG_RTW_ACS*/
  12158. scan_ms = ss->scan_ch_ms;
  12159. #endif
  12160. #if defined(CONFIG_ANTENNA_DIVERSITY) || defined(DBG_SCAN_SW_ANTDIV_BL)
  12161. if (ss->is_sw_antdiv_bl_scan)
  12162. scan_ms = scan_ms / 2;
  12163. #endif
  12164. #ifdef CONFIG_RTW_ACS
  12165. if (IS_ACS_ENABLE(padapter)) {
  12166. if (pparm->token)
  12167. rtw_acs_trigger(padapter, scan_ms, scan_ch, NHM_PID_IEEE_11K_HIGH);
  12168. else
  12169. rtw_acs_trigger(padapter, scan_ms, scan_ch, NHM_PID_ACS);
  12170. }
  12171. #endif
  12172. #ifdef CONFIG_BACKGROUND_NOISE_MONITOR
  12173. if (IS_NM_ENABLE(padapter))
  12174. rtw_noise_measure(padapter, scan_ch, _FALSE, 0, scan_ms / 2);
  12175. #endif
  12176. set_survey_timer(pmlmeext, scan_ms);
  12177. break;
  12178. }
  12179. #ifdef CONFIG_SCAN_BACKOP
  12180. case SCAN_BACKING_OP: {
  12181. u8 back_ch, back_bw, back_ch_offset;
  12182. u8 need_ch_setting_union = _TRUE;
  12183. #ifdef CONFIG_MCC_MODE
  12184. need_ch_setting_union = rtw_hal_mcc_change_scan_flag(padapter,
  12185. &back_ch, &back_bw, &back_ch_offset);
  12186. #endif /* CONFIG_MCC_MODE */
  12187. if (need_ch_setting_union) {
  12188. if (rtw_mi_get_ch_setting_union(padapter, &back_ch, &back_bw, &back_ch_offset) == 0)
  12189. rtw_warn_on(1);
  12190. }
  12191. #ifdef DBG_SITESURVEY
  12192. RTW_INFO(FUNC_ADPT_FMT" %s ch:%u, bw:%u, offset:%u at %dms\n"
  12193. , FUNC_ADPT_ARG(padapter)
  12194. , mlmeext_scan_state_str(pmlmeext)
  12195. , back_ch, back_bw, back_ch_offset
  12196. , rtw_get_passing_time_ms(padapter->mlmepriv.scan_start_time)
  12197. );
  12198. #endif /*DBG_SITESURVEY*/
  12199. set_channel_bwmode(padapter, back_ch, back_ch_offset, back_bw);
  12200. sitesurvey_set_msr(padapter, _FALSE);
  12201. val8 = 0; /* survey done */
  12202. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_SITESURVEY, (u8 *)(&val8));
  12203. if (mlmeext_chk_scan_backop_flags(pmlmeext, SS_BACKOP_PS_ANNC)) {
  12204. sitesurvey_set_igi(padapter);
  12205. rtw_hal_macid_wakeup_all_used(padapter);
  12206. rtw_ps_annc(padapter, 0);
  12207. }
  12208. mlmeext_set_scan_state(pmlmeext, SCAN_BACK_OP);
  12209. ss->backop_time = rtw_get_current_time();
  12210. if (mlmeext_chk_scan_backop_flags(pmlmeext, SS_BACKOP_TX_RESUME))
  12211. rtw_mi_os_xmit_schedule(padapter);
  12212. goto operation_by_state;
  12213. }
  12214. case SCAN_BACK_OP:
  12215. if (rtw_get_passing_time_ms(ss->backop_time) >= ss->backop_ms
  12216. || pmlmeext->scan_abort
  12217. ) {
  12218. mlmeext_set_scan_state(pmlmeext, SCAN_LEAVING_OP);
  12219. goto operation_by_state;
  12220. }
  12221. set_survey_timer(pmlmeext, 50);
  12222. break;
  12223. case SCAN_LEAVING_OP:
  12224. /*
  12225. * prepare to leave operating channel
  12226. */
  12227. /* clear HW TX queue before scan */
  12228. rtw_hal_set_hwreg(padapter, HW_VAR_CHECK_TXBUF, 0);
  12229. rtw_hal_macid_sleep_all_used(padapter);
  12230. if (mlmeext_chk_scan_backop_flags(pmlmeext, SS_BACKOP_PS_ANNC)
  12231. && rtw_ps_annc(padapter, 1)
  12232. ) {
  12233. mlmeext_set_scan_state(pmlmeext, SCAN_PS_ANNC_WAIT);
  12234. mlmeext_set_scan_next_state(pmlmeext, SCAN_LEAVE_OP);
  12235. set_survey_timer(pmlmeext, 50); /* delay 50ms to protect nulldata(1) */
  12236. } else {
  12237. mlmeext_set_scan_state(pmlmeext, SCAN_LEAVE_OP);
  12238. goto operation_by_state;
  12239. }
  12240. break;
  12241. case SCAN_LEAVE_OP:
  12242. /*
  12243. * HW register and DM setting for enter scan
  12244. */
  12245. if (mlmeext_chk_scan_backop_flags(pmlmeext, SS_BACKOP_PS_ANNC))
  12246. sitesurvey_set_igi(padapter);
  12247. sitesurvey_set_msr(padapter, _TRUE);
  12248. val8 = 1; /* under site survey */
  12249. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_SITESURVEY, (u8 *)(&val8));
  12250. mlmeext_set_scan_state(pmlmeext, SCAN_PROCESS);
  12251. goto operation_by_state;
  12252. #endif /* CONFIG_SCAN_BACKOP */
  12253. #if defined(CONFIG_ANTENNA_DIVERSITY) || defined(DBG_SCAN_SW_ANTDIV_BL)
  12254. case SCAN_SW_ANTDIV_BL:
  12255. /*
  12256. * 20100721
  12257. * For SW antenna diversity before link, it needs to switch to another antenna and scan again.
  12258. * It compares the scan result and select better one to do connection.
  12259. */
  12260. ss->bss_cnt = 0;
  12261. ss->channel_idx = 0;
  12262. ss->is_sw_antdiv_bl_scan = 1;
  12263. mlmeext_set_scan_next_state(pmlmeext, SCAN_PROCESS);
  12264. set_survey_timer(pmlmeext, ss->scan_ch_ms);
  12265. break;
  12266. #endif
  12267. #ifdef CONFIG_P2P
  12268. case SCAN_TO_P2P_LISTEN:
  12269. /*
  12270. * Set the P2P State to the listen state of find phase
  12271. * and set the current channel to the listen channel
  12272. */
  12273. set_channel_bwmode(padapter, pwdinfo->listen_channel, HAL_PRIME_CHNL_OFFSET_DONT_CARE, CHANNEL_WIDTH_20);
  12274. rtw_p2p_set_state(pwdinfo, P2P_STATE_FIND_PHASE_LISTEN);
  12275. /* turn on phy-dynamic functions */
  12276. rtw_phydm_ability_restore(padapter);
  12277. sitesurvey_set_igi(padapter);
  12278. mlmeext_set_scan_state(pmlmeext, SCAN_P2P_LISTEN);
  12279. _set_timer(&pwdinfo->find_phase_timer, (u32)((u32)pwdinfo->listen_dwell * 100));
  12280. break;
  12281. case SCAN_P2P_LISTEN:
  12282. mlmeext_set_scan_state(pmlmeext, SCAN_PROCESS);
  12283. ss->channel_idx = 0;
  12284. goto operation_by_state;
  12285. #endif /* CONFIG_P2P */
  12286. case SCAN_COMPLETE:
  12287. #ifdef CONFIG_RTW_CFGVENDOR_RANDOM_MAC_OUI
  12288. rtw_hal_set_hw_mac_addr(padapter, adapter_mac_addr(padapter));
  12289. #endif
  12290. #ifdef CONFIG_P2P
  12291. if (rtw_p2p_chk_state(pwdinfo, P2P_STATE_SCAN)
  12292. || rtw_p2p_chk_state(pwdinfo, P2P_STATE_FIND_PHASE_SEARCH)
  12293. ) {
  12294. #ifdef CONFIG_CONCURRENT_MODE
  12295. if (pwdinfo->driver_interface == DRIVER_WEXT) {
  12296. if (rtw_mi_check_status(padapter, MI_LINKED))
  12297. _set_timer(&pwdinfo->ap_p2p_switch_timer, 500);
  12298. }
  12299. #endif
  12300. rtw_p2p_set_state(pwdinfo, rtw_p2p_pre_state(pwdinfo));
  12301. }
  12302. rtw_p2p_findphase_ex_set(pwdinfo, P2P_FINDPHASE_EX_NONE);
  12303. #endif /* CONFIG_P2P */
  12304. /* switch channel */
  12305. survey_done_set_ch_bw(padapter);
  12306. sitesurvey_set_msr(padapter, _FALSE);
  12307. val8 = 0; /* survey done */
  12308. rtw_hal_set_hwreg(padapter, HW_VAR_MLME_SITESURVEY, (u8 *)(&val8));
  12309. /* turn on phy-dynamic functions */
  12310. rtw_phydm_ability_restore(padapter);
  12311. sitesurvey_set_igi(padapter);
  12312. #ifdef CONFIG_MCC_MODE
  12313. /* start MCC fail, then tx null data */
  12314. if (!rtw_hal_set_mcc_setting_scan_complete(padapter))
  12315. #endif
  12316. {
  12317. rtw_hal_macid_wakeup_all_used(padapter);
  12318. rtw_ps_annc(padapter, 0);
  12319. }
  12320. /* apply rx ampdu setting */
  12321. rtw_rx_ampdu_apply(padapter);
  12322. mlmeext_set_scan_state(pmlmeext, SCAN_DISABLE);
  12323. report_surveydone_event(padapter);
  12324. #ifdef CONFIG_RTW_ACS
  12325. if (IS_ACS_ENABLE(padapter))
  12326. rtw_acs_select_best_chan(padapter);
  12327. #endif
  12328. #if defined(CONFIG_BACKGROUND_NOISE_MONITOR) && defined(DBG_NOISE_MONITOR)
  12329. if (IS_NM_ENABLE(padapter))
  12330. rtw_noise_info_dump(RTW_DBGDUMP, padapter);
  12331. #endif
  12332. issue_action_BSSCoexistPacket(padapter);
  12333. issue_action_BSSCoexistPacket(padapter);
  12334. issue_action_BSSCoexistPacket(padapter);
  12335. #ifdef CONFIG_RTW_80211K
  12336. if (ss->token)
  12337. rm_post_event(padapter, ss->token, RM_EV_survey_done);
  12338. #endif /* CONFIG_RTW_80211K */
  12339. break;
  12340. }
  12341. return H2C_SUCCESS;
  12342. }
  12343. u8 setauth_hdl(_adapter *padapter, unsigned char *pbuf)
  12344. {
  12345. struct setauth_parm *pparm = (struct setauth_parm *)pbuf;
  12346. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12347. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  12348. if (pparm->mode < 4)
  12349. pmlmeinfo->auth_algo = pparm->mode;
  12350. return H2C_SUCCESS;
  12351. }
  12352. /*
  12353. SEC CAM Entry format (32 bytes)
  12354. DW0 - MAC_ADDR[15:0] | Valid[15] | MFB[14:8] | RSVD[7] | GK[6] | MIC_KEY[5] | SEC_TYPE[4:2] | KID[1:0]
  12355. DW0 - MAC_ADDR[15:0] | Valid[15] |RSVD[14:9] | RPT_MODE[8] | SPP_MODE[7] | GK[6] | MIC_KEY[5] | SEC_TYPE[4:2] | KID[1:0] (92E/8812A/8814A)
  12356. DW1 - MAC_ADDR[47:16]
  12357. DW2 - KEY[31:0]
  12358. DW3 - KEY[63:32]
  12359. DW4 - KEY[95:64]
  12360. DW5 - KEY[127:96]
  12361. DW6 - RSVD
  12362. DW7 - RSVD
  12363. */
  12364. /*Set WEP key or Group Key*/
  12365. u8 setkey_hdl(_adapter *padapter, u8 *pbuf)
  12366. {
  12367. u16 ctrl = 0;
  12368. s16 cam_id = 0;
  12369. struct setkey_parm *pparm = (struct setkey_parm *)pbuf;
  12370. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12371. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  12372. unsigned char null_addr[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
  12373. u8 *addr;
  12374. bool used = _FALSE;
  12375. /* main tx key for wep. */
  12376. if (pparm->set_tx)
  12377. pmlmeinfo->key_index = pparm->keyid;
  12378. #ifdef CONFIG_CONCURRENT_MODE
  12379. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) || ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE))
  12380. cam_id = rtw_iface_bcmc_id_get(padapter);
  12381. else
  12382. #endif
  12383. cam_id = rtw_camid_alloc(padapter, NULL, pparm->keyid, 1, &used);
  12384. if (cam_id < 0)
  12385. goto enable_mc;
  12386. #ifndef CONFIG_CONCURRENT_MODE
  12387. if (cam_id >= 0 && cam_id <= 3) {
  12388. /* default key camid */
  12389. addr = null_addr;
  12390. } else
  12391. #endif
  12392. {
  12393. /* not default key camid */
  12394. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) || ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE)) {
  12395. /* group TX, force sec cam entry_id */
  12396. addr = adapter_mac_addr(padapter);
  12397. } else {
  12398. /* group RX, searched by A2 (TA) */
  12399. addr = get_bssid(&padapter->mlmepriv);
  12400. }
  12401. }
  12402. /* cam entry searched is pairwise key */
  12403. if (used == _TRUE && rtw_camid_is_gk(padapter, cam_id) == _FALSE) {
  12404. s16 camid_clr;
  12405. RTW_PRINT(FUNC_ADPT_FMT" group key with "MAC_FMT" id:%u the same key id as pairwise key\n"
  12406. , FUNC_ADPT_ARG(padapter), MAC_ARG(addr), pparm->keyid);
  12407. /* HW has problem to distinguish this group key with existing pairwise key, stop HW enc and dec for BMC */
  12408. rtw_camctl_set_flags(padapter, SEC_STATUS_STA_PK_GK_CONFLICT_DIS_BMC_SEARCH);
  12409. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_CFG, NULL);
  12410. /* clear group key */
  12411. while ((camid_clr = rtw_camid_search(padapter, addr, -1, 1)) >= 0) {
  12412. RTW_PRINT("clear group key for addr:"MAC_FMT", camid:%d\n", MAC_ARG(addr), camid_clr);
  12413. clear_cam_entry(padapter, camid_clr);
  12414. rtw_camid_free(padapter, camid_clr);
  12415. }
  12416. goto enable_mc;
  12417. }
  12418. ctrl = BIT(15) | BIT(6) | ((pparm->algorithm) << 2) | pparm->keyid;
  12419. RTW_PRINT("set group key camid:%d, addr:"MAC_FMT", kid:%d, type:%s\n"
  12420. , cam_id, MAC_ARG(addr), pparm->keyid, security_type_str(pparm->algorithm));
  12421. write_cam(padapter, cam_id, ctrl, addr, pparm->key);
  12422. /* if ((cam_id > 3) && (((pmlmeinfo->state&0x03) == WIFI_FW_AP_STATE) || ((pmlmeinfo->state&0x03) == WIFI_FW_ADHOC_STATE)))*/
  12423. #ifdef CONFIG_CONCURRENT_MODE
  12424. if (((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) || ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE)) {
  12425. if (is_wep_enc(pparm->algorithm)) {
  12426. padapter->securitypriv.dot11Def_camid[pparm->keyid] = cam_id;
  12427. padapter->securitypriv.dot118021x_bmc_cam_id =
  12428. padapter->securitypriv.dot11Def_camid[padapter->securitypriv.dot11PrivacyKeyIndex];
  12429. RTW_PRINT("wep group key - force camid:%d\n", padapter->securitypriv.dot118021x_bmc_cam_id);
  12430. } else {
  12431. /*u8 org_cam_id = padapter->securitypriv.dot118021x_bmc_cam_id;*/
  12432. /*force GK's cam id*/
  12433. padapter->securitypriv.dot118021x_bmc_cam_id = cam_id;
  12434. /* for GTK rekey
  12435. if ((org_cam_id != INVALID_SEC_MAC_CAM_ID) &&
  12436. (org_cam_id != cam_id)) {
  12437. RTW_PRINT("clear group key for addr:"MAC_FMT", org_camid:%d new_camid:%d\n", MAC_ARG(addr), org_cam_id, cam_id);
  12438. clear_cam_entry(padapter, org_cam_id);
  12439. rtw_camid_free(padapter, org_cam_id);
  12440. }*/
  12441. }
  12442. }
  12443. #endif
  12444. #ifndef CONFIG_CONCURRENT_MODE
  12445. if (cam_id >= 0 && cam_id <= 3)
  12446. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_DK_CFG, (u8 *)_TRUE);
  12447. #endif
  12448. /* 8814au should set both broadcast and unicast CAM entry for WEP key in STA mode */
  12449. if (is_wep_enc(pparm->algorithm) && check_mlmeinfo_state(pmlmeext, WIFI_FW_STATION_STATE) &&
  12450. _rtw_camctl_chk_cap(padapter, SEC_CAP_CHK_BMC)) {
  12451. struct set_stakey_parm sta_pparm;
  12452. _rtw_memset(&sta_pparm, 0, sizeof(struct set_stakey_parm));
  12453. sta_pparm.algorithm = pparm->algorithm;
  12454. sta_pparm.keyid = pparm->keyid;
  12455. _rtw_memcpy(sta_pparm.key, pparm->key, 16);
  12456. _rtw_memcpy(sta_pparm.addr, get_bssid(&padapter->mlmepriv), ETH_ALEN);
  12457. set_stakey_hdl(padapter, (u8 *)&sta_pparm);
  12458. }
  12459. enable_mc:
  12460. /* allow multicast packets to driver */
  12461. rtw_hal_set_hwreg(padapter, HW_VAR_ON_RCR_AM, null_addr);
  12462. return H2C_SUCCESS;
  12463. }
  12464. void rtw_ap_wep_pk_setting(_adapter *adapter, struct sta_info *psta)
  12465. {
  12466. struct security_priv *psecuritypriv = &(adapter->securitypriv);
  12467. struct set_stakey_parm sta_pparm;
  12468. sint keyid;
  12469. if (!is_wep_enc(psecuritypriv->dot11PrivacyAlgrthm))
  12470. return;
  12471. for (keyid = 0; keyid < 4; keyid++) {
  12472. if ((psecuritypriv->key_mask & BIT(keyid)) && (keyid == psecuritypriv->dot11PrivacyKeyIndex)) {
  12473. sta_pparm.algorithm = psecuritypriv->dot11PrivacyAlgrthm;
  12474. sta_pparm.keyid = keyid;
  12475. sta_pparm.gk = 0;
  12476. _rtw_memcpy(sta_pparm.key, &(psecuritypriv->dot11DefKey[keyid].skey[0]), 16);
  12477. _rtw_memcpy(sta_pparm.addr, psta->cmn.mac_addr, ETH_ALEN);
  12478. RTW_PRINT(FUNC_ADPT_FMT"set WEP - PK with "MAC_FMT" keyid:%u\n"
  12479. , FUNC_ADPT_ARG(adapter), MAC_ARG(psta->cmn.mac_addr), keyid);
  12480. set_stakey_hdl(adapter, (u8 *)&sta_pparm);
  12481. }
  12482. }
  12483. }
  12484. u8 set_stakey_hdl(_adapter *padapter, u8 *pbuf)
  12485. {
  12486. u16 ctrl = 0;
  12487. s16 cam_id = 0;
  12488. bool used;
  12489. u8 ret = H2C_SUCCESS;
  12490. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12491. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  12492. struct set_stakey_parm *pparm = (struct set_stakey_parm *)pbuf;
  12493. struct sta_priv *pstapriv = &padapter->stapriv;
  12494. struct sta_info *psta;
  12495. if (pparm->algorithm == _NO_PRIVACY_)
  12496. goto write_to_cam;
  12497. psta = rtw_get_stainfo(pstapriv, pparm->addr);
  12498. if (!psta) {
  12499. RTW_PRINT("%s sta:"MAC_FMT" not found\n", __func__, MAC_ARG(pparm->addr));
  12500. ret = H2C_REJECTED;
  12501. goto exit;
  12502. }
  12503. pmlmeinfo->enc_algo = pparm->algorithm;
  12504. cam_id = rtw_camid_alloc(padapter, psta, pparm->keyid, pparm->gk, &used);
  12505. if (cam_id < 0)
  12506. goto exit;
  12507. /* cam entry searched is group key when setting pariwise key */
  12508. if (!pparm->gk && used == _TRUE && rtw_camid_is_gk(padapter, cam_id) == _TRUE) {
  12509. s16 camid_clr;
  12510. RTW_PRINT(FUNC_ADPT_FMT" pairwise key with "MAC_FMT" id:%u the same key id as group key\n"
  12511. , FUNC_ADPT_ARG(padapter), MAC_ARG(pparm->addr), pparm->keyid);
  12512. /* HW has problem to distinguish this pairwise key with existing group key, stop HW enc and dec for BMC */
  12513. rtw_camctl_set_flags(padapter, SEC_STATUS_STA_PK_GK_CONFLICT_DIS_BMC_SEARCH);
  12514. rtw_hal_set_hwreg(padapter, HW_VAR_SEC_CFG, NULL);
  12515. /* clear group key */
  12516. while ((camid_clr = rtw_camid_search(padapter, pparm->addr, -1, 1)) >= 0) {
  12517. RTW_PRINT("clear group key for addr:"MAC_FMT", camid:%d\n", MAC_ARG(pparm->addr), camid_clr);
  12518. clear_cam_entry(padapter, camid_clr);
  12519. rtw_camid_free(padapter, camid_clr);
  12520. }
  12521. }
  12522. write_to_cam:
  12523. if (pparm->algorithm == _NO_PRIVACY_) {
  12524. while ((cam_id = rtw_camid_search(padapter, pparm->addr, -1, -1)) >= 0) {
  12525. RTW_PRINT("clear key for addr:"MAC_FMT", camid:%d\n", MAC_ARG(pparm->addr), cam_id);
  12526. clear_cam_entry(padapter, cam_id);
  12527. rtw_camid_free(padapter, cam_id);
  12528. }
  12529. } else {
  12530. RTW_PRINT("set %s key camid:%d, addr:"MAC_FMT", kid:%d, type:%s\n"
  12531. , pparm->gk ? "group" : "pairwise"
  12532. , cam_id, MAC_ARG(pparm->addr), pparm->keyid, security_type_str(pparm->algorithm));
  12533. ctrl = BIT(15) | ((pparm->algorithm) << 2) | pparm->keyid;
  12534. if (pparm->gk)
  12535. ctrl |= BIT(6);
  12536. write_cam(padapter, cam_id, ctrl, pparm->addr, pparm->key);
  12537. }
  12538. ret = H2C_SUCCESS_RSP;
  12539. exit:
  12540. return ret;
  12541. }
  12542. u8 add_ba_hdl(_adapter *padapter, unsigned char *pbuf)
  12543. {
  12544. struct addBaReq_parm *pparm = (struct addBaReq_parm *)pbuf;
  12545. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12546. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  12547. struct sta_info *psta = rtw_get_stainfo(&padapter->stapriv, pparm->addr);
  12548. if (!psta)
  12549. return H2C_SUCCESS;
  12550. #ifdef CONFIG_80211N_HT
  12551. if (((pmlmeinfo->state & WIFI_FW_ASSOC_SUCCESS) && (pmlmeinfo->HT_enable)) ||
  12552. ((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE)) {
  12553. /* pmlmeinfo->ADDBA_retry_count = 0; */
  12554. /* pmlmeinfo->candidate_tid_bitmap |= (0x1 << pparm->tid); */
  12555. /* psta->htpriv.candidate_tid_bitmap |= BIT(pparm->tid); */
  12556. issue_addba_req(padapter, pparm->addr, (u8)pparm->tid);
  12557. _set_timer(&psta->addba_retry_timer, ADDBA_TO);
  12558. }
  12559. #ifdef CONFIG_TDLS
  12560. else if ((psta->tdls_sta_state & TDLS_LINKED_STATE) &&
  12561. (psta->htpriv.ht_option == _TRUE) &&
  12562. (psta->htpriv.ampdu_enable == _TRUE)) {
  12563. issue_addba_req(padapter, pparm->addr, (u8)pparm->tid);
  12564. _set_timer(&psta->addba_retry_timer, ADDBA_TO);
  12565. }
  12566. #endif /* CONFIG */
  12567. else
  12568. psta->htpriv.candidate_tid_bitmap &= ~BIT(pparm->tid);
  12569. #endif /* CONFIG_80211N_HT */
  12570. return H2C_SUCCESS;
  12571. }
  12572. u8 add_ba_rsp_hdl(_adapter *padapter, unsigned char *pbuf)
  12573. {
  12574. struct addBaRsp_parm *pparm = (struct addBaRsp_parm *)pbuf;
  12575. struct recv_reorder_ctrl *preorder_ctrl;
  12576. struct sta_priv *pstapriv = &padapter->stapriv;
  12577. struct sta_info *psta;
  12578. u8 ret = _TRUE;
  12579. psta = rtw_get_stainfo(pstapriv, pparm->addr);
  12580. if (!psta)
  12581. goto exit;
  12582. preorder_ctrl = &psta->recvreorder_ctrl[pparm->tid];
  12583. ret = issue_addba_rsp_wait_ack(padapter, pparm->addr, pparm->tid, pparm->status, pparm->size, 3, 50);
  12584. #ifdef CONFIG_UPDATE_INDICATE_SEQ_WHILE_PROCESS_ADDBA_REQ
  12585. /* status = 0 means accept this addba req, so update indicate seq = start_seq under this compile flag */
  12586. if (pparm->status == 0) {
  12587. preorder_ctrl->indicate_seq = pparm->start_seq;
  12588. #ifdef DBG_RX_SEQ
  12589. RTW_INFO("DBG_RX_SEQ "FUNC_ADPT_FMT" tid:%u SN_UPDATE indicate_seq:%d, start_seq:%d\n"
  12590. , FUNC_ADPT_ARG(padapter), preorder_ctrl->tid, preorder_ctrl->indicate_seq, pparm->start_seq);
  12591. #endif
  12592. }
  12593. #else
  12594. preorder_ctrl->indicate_seq = 0xffff;
  12595. #ifdef DBG_RX_SEQ
  12596. RTW_INFO("DBG_RX_SEQ "FUNC_ADPT_FMT" tid:%u SN_CLEAR indicate_seq:%d, start_seq:%d\n"
  12597. , FUNC_ADPT_ARG(padapter), preorder_ctrl->tid, preorder_ctrl->indicate_seq, pparm->start_seq);
  12598. #endif
  12599. #endif
  12600. /*
  12601. * status = 0 means accept this addba req
  12602. * status = 37 means reject this addba req
  12603. */
  12604. if (pparm->status == 0) {
  12605. preorder_ctrl->enable = _TRUE;
  12606. preorder_ctrl->ampdu_size = pparm->size;
  12607. } else if (pparm->status == 37)
  12608. preorder_ctrl->enable = _FALSE;
  12609. exit:
  12610. return H2C_SUCCESS;
  12611. }
  12612. u8 chk_bmc_sleepq_cmd(_adapter *padapter)
  12613. {
  12614. struct cmd_obj *ph2c;
  12615. struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
  12616. u8 res = _SUCCESS;
  12617. ph2c = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  12618. if (ph2c == NULL) {
  12619. res = _FAIL;
  12620. goto exit;
  12621. }
  12622. init_h2fwcmd_w_parm_no_parm_rsp(ph2c, GEN_CMD_CODE(_ChkBMCSleepq));
  12623. res = rtw_enqueue_cmd(pcmdpriv, ph2c);
  12624. exit:
  12625. return res;
  12626. }
  12627. u8 set_tx_beacon_cmd(_adapter *padapter)
  12628. {
  12629. struct cmd_obj *ph2c;
  12630. struct Tx_Beacon_param *ptxBeacon_parm;
  12631. struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
  12632. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  12633. struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
  12634. u8 res = _SUCCESS;
  12635. int len_diff = 0;
  12636. ph2c = (struct cmd_obj *)rtw_zmalloc(sizeof(struct cmd_obj));
  12637. if (ph2c == NULL) {
  12638. res = _FAIL;
  12639. goto exit;
  12640. }
  12641. ptxBeacon_parm = (struct Tx_Beacon_param *)rtw_zmalloc(sizeof(struct Tx_Beacon_param));
  12642. if (ptxBeacon_parm == NULL) {
  12643. rtw_mfree((unsigned char *)ph2c, sizeof(struct cmd_obj));
  12644. res = _FAIL;
  12645. goto exit;
  12646. }
  12647. _rtw_memcpy(&(ptxBeacon_parm->network), &(pmlmeinfo->network), sizeof(WLAN_BSSID_EX));
  12648. len_diff = update_hidden_ssid(
  12649. ptxBeacon_parm->network.IEs + _BEACON_IE_OFFSET_
  12650. , ptxBeacon_parm->network.IELength - _BEACON_IE_OFFSET_
  12651. , pmlmeinfo->hidden_ssid_mode
  12652. );
  12653. ptxBeacon_parm->network.IELength += len_diff;
  12654. init_h2fwcmd_w_parm_no_rsp(ph2c, ptxBeacon_parm, GEN_CMD_CODE(_TX_Beacon));
  12655. res = rtw_enqueue_cmd(pcmdpriv, ph2c);
  12656. exit:
  12657. return res;
  12658. }
  12659. u8 mlme_evt_hdl(_adapter *padapter, unsigned char *pbuf)
  12660. {
  12661. u8 evt_code, evt_seq;
  12662. u16 evt_sz;
  12663. uint *peventbuf;
  12664. void (*event_callback)(_adapter *dev, u8 *pbuf);
  12665. struct evt_priv *pevt_priv = &(padapter->evtpriv);
  12666. if (pbuf == NULL)
  12667. goto _abort_event_;
  12668. peventbuf = (uint *)pbuf;
  12669. evt_sz = (u16)(*peventbuf & 0xffff);
  12670. evt_seq = (u8)((*peventbuf >> 24) & 0x7f);
  12671. evt_code = (u8)((*peventbuf >> 16) & 0xff);
  12672. #ifdef CHECK_EVENT_SEQ
  12673. /* checking event sequence... */
  12674. if (evt_seq != (ATOMIC_READ(&pevt_priv->event_seq) & 0x7f)) {
  12675. pevt_priv->event_seq = (evt_seq + 1) & 0x7f;
  12676. goto _abort_event_;
  12677. }
  12678. #endif
  12679. /* checking if event code is valid */
  12680. if (evt_code >= MAX_C2HEVT) {
  12681. goto _abort_event_;
  12682. }
  12683. /* checking if event size match the event parm size */
  12684. if ((wlanevents[evt_code].parmsize != 0) &&
  12685. (wlanevents[evt_code].parmsize != evt_sz)) {
  12686. goto _abort_event_;
  12687. }
  12688. ATOMIC_INC(&pevt_priv->event_seq);
  12689. peventbuf += 2;
  12690. if (peventbuf) {
  12691. event_callback = wlanevents[evt_code].event_callback;
  12692. event_callback(padapter, (u8 *)peventbuf);
  12693. pevt_priv->evt_done_cnt++;
  12694. }
  12695. _abort_event_:
  12696. return H2C_SUCCESS;
  12697. }
  12698. u8 h2c_msg_hdl(_adapter *padapter, unsigned char *pbuf)
  12699. {
  12700. if (!pbuf)
  12701. return H2C_PARAMETERS_ERROR;
  12702. return H2C_SUCCESS;
  12703. }
  12704. u8 chk_bmc_sleepq_hdl(_adapter *padapter, unsigned char *pbuf)
  12705. {
  12706. #ifdef CONFIG_AP_MODE
  12707. _irqL irqL;
  12708. struct sta_info *psta_bmc;
  12709. _list *xmitframe_plist, *xmitframe_phead;
  12710. struct xmit_frame *pxmitframe = NULL;
  12711. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  12712. struct sta_priv *pstapriv = &padapter->stapriv;
  12713. /* for BC/MC Frames */
  12714. psta_bmc = rtw_get_bcmc_stainfo(padapter);
  12715. if (!psta_bmc)
  12716. return H2C_SUCCESS;
  12717. if ((rtw_tim_map_is_set(padapter, pstapriv->tim_bitmap, 0)) && (psta_bmc->sleepq_len > 0)) {
  12718. #ifndef CONFIG_PCI_HCI
  12719. rtw_msleep_os(10);/* 10ms, ATIM(HIQ) Windows */
  12720. #endif
  12721. /* _enter_critical_bh(&psta_bmc->sleep_q.lock, &irqL); */
  12722. _enter_critical_bh(&pxmitpriv->lock, &irqL);
  12723. xmitframe_phead = get_list_head(&psta_bmc->sleep_q);
  12724. xmitframe_plist = get_next(xmitframe_phead);
  12725. while ((rtw_end_of_queue_search(xmitframe_phead, xmitframe_plist)) == _FALSE) {
  12726. pxmitframe = LIST_CONTAINOR(xmitframe_plist, struct xmit_frame, list);
  12727. xmitframe_plist = get_next(xmitframe_plist);
  12728. rtw_list_delete(&pxmitframe->list);
  12729. psta_bmc->sleepq_len--;
  12730. if (psta_bmc->sleepq_len > 0)
  12731. pxmitframe->attrib.mdata = 1;
  12732. else
  12733. pxmitframe->attrib.mdata = 0;
  12734. pxmitframe->attrib.triggered = 1;
  12735. if (xmitframe_hiq_filter(pxmitframe) == _TRUE)
  12736. pxmitframe->attrib.qsel = QSLT_HIGH;/* HIQ */
  12737. #if 0
  12738. _exit_critical_bh(&psta_bmc->sleep_q.lock, &irqL);
  12739. if (rtw_hal_xmit(padapter, pxmitframe) == _TRUE)
  12740. rtw_os_xmit_complete(padapter, pxmitframe);
  12741. _enter_critical_bh(&psta_bmc->sleep_q.lock, &irqL);
  12742. #endif
  12743. rtw_hal_xmitframe_enqueue(padapter, pxmitframe);
  12744. }
  12745. /* _exit_critical_bh(&psta_bmc->sleep_q.lock, &irqL); */
  12746. _exit_critical_bh(&pxmitpriv->lock, &irqL);
  12747. if (rtw_get_intf_type(padapter) != RTW_PCIE) {
  12748. /* check hi queue and bmc_sleepq */
  12749. rtw_chk_hi_queue_cmd(padapter);
  12750. }
  12751. }
  12752. #endif
  12753. return H2C_SUCCESS;
  12754. }
  12755. u8 tx_beacon_hdl(_adapter *padapter, unsigned char *pbuf)
  12756. {
  12757. /*RTW_INFO(FUNC_ADPT_FMT, FUNC_ADPT_ARG(padapter));*/
  12758. #ifdef CONFIG_SWTIMER_BASED_TXBCN
  12759. tx_beacon_handlder(padapter->dvobj);
  12760. #else
  12761. if (send_beacon(padapter) == _FAIL) {
  12762. RTW_INFO("issue_beacon, fail!\n");
  12763. return H2C_PARAMETERS_ERROR;
  12764. }
  12765. /* tx bc/mc frames after update TIM */
  12766. chk_bmc_sleepq_hdl(padapter, NULL);
  12767. #endif
  12768. return H2C_SUCCESS;
  12769. }
  12770. /*
  12771. * according to channel
  12772. * add/remove WLAN_BSSID_EX.IEs's ERP ie
  12773. * set WLAN_BSSID_EX.SupportedRates
  12774. * update WLAN_BSSID_EX.IEs's Supported Rate and Extended Supported Rate ie
  12775. */
  12776. void change_band_update_ie(_adapter *padapter, WLAN_BSSID_EX *pnetwork, u8 ch)
  12777. {
  12778. u8 network_type, rate_len, total_rate_len, remainder_rate_len;
  12779. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  12780. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  12781. u8 erpinfo = 0x4;
  12782. if (ch >= 36) {
  12783. network_type = WIRELESS_11A;
  12784. total_rate_len = IEEE80211_NUM_OFDM_RATESLEN;
  12785. rtw_remove_bcn_ie(padapter, pnetwork, _ERPINFO_IE_);
  12786. #ifdef CONFIG_80211AC_VHT
  12787. /* if channel in 5G band, then add vht ie . */
  12788. if ((pmlmepriv->htpriv.ht_option == _TRUE)
  12789. && REGSTY_IS_11AC_ENABLE(&padapter->registrypriv)
  12790. && is_supported_vht(padapter->registrypriv.wireless_mode)
  12791. && (!rfctl->country_ent || COUNTRY_CHPLAN_EN_11AC(rfctl->country_ent))
  12792. ) {
  12793. if (REGSTY_IS_11AC_AUTO(&padapter->registrypriv)
  12794. || pmlmepriv->ori_vht_en)
  12795. rtw_vht_ies_attach(padapter, pnetwork);
  12796. }
  12797. #endif
  12798. } else {
  12799. network_type = WIRELESS_11BG;
  12800. total_rate_len = IEEE80211_CCK_RATE_LEN + IEEE80211_NUM_OFDM_RATESLEN;
  12801. rtw_add_bcn_ie(padapter, pnetwork, _ERPINFO_IE_, &erpinfo, 1);
  12802. #ifdef CONFIG_80211AC_VHT
  12803. rtw_vht_ies_detach(padapter, pnetwork);
  12804. #endif
  12805. }
  12806. rtw_set_supported_rate(pnetwork->SupportedRates, network_type);
  12807. UpdateBrateTbl(padapter, pnetwork->SupportedRates);
  12808. if (total_rate_len > 8) {
  12809. rate_len = 8;
  12810. remainder_rate_len = total_rate_len - 8;
  12811. } else {
  12812. rate_len = total_rate_len;
  12813. remainder_rate_len = 0;
  12814. }
  12815. rtw_add_bcn_ie(padapter, pnetwork, _SUPPORTEDRATES_IE_, pnetwork->SupportedRates, rate_len);
  12816. if (remainder_rate_len)
  12817. rtw_add_bcn_ie(padapter, pnetwork, _EXT_SUPPORTEDRATES_IE_, (pnetwork->SupportedRates + 8), remainder_rate_len);
  12818. else
  12819. rtw_remove_bcn_ie(padapter, pnetwork, _EXT_SUPPORTEDRATES_IE_);
  12820. pnetwork->Length = get_WLAN_BSSID_EX_sz(pnetwork);
  12821. }
  12822. void rtw_join_done_chk_ch(_adapter *adapter, int join_res)
  12823. {
  12824. #define DUMP_ADAPTERS_STATUS 0
  12825. struct dvobj_priv *dvobj;
  12826. _adapter *iface;
  12827. struct mlme_priv *mlme;
  12828. struct mlme_ext_priv *mlmeext;
  12829. u8 u_ch, u_offset, u_bw;
  12830. int i;
  12831. dvobj = adapter_to_dvobj(adapter);
  12832. if (DUMP_ADAPTERS_STATUS) {
  12833. RTW_INFO(FUNC_ADPT_FMT" enter\n", FUNC_ADPT_ARG(adapter));
  12834. dump_adapters_status(RTW_DBGDUMP , dvobj);
  12835. }
  12836. if (join_res >= 0) {
  12837. #ifdef CONFIG_MCC_MODE
  12838. /* MCC setting success, don't go to ch union process */
  12839. if (rtw_hal_set_mcc_setting_join_done_chk_ch(adapter))
  12840. return;
  12841. #endif /* CONFIG_MCC_MODE */
  12842. if (rtw_mi_get_ch_setting_union(adapter, &u_ch, &u_bw, &u_offset) <= 0) {
  12843. dump_adapters_status(RTW_DBGDUMP , dvobj);
  12844. rtw_warn_on(1);
  12845. }
  12846. for (i = 0; i < dvobj->iface_nums; i++) {
  12847. iface = dvobj->padapters[i];
  12848. mlme = &iface->mlmepriv;
  12849. mlmeext = &iface->mlmeextpriv;
  12850. if (!iface || iface == adapter)
  12851. continue;
  12852. if ((MLME_IS_AP(iface) || MLME_IS_MESH(iface))
  12853. && check_fwstate(mlme, WIFI_ASOC_STATE)
  12854. ) {
  12855. u8 ori_ch, ori_bw, ori_offset;
  12856. bool is_grouped = rtw_is_chbw_grouped(u_ch, u_bw, u_offset
  12857. , mlmeext->cur_channel, mlmeext->cur_bwmode, mlmeext->cur_ch_offset);
  12858. if (is_grouped == _FALSE) {
  12859. /* handle AP which need to switch ch setting */
  12860. ori_ch = mlmeext->cur_channel;
  12861. ori_bw = mlmeext->cur_bwmode;
  12862. ori_offset = mlmeext->cur_ch_offset;
  12863. /* restore original bw, adjust bw by registry setting on target ch */
  12864. mlmeext->cur_bwmode = mlme->ori_bw;
  12865. mlmeext->cur_channel = u_ch;
  12866. rtw_adjust_chbw(iface, mlmeext->cur_channel, &mlmeext->cur_bwmode, &mlmeext->cur_ch_offset);
  12867. #ifdef CONFIG_RTW_MESH
  12868. if (MLME_IS_MESH(iface))
  12869. rtw_mesh_adjust_chbw(mlmeext->cur_channel, &mlmeext->cur_bwmode, &mlmeext->cur_ch_offset);
  12870. #endif
  12871. rtw_chset_sync_chbw(adapter_to_chset(adapter)
  12872. , &mlmeext->cur_channel, &mlmeext->cur_bwmode, &mlmeext->cur_ch_offset
  12873. , &u_ch, &u_bw, &u_offset);
  12874. RTW_INFO(FUNC_ADPT_FMT" %u,%u,%u => %u,%u,%u\n", FUNC_ADPT_ARG(iface)
  12875. , ori_ch, ori_bw, ori_offset
  12876. , mlmeext->cur_channel, mlmeext->cur_bwmode, mlmeext->cur_ch_offset);
  12877. rtw_ap_update_bss_chbw(iface, &(mlmeext->mlmext_info.network)
  12878. , mlmeext->cur_channel, mlmeext->cur_bwmode, mlmeext->cur_ch_offset);
  12879. _rtw_memcpy(&(mlme->cur_network.network), &(mlmeext->mlmext_info.network), sizeof(WLAN_BSSID_EX));
  12880. rtw_start_bss_hdl_after_chbw_decided(iface);
  12881. if (MLME_IS_GO(iface) || MLME_IS_MESH(iface)) { /* pure AP is not needed*/
  12882. #if defined(CONFIG_IOCTL_CFG80211) && (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 5, 0))
  12883. u8 ht_option = 0;
  12884. #ifdef CONFIG_80211N_HT
  12885. ht_option = mlme->htpriv.ht_option;
  12886. #endif
  12887. rtw_cfg80211_ch_switch_notify(iface
  12888. , mlmeext->cur_channel, mlmeext->cur_bwmode, mlmeext->cur_ch_offset
  12889. , ht_option);
  12890. #endif
  12891. }
  12892. }
  12893. clr_fwstate(mlme, WIFI_OP_CH_SWITCHING);
  12894. update_beacon(iface, 0xFF, NULL, _TRUE);
  12895. }
  12896. }
  12897. #ifdef CONFIG_DFS_MASTER
  12898. rtw_dfs_rd_en_decision(adapter, MLME_STA_CONNECTED, 0);
  12899. #endif
  12900. } else {
  12901. for (i = 0; i < dvobj->iface_nums; i++) {
  12902. iface = dvobj->padapters[i];
  12903. mlme = &iface->mlmepriv;
  12904. mlmeext = &iface->mlmeextpriv;
  12905. if (!iface || iface == adapter)
  12906. continue;
  12907. if ((MLME_IS_AP(iface) || MLME_IS_MESH(iface))
  12908. && check_fwstate(mlme, WIFI_ASOC_STATE)
  12909. ) {
  12910. clr_fwstate(mlme, WIFI_OP_CH_SWITCHING);
  12911. update_beacon(iface, 0xFF, NULL, _TRUE);
  12912. }
  12913. }
  12914. #ifdef CONFIG_DFS_MASTER
  12915. rtw_dfs_rd_en_decision(adapter, MLME_STA_DISCONNECTED, 0);
  12916. #endif
  12917. }
  12918. if (rtw_mi_get_ch_setting_union(adapter, &u_ch, &u_bw, &u_offset)) {
  12919. RTW_INFO(FUNC_ADPT_FMT" union:%u,%u,%u\n", FUNC_ADPT_ARG(adapter), u_ch, u_bw, u_offset);
  12920. set_channel_bwmode(adapter, u_ch, u_offset, u_bw);
  12921. rtw_mi_update_union_chan_inf(adapter, u_ch, u_offset, u_bw);
  12922. }
  12923. if (DUMP_ADAPTERS_STATUS) {
  12924. RTW_INFO(FUNC_ADPT_FMT" exit\n", FUNC_ADPT_ARG(adapter));
  12925. dump_adapters_status(RTW_DBGDUMP , dvobj);
  12926. }
  12927. }
  12928. int rtw_chk_start_clnt_join(_adapter *adapter, u8 *ch, u8 *bw, u8 *offset)
  12929. {
  12930. #ifdef CONFIG_CONCURRENT_MODE
  12931. bool chbw_allow = _TRUE;
  12932. #endif
  12933. bool connect_allow = _TRUE;
  12934. struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
  12935. u8 cur_ch, cur_bw, cur_ch_offset;
  12936. u8 u_ch, u_offset, u_bw;
  12937. u_ch = cur_ch = pmlmeext->cur_channel;
  12938. u_bw = cur_bw = pmlmeext->cur_bwmode;
  12939. u_offset = cur_ch_offset = pmlmeext->cur_ch_offset;
  12940. if (!ch || !bw || !offset) {
  12941. connect_allow = _FALSE;
  12942. rtw_warn_on(1);
  12943. goto exit;
  12944. }
  12945. if (cur_ch == 0) {
  12946. connect_allow = _FALSE;
  12947. RTW_ERR(FUNC_ADPT_FMT" cur_ch:%u\n"
  12948. , FUNC_ADPT_ARG(adapter), cur_ch);
  12949. rtw_warn_on(1);
  12950. goto exit;
  12951. }
  12952. RTW_INFO(FUNC_ADPT_FMT" req: %u,%u,%u\n", FUNC_ADPT_ARG(adapter), u_ch, u_bw, u_offset);
  12953. #ifdef CONFIG_CONCURRENT_MODE
  12954. {
  12955. struct dvobj_priv *dvobj;
  12956. _adapter *iface;
  12957. struct mlme_priv *mlme;
  12958. struct mlme_ext_priv *mlmeext;
  12959. struct mi_state mstate;
  12960. int i;
  12961. dvobj = adapter_to_dvobj(adapter);
  12962. rtw_mi_status_no_self(adapter, &mstate);
  12963. RTW_INFO(FUNC_ADPT_FMT" others ld_sta_num:%u, ap_num:%u, mesh_num:%u\n"
  12964. , FUNC_ADPT_ARG(adapter), MSTATE_STA_LD_NUM(&mstate)
  12965. , MSTATE_AP_NUM(&mstate), MSTATE_MESH_NUM(&mstate));
  12966. if (!MSTATE_STA_LD_NUM(&mstate) && !MSTATE_AP_NUM(&mstate) && !MSTATE_MESH_NUM(&mstate)) {
  12967. /* consider linking STA? */
  12968. goto connect_allow_hdl;
  12969. }
  12970. if (rtw_mi_get_ch_setting_union_no_self(adapter, &u_ch, &u_bw, &u_offset) <= 0) {
  12971. dump_adapters_status(RTW_DBGDUMP , dvobj);
  12972. rtw_warn_on(1);
  12973. }
  12974. RTW_INFO(FUNC_ADPT_FMT" others union:%u,%u,%u\n"
  12975. , FUNC_ADPT_ARG(adapter), u_ch, u_bw, u_offset);
  12976. /* chbw_allow? */
  12977. chbw_allow = rtw_is_chbw_grouped(pmlmeext->cur_channel, pmlmeext->cur_bwmode, pmlmeext->cur_ch_offset
  12978. , u_ch, u_bw, u_offset);
  12979. RTW_INFO(FUNC_ADPT_FMT" chbw_allow:%d\n"
  12980. , FUNC_ADPT_ARG(adapter), chbw_allow);
  12981. #ifdef CONFIG_MCC_MODE
  12982. /* check setting success, don't go to ch union process */
  12983. if (rtw_hal_set_mcc_setting_chk_start_clnt_join(adapter, &u_ch, &u_bw, &u_offset, chbw_allow))
  12984. goto exit;
  12985. #endif
  12986. if (chbw_allow == _TRUE) {
  12987. rtw_sync_chbw(&cur_ch, &cur_bw, &cur_ch_offset, &u_ch, &u_bw, &u_offset);
  12988. rtw_warn_on(cur_ch != pmlmeext->cur_channel);
  12989. rtw_warn_on(cur_bw != pmlmeext->cur_bwmode);
  12990. rtw_warn_on(cur_ch_offset != pmlmeext->cur_ch_offset);
  12991. goto connect_allow_hdl;
  12992. }
  12993. #ifdef CONFIG_CFG80211_ONECHANNEL_UNDER_CONCURRENT
  12994. /* chbw_allow is _FALSE, connect allow? */
  12995. for (i = 0; i < dvobj->iface_nums; i++) {
  12996. iface = dvobj->padapters[i];
  12997. mlme = &iface->mlmepriv;
  12998. mlmeext = &iface->mlmeextpriv;
  12999. if (check_fwstate(mlme, WIFI_STATION_STATE)
  13000. && check_fwstate(mlme, WIFI_ASOC_STATE)
  13001. #if defined(CONFIG_P2P)
  13002. && rtw_p2p_chk_state(&(iface->wdinfo), P2P_STATE_NONE)
  13003. #endif
  13004. ) {
  13005. connect_allow = _FALSE;
  13006. break;
  13007. }
  13008. }
  13009. #endif /* CONFIG_CFG80211_ONECHANNEL_UNDER_CONCURRENT */
  13010. if (MSTATE_STA_LD_NUM(&mstate) + MSTATE_AP_LD_NUM(&mstate) + MSTATE_MESH_LD_NUM(&mstate) >= 4)
  13011. connect_allow = _FALSE;
  13012. RTW_INFO(FUNC_ADPT_FMT" connect_allow:%d\n"
  13013. , FUNC_ADPT_ARG(adapter), connect_allow);
  13014. if (connect_allow == _FALSE)
  13015. goto exit;
  13016. connect_allow_hdl:
  13017. /* connect_allow == _TRUE */
  13018. if (chbw_allow == _FALSE) {
  13019. u_ch = cur_ch;
  13020. u_bw = cur_bw;
  13021. u_offset = cur_ch_offset;
  13022. for (i = 0; i < dvobj->iface_nums; i++) {
  13023. iface = dvobj->padapters[i];
  13024. mlme = &iface->mlmepriv;
  13025. mlmeext = &iface->mlmeextpriv;
  13026. if (!iface || iface == adapter)
  13027. continue;
  13028. if ((MLME_IS_AP(iface) || MLME_IS_MESH(iface))
  13029. && check_fwstate(mlme, WIFI_ASOC_STATE)
  13030. ) {
  13031. #ifdef CONFIG_SPCT_CH_SWITCH
  13032. if (1)
  13033. rtw_ap_inform_ch_switch(iface, pmlmeext->cur_channel , pmlmeext->cur_ch_offset);
  13034. else
  13035. #endif
  13036. rtw_sta_flush(iface, _FALSE);
  13037. rtw_hal_set_hwreg(iface, HW_VAR_CHECK_TXBUF, 0);
  13038. set_fwstate(mlme, WIFI_OP_CH_SWITCHING);
  13039. } else if (check_fwstate(mlme, WIFI_STATION_STATE)
  13040. && check_fwstate(mlme, WIFI_ASOC_STATE)
  13041. ) {
  13042. rtw_disassoc_cmd(iface, 500, RTW_CMDF_DIRECTLY);
  13043. rtw_indicate_disconnect(iface, 0, _FALSE);
  13044. rtw_free_assoc_resources(iface, _TRUE);
  13045. }
  13046. }
  13047. }
  13048. #ifdef CONFIG_DFS_MASTER
  13049. rtw_dfs_rd_en_decision(adapter, MLME_STA_CONNECTING, 0);
  13050. #endif
  13051. }
  13052. #endif /* CONFIG_CONCURRENT_MODE */
  13053. exit:
  13054. if (connect_allow == _TRUE) {
  13055. RTW_INFO(FUNC_ADPT_FMT" union: %u,%u,%u\n", FUNC_ADPT_ARG(adapter), u_ch, u_bw, u_offset);
  13056. *ch = u_ch;
  13057. *bw = u_bw;
  13058. *offset = u_offset;
  13059. }
  13060. return connect_allow == _TRUE ? _SUCCESS : _FAIL;
  13061. }
  13062. u8 rtw_set_chbw_hdl(_adapter *padapter, u8 *pbuf)
  13063. {
  13064. struct set_ch_parm *set_ch_parm;
  13065. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  13066. if (!pbuf)
  13067. return H2C_PARAMETERS_ERROR;
  13068. set_ch_parm = (struct set_ch_parm *)pbuf;
  13069. RTW_INFO(FUNC_NDEV_FMT" ch:%u, bw:%u, ch_offset:%u\n",
  13070. FUNC_NDEV_ARG(padapter->pnetdev),
  13071. set_ch_parm->ch, set_ch_parm->bw, set_ch_parm->ch_offset);
  13072. pmlmeext->cur_channel = set_ch_parm->ch;
  13073. pmlmeext->cur_ch_offset = set_ch_parm->ch_offset;
  13074. pmlmeext->cur_bwmode = set_ch_parm->bw;
  13075. set_channel_bwmode(padapter, set_ch_parm->ch, set_ch_parm->ch_offset, set_ch_parm->bw);
  13076. return H2C_SUCCESS;
  13077. }
  13078. u8 set_chplan_hdl(_adapter *padapter, unsigned char *pbuf)
  13079. {
  13080. struct SetChannelPlan_param *setChannelPlan_param;
  13081. struct rf_ctl_t *rfctl = adapter_to_rfctl(padapter);
  13082. if (!pbuf)
  13083. return H2C_PARAMETERS_ERROR;
  13084. setChannelPlan_param = (struct SetChannelPlan_param *)pbuf;
  13085. if (!rtw_is_channel_plan_valid(setChannelPlan_param->channel_plan))
  13086. return H2C_PARAMETERS_ERROR;
  13087. rfctl->country_ent = setChannelPlan_param->country_ent;
  13088. rfctl->ChannelPlan = setChannelPlan_param->channel_plan;
  13089. rfctl->max_chan_nums = init_channel_set(padapter, rfctl->ChannelPlan, rfctl->channel_set);
  13090. init_channel_list(padapter, rfctl->channel_set, &rfctl->channel_list);
  13091. #ifdef CONFIG_TXPWR_LIMIT
  13092. rtw_txpwr_init_regd(rfctl);
  13093. #endif
  13094. rtw_hal_set_odm_var(padapter, HAL_ODM_REGULATION, NULL, _TRUE);
  13095. #ifdef CONFIG_IOCTL_CFG80211
  13096. rtw_regd_apply_flags(adapter_to_wiphy(padapter));
  13097. #endif
  13098. return H2C_SUCCESS;
  13099. }
  13100. u8 led_blink_hdl(_adapter *padapter, unsigned char *pbuf)
  13101. {
  13102. struct LedBlink_param *ledBlink_param;
  13103. if (!pbuf)
  13104. return H2C_PARAMETERS_ERROR;
  13105. ledBlink_param = (struct LedBlink_param *)pbuf;
  13106. #ifdef CONFIG_RTW_LED_HANDLED_BY_CMD_THREAD
  13107. BlinkHandler((PLED_DATA)ledBlink_param->pLed);
  13108. #endif
  13109. return H2C_SUCCESS;
  13110. }
  13111. u8 set_csa_hdl(_adapter *adapter, unsigned char *pbuf)
  13112. {
  13113. #ifdef CONFIG_DFS
  13114. struct rf_ctl_t *rfctl = adapter_to_rfctl(adapter);
  13115. if (rfctl->csa_ch)
  13116. rtw_dfs_ch_switch_hdl(adapter_to_dvobj(adapter));
  13117. #endif
  13118. return H2C_SUCCESS;
  13119. }
  13120. u8 tdls_hdl(_adapter *padapter, unsigned char *pbuf)
  13121. {
  13122. #ifdef CONFIG_TDLS
  13123. _irqL irqL;
  13124. HAL_DATA_TYPE *pHalData = GET_HAL_DATA(padapter);
  13125. struct tdls_info *ptdlsinfo = &padapter->tdlsinfo;
  13126. #ifdef CONFIG_TDLS_CH_SW
  13127. struct tdls_ch_switch *pchsw_info = &ptdlsinfo->chsw_info;
  13128. #endif
  13129. struct TDLSoption_param *TDLSoption;
  13130. struct sta_info *ptdls_sta = NULL;
  13131. struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
  13132. struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
  13133. struct sta_info *ap_sta = rtw_get_stainfo(&padapter->stapriv, get_my_bssid(&(pmlmeinfo->network)));
  13134. u8 survey_channel, i, min, option;
  13135. struct tdls_txmgmt txmgmt;
  13136. u32 setchtime, resp_sleep = 0, wait_time;
  13137. u8 zaddr[ETH_ALEN] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
  13138. u8 ret;
  13139. u8 doiqk;
  13140. u64 tx_ra_bitmap = 0;
  13141. if (!pbuf)
  13142. return H2C_PARAMETERS_ERROR;
  13143. TDLSoption = (struct TDLSoption_param *)pbuf;
  13144. option = TDLSoption->option;
  13145. if (!_rtw_memcmp(TDLSoption->addr, zaddr, ETH_ALEN)) {
  13146. ptdls_sta = rtw_get_stainfo(&(padapter->stapriv), TDLSoption->addr);
  13147. if (ptdls_sta == NULL)
  13148. return H2C_REJECTED;
  13149. } else {
  13150. if (!(option == TDLS_RS_RCR))
  13151. return H2C_REJECTED;
  13152. }
  13153. /* _enter_critical_bh(&(ptdlsinfo->hdl_lock), &irqL); */
  13154. /* RTW_INFO("[%s] option:%d\n", __FUNCTION__, option); */
  13155. switch (option) {
  13156. case TDLS_ESTABLISHED: {
  13157. /* As long as TDLS handshake success, we should set RCR_CBSSID_DATA bit to 0 */
  13158. /* So we can receive all kinds of data frames. */
  13159. u8 sta_band = 0;
  13160. /* leave ALL PS when TDLS is established */
  13161. rtw_pwr_wakeup(padapter);
  13162. rtw_hal_rcr_set_chk_bssid(padapter, MLME_TDLS_LINKED);
  13163. RTW_INFO("Created Direct Link with "MAC_FMT"\n", MAC_ARG(ptdls_sta->cmn.mac_addr));
  13164. /* Set TDLS sta rate. */
  13165. /* Update station supportRate */
  13166. rtw_hal_update_sta_ra_info(padapter, ptdls_sta);
  13167. tx_ra_bitmap = ptdls_sta->cmn.ra_info.ramask;
  13168. if (pmlmeext->cur_channel > 14) {
  13169. if (tx_ra_bitmap & 0xffff000)
  13170. sta_band |= WIRELESS_11_5N ;
  13171. if (tx_ra_bitmap & 0xff0)
  13172. sta_band |= WIRELESS_11A;
  13173. /* 5G band */
  13174. #ifdef CONFIG_80211AC_VHT
  13175. if (ptdls_sta->vhtpriv.vht_option)
  13176. sta_band = WIRELESS_11_5AC;
  13177. #endif
  13178. } else {
  13179. if (tx_ra_bitmap & 0xffff000)
  13180. sta_band |= WIRELESS_11_24N;
  13181. if (tx_ra_bitmap & 0xff0)
  13182. sta_band |= WIRELESS_11G;
  13183. if (tx_ra_bitmap & 0x0f)
  13184. sta_band |= WIRELESS_11B;
  13185. }
  13186. ptdls_sta->wireless_mode = sta_band;
  13187. rtw_hal_update_sta_wset(padapter, ptdls_sta);
  13188. /* Sta mode */
  13189. rtw_hal_set_odm_var(padapter, HAL_ODM_STA_INFO, ptdls_sta, _TRUE);
  13190. set_sta_rate(padapter, ptdls_sta);
  13191. rtw_sta_media_status_rpt(padapter, ptdls_sta, 1);
  13192. break;
  13193. }
  13194. case TDLS_ISSUE_PTI:
  13195. ptdls_sta->tdls_sta_state |= TDLS_WAIT_PTR_STATE;
  13196. issue_tdls_peer_traffic_indication(padapter, ptdls_sta);
  13197. _set_timer(&ptdls_sta->pti_timer, TDLS_PTI_TIME);
  13198. break;
  13199. #ifdef CONFIG_TDLS_CH_SW
  13200. case TDLS_CH_SW_RESP:
  13201. _rtw_memset(&txmgmt, 0x00, sizeof(struct tdls_txmgmt));
  13202. txmgmt.status_code = 0;
  13203. _rtw_memcpy(txmgmt.peer, ptdls_sta->cmn.mac_addr, ETH_ALEN);
  13204. if (ap_sta)
  13205. rtw_hal_macid_sleep(padapter, ap_sta->cmn.mac_id);
  13206. issue_nulldata(padapter, NULL, 1, 3, 3);
  13207. RTW_INFO("[TDLS ] issue tdls channel switch response\n");
  13208. ret = issue_tdls_ch_switch_rsp(padapter, &txmgmt, _TRUE);
  13209. /* If we receive TDLS_CH_SW_REQ at off channel which it's target is AP's channel */
  13210. /* then we just switch to AP's channel*/
  13211. if (padapter->mlmeextpriv.cur_channel == pchsw_info->off_ch_num) {
  13212. rtw_tdls_cmd(padapter, ptdls_sta->cmn.mac_addr, TDLS_CH_SW_END_TO_BASE_CHNL);
  13213. break;
  13214. }
  13215. if (ret == _SUCCESS)
  13216. rtw_tdls_cmd(padapter, ptdls_sta->cmn.mac_addr, TDLS_CH_SW_TO_OFF_CHNL);
  13217. else
  13218. RTW_INFO("[TDLS] issue_tdls_ch_switch_rsp wait ack fail !!!!!!!!!!\n");
  13219. break;
  13220. case TDLS_CH_SW_PREPARE:
  13221. pchsw_info->ch_sw_state |= TDLS_CH_SWITCH_PREPARE_STATE;
  13222. /* to collect IQK info of off-chnl */
  13223. doiqk = _TRUE;
  13224. rtw_hal_set_hwreg(padapter, HW_VAR_DO_IQK, &doiqk);
  13225. set_channel_bwmode(padapter, pchsw_info->off_ch_num, pchsw_info->ch_offset, (pchsw_info->ch_offset) ? CHANNEL_WIDTH_40 : CHANNEL_WIDTH_20);
  13226. doiqk = _FALSE;
  13227. rtw_hal_set_hwreg(padapter, HW_VAR_DO_IQK, &doiqk);
  13228. /* switch back to base-chnl */
  13229. set_channel_bwmode(padapter, pmlmeext->cur_channel, pmlmeext->cur_ch_offset, pmlmeext->cur_bwmode);
  13230. rtw_tdls_cmd(padapter, ptdls_sta->cmn.mac_addr, TDLS_CH_SW_START);
  13231. pchsw_info->ch_sw_state &= ~(TDLS_CH_SWITCH_PREPARE_STATE);
  13232. break;
  13233. case TDLS_CH_SW_START:
  13234. rtw_tdls_set_ch_sw_oper_control(padapter, _TRUE);
  13235. break;
  13236. case TDLS_CH_SW_TO_OFF_CHNL:
  13237. if (ap_sta)
  13238. rtw_hal_macid_sleep(padapter, ap_sta->cmn.mac_id);
  13239. issue_nulldata(padapter, NULL, 1, 3, 3);
  13240. if (padapter->registrypriv.wifi_spec == 0) {
  13241. if (!(pchsw_info->ch_sw_state & TDLS_CH_SW_INITIATOR_STATE))
  13242. _set_timer(&ptdls_sta->ch_sw_timer, (u32)(ptdls_sta->ch_switch_timeout) / 1000);
  13243. }
  13244. if (rtw_tdls_do_ch_sw(padapter, ptdls_sta, TDLS_CH_SW_OFF_CHNL, pchsw_info->off_ch_num,
  13245. pchsw_info->ch_offset, (pchsw_info->ch_offset) ? CHANNEL_WIDTH_40 : CHANNEL_WIDTH_20, ptdls_sta->ch_switch_time) == _SUCCESS) {
  13246. pchsw_info->ch_sw_state &= ~(TDLS_PEER_AT_OFF_STATE);
  13247. if (pchsw_info->ch_sw_state & TDLS_CH_SW_INITIATOR_STATE) {
  13248. if (issue_nulldata_to_TDLS_peer_STA(ptdls_sta->padapter, ptdls_sta->cmn.mac_addr, 0, 1,
  13249. (padapter->registrypriv.wifi_spec == 0) ? 3 : 0) == _FAIL)
  13250. rtw_tdls_cmd(padapter, ptdls_sta->cmn.mac_addr, TDLS_CH_SW_TO_BASE_CHNL);
  13251. }
  13252. } else {
  13253. if (!(pchsw_info->ch_sw_state & TDLS_CH_SW_INITIATOR_STATE))
  13254. _cancel_timer_ex(&ptdls_sta->ch_sw_timer);
  13255. }
  13256. break;
  13257. case TDLS_CH_SW_END:
  13258. case TDLS_CH_SW_END_TO_BASE_CHNL:
  13259. rtw_tdls_set_ch_sw_oper_control(padapter, _FALSE);
  13260. _cancel_timer_ex(&ptdls_sta->ch_sw_timer);
  13261. _cancel_timer_ex(&ptdls_sta->stay_on_base_chnl_timer);
  13262. _cancel_timer_ex(&ptdls_sta->ch_sw_monitor_timer);
  13263. #if 0
  13264. _rtw_memset(pHalData->tdls_ch_sw_iqk_info_base_chnl, 0x00, sizeof(pHalData->tdls_ch_sw_iqk_info_base_chnl));
  13265. _rtw_memset(pHalData->tdls_ch_sw_iqk_info_off_chnl, 0x00, sizeof(pHalData->tdls_ch_sw_iqk_info_off_chnl));
  13266. #endif
  13267. if (option == TDLS_CH_SW_END_TO_BASE_CHNL)
  13268. rtw_tdls_cmd(padapter, ptdls_sta->cmn.mac_addr, TDLS_CH_SW_TO_BASE_CHNL);
  13269. break;
  13270. case TDLS_CH_SW_TO_BASE_CHNL_UNSOLICITED:
  13271. case TDLS_CH_SW_TO_BASE_CHNL:
  13272. pchsw_info->ch_sw_state &= ~(TDLS_PEER_AT_OFF_STATE | TDLS_WAIT_CH_RSP_STATE);
  13273. if (option == TDLS_CH_SW_TO_BASE_CHNL_UNSOLICITED) {
  13274. if (ptdls_sta != NULL) {
  13275. /* Send unsolicited channel switch rsp. to peer */
  13276. _rtw_memset(&txmgmt, 0x00, sizeof(struct tdls_txmgmt));
  13277. txmgmt.status_code = 0;
  13278. _rtw_memcpy(txmgmt.peer, ptdls_sta->cmn.mac_addr, ETH_ALEN);
  13279. issue_tdls_ch_switch_rsp(padapter, &txmgmt, _FALSE);
  13280. }
  13281. }
  13282. if (rtw_tdls_do_ch_sw(padapter, ptdls_sta, TDLS_CH_SW_BASE_CHNL, pmlmeext->cur_channel,
  13283. pmlmeext->cur_ch_offset, pmlmeext->cur_bwmode, ptdls_sta->ch_switch_time) == _SUCCESS) {
  13284. if (ap_sta)
  13285. rtw_hal_macid_wakeup(padapter, ap_sta->cmn.mac_id);
  13286. issue_nulldata(padapter, NULL, 0, 3, 3);
  13287. /* set ch sw monitor timer for responder */
  13288. if (!(pchsw_info->ch_sw_state & TDLS_CH_SW_INITIATOR_STATE))
  13289. _set_timer(&ptdls_sta->ch_sw_monitor_timer, TDLS_CH_SW_MONITOR_TIMEOUT);
  13290. }
  13291. break;
  13292. #endif
  13293. case TDLS_RS_RCR:
  13294. rtw_hal_rcr_set_chk_bssid(padapter, MLME_TDLS_NOLINK);
  13295. break;
  13296. case TDLS_TEARDOWN_STA:
  13297. case TDLS_TEARDOWN_STA_NO_WAIT:
  13298. _rtw_memset(&txmgmt, 0x00, sizeof(struct tdls_txmgmt));
  13299. txmgmt.status_code = _RSON_TDLS_TEAR_UN_RSN_;
  13300. _rtw_memcpy(txmgmt.peer, ptdls_sta->cmn.mac_addr, ETH_ALEN);
  13301. issue_tdls_teardown(padapter, &txmgmt, (option == TDLS_TEARDOWN_STA) ? _TRUE : _FALSE);
  13302. break;
  13303. case TDLS_TEARDOWN_STA_LOCALLY:
  13304. case TDLS_TEARDOWN_STA_LOCALLY_POST:
  13305. #ifdef CONFIG_TDLS_CH_SW
  13306. if (_rtw_memcmp(TDLSoption->addr, pchsw_info->addr, ETH_ALEN) == _TRUE) {
  13307. pchsw_info->ch_sw_state &= ~(TDLS_CH_SW_INITIATOR_STATE |
  13308. TDLS_CH_SWITCH_ON_STATE |
  13309. TDLS_PEER_AT_OFF_STATE);
  13310. rtw_tdls_set_ch_sw_oper_control(padapter, _FALSE);
  13311. _rtw_memset(pchsw_info->addr, 0x00, ETH_ALEN);
  13312. }
  13313. #endif
  13314. if (option == TDLS_TEARDOWN_STA_LOCALLY)
  13315. rtw_tdls_teardown_pre_hdl(padapter, ptdls_sta);
  13316. rtw_tdls_teardown_post_hdl(padapter, ptdls_sta, _FALSE);
  13317. if (ptdlsinfo->tdls_sctx != NULL)
  13318. rtw_sctx_done(&(ptdlsinfo->tdls_sctx));
  13319. break;
  13320. }
  13321. /* _exit_critical_bh(&(ptdlsinfo->hdl_lock), &irqL); */
  13322. return H2C_SUCCESS;
  13323. #else
  13324. return H2C_REJECTED;
  13325. #endif /* CONFIG_TDLS */
  13326. }
  13327. u8 run_in_thread_hdl(_adapter *padapter, u8 *pbuf)
  13328. {
  13329. struct RunInThread_param *p;
  13330. if (NULL == pbuf)
  13331. return H2C_PARAMETERS_ERROR;
  13332. p = (struct RunInThread_param *)pbuf;
  13333. if (p->func)
  13334. p->func(p->context);
  13335. return H2C_SUCCESS;
  13336. }
  13337. u8 rtw_getmacreg_hdl(_adapter *padapter, u8 *pbuf)
  13338. {
  13339. struct readMAC_parm *preadmacparm = NULL;
  13340. u8 sz = 0;
  13341. u32 addr = 0;
  13342. u32 value = 0;
  13343. if (!pbuf)
  13344. return H2C_PARAMETERS_ERROR;
  13345. preadmacparm = (struct readMAC_parm *) pbuf;
  13346. sz = preadmacparm->len;
  13347. addr = preadmacparm->addr;
  13348. value = 0;
  13349. switch (sz) {
  13350. case 1:
  13351. value = rtw_read8(padapter, addr);
  13352. break;
  13353. case 2:
  13354. value = rtw_read16(padapter, addr);
  13355. break;
  13356. case 4:
  13357. value = rtw_read32(padapter, addr);
  13358. break;
  13359. default:
  13360. RTW_INFO("%s: Unknown size\n", __func__);
  13361. break;
  13362. }
  13363. RTW_INFO("%s: addr:0x%02x valeu:0x%02x\n", __func__, addr, value);
  13364. return H2C_SUCCESS;
  13365. }