1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460 |
- /**
- * include/enet.h - a Single-Header auto-generated variant of enet.h library.
- *
- * Usage:
- * #define ENET_IMPLEMENTATION exactly in ONE source file right BEFORE including the library, like:
- *
- * #define ENET_IMPLEMENTATION
- * #include <enet.h>
- *
- */
- #ifdef ENET_IMPLEMENTATION
- # define ENET_BUILDING_LIB 1
- #endif
- /**
- * @file enet.h
- * @brief ENet public header file
- */
- #ifndef __ENET_ENET_H__
- # define __ENET_ENET_H__
- # ifdef __cplusplus
- extern "C"
- {
- # endif
- # include <stdlib.h>
- # ifdef _WIN32
- /**
- * @file win32.h
- * @brief ENet Win32 header
- */
- # ifndef __ENET_WIN32_H__
- # define __ENET_WIN32_H__
- # ifdef _MSC_VER
- # ifdef ENET_BUILDING_LIB
- # pragma warning (disable: 4267)// size_t to int conversion
- # pragma warning (disable: 4244)// 64bit to 32bit int
- # pragma warning (disable: 4018)// signed/unsigned mismatch
- # pragma warning (disable: 4146)// unary minus operator applied to unsigned type
- # endif
- # endif
- # include <winsock2.h>
- typedef SOCKET ENetSocket;
- # define ENET_SOCKET_NULL INVALID_SOCKET
- # define ENET_HOST_TO_NET_16(value) (htons(value))
- # define ENET_HOST_TO_NET_32(value) (htonl(value))
- # define ENET_NET_TO_HOST_16(value) (ntohs(value))
- # define ENET_NET_TO_HOST_32(value) (ntohl(value))
- typedef struct {
- size_t dataLength;
- void * data;
- } ENetBuffer;
- # define ENET_CALLBACK __cdecl
- # ifdef ENET_DLL
- # ifdef ENET_BUILDING_LIB
- # define ENET_API __declspec(dllexport)
- # else
- # define ENET_API __declspec(dllimport)
- # endif /* ENET_BUILDING_LIB */
- # else /* !ENET_DLL */
- # define ENET_API extern
- # endif /* ENET_DLL */
- typedef fd_set ENetSocketSet;
- # define ENET_SOCKETSET_EMPTY(sockset) FD_ZERO(&(sockset))
- # define ENET_SOCKETSET_ADD(sockset, socket) FD_SET(socket, &(sockset))
- # define ENET_SOCKETSET_REMOVE(sockset, socket) FD_CLR(socket, &(sockset))
- # define ENET_SOCKETSET_CHECK(sockset, socket) FD_ISSET(socket, &(sockset))
- # endif /* __ENET_WIN32_H__ */
- # else // ifdef _WIN32
- /**
- * @file unix.h
- * @brief ENet Unix header
- */
- # ifndef __ENET_UNIX_H__
- # define __ENET_UNIX_H__
- # include <sys/time.h>
- # include <sys/types.h>
- # include <sys/socket.h>
- # include <netinet/in.h>
- # include <unistd.h>
- # ifdef MSG_MAXIOVLEN
- # define ENET_BUFFER_MAXIMUM MSG_MAXIOVLEN
- # endif
- typedef int ENetSocket;
- # define ENET_SOCKET_NULL -1
- # define ENET_HOST_TO_NET_16(value) (htons(value)) /**< macro that converts host to net byte-order of a 16-bit value */
- # define ENET_HOST_TO_NET_32(value) (htonl(value)) /**< macro that converts host to net byte-order of a 32-bit value */
- # define ENET_NET_TO_HOST_16(value) (ntohs(value)) /**< macro that converts net to host byte-order of a 16-bit value */
- # define ENET_NET_TO_HOST_32(value) (ntohl(value)) /**< macro that converts net to host byte-order of a 32-bit value */
- typedef struct {
- void * data;
- size_t dataLength;
- } ENetBuffer;
- # define ENET_CALLBACK
- # define ENET_API extern
- typedef fd_set ENetSocketSet;
- # define ENET_SOCKETSET_EMPTY(sockset) FD_ZERO(&(sockset))
- # define ENET_SOCKETSET_ADD(sockset, socket) FD_SET(socket, &(sockset))
- # define ENET_SOCKETSET_REMOVE(sockset, socket) FD_CLR(socket, &(sockset))
- # define ENET_SOCKETSET_CHECK(sockset, socket) FD_ISSET(socket, &(sockset))
- # endif /* __ENET_UNIX_H__ */
- # endif // ifdef _WIN32
- /**
- * @file types.h
- * @brief type definitions for ENet
- */
- # ifndef __ENET_TYPES_H__
- # define __ENET_TYPES_H__
- # include <stdint.h>
- typedef unsigned char enet_uint8; /**< unsigned 8-bit type */
- typedef unsigned short enet_uint16; /**< unsigned 16-bit type */
- typedef unsigned int enet_uint32; /**< unsigned 32-bit type */
- typedef uint64_t enet_uint64;
- # endif /* __ENET_TYPES_H__ */
- /**
- * @file protocol.h
- * @brief ENet protocol
- */
- # ifndef __ENET_PROTOCOL_H__
- # define __ENET_PROTOCOL_H__
- enum {
- ENET_PROTOCOL_MINIMUM_MTU = 576,
- ENET_PROTOCOL_MAXIMUM_MTU = 4096,
- ENET_PROTOCOL_MAXIMUM_PACKET_COMMANDS = 32,
- ENET_PROTOCOL_MINIMUM_WINDOW_SIZE = 4096,
- ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE = 65536,
- ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT = 1,
- ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT = 255,
- ENET_PROTOCOL_MAXIMUM_PEER_ID = 0xFFF,
- ENET_PROTOCOL_MAXIMUM_FRAGMENT_COUNT = 1024 * 1024
- };
- typedef enum _ENetProtocolCommand {
- ENET_PROTOCOL_COMMAND_NONE = 0,
- ENET_PROTOCOL_COMMAND_ACKNOWLEDGE = 1,
- ENET_PROTOCOL_COMMAND_CONNECT = 2,
- ENET_PROTOCOL_COMMAND_VERIFY_CONNECT = 3,
- ENET_PROTOCOL_COMMAND_DISCONNECT = 4,
- ENET_PROTOCOL_COMMAND_PING = 5,
- ENET_PROTOCOL_COMMAND_SEND_RELIABLE = 6,
- ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE = 7,
- ENET_PROTOCOL_COMMAND_SEND_FRAGMENT = 8,
- ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED = 9,
- ENET_PROTOCOL_COMMAND_BANDWIDTH_LIMIT = 10,
- ENET_PROTOCOL_COMMAND_THROTTLE_CONFIGURE = 11,
- ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE_FRAGMENT = 12,
- ENET_PROTOCOL_COMMAND_COUNT = 13,
- ENET_PROTOCOL_COMMAND_MASK = 0x0F
- } ENetProtocolCommand;
- typedef enum _ENetProtocolFlag {
- ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE = (1 << 7),
- ENET_PROTOCOL_COMMAND_FLAG_UNSEQUENCED = (1 << 6),
- ENET_PROTOCOL_HEADER_FLAG_COMPRESSED = (1 << 14),
- ENET_PROTOCOL_HEADER_FLAG_SENT_TIME = (1 << 15),
- ENET_PROTOCOL_HEADER_FLAG_MASK = ENET_PROTOCOL_HEADER_FLAG_COMPRESSED | ENET_PROTOCOL_HEADER_FLAG_SENT_TIME,
- ENET_PROTOCOL_HEADER_SESSION_MASK = (3 << 12),
- ENET_PROTOCOL_HEADER_SESSION_SHIFT = 12
- } ENetProtocolFlag;
- # ifdef _MSC_VER
- # pragma pack(push, 1)
- # define ENET_PACKED
- # elif defined(__GNUC__) || defined(__clang__)
- # define ENET_PACKED __attribute__((packed))
- # else
- # define ENET_PACKED
- # endif
- typedef struct _ENetProtocolHeader {
- enet_uint16 peerID;
- enet_uint16 sentTime;
- } ENET_PACKED ENetProtocolHeader;
- typedef struct _ENetProtocolCommandHeader {
- enet_uint8 command;
- enet_uint8 channelID;
- enet_uint16 reliableSequenceNumber;
- } ENET_PACKED ENetProtocolCommandHeader;
- typedef struct _ENetProtocolAcknowledge {
- ENetProtocolCommandHeader header;
- enet_uint16 receivedReliableSequenceNumber;
- enet_uint16 receivedSentTime;
- } ENET_PACKED ENetProtocolAcknowledge;
- typedef struct _ENetProtocolConnect {
- ENetProtocolCommandHeader header;
- enet_uint16 outgoingPeerID;
- enet_uint8 incomingSessionID;
- enet_uint8 outgoingSessionID;
- enet_uint32 mtu;
- enet_uint32 windowSize;
- enet_uint32 channelCount;
- enet_uint32 incomingBandwidth;
- enet_uint32 outgoingBandwidth;
- enet_uint32 packetThrottleInterval;
- enet_uint32 packetThrottleAcceleration;
- enet_uint32 packetThrottleDeceleration;
- enet_uint32 connectID;
- enet_uint32 data;
- } ENET_PACKED ENetProtocolConnect;
- typedef struct _ENetProtocolVerifyConnect {
- ENetProtocolCommandHeader header;
- enet_uint16 outgoingPeerID;
- enet_uint8 incomingSessionID;
- enet_uint8 outgoingSessionID;
- enet_uint32 mtu;
- enet_uint32 windowSize;
- enet_uint32 channelCount;
- enet_uint32 incomingBandwidth;
- enet_uint32 outgoingBandwidth;
- enet_uint32 packetThrottleInterval;
- enet_uint32 packetThrottleAcceleration;
- enet_uint32 packetThrottleDeceleration;
- enet_uint32 connectID;
- } ENET_PACKED ENetProtocolVerifyConnect;
- typedef struct _ENetProtocolBandwidthLimit {
- ENetProtocolCommandHeader header;
- enet_uint32 incomingBandwidth;
- enet_uint32 outgoingBandwidth;
- } ENET_PACKED ENetProtocolBandwidthLimit;
- typedef struct _ENetProtocolThrottleConfigure {
- ENetProtocolCommandHeader header;
- enet_uint32 packetThrottleInterval;
- enet_uint32 packetThrottleAcceleration;
- enet_uint32 packetThrottleDeceleration;
- } ENET_PACKED ENetProtocolThrottleConfigure;
- typedef struct _ENetProtocolDisconnect {
- ENetProtocolCommandHeader header;
- enet_uint32 data;
- } ENET_PACKED ENetProtocolDisconnect;
- typedef struct _ENetProtocolPing {
- ENetProtocolCommandHeader header;
- } ENET_PACKED ENetProtocolPing;
- typedef struct _ENetProtocolSendReliable {
- ENetProtocolCommandHeader header;
- enet_uint16 dataLength;
- } ENET_PACKED ENetProtocolSendReliable;
- typedef struct _ENetProtocolSendUnreliable {
- ENetProtocolCommandHeader header;
- enet_uint16 unreliableSequenceNumber;
- enet_uint16 dataLength;
- } ENET_PACKED ENetProtocolSendUnreliable;
- typedef struct _ENetProtocolSendUnsequenced {
- ENetProtocolCommandHeader header;
- enet_uint16 unsequencedGroup;
- enet_uint16 dataLength;
- } ENET_PACKED ENetProtocolSendUnsequenced;
- typedef struct _ENetProtocolSendFragment {
- ENetProtocolCommandHeader header;
- enet_uint16 startSequenceNumber;
- enet_uint16 dataLength;
- enet_uint32 fragmentCount;
- enet_uint32 fragmentNumber;
- enet_uint32 totalLength;
- enet_uint32 fragmentOffset;
- } ENET_PACKED ENetProtocolSendFragment;
- typedef union _ENetProtocol {
- ENetProtocolCommandHeader header;
- ENetProtocolAcknowledge acknowledge;
- ENetProtocolConnect connect;
- ENetProtocolVerifyConnect verifyConnect;
- ENetProtocolDisconnect disconnect;
- ENetProtocolPing ping;
- ENetProtocolSendReliable sendReliable;
- ENetProtocolSendUnreliable sendUnreliable;
- ENetProtocolSendUnsequenced sendUnsequenced;
- ENetProtocolSendFragment sendFragment;
- ENetProtocolBandwidthLimit bandwidthLimit;
- ENetProtocolThrottleConfigure throttleConfigure;
- } ENET_PACKED ENetProtocol;
- # ifdef _MSC_VER
- # pragma pack(pop)
- # endif
- # endif /* __ENET_PROTOCOL_H__ */
- /**
- * @file list.h
- * @brief ENet list management
- */
- # ifndef __ENET_LIST_H__
- # define __ENET_LIST_H__
- typedef struct _ENetListNode {
- struct _ENetListNode *next;
- struct _ENetListNode *previous;
- } ENetListNode;
- typedef ENetListNode *ENetListIterator;
- typedef struct _ENetList {
- ENetListNode sentinel;
- } ENetList;
- extern void
- enet_list_clear(ENetList *);
- extern ENetListIterator enet_list_insert(ENetListIterator, void *);
- extern void *enet_list_remove(ENetListIterator);
- extern ENetListIterator enet_list_move(ENetListIterator, void *, void *);
- extern size_t
- enet_list_size(ENetList *);
- # define enet_list_begin(list) ((list)->sentinel.next)
- # define enet_list_end(list) (&(list)->sentinel)
- # define enet_list_empty(list) (enet_list_begin(list) == enet_list_end(list))
- # define enet_list_next(iterator) ((iterator)->next)
- # define enet_list_previous(iterator) ((iterator)->previous)
- # define enet_list_front(list) ((void *) (list)->sentinel.next)
- # define enet_list_back(list) ((void *) (list)->sentinel.previous)
- # endif /* __ENET_LIST_H__ */
- /**
- * @file time.h
- * @brief ENet time constants and macros
- */
- # ifndef __ENET_TIME_H__
- # define __ENET_TIME_H__
- # define ENET_TIME_OVERFLOW 86400000
- # define ENET_TIME_LESS(a, b) ((a) - (b) >= ENET_TIME_OVERFLOW)
- # define ENET_TIME_GREATER(a, b) ((b) - (a) >= ENET_TIME_OVERFLOW)
- # define ENET_TIME_LESS_EQUAL(a, b) (!ENET_TIME_GREATER(a, b))
- # define ENET_TIME_GREATER_EQUAL(a, b) (!ENET_TIME_LESS(a, b))
- # define ENET_TIME_DIFFERENCE(a, b) ((a) - (b) >= ENET_TIME_OVERFLOW ? (b) - (a) : (a) - (b))
- # endif /* __ENET_TIME_H__ */
- /**
- * @file utility.h
- * @brief ENet utility header
- */
- # ifndef __ENET_UTILITY_H__
- # define __ENET_UTILITY_H__
- # define ENET_MAX(x, y) ((x) > (y) ? (x) : (y))
- # define ENET_MIN(x, y) ((x) < (y) ? (x) : (y))
- # endif /* __ENET_UTILITY_H__ */
- /**
- * @file callbacks.h
- * @brief ENet callbacks
- */
- # ifndef __ENET_CALLBACKS_H__
- # define __ENET_CALLBACKS_H__
- typedef struct _ENetCallbacks {
- void * (ENET_CALLBACK * malloc)(size_t size);
- void (ENET_CALLBACK * free)(void *memory);
- void (ENET_CALLBACK * no_memory)(void);
- } ENetCallbacks;
- /** @defgroup callbacks ENet internal callbacks
- * @{
- * @ingroup private
- */
- extern void *enet_malloc(size_t);
- extern void
- enet_free(void *);
- /** @} */
- # endif /* __ENET_CALLBACKS_H__ */
- # define ENET_VERSION_MAJOR 1
- # define ENET_VERSION_MINOR 4
- # define ENET_VERSION_PATCH 4
- # define ENET_VERSION_CREATE(major, minor, patch) (((major) << 16) | ((minor) << 8) | (patch))
- # define ENET_VERSION_GET_MAJOR(version) (((version) >> 16) & 0xFF)
- # define ENET_VERSION_GET_MINOR(version) (((version) >> 8) & 0xFF)
- # define ENET_VERSION_GET_PATCH(version) ((version) & 0xFF)
- # define ENET_VERSION ENET_VERSION_CREATE(ENET_VERSION_MAJOR, ENET_VERSION_MINOR, ENET_VERSION_PATCH)
- typedef enet_uint32 ENetVersion;
- struct _ENetHost;
- struct _ENetEvent;
- struct _ENetPacket;
- typedef enum _ENetSocketType {
- ENET_SOCKET_TYPE_STREAM = 1,
- ENET_SOCKET_TYPE_DATAGRAM = 2
- } ENetSocketType;
- typedef enum _ENetSocketWait {
- ENET_SOCKET_WAIT_NONE = 0,
- ENET_SOCKET_WAIT_SEND = (1 << 0),
- ENET_SOCKET_WAIT_RECEIVE = (1 << 1),
- ENET_SOCKET_WAIT_INTERRUPT = (1 << 2)
- } ENetSocketWait;
- typedef enum _ENetSocketOption {
- ENET_SOCKOPT_NONBLOCK = 1,
- ENET_SOCKOPT_BROADCAST = 2,
- ENET_SOCKOPT_RCVBUF = 3,
- ENET_SOCKOPT_SNDBUF = 4,
- ENET_SOCKOPT_REUSEADDR = 5,
- ENET_SOCKOPT_RCVTIMEO = 6,
- ENET_SOCKOPT_SNDTIMEO = 7,
- ENET_SOCKOPT_ERROR = 8,
- ENET_SOCKOPT_NODELAY = 9
- } ENetSocketOption;
- typedef enum _ENetSocketShutdown {
- ENET_SOCKET_SHUTDOWN_READ = 0,
- ENET_SOCKET_SHUTDOWN_WRITE = 1,
- ENET_SOCKET_SHUTDOWN_READ_WRITE = 2
- } ENetSocketShutdown;
- # define ENET_HOST_ANY 0
- # define ENET_HOST_BROADCAST 0xFFFFFFFFU
- # define ENET_PORT_ANY 0
- /**
- * Portable internet address structure.
- *
- * The host must be specified in network byte-order, and the port must be in host
- * byte-order. The constant ENET_HOST_ANY may be used to specify the default
- * server host. The constant ENET_HOST_BROADCAST may be used to specify the
- * broadcast address (255.255.255.255). This makes sense for enet_host_connect,
- * but not for enet_host_create. Once a server responds to a broadcast, the
- * address is updated from ENET_HOST_BROADCAST to the server's actual IP address.
- */
- typedef struct _ENetAddress {
- enet_uint32 host;
- enet_uint16 port;
- } ENetAddress;
- /**
- * Packet flag bit constants.
- *
- * The host must be specified in network byte-order, and the port must be in
- * host byte-order. The constant ENET_HOST_ANY may be used to specify the
- * default server host.
- *
- * @sa ENetPacket
- */
- typedef enum _ENetPacketFlag {
- /** packet must be received by the target peer and resend attempts should be
- * made until the packet is delivered */
- ENET_PACKET_FLAG_RELIABLE = (1 << 0),
- /** packet will not be sequenced with other packets
- * not supported for reliable packets
- */
- ENET_PACKET_FLAG_UNSEQUENCED = (1 << 1),
- /** packet will not allocate data, and user must supply it instead */
- ENET_PACKET_FLAG_NO_ALLOCATE = (1 << 2),
- /** packet will be fragmented using unreliable (instead of reliable) sends
- * if it exceeds the MTU */
- ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT = (1 << 3),
- /** whether the packet has been sent from all queues it has been entered into */
- ENET_PACKET_FLAG_SENT = (1 << 8)
- } ENetPacketFlag;
- typedef void (ENET_CALLBACK * ENetPacketFreeCallback)(struct _ENetPacket *);
- /**
- * ENet packet structure.
- *
- * An ENet data packet that may be sent to or received from a peer. The shown
- * fields should only be read and never modified. The data field contains the
- * allocated data for the packet. The dataLength fields specifies the length
- * of the allocated data. The flags field is either 0 (specifying no flags),
- * or a bitwise-or of any combination of the following flags:
- *
- * ENET_PACKET_FLAG_RELIABLE - packet must be received by the target peer
- * and resend attempts should be made until the packet is delivered
- *
- * ENET_PACKET_FLAG_UNSEQUENCED - packet will not be sequenced with other packets
- * (not supported for reliable packets)
- *
- * ENET_PACKET_FLAG_NO_ALLOCATE - packet will not allocate data, and user must supply it instead
- *
- * ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT - packet will be fragmented using unreliable
- * (instead of reliable) sends if it exceeds the MTU
- *
- * ENET_PACKET_FLAG_SENT - whether the packet has been sent from all queues it has been entered into
- * @sa ENetPacketFlag
- */
- typedef struct _ENetPacket {
- size_t referenceCount; /**< internal use only */
- enet_uint32 flags; /**< bitwise-or of ENetPacketFlag constants */
- enet_uint8 * data; /**< allocated data for packet */
- size_t dataLength; /**< length of data */
- ENetPacketFreeCallback freeCallback; /**< function to be called when the packet is no longer in use */
- void * userData; /**< application private data, may be freely modified */
- } ENetPacket;
- typedef struct _ENetAcknowledgement {
- ENetListNode acknowledgementList;
- enet_uint32 sentTime;
- ENetProtocol command;
- } ENetAcknowledgement;
- typedef struct _ENetOutgoingCommand {
- ENetListNode outgoingCommandList;
- enet_uint16 reliableSequenceNumber;
- enet_uint16 unreliableSequenceNumber;
- enet_uint32 sentTime;
- enet_uint32 roundTripTimeout;
- enet_uint32 roundTripTimeoutLimit;
- enet_uint32 fragmentOffset;
- enet_uint16 fragmentLength;
- enet_uint16 sendAttempts;
- ENetProtocol command;
- ENetPacket * packet;
- } ENetOutgoingCommand;
- typedef struct _ENetIncomingCommand {
- ENetListNode incomingCommandList;
- enet_uint16 reliableSequenceNumber;
- enet_uint16 unreliableSequenceNumber;
- ENetProtocol command;
- enet_uint32 fragmentCount;
- enet_uint32 fragmentsRemaining;
- enet_uint32 *fragments;
- ENetPacket * packet;
- } ENetIncomingCommand;
- typedef enum _ENetPeerState {
- ENET_PEER_STATE_DISCONNECTED = 0,
- ENET_PEER_STATE_CONNECTING = 1,
- ENET_PEER_STATE_ACKNOWLEDGING_CONNECT = 2,
- ENET_PEER_STATE_CONNECTION_PENDING = 3,
- ENET_PEER_STATE_CONNECTION_SUCCEEDED = 4,
- ENET_PEER_STATE_CONNECTED = 5,
- ENET_PEER_STATE_DISCONNECT_LATER = 6,
- ENET_PEER_STATE_DISCONNECTING = 7,
- ENET_PEER_STATE_ACKNOWLEDGING_DISCONNECT = 8,
- ENET_PEER_STATE_ZOMBIE = 9
- } ENetPeerState;
- # ifndef ENET_BUFFER_MAXIMUM
- # define ENET_BUFFER_MAXIMUM (1 + 2 * ENET_PROTOCOL_MAXIMUM_PACKET_COMMANDS)
- # endif
- enum {
- ENET_HOST_RECEIVE_BUFFER_SIZE = 256 * 1024,
- ENET_HOST_SEND_BUFFER_SIZE = 256 * 1024,
- ENET_HOST_BANDWIDTH_THROTTLE_INTERVAL = 1000,
- ENET_HOST_DEFAULT_MTU = 1400,
- ENET_HOST_DEFAULT_MAXIMUM_PACKET_SIZE = 32 * 1024 * 1024,
- ENET_HOST_DEFAULT_MAXIMUM_WAITING_DATA = 32 * 1024 * 1024,
- ENET_PEER_DEFAULT_ROUND_TRIP_TIME = 500,
- ENET_PEER_DEFAULT_PACKET_THROTTLE = 32,
- ENET_PEER_PACKET_THROTTLE_SCALE = 32,
- ENET_PEER_PACKET_THROTTLE_COUNTER = 7,
- ENET_PEER_PACKET_THROTTLE_ACCELERATION = 2,
- ENET_PEER_PACKET_THROTTLE_DECELERATION = 2,
- ENET_PEER_PACKET_THROTTLE_INTERVAL = 5000,
- ENET_PEER_PACKET_LOSS_SCALE = (1 << 16),
- ENET_PEER_PACKET_LOSS_INTERVAL = 10000,
- ENET_PEER_WINDOW_SIZE_SCALE = 64 * 1024,
- ENET_PEER_TIMEOUT_LIMIT = 32,
- ENET_PEER_TIMEOUT_MINIMUM = 5000,
- ENET_PEER_TIMEOUT_MAXIMUM = 30000,
- ENET_PEER_PING_INTERVAL = 500,
- ENET_PEER_UNSEQUENCED_WINDOWS = 64,
- ENET_PEER_UNSEQUENCED_WINDOW_SIZE = 1024,
- ENET_PEER_FREE_UNSEQUENCED_WINDOWS = 32,
- ENET_PEER_RELIABLE_WINDOWS = 16,
- ENET_PEER_RELIABLE_WINDOW_SIZE = 0x1000,
- ENET_PEER_FREE_RELIABLE_WINDOWS = 8
- };
- typedef struct _ENetChannel {
- enet_uint16 outgoingReliableSequenceNumber;
- enet_uint16 outgoingUnreliableSequenceNumber;
- enet_uint16 usedReliableWindows;
- enet_uint16 reliableWindows [ENET_PEER_RELIABLE_WINDOWS];
- enet_uint16 incomingReliableSequenceNumber;
- enet_uint16 incomingUnreliableSequenceNumber;
- ENetList incomingReliableCommands;
- ENetList incomingUnreliableCommands;
- } ENetChannel;
- /**
- * An ENet peer which data packets may be sent or received from.
- *
- * No fields should be modified unless otherwise specified.
- */
- typedef struct _ENetPeer {
- ENetListNode dispatchList;
- struct _ENetHost *host;
- enet_uint16 outgoingPeerID;
- enet_uint16 incomingPeerID;
- enet_uint32 connectID;
- enet_uint8 outgoingSessionID;
- enet_uint8 incomingSessionID;
- ENetAddress address; /**< Internet address of the peer */
- void * data; /**< Application private data, may be freely modified */
- ENetPeerState state;
- ENetChannel * channels;
- size_t channelCount; /**< Number of channels allocated for communication with peer */
- enet_uint32 incomingBandwidth; /**< Downstream bandwidth of the client in bytes/second */
- enet_uint32 outgoingBandwidth; /**< Upstream bandwidth of the client in bytes/second */
- enet_uint32 incomingBandwidthThrottleEpoch;
- enet_uint32 outgoingBandwidthThrottleEpoch;
- enet_uint32 incomingDataTotal;
- enet_uint32 outgoingDataTotal;
- enet_uint32 lastSendTime;
- enet_uint32 lastReceiveTime;
- enet_uint32 nextTimeout;
- enet_uint32 earliestTimeout;
- enet_uint32 packetLossEpoch;
- enet_uint32 packetsSent;
- enet_uint32 packetsLost;
- enet_uint32 packetLoss; /**< mean packet loss of reliable packets as a ratio with respect to the constant ENET_PEER_PACKET_LOSS_SCALE */
- enet_uint32 packetLossVariance;
- enet_uint32 packetThrottle;
- enet_uint32 packetThrottleLimit;
- enet_uint32 packetThrottleCounter;
- enet_uint32 packetThrottleEpoch;
- enet_uint32 packetThrottleAcceleration;
- enet_uint32 packetThrottleDeceleration;
- enet_uint32 packetThrottleInterval;
- enet_uint32 pingInterval;
- enet_uint32 timeoutLimit;
- enet_uint32 timeoutMinimum;
- enet_uint32 timeoutMaximum;
- enet_uint32 lastRoundTripTime;
- enet_uint32 lowestRoundTripTime;
- enet_uint32 lastRoundTripTimeVariance;
- enet_uint32 highestRoundTripTimeVariance;
- enet_uint32 roundTripTime; /**< mean round trip time (RTT), in milliseconds, between sending a reliable packet and receiving its acknowledgement */
- enet_uint32 roundTripTimeVariance;
- enet_uint32 mtu;
- enet_uint32 windowSize;
- enet_uint32 reliableDataInTransit;
- enet_uint16 outgoingReliableSequenceNumber;
- ENetList acknowledgements;
- ENetList sentReliableCommands;
- ENetList sentUnreliableCommands;
- ENetList outgoingReliableCommands;
- ENetList outgoingUnreliableCommands;
- ENetList dispatchedCommands;
- int needsDispatch;
- enet_uint16 incomingUnsequencedGroup;
- enet_uint16 outgoingUnsequencedGroup;
- enet_uint32 unsequencedWindow [ENET_PEER_UNSEQUENCED_WINDOW_SIZE / 32];
- enet_uint32 eventData;
- size_t totalWaitingData;
- } ENetPeer;
- /** An ENet packet compressor for compressing UDP packets before socket sends or receives.
- */
- typedef struct _ENetCompressor {
- /** Context data for the compressor. Must be non-NULL. */
- void *context;
- /** Compresses from inBuffers[0:inBufferCount-1], containing inLimit bytes, to outData, outputting at most outLimit bytes. Should return 0 on failure. */
- size_t(ENET_CALLBACK * compress) (void *context, const ENetBuffer * inBuffers, size_t inBufferCount, size_t inLimit,
- enet_uint8 * outData, size_t outLimit);
- /** Decompresses from inData, containing inLimit bytes, to outData, outputting at most outLimit bytes. Should return 0 on failure. */
- size_t(ENET_CALLBACK * decompress) (void *context, const enet_uint8 * inData, size_t inLimit, enet_uint8 * outData,
- size_t outLimit);
- /** Destroys the context when compression is disabled or the host is destroyed. May be NULL. */
- void (ENET_CALLBACK * destroy)(void *context);
- } ENetCompressor;
- /** Callback that computes the checksum of the data held in buffers[0:bufferCount-1] */
- typedef enet_uint32 (ENET_CALLBACK * ENetChecksumCallback)(const ENetBuffer *buffers, size_t bufferCount);
- /** Callback for intercepting received raw UDP packets. Should return 1 to intercept, 0 to ignore, or -1 to propagate an error. */
- typedef int (ENET_CALLBACK * ENetInterceptCallback)(struct _ENetHost *host, struct _ENetEvent *event);
- /** An ENet host for communicating with peers.
- *
- * No fields should be modified unless otherwise stated.
- *
- * @sa enet_host_create()
- * @sa enet_host_destroy()
- * @sa enet_host_connect()
- * @sa enet_host_service()
- * @sa enet_host_flush()
- * @sa enet_host_broadcast()
- * @sa enet_host_compress()
- * @sa enet_host_compress_with_range_coder()
- * @sa enet_host_channel_limit()
- * @sa enet_host_bandwidth_limit()
- * @sa enet_host_bandwidth_throttle()
- */
- typedef struct _ENetHost {
- ENetSocket socket;
- ENetAddress address; /**< Internet address of the host */
- enet_uint32 incomingBandwidth; /**< downstream bandwidth of the host */
- enet_uint32 outgoingBandwidth; /**< upstream bandwidth of the host */
- enet_uint32 bandwidthThrottleEpoch;
- enet_uint32 mtu;
- enet_uint32 randomSeed;
- int recalculateBandwidthLimits;
- ENetPeer * peers; /**< array of peers allocated for this host */
- size_t peerCount; /**< number of peers allocated for this host */
- size_t channelLimit; /**< maximum number of channels allowed for connected peers */
- enet_uint32 serviceTime;
- ENetList dispatchQueue;
- int continueSending;
- size_t packetSize;
- enet_uint16 headerFlags;
- ENetProtocol commands [ENET_PROTOCOL_MAXIMUM_PACKET_COMMANDS];
- size_t commandCount;
- ENetBuffer buffers [ENET_BUFFER_MAXIMUM];
- size_t bufferCount;
- ENetChecksumCallback checksum; /**< callback the user can set to enable packet checksums for this host */
- ENetCompressor compressor;
- enet_uint8 packetData [2][ENET_PROTOCOL_MAXIMUM_MTU];
- ENetAddress receivedAddress;
- enet_uint8 * receivedData;
- size_t receivedDataLength;
- enet_uint32 totalSentData; /**< total data sent, user should reset to 0 as needed to prevent overflow */
- enet_uint32 totalSentPackets; /**< total UDP packets sent, user should reset to 0 as needed to prevent overflow */
- enet_uint32 totalReceivedData; /**< total data received, user should reset to 0 as needed to prevent overflow */
- enet_uint32 totalReceivedPackets; /**< total UDP packets received, user should reset to 0 as needed to prevent overflow */
- ENetInterceptCallback intercept; /**< callback the user can set to intercept received raw UDP packets */
- size_t connectedPeers;
- size_t bandwidthLimitedPeers;
- size_t duplicatePeers; /**< optional number of allowed peers from duplicate IPs, defaults to ENET_PROTOCOL_MAXIMUM_PEER_ID */
- size_t maximumPacketSize; /**< the maximum allowable packet size that may be sent or received on a peer */
- size_t maximumWaitingData; /**< the maximum aggregate amount of buffer space a peer may use waiting for packets to be delivered */
- } ENetHost;
- /**
- * An ENet event type, as specified in @ref ENetEvent.
- */
- typedef enum _ENetEventType {
- /** no event occurred within the specified time limit */
- ENET_EVENT_TYPE_NONE = 0,
- /** a connection request initiated by enet_host_connect has completed.
- * The peer field contains the peer which successfully connected.
- */
- ENET_EVENT_TYPE_CONNECT = 1,
- /** a peer has disconnected. This event is generated on a successful
- * completion of a disconnect initiated by enet_peer_disconnect, if
- * a peer has timed out, or if a connection request intialized by
- * enet_host_connect has timed out. The peer field contains the peer
- * which disconnected. The data field contains user supplied data
- * describing the disconnection, or 0, if none is available.
- */
- ENET_EVENT_TYPE_DISCONNECT = 2,
- /** a packet has been received from a peer. The peer field specifies the
- * peer which sent the packet. The channelID field specifies the channel
- * number upon which the packet was received. The packet field contains
- * the packet that was received; this packet must be destroyed with
- * enet_packet_destroy after use.
- */
- ENET_EVENT_TYPE_RECEIVE = 3
- } ENetEventType;
- /**
- * An ENet event as returned by enet_host_service().
- *
- * @sa enet_host_service
- */
- typedef struct _ENetEvent {
- ENetEventType type; /**< type of the event */
- ENetPeer * peer; /**< peer that generated a connect, disconnect or receive event */
- enet_uint8 channelID; /**< channel on the peer that generated the event, if appropriate */
- enet_uint32 data; /**< data associated with the event, if appropriate */
- ENetPacket * packet; /**< packet associated with the event, if appropriate */
- } ENetEvent;
- /** @defgroup global ENet global functions
- * @{
- */
- /**
- * Initializes ENet globally. Must be called prior to using any functions in
- * ENet.
- * @returns 0 on success, < 0 on failure
- */
- ENET_API int
- enet_initialize(void);
- /**
- * Initializes ENet globally and supplies user-overridden callbacks. Must be called prior to using any functions in ENet. Do not use enet_initialize() if you use this variant. Make sure the ENetCallbacks structure is zeroed out so that any additional callbacks added in future versions will be properly ignored.
- *
- * @param version the constant ENET_VERSION should be supplied so ENet knows which version of ENetCallbacks struct to use
- * @param inits user-overridden callbacks where any NULL callbacks will use ENet's defaults
- * @returns 0 on success, < 0 on failure
- */
- ENET_API int
- enet_initialize_with_callbacks(ENetVersion version, const ENetCallbacks *inits);
- /**
- * Shuts down ENet globally. Should be called when a program that has
- * initialized ENet exits.
- */
- ENET_API void
- enet_deinitialize(void);
- /**
- * Gives the linked version of the ENet library.
- * @returns the version number
- */
- ENET_API ENetVersion
- enet_linked_version(void);
- /** @} */
- /** @defgroup private ENet private implementation functions */
- /**
- * Returns the wall-time in milliseconds. Its initial value is unspecified
- * unless otherwise set.
- */
- ENET_API enet_uint64
- enet_time_get(void);
- /**
- * Sets the current wall-time in milliseconds.
- */
- ENET_API void enet_time_set(enet_uint64);
- /** @defgroup socket ENet socket functions
- * @{
- */
- ENET_API ENetSocket enet_socket_create(ENetSocketType);
- ENET_API int enet_socket_bind(ENetSocket, const ENetAddress *);
- ENET_API int enet_socket_get_address(ENetSocket, ENetAddress *);
- ENET_API int enet_socket_listen(ENetSocket, int);
- ENET_API ENetSocket enet_socket_accept(ENetSocket, ENetAddress *);
- ENET_API int enet_socket_connect(ENetSocket, const ENetAddress *);
- ENET_API int enet_socket_send(ENetSocket, const ENetAddress *, const ENetBuffer *, size_t);
- ENET_API int enet_socket_receive(ENetSocket, ENetAddress *, ENetBuffer *, size_t);
- ENET_API int enet_socket_wait(ENetSocket, enet_uint32 *, enet_uint64);
- ENET_API int enet_socket_set_option(ENetSocket, ENetSocketOption, int);
- ENET_API int enet_socket_get_option(ENetSocket, ENetSocketOption, int *);
- ENET_API int enet_socket_shutdown(ENetSocket, ENetSocketShutdown);
- ENET_API void enet_socket_destroy(ENetSocket);
- ENET_API int enet_socketset_select(ENetSocket, ENetSocketSet *, ENetSocketSet *, enet_uint32);
- /** @} */
- /** @defgroup Address ENet address functions
- * @{
- */
- /** Attempts to parse the printable form of the IP address in the parameter hostName
- * and sets the host field in the address parameter if successful.
- * @param address destination to store the parsed IP address
- * @param hostName IP address to parse
- * @retval 0 on success
- * @retval < 0 on failure
- * @returns the address of the given hostName in address on success
- */
- ENET_API int
- enet_address_set_host_ip(ENetAddress *address, const char *hostName);
- /** Attempts to resolve the host named by the parameter hostName and sets
- * the host field in the address parameter if successful.
- * @param address destination to store resolved address
- * @param hostName host name to lookup
- * @retval 0 on success
- * @retval < 0 on failure
- * @returns the address of the given hostName in address on success
- */
- ENET_API int
- enet_address_set_host(ENetAddress *address, const char *hostName);
- /** Gives the printable form of the IP address specified in the address parameter.
- * @param address address printed
- * @param hostName destination for name, must not be NULL
- * @param nameLength maximum length of hostName.
- * @returns the null-terminated name of the host in hostName on success
- * @retval 0 on success
- * @retval < 0 on failure
- */
- ENET_API int
- enet_address_get_host_ip(const ENetAddress *address, char *hostName, size_t nameLength);
- /** Attempts to do a reverse lookup of the host field in the address parameter.
- * @param address address used for reverse lookup
- * @param hostName destination for name, must not be NULL
- * @param nameLength maximum length of hostName.
- * @returns the null-terminated name of the host in hostName on success
- * @retval 0 on success
- * @retval < 0 on failure
- */
- ENET_API int
- enet_address_get_host(const ENetAddress *address, char *hostName, size_t nameLength);
- /** @} */
- ENET_API ENetPacket *enet_packet_create(const void *, size_t, enet_uint32);
- ENET_API ENetPacket *enet_packet_create_offset(const void *, size_t, size_t, enet_uint32);
- ENET_API void
- enet_packet_destroy(ENetPacket *);
- ENET_API int enet_packet_resize(ENetPacket *, size_t);
- ENET_API enet_uint32 enet_crc32(const ENetBuffer *, size_t);
- ENET_API ENetHost *enet_host_create(const ENetAddress *, size_t, size_t, enet_uint32, enet_uint32);
- ENET_API void
- enet_host_destroy(ENetHost *);
- ENET_API ENetPeer *enet_host_connect(ENetHost *, const ENetAddress *, size_t, enet_uint32);
- ENET_API int
- enet_host_check_events(ENetHost *, ENetEvent *);
- ENET_API int enet_host_service(ENetHost *, ENetEvent *, enet_uint32);
- ENET_API void
- enet_host_flush(ENetHost *);
- ENET_API void enet_host_broadcast(ENetHost *, enet_uint8, ENetPacket *);
- ENET_API void
- enet_host_compress(ENetHost *, const ENetCompressor *);
- ENET_API int
- enet_host_compress_with_range_coder(ENetHost *host);
- ENET_API void enet_host_channel_limit(ENetHost *, size_t);
- ENET_API void enet_host_bandwidth_limit(ENetHost *, enet_uint32, enet_uint32);
- extern void
- enet_host_bandwidth_throttle(ENetHost *);
- extern enet_uint64
- enet_host_random_seed(void);
- ENET_API int enet_peer_send(ENetPeer *, enet_uint8, ENetPacket *);
- ENET_API ENetPacket *
- enet_peer_receive(ENetPeer *, enet_uint8 *channelID);
- ENET_API void
- enet_peer_ping(ENetPeer *);
- ENET_API void enet_peer_ping_interval(ENetPeer *, enet_uint32);
- ENET_API void enet_peer_timeout(ENetPeer *, enet_uint32, enet_uint32, enet_uint32);
- ENET_API void
- enet_peer_reset(ENetPeer *);
- ENET_API void enet_peer_disconnect(ENetPeer *, enet_uint32);
- ENET_API void enet_peer_disconnect_now(ENetPeer *, enet_uint32);
- ENET_API void enet_peer_disconnect_later(ENetPeer *, enet_uint32);
- ENET_API void enet_peer_throttle_configure(ENetPeer *, enet_uint32, enet_uint32, enet_uint32);
- extern int enet_peer_throttle(ENetPeer *, enet_uint32);
- extern void
- enet_peer_reset_queues(ENetPeer *);
- extern void
- enet_peer_setup_outgoing_command(ENetPeer *, ENetOutgoingCommand *);
- extern ENetOutgoingCommand *enet_peer_queue_outgoing_command(ENetPeer *, const ENetProtocol *, ENetPacket *,
- enet_uint32, enet_uint16);
- extern ENetIncomingCommand *enet_peer_queue_incoming_command(ENetPeer *, const ENetProtocol *, const void *, size_t,
- enet_uint32, enet_uint32);
- extern ENetAcknowledgement *enet_peer_queue_acknowledgement(ENetPeer *, const ENetProtocol *, enet_uint16);
- extern void
- enet_peer_dispatch_incoming_unreliable_commands(ENetPeer *, ENetChannel *);
- extern void
- enet_peer_dispatch_incoming_reliable_commands(ENetPeer *, ENetChannel *);
- extern void
- enet_peer_on_connect(ENetPeer *);
- extern void
- enet_peer_on_disconnect(ENetPeer *);
- ENET_API void *
- enet_range_coder_create(void);
- ENET_API void
- enet_range_coder_destroy(void *);
- ENET_API size_t enet_range_coder_compress(void *, const ENetBuffer *, size_t, size_t, enet_uint8 *, size_t);
- ENET_API size_t enet_range_coder_decompress(void *, const enet_uint8 *, size_t, enet_uint8 *, size_t);
- extern size_t enet_protocol_command_size(enet_uint8);
- # ifdef __cplusplus
- }
- # endif
- # if defined(ENET_IMPLEMENTATION) && !defined(ENET_IMPLEMENTATION_DONE)
- # define ENET_IMPLEMENTATION_DONE
- # ifdef __cplusplus
- extern "C"
- {
- # endif
- // @from_file: src/callbacks.c
- /**
- * @file callbacks.c
- * @brief ENet callback functions
- */
- # define ENET_BUILDING_LIB 1
- static ENetCallbacks callbacks = { malloc, free, abort };
- int enet_initialize_with_callbacks(ENetVersion version, const ENetCallbacks *inits) {
- if (version < ENET_VERSION_CREATE(1, 3, 0)) {
- return -1;
- }
- if (inits->malloc != NULL || inits->free != NULL) {
- if (inits->malloc == NULL || inits->free == NULL) {
- return -1;
- }
- callbacks.malloc = inits->malloc;
- callbacks.free = inits->free;
- }
- if (inits->no_memory != NULL) {
- callbacks.no_memory = inits->no_memory;
- }
- return enet_initialize();
- }
- ENetVersion enet_linked_version(void) {
- return ENET_VERSION;
- }
- void * enet_malloc(size_t size) {
- void *memory = callbacks.malloc(size);
- if (memory == NULL) {
- callbacks.no_memory();
- }
- return memory;
- }
- void enet_free(void *memory) {
- callbacks.free(memory);
- }
- // @from_file: src/compress.c
- /**
- * @file compress.c
- * @brief An adaptive order-2 PPM range coder
- */
- # define ENET_BUILDING_LIB 1
- # include <string.h>
- typedef struct _ENetSymbol {
- /* binary indexed tree of symbols */
- enet_uint8 value;
- enet_uint8 count;
- enet_uint16 under;
- enet_uint16 left, right;
- /* context defined by this symbol */
- enet_uint16 symbols;
- enet_uint16 escapes;
- enet_uint16 total;
- enet_uint16 parent;
- } ENetSymbol;
- /* adaptation constants tuned aggressively for small packet sizes rather than large file compression */
- enum {
- ENET_RANGE_CODER_TOP = 1 << 24,
- ENET_RANGE_CODER_BOTTOM = 1 << 16,
- ENET_CONTEXT_SYMBOL_DELTA = 3,
- ENET_CONTEXT_SYMBOL_MINIMUM = 1,
- ENET_CONTEXT_ESCAPE_MINIMUM = 1,
- ENET_SUBCONTEXT_ORDER = 2,
- ENET_SUBCONTEXT_SYMBOL_DELTA = 2,
- ENET_SUBCONTEXT_ESCAPE_DELTA = 5
- };
- /* context exclusion roughly halves compression speed, so disable for now */
- # undef ENET_CONTEXT_EXCLUSION
- typedef struct _ENetRangeCoder {
- /* only allocate enough symbols for reasonable MTUs, would need to be larger for large file compression */
- ENetSymbol symbols[4096];
- } ENetRangeCoder;
- void * enet_range_coder_create(void) {
- ENetRangeCoder *rangeCoder = (ENetRangeCoder *) enet_malloc(sizeof(ENetRangeCoder));
- if (rangeCoder == NULL) {
- return NULL;
- }
- return rangeCoder;
- }
- void enet_range_coder_destroy(void *context) {
- ENetRangeCoder *rangeCoder = (ENetRangeCoder *) context;
- if (rangeCoder == NULL) {
- return;
- }
- enet_free(rangeCoder);
- }
- # define ENET_SYMBOL_CREATE(symbol, value_, count_) \
- { \
- symbol = &rangeCoder->symbols [nextSymbol++]; \
- symbol->value = value_; \
- symbol->count = count_; \
- symbol->under = count_; \
- symbol->left = 0; \
- symbol->right = 0; \
- symbol->symbols = 0; \
- symbol->escapes = 0; \
- symbol->total = 0; \
- symbol->parent = 0; \
- }
- # define ENET_CONTEXT_CREATE(context, escapes_, minimum) \
- { \
- ENET_SYMBOL_CREATE(context, 0, 0); \
- (context)->escapes = escapes_; \
- (context)->total = escapes_ + 256 * minimum; \
- (context)->symbols = 0; \
- }
- static enet_uint16 enet_symbol_rescale(ENetSymbol *symbol) {
- enet_uint16 total = 0;
- for (;;) {
- symbol->count -= symbol->count >> 1;
- symbol->under = symbol->count;
- if (symbol->left) {
- symbol->under += enet_symbol_rescale(symbol + symbol->left);
- }
- total += symbol->under;
- if (!symbol->right) { break; }
- symbol += symbol->right;
- }
- return total;
- }
- # define ENET_CONTEXT_RESCALE(context, minimum) \
- { \
- (context)->total = (context)->symbols ? enet_symbol_rescale((context) + (context)->symbols) : 0; \
- (context)->escapes -= (context)->escapes >> 1; \
- (context)->total += (context)->escapes + 256 * minimum; \
- }
- # define ENET_RANGE_CODER_OUTPUT(value) \
- { \
- if (outData >= outEnd) { \
- return 0; } \
- *outData++ = value; \
- }
- # define ENET_RANGE_CODER_ENCODE(under, count, total) \
- { \
- encodeRange /= (total); \
- encodeLow += (under) * encodeRange; \
- encodeRange *= (count); \
- for (;;) \
- { \
- if ((encodeLow ^ (encodeLow + encodeRange)) >= ENET_RANGE_CODER_TOP) \
- { \
- if (encodeRange >= ENET_RANGE_CODER_BOTTOM) { break; } \
- encodeRange = -encodeLow & (ENET_RANGE_CODER_BOTTOM - 1); \
- } \
- ENET_RANGE_CODER_OUTPUT(encodeLow >> 24); \
- encodeRange <<= 8; \
- encodeLow <<= 8; \
- } \
- }
- # define ENET_RANGE_CODER_FLUSH \
- { \
- while (encodeLow) \
- { \
- ENET_RANGE_CODER_OUTPUT(encodeLow >> 24); \
- encodeLow <<= 8; \
- } \
- }
- # define ENET_RANGE_CODER_FREE_SYMBOLS \
- { \
- if (nextSymbol >= sizeof(rangeCoder->symbols) / sizeof(ENetSymbol) - ENET_SUBCONTEXT_ORDER) \
- { \
- nextSymbol = 0; \
- ENET_CONTEXT_CREATE(root, ENET_CONTEXT_ESCAPE_MINIMUM, ENET_CONTEXT_SYMBOL_MINIMUM); \
- predicted = 0; \
- order = 0; \
- } \
- }
- # define ENET_CONTEXT_ENCODE(context, symbol_, value_, under_, count_, update, minimum) \
- { \
- under_ = value * minimum; \
- count_ = minimum; \
- if (!(context)->symbols) \
- { \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- (context)->symbols = symbol_ - (context); \
- } \
- else \
- { \
- ENetSymbol *node = (context) + (context)->symbols; \
- for (;;) \
- { \
- if (value_ < node->value) \
- { \
- node->under += update; \
- if (node->left) { node += node->left; continue; } \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- node->left = symbol_ - node; \
- } \
- else if (value_ > node->value) \
- { \
- under_ += node->under; \
- if (node->right) { node += node->right; continue; } \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- node->right = symbol_ - node; \
- } \
- else \
- { \
- count_ += node->count; \
- under_ += node->under - node->count; \
- node->under += update; \
- node->count += update; \
- symbol_ = node; \
- } \
- break; \
- } \
- } \
- }
- # ifdef ENET_CONTEXT_EXCLUSION
- static const ENetSymbol emptyContext = { 0, 0, 0, 0, 0, 0, 0, 0, 0 };
- # define ENET_CONTEXT_WALK(context, body) \
- { \
- const ENetSymbol *node = (context) + (context)->symbols; \
- const ENetSymbol *stack [256]; \
- size_t stackSize = 0; \
- while (node->left) \
- { \
- stack [stackSize++] = node; \
- node += node->left; \
- } \
- for (;;) \
- { \
- body; \
- if (node->right) \
- { \
- node += node->right; \
- while (node->left) \
- { \
- stack [stackSize++] = node; \
- node += node->left; \
- } \
- } \
- else if (stackSize <= 0) { \
- break; } \
- else { \
- node = stack [--stackSize]; } \
- } \
- }
- # define ENET_CONTEXT_ENCODE_EXCLUDE(context, value_, under, total, minimum) \
- ENET_CONTEXT_WALK(context, { \
- if (node->value != value_) \
- { \
- enet_uint16 parentCount = rangeCoder->symbols [node->parent].count + minimum; \
- if (node->value < value_) { \
- under -= parentCount; } \
- total -= parentCount; \
- } \
- })
- # endif // ifdef ENET_CONTEXT_EXCLUSION
- size_t enet_range_coder_compress(void *context, const ENetBuffer *inBuffers, size_t inBufferCount, size_t inLimit,
- enet_uint8 *outData, size_t outLimit) {
- ENetRangeCoder *rangeCoder = (ENetRangeCoder *) context;
- enet_uint8 *outStart = outData, *outEnd = &outData [outLimit];
- const enet_uint8 *inData, *inEnd;
- enet_uint32 encodeLow = 0, encodeRange = ~0;
- ENetSymbol *root;
- enet_uint16 predicted = 0;
- size_t order = 0, nextSymbol = 0;
- if (rangeCoder == NULL || inBufferCount <= 0 || inLimit <= 0) {
- return 0;
- }
- inData = (const enet_uint8 *) inBuffers->data;
- inEnd = &inData [inBuffers->dataLength];
- inBuffers++;
- inBufferCount--;
- ENET_CONTEXT_CREATE(root, ENET_CONTEXT_ESCAPE_MINIMUM, ENET_CONTEXT_SYMBOL_MINIMUM);
- for (;;) {
- ENetSymbol *subcontext, *symbol;
- # ifdef ENET_CONTEXT_EXCLUSION
- const ENetSymbol *childContext = &emptyContext;
- # endif
- enet_uint8 value;
- enet_uint16 count, under, *parent = &predicted, total;
- if (inData >= inEnd) {
- if (inBufferCount <= 0) {
- break;
- }
- inData = (const enet_uint8 *) inBuffers->data;
- inEnd = &inData [inBuffers->dataLength];
- inBuffers++;
- inBufferCount--;
- }
- value = *inData++;
- for (subcontext = &rangeCoder->symbols [predicted];
- subcontext != root;
- # ifdef ENET_CONTEXT_EXCLUSION
- childContext = subcontext,
- # endif
- subcontext = &rangeCoder->symbols [subcontext->parent])
- {
- ENET_CONTEXT_ENCODE(subcontext, symbol, value, under, count, ENET_SUBCONTEXT_SYMBOL_DELTA, 0);
- *parent = symbol - rangeCoder->symbols;
- parent = &symbol->parent;
- total = subcontext->total;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > ENET_SUBCONTEXT_SYMBOL_DELTA + ENET_SUBCONTEXT_ESCAPE_DELTA) {
- ENET_CONTEXT_ENCODE_EXCLUDE(childContext, value, under, total, 0);
- }
- # endif
- if (count > 0) {
- ENET_RANGE_CODER_ENCODE(subcontext->escapes + under, count, total);
- } else {
- if (subcontext->escapes > 0 && subcontext->escapes < total) {
- ENET_RANGE_CODER_ENCODE(0, subcontext->escapes, total);
- }
- subcontext->escapes += ENET_SUBCONTEXT_ESCAPE_DELTA;
- subcontext->total += ENET_SUBCONTEXT_ESCAPE_DELTA;
- }
- subcontext->total += ENET_SUBCONTEXT_SYMBOL_DELTA;
- if (count > 0xFF - 2 * ENET_SUBCONTEXT_SYMBOL_DELTA ||
- subcontext->total > ENET_RANGE_CODER_BOTTOM - 0x100)
- {
- ENET_CONTEXT_RESCALE(subcontext, 0);
- }
- if (count > 0) { goto nextInput; }
- }
- ENET_CONTEXT_ENCODE(root, symbol, value, under, count, ENET_CONTEXT_SYMBOL_DELTA, ENET_CONTEXT_SYMBOL_MINIMUM);
- *parent = symbol - rangeCoder->symbols;
- parent = &symbol->parent;
- total = root->total;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > ENET_SUBCONTEXT_SYMBOL_DELTA + ENET_SUBCONTEXT_ESCAPE_DELTA) {
- ENET_CONTEXT_ENCODE_EXCLUDE(childContext, value, under, total, ENET_CONTEXT_SYMBOL_MINIMUM);
- }
- # endif
- ENET_RANGE_CODER_ENCODE(root->escapes + under, count, total);
- root->total += ENET_CONTEXT_SYMBOL_DELTA;
- if (count > 0xFF - 2 * ENET_CONTEXT_SYMBOL_DELTA + ENET_CONTEXT_SYMBOL_MINIMUM ||
- root->total > ENET_RANGE_CODER_BOTTOM - 0x100)
- {
- ENET_CONTEXT_RESCALE(root, ENET_CONTEXT_SYMBOL_MINIMUM);
- }
- nextInput:
- if (order >= ENET_SUBCONTEXT_ORDER) {
- predicted = rangeCoder->symbols [predicted].parent;
- } else {
- order++;
- }
- ENET_RANGE_CODER_FREE_SYMBOLS;
- }
- ENET_RANGE_CODER_FLUSH;
- return (size_t) (outData - outStart);
- } // enet_range_coder_compress
- # define ENET_RANGE_CODER_SEED \
- { \
- if (inData < inEnd) { decodeCode |= *inData++ << 24; } \
- if (inData < inEnd) { decodeCode |= *inData++ << 16; } \
- if (inData < inEnd) { decodeCode |= *inData++ << 8; } \
- if (inData < inEnd) { decodeCode |= *inData++; } \
- }
- # define ENET_RANGE_CODER_READ(total) ((decodeCode - decodeLow) / (decodeRange /= (total)))
- # define ENET_RANGE_CODER_DECODE(under, count, total) \
- { \
- decodeLow += (under) * decodeRange; \
- decodeRange *= (count); \
- for (;;) \
- { \
- if ((decodeLow ^ (decodeLow + decodeRange)) >= ENET_RANGE_CODER_TOP) \
- { \
- if (decodeRange >= ENET_RANGE_CODER_BOTTOM) { break; } \
- decodeRange = -decodeLow & (ENET_RANGE_CODER_BOTTOM - 1); \
- } \
- decodeCode <<= 8; \
- if (inData < inEnd) { \
- decodeCode |= *inData++; } \
- decodeRange <<= 8; \
- decodeLow <<= 8; \
- } \
- }
- # define ENET_CONTEXT_DECODE(context, symbol_, code, value_, under_, count_, update, minimum, createRoot, visitNode, \
- createRight, createLeft) \
- { \
- under_ = 0; \
- count_ = minimum; \
- if (!(context)->symbols) \
- { \
- createRoot; \
- } \
- else \
- { \
- ENetSymbol *node = (context) + (context)->symbols; \
- for (;;) \
- { \
- enet_uint16 after = under_ + node->under + (node->value + 1) * minimum, before = node->count + minimum; \
- visitNode; \
- if (code >= after) \
- { \
- under_ += node->under; \
- if (node->right) { node += node->right; continue; } \
- createRight; \
- } \
- else if (code < after - before) \
- { \
- node->under += update; \
- if (node->left) { node += node->left; continue; } \
- createLeft; \
- } \
- else \
- { \
- value_ = node->value; \
- count_ += node->count; \
- under_ = after - before; \
- node->under += update; \
- node->count += update; \
- symbol_ = node; \
- } \
- break; \
- } \
- } \
- }
- # define ENET_CONTEXT_TRY_DECODE(context, symbol_, code, value_, under_, count_, update, minimum, exclude) \
- ENET_CONTEXT_DECODE(context, symbol_, code, value_, under_, count_, update, minimum, return 0, \
- exclude(node->value, after, before), return 0, return 0)
- # define ENET_CONTEXT_ROOT_DECODE(context, symbol_, code, value_, under_, count_, update, minimum, exclude) \
- ENET_CONTEXT_DECODE(context, symbol_, code, value_, under_, count_, update, minimum, \
- { \
- value_ = code / minimum; \
- under_ = code - code % minimum; \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- (context)->symbols = symbol_ - (context); \
- }, \
- exclude(node->value, after, before), \
- { \
- value_ = node->value + 1 + (code - after) / minimum; \
- under_ = code - (code - after) % minimum; \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- node->right = symbol_ - node; \
- }, \
- { \
- value_ = node->value - 1 - (after - before - code - 1) / minimum; \
- under_ = code - (after - before - code - 1) % minimum; \
- ENET_SYMBOL_CREATE(symbol_, value_, update); \
- node->left = symbol_ - node; \
- }) \
- # ifdef ENET_CONTEXT_EXCLUSION
- typedef struct _ENetExclude {
- enet_uint8 value;
- enet_uint16 under;
- } ENetExclude;
- # define ENET_CONTEXT_DECODE_EXCLUDE(context, total, minimum) \
- { \
- enet_uint16 under = 0; \
- nextExclude = excludes; \
- ENET_CONTEXT_WALK(context, { \
- under += rangeCoder->symbols [node->parent].count + minimum; \
- nextExclude->value = node->value; \
- nextExclude->under = under; \
- nextExclude++; \
- }); \
- total -= under; \
- }
- # define ENET_CONTEXT_EXCLUDED(value_, after, before) \
- { \
- size_t low = 0, high = nextExclude - excludes; \
- for (;;) \
- { \
- size_t mid = (low + high) >> 1; \
- const ENetExclude *exclude = &excludes [mid]; \
- if (value_ < exclude->value) \
- { \
- if (low + 1 < high) \
- { \
- high = mid; \
- continue; \
- } \
- if (exclude > excludes) { \
- after -= exclude [-1].under; } \
- } \
- else \
- { \
- if (value_ > exclude->value) \
- { \
- if (low + 1 < high) \
- { \
- low = mid; \
- continue; \
- } \
- } \
- else { \
- before = 0; } \
- after -= exclude->under; \
- } \
- break; \
- } \
- }
- # endif // ifdef ENET_CONTEXT_EXCLUSION
- # define ENET_CONTEXT_NOT_EXCLUDED(value_, after, before)
- size_t enet_range_coder_decompress(void *context, const enet_uint8 *inData, size_t inLimit, enet_uint8 *outData,
- size_t outLimit) {
- ENetRangeCoder *rangeCoder = (ENetRangeCoder *) context;
- enet_uint8 *outStart = outData, *outEnd = &outData [outLimit];
- const enet_uint8 *inEnd = &inData [inLimit];
- enet_uint32 decodeLow = 0, decodeCode = 0, decodeRange = ~0;
- ENetSymbol *root;
- enet_uint16 predicted = 0;
- size_t order = 0, nextSymbol = 0;
- # ifdef ENET_CONTEXT_EXCLUSION
- ENetExclude excludes [256];
- ENetExclude *nextExclude = excludes;
- # endif
- if (rangeCoder == NULL || inLimit <= 0) {
- return 0;
- }
- ENET_CONTEXT_CREATE(root, ENET_CONTEXT_ESCAPE_MINIMUM, ENET_CONTEXT_SYMBOL_MINIMUM);
- ENET_RANGE_CODER_SEED;
- for (;;) {
- ENetSymbol *subcontext, *symbol, *patch;
- # ifdef ENET_CONTEXT_EXCLUSION
- const ENetSymbol *childContext = &emptyContext;
- # endif
- enet_uint8 value = 0;
- enet_uint16 code, under, count, bottom, *parent = &predicted, total;
- for (subcontext = &rangeCoder->symbols [predicted];
- subcontext != root;
- # ifdef ENET_CONTEXT_EXCLUSION
- childContext = subcontext,
- # endif
- subcontext = &rangeCoder->symbols [subcontext->parent])
- {
- if (subcontext->escapes <= 0) {
- continue;
- }
- total = subcontext->total;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > 0) {
- ENET_CONTEXT_DECODE_EXCLUDE(childContext, total, 0);
- }
- # endif
- if (subcontext->escapes >= total) {
- continue;
- }
- code = ENET_RANGE_CODER_READ(total);
- if (code < subcontext->escapes) {
- ENET_RANGE_CODER_DECODE(0, subcontext->escapes, total);
- continue;
- }
- code -= subcontext->escapes;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > 0) {
- ENET_CONTEXT_TRY_DECODE(subcontext, symbol, code, value, under, count, ENET_SUBCONTEXT_SYMBOL_DELTA, 0,
- ENET_CONTEXT_EXCLUDED);
- } else
- # endif
- {
- ENET_CONTEXT_TRY_DECODE(subcontext, symbol, code, value, under, count, ENET_SUBCONTEXT_SYMBOL_DELTA, 0,
- ENET_CONTEXT_NOT_EXCLUDED);
- }
- bottom = symbol - rangeCoder->symbols;
- ENET_RANGE_CODER_DECODE(subcontext->escapes + under, count, total);
- subcontext->total += ENET_SUBCONTEXT_SYMBOL_DELTA;
- if (count > 0xFF - 2 * ENET_SUBCONTEXT_SYMBOL_DELTA ||
- subcontext->total > ENET_RANGE_CODER_BOTTOM - 0x100)
- {
- ENET_CONTEXT_RESCALE(subcontext, 0);
- }
- goto patchContexts;
- }
- total = root->total;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > 0) {
- ENET_CONTEXT_DECODE_EXCLUDE(childContext, total, ENET_CONTEXT_SYMBOL_MINIMUM);
- }
- # endif
- code = ENET_RANGE_CODER_READ(total);
- if (code < root->escapes) {
- ENET_RANGE_CODER_DECODE(0, root->escapes, total);
- break;
- }
- code -= root->escapes;
- # ifdef ENET_CONTEXT_EXCLUSION
- if (childContext->total > 0) {
- ENET_CONTEXT_ROOT_DECODE(root, symbol, code, value, under, count, ENET_CONTEXT_SYMBOL_DELTA,
- ENET_CONTEXT_SYMBOL_MINIMUM, ENET_CONTEXT_EXCLUDED);
- } else
- # endif
- {
- ENET_CONTEXT_ROOT_DECODE(root, symbol, code, value, under, count, ENET_CONTEXT_SYMBOL_DELTA,
- ENET_CONTEXT_SYMBOL_MINIMUM, ENET_CONTEXT_NOT_EXCLUDED);
- }
- bottom = symbol - rangeCoder->symbols;
- ENET_RANGE_CODER_DECODE(root->escapes + under, count, total);
- root->total += ENET_CONTEXT_SYMBOL_DELTA;
- if (count > 0xFF - 2 * ENET_CONTEXT_SYMBOL_DELTA + ENET_CONTEXT_SYMBOL_MINIMUM ||
- root->total > ENET_RANGE_CODER_BOTTOM - 0x100)
- {
- ENET_CONTEXT_RESCALE(root, ENET_CONTEXT_SYMBOL_MINIMUM);
- }
- patchContexts:
- for (patch = &rangeCoder->symbols [predicted];
- patch != subcontext;
- patch = &rangeCoder->symbols [patch->parent])
- {
- ENET_CONTEXT_ENCODE(patch, symbol, value, under, count, ENET_SUBCONTEXT_SYMBOL_DELTA, 0);
- *parent = symbol - rangeCoder->symbols;
- parent = &symbol->parent;
- if (count <= 0) {
- patch->escapes += ENET_SUBCONTEXT_ESCAPE_DELTA;
- patch->total += ENET_SUBCONTEXT_ESCAPE_DELTA;
- }
- patch->total += ENET_SUBCONTEXT_SYMBOL_DELTA;
- if (count > 0xFF - 2 * ENET_SUBCONTEXT_SYMBOL_DELTA || patch->total > ENET_RANGE_CODER_BOTTOM - 0x100) {
- ENET_CONTEXT_RESCALE(patch, 0);
- }
- }
- *parent = bottom;
- ENET_RANGE_CODER_OUTPUT(value);
- if (order >= ENET_SUBCONTEXT_ORDER) {
- predicted = rangeCoder->symbols [predicted].parent;
- } else {
- order++;
- }
- ENET_RANGE_CODER_FREE_SYMBOLS;
- }
- return (size_t) (outData - outStart);
- } // enet_range_coder_decompress
- /** @defgroup host ENet host functions
- * @{
- */
- /** Sets the packet compressor the host should use to the default range coder.
- * @param host host to enable the range coder for
- * @returns 0 on success, < 0 on failure
- */
- int enet_host_compress_with_range_coder(ENetHost *host) {
- ENetCompressor compressor;
- memset(&compressor, 0, sizeof(compressor));
- compressor.context = enet_range_coder_create();
- if (compressor.context == NULL) {
- return -1;
- }
- compressor.compress = enet_range_coder_compress;
- compressor.decompress = enet_range_coder_decompress;
- compressor.destroy = enet_range_coder_destroy;
- enet_host_compress(host, &compressor);
- return 0;
- }
- /** @} */
- // @from_file: src/host.c
- /**
- * @file host.c
- * @brief ENet host management functions
- */
- # define ENET_BUILDING_LIB 1
- /** @defgroup host ENet host functions
- * @{
- */
- /** Creates a host for communicating to peers.
- *
- * @param address the address at which other peers may connect to this host. If NULL, then no peers may connect to the host.
- * @param peerCount the maximum number of peers that should be allocated for the host.
- * @param channelLimit the maximum number of channels allowed; if 0, then this is equivalent to ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT
- * @param incomingBandwidth downstream bandwidth of the host in bytes/second; if 0, ENet will assume unlimited bandwidth.
- * @param outgoingBandwidth upstream bandwidth of the host in bytes/second; if 0, ENet will assume unlimited bandwidth.
- *
- * @returns the host on success and NULL on failure
- *
- * @remarks ENet will strategically drop packets on specific sides of a connection between hosts
- * to ensure the host's bandwidth is not overwhelmed. The bandwidth parameters also determine
- * the window size of a connection which limits the amount of reliable packets that may be in transit
- * at any given time.
- */
- ENetHost * enet_host_create(const ENetAddress *address, size_t peerCount, size_t channelLimit,
- enet_uint32 incomingBandwidth, enet_uint32 outgoingBandwidth) {
- ENetHost *host;
- ENetPeer *currentPeer;
- if (peerCount > ENET_PROTOCOL_MAXIMUM_PEER_ID) {
- return NULL;
- }
- host = (ENetHost *) enet_malloc(sizeof(ENetHost));
- if (host == NULL) {
- return NULL;
- }
- memset(host, 0, sizeof(ENetHost));
- host->peers = (ENetPeer *) enet_malloc(peerCount * sizeof(ENetPeer));
- if (host->peers == NULL) {
- enet_free(host);
- return NULL;
- }
- memset(host->peers, 0, peerCount * sizeof(ENetPeer));
- host->socket = enet_socket_create(ENET_SOCKET_TYPE_DATAGRAM);
- if (host->socket == ENET_SOCKET_NULL || (address != NULL && enet_socket_bind(host->socket, address) < 0)) {
- if (host->socket != ENET_SOCKET_NULL) {
- enet_socket_destroy(host->socket);
- }
- enet_free(host->peers);
- enet_free(host);
- return NULL;
- }
- enet_socket_set_option(host->socket, ENET_SOCKOPT_NONBLOCK, 1);
- enet_socket_set_option(host->socket, ENET_SOCKOPT_BROADCAST, 1);
- enet_socket_set_option(host->socket, ENET_SOCKOPT_RCVBUF, ENET_HOST_RECEIVE_BUFFER_SIZE);
- enet_socket_set_option(host->socket, ENET_SOCKOPT_SNDBUF, ENET_HOST_SEND_BUFFER_SIZE);
- if (address != NULL && enet_socket_get_address(host->socket, &host->address) < 0) {
- host->address = *address;
- }
- if (!channelLimit || channelLimit > ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT) {
- channelLimit = ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT;
- } else if (channelLimit < ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT) {
- channelLimit = ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT;
- }
- host->randomSeed = (enet_uint32) (size_t) host;
- host->randomSeed += enet_host_random_seed();
- host->randomSeed = (host->randomSeed << 16) | (host->randomSeed >> 16);
- host->channelLimit = channelLimit;
- host->incomingBandwidth = incomingBandwidth;
- host->outgoingBandwidth = outgoingBandwidth;
- host->bandwidthThrottleEpoch = 0;
- host->recalculateBandwidthLimits = 0;
- host->mtu = ENET_HOST_DEFAULT_MTU;
- host->peerCount = peerCount;
- host->commandCount = 0;
- host->bufferCount = 0;
- host->checksum = NULL;
- host->receivedAddress.host = ENET_HOST_ANY;
- host->receivedAddress.port = 0;
- host->receivedData = NULL;
- host->receivedDataLength = 0;
- host->totalSentData = 0;
- host->totalSentPackets = 0;
- host->totalReceivedData = 0;
- host->totalReceivedPackets = 0;
- host->connectedPeers = 0;
- host->bandwidthLimitedPeers = 0;
- host->duplicatePeers = ENET_PROTOCOL_MAXIMUM_PEER_ID;
- host->maximumPacketSize = ENET_HOST_DEFAULT_MAXIMUM_PACKET_SIZE;
- host->maximumWaitingData = ENET_HOST_DEFAULT_MAXIMUM_WAITING_DATA;
- host->compressor.context = NULL;
- host->compressor.compress = NULL;
- host->compressor.decompress = NULL;
- host->compressor.destroy = NULL;
- host->intercept = NULL;
- enet_list_clear(&host->dispatchQueue);
- for (currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- currentPeer->host = host;
- currentPeer->incomingPeerID = currentPeer - host->peers;
- currentPeer->outgoingSessionID = currentPeer->incomingSessionID = 0xFF;
- currentPeer->data = NULL;
- enet_list_clear(¤tPeer->acknowledgements);
- enet_list_clear(¤tPeer->sentReliableCommands);
- enet_list_clear(¤tPeer->sentUnreliableCommands);
- enet_list_clear(¤tPeer->outgoingReliableCommands);
- enet_list_clear(¤tPeer->outgoingUnreliableCommands);
- enet_list_clear(¤tPeer->dispatchedCommands);
- enet_peer_reset(currentPeer);
- }
- return host;
- } // enet_host_create
- /** Destroys the host and all resources associated with it.
- * @param host pointer to the host to destroy
- */
- void enet_host_destroy(ENetHost *host) {
- ENetPeer *currentPeer;
- if (host == NULL) {
- return;
- }
- enet_socket_destroy(host->socket);
- for (currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- enet_peer_reset(currentPeer);
- }
- if (host->compressor.context != NULL && host->compressor.destroy) {
- (*host->compressor.destroy)(host->compressor.context);
- }
- enet_free(host->peers);
- enet_free(host);
- }
- /** Initiates a connection to a foreign host.
- * @param host host seeking the connection
- * @param address destination for the connection
- * @param channelCount number of channels to allocate
- * @param data user data supplied to the receiving host
- * @returns a peer representing the foreign host on success, NULL on failure
- * @remarks The peer returned will have not completed the connection until enet_host_service()
- * notifies of an ENET_EVENT_TYPE_CONNECT event for the peer.
- */
- ENetPeer * enet_host_connect(ENetHost *host, const ENetAddress *address, size_t channelCount, enet_uint32 data) {
- ENetPeer *currentPeer;
- ENetChannel *channel;
- ENetProtocol command;
- if (channelCount < ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT) {
- channelCount = ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT;
- } else if (channelCount > ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT) {
- channelCount = ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT;
- }
- for (currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- if (currentPeer->state == ENET_PEER_STATE_DISCONNECTED) {
- break;
- }
- }
- if (currentPeer >= &host->peers [host->peerCount]) {
- return NULL;
- }
- currentPeer->channels = (ENetChannel *) enet_malloc(channelCount * sizeof(ENetChannel));
- if (currentPeer->channels == NULL) {
- return NULL;
- }
- currentPeer->channelCount = channelCount;
- currentPeer->state = ENET_PEER_STATE_CONNECTING;
- currentPeer->address = *address;
- currentPeer->connectID = ++host->randomSeed;
- if (host->outgoingBandwidth == 0) {
- currentPeer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- } else {
- currentPeer->windowSize = (host->outgoingBandwidth
- / ENET_PEER_WINDOW_SIZE_SCALE)
- * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- }
- if (currentPeer->windowSize < ENET_PROTOCOL_MINIMUM_WINDOW_SIZE) {
- currentPeer->windowSize = ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else if (currentPeer->windowSize > ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE) {
- currentPeer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- }
- for (channel = currentPeer->channels;
- channel < ¤tPeer->channels [channelCount];
- ++channel)
- {
- channel->outgoingReliableSequenceNumber = 0;
- channel->outgoingUnreliableSequenceNumber = 0;
- channel->incomingReliableSequenceNumber = 0;
- channel->incomingUnreliableSequenceNumber = 0;
- enet_list_clear(&channel->incomingReliableCommands);
- enet_list_clear(&channel->incomingUnreliableCommands);
- channel->usedReliableWindows = 0;
- memset(channel->reliableWindows, 0, sizeof(channel->reliableWindows));
- }
- command.header.command = ENET_PROTOCOL_COMMAND_CONNECT | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- command.header.channelID = 0xFF;
- command.connect.outgoingPeerID = ENET_HOST_TO_NET_16(currentPeer->incomingPeerID);
- command.connect.incomingSessionID = currentPeer->incomingSessionID;
- command.connect.outgoingSessionID = currentPeer->outgoingSessionID;
- command.connect.mtu = ENET_HOST_TO_NET_32(currentPeer->mtu);
- command.connect.windowSize = ENET_HOST_TO_NET_32(currentPeer->windowSize);
- command.connect.channelCount = ENET_HOST_TO_NET_32(channelCount);
- command.connect.incomingBandwidth = ENET_HOST_TO_NET_32(host->incomingBandwidth);
- command.connect.outgoingBandwidth = ENET_HOST_TO_NET_32(host->outgoingBandwidth);
- command.connect.packetThrottleInterval = ENET_HOST_TO_NET_32(currentPeer->packetThrottleInterval);
- command.connect.packetThrottleAcceleration = ENET_HOST_TO_NET_32(currentPeer->packetThrottleAcceleration);
- command.connect.packetThrottleDeceleration = ENET_HOST_TO_NET_32(currentPeer->packetThrottleDeceleration);
- command.connect.connectID = currentPeer->connectID;
- command.connect.data = ENET_HOST_TO_NET_32(data);
- enet_peer_queue_outgoing_command(currentPeer, &command, NULL, 0, 0);
- return currentPeer;
- } // enet_host_connect
- /** Queues a packet to be sent to all peers associated with the host.
- * @param host host on which to broadcast the packet
- * @param channelID channel on which to broadcast
- * @param packet packet to broadcast
- */
- void enet_host_broadcast(ENetHost *host, enet_uint8 channelID, ENetPacket *packet) {
- ENetPeer *currentPeer;
- for (currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- if (currentPeer->state != ENET_PEER_STATE_CONNECTED) {
- continue;
- }
- enet_peer_send(currentPeer, channelID, packet);
- }
- if (packet->referenceCount == 0) {
- enet_packet_destroy(packet);
- }
- }
- /** Sets the packet compressor the host should use to compress and decompress packets.
- * @param host host to enable or disable compression for
- * @param compressor callbacks for for the packet compressor; if NULL, then compression is disabled
- */
- void enet_host_compress(ENetHost *host, const ENetCompressor *compressor) {
- if (host->compressor.context != NULL && host->compressor.destroy) {
- (*host->compressor.destroy)(host->compressor.context);
- }
- if (compressor) {
- host->compressor = *compressor;
- } else {
- host->compressor.context = NULL;
- }
- }
- /** Limits the maximum allowed channels of future incoming connections.
- * @param host host to limit
- * @param channelLimit the maximum number of channels allowed; if 0, then this is equivalent to ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT
- */
- void enet_host_channel_limit(ENetHost *host, size_t channelLimit) {
- if (!channelLimit || channelLimit > ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT) {
- channelLimit = ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT;
- } else if (channelLimit < ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT) {
- channelLimit = ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT;
- }
- host->channelLimit = channelLimit;
- }
- /** Adjusts the bandwidth limits of a host.
- * @param host host to adjust
- * @param incomingBandwidth new incoming bandwidth
- * @param outgoingBandwidth new outgoing bandwidth
- * @remarks the incoming and outgoing bandwidth parameters are identical in function to those
- * specified in enet_host_create().
- */
- void enet_host_bandwidth_limit(ENetHost *host, enet_uint32 incomingBandwidth, enet_uint32 outgoingBandwidth) {
- host->incomingBandwidth = incomingBandwidth;
- host->outgoingBandwidth = outgoingBandwidth;
- host->recalculateBandwidthLimits = 1;
- }
- void enet_host_bandwidth_throttle(ENetHost *host) {
- enet_uint32 timeCurrent = enet_time_get(),
- elapsedTime = timeCurrent - host->bandwidthThrottleEpoch,
- peersRemaining = (enet_uint32) host->connectedPeers,
- dataTotal = ~0,
- bandwidth = ~0,
- throttle = 0,
- bandwidthLimit = 0;
- int needsAdjustment = host->bandwidthLimitedPeers > 0 ? 1 : 0;
- ENetPeer *peer;
- ENetProtocol command;
- if (elapsedTime < ENET_HOST_BANDWIDTH_THROTTLE_INTERVAL) {
- return;
- }
- host->bandwidthThrottleEpoch = timeCurrent;
- if (peersRemaining == 0) {
- return;
- }
- if (host->outgoingBandwidth != 0) {
- dataTotal = 0;
- bandwidth = (host->outgoingBandwidth * elapsedTime) / 1000;
- for (peer = host->peers;
- peer < &host->peers [host->peerCount];
- ++peer)
- {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- continue;
- }
- dataTotal += peer->outgoingDataTotal;
- }
- }
- while (peersRemaining > 0 && needsAdjustment != 0) {
- needsAdjustment = 0;
- if (dataTotal <= bandwidth) {
- throttle = ENET_PEER_PACKET_THROTTLE_SCALE;
- } else {
- throttle = (bandwidth * ENET_PEER_PACKET_THROTTLE_SCALE) / dataTotal;
- }
- for (peer = host->peers;
- peer < &host->peers [host->peerCount];
- ++peer)
- {
- enet_uint32 peerBandwidth;
- if ((peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) ||
- peer->incomingBandwidth == 0 ||
- peer->outgoingBandwidthThrottleEpoch == timeCurrent)
- {
- continue;
- }
- peerBandwidth = (peer->incomingBandwidth * elapsedTime) / 1000;
- if ((throttle * peer->outgoingDataTotal) / ENET_PEER_PACKET_THROTTLE_SCALE <= peerBandwidth) {
- continue;
- }
- peer->packetThrottleLimit = (peerBandwidth
- * ENET_PEER_PACKET_THROTTLE_SCALE) / peer->outgoingDataTotal;
- if (peer->packetThrottleLimit == 0) {
- peer->packetThrottleLimit = 1;
- }
- if (peer->packetThrottle > peer->packetThrottleLimit) {
- peer->packetThrottle = peer->packetThrottleLimit;
- }
- peer->outgoingBandwidthThrottleEpoch = timeCurrent;
- peer->incomingDataTotal = 0;
- peer->outgoingDataTotal = 0;
- needsAdjustment = 1;
- --peersRemaining;
- bandwidth -= peerBandwidth;
- dataTotal -= peerBandwidth;
- }
- }
- if (peersRemaining > 0) {
- if (dataTotal <= bandwidth) {
- throttle = ENET_PEER_PACKET_THROTTLE_SCALE;
- } else {
- throttle = (bandwidth * ENET_PEER_PACKET_THROTTLE_SCALE) / dataTotal;
- }
- for (peer = host->peers;
- peer < &host->peers [host->peerCount];
- ++peer)
- {
- if ((peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) ||
- peer->outgoingBandwidthThrottleEpoch == timeCurrent)
- {
- continue;
- }
- peer->packetThrottleLimit = throttle;
- if (peer->packetThrottle > peer->packetThrottleLimit) {
- peer->packetThrottle = peer->packetThrottleLimit;
- }
- peer->incomingDataTotal = 0;
- peer->outgoingDataTotal = 0;
- }
- }
- if (host->recalculateBandwidthLimits) {
- host->recalculateBandwidthLimits = 0;
- peersRemaining = (enet_uint32) host->connectedPeers;
- bandwidth = host->incomingBandwidth;
- needsAdjustment = 1;
- if (bandwidth == 0) {
- bandwidthLimit = 0;
- } else {
- while (peersRemaining > 0 && needsAdjustment != 0) {
- needsAdjustment = 0;
- bandwidthLimit = bandwidth / peersRemaining;
- for (peer = host->peers;
- peer < &host->peers [host->peerCount];
- ++peer)
- {
- if ((peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) ||
- peer->incomingBandwidthThrottleEpoch == timeCurrent)
- {
- continue;
- }
- if (peer->outgoingBandwidth > 0 &&
- peer->outgoingBandwidth >= bandwidthLimit)
- {
- continue;
- }
- peer->incomingBandwidthThrottleEpoch = timeCurrent;
- needsAdjustment = 1;
- --peersRemaining;
- bandwidth -= peer->outgoingBandwidth;
- }
- }
- }
- for (peer = host->peers;
- peer < &host->peers [host->peerCount];
- ++peer)
- {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- continue;
- }
- command.header.command = ENET_PROTOCOL_COMMAND_BANDWIDTH_LIMIT | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- command.header.channelID = 0xFF;
- command.bandwidthLimit.outgoingBandwidth = ENET_HOST_TO_NET_32(host->outgoingBandwidth);
- if (peer->incomingBandwidthThrottleEpoch == timeCurrent) {
- command.bandwidthLimit.incomingBandwidth = ENET_HOST_TO_NET_32(peer->outgoingBandwidth);
- } else {
- command.bandwidthLimit.incomingBandwidth = ENET_HOST_TO_NET_32(bandwidthLimit);
- }
- enet_peer_queue_outgoing_command(peer, &command, NULL, 0, 0);
- }
- }
- } // enet_host_bandwidth_throttle
- /** @} */
- // @from_file: src/list.c
- /**
- * @file list.c
- * @brief ENet linked list functions
- */
- # define ENET_BUILDING_LIB 1
- /**
- * @defgroup list ENet linked list utility functions
- * @ingroup private
- * @{
- */
- void enet_list_clear(ENetList *list) {
- list->sentinel.next = &list->sentinel;
- list->sentinel.previous = &list->sentinel;
- }
- ENetListIterator enet_list_insert(ENetListIterator position, void *data) {
- ENetListIterator result = (ENetListIterator) data;
- result->previous = position->previous;
- result->next = position;
- result->previous->next = result;
- position->previous = result;
- return result;
- }
- void * enet_list_remove(ENetListIterator position) {
- position->previous->next = position->next;
- position->next->previous = position->previous;
- return position;
- }
- ENetListIterator enet_list_move(ENetListIterator position, void *dataFirst, void *dataLast) {
- ENetListIterator first = (ENetListIterator) dataFirst,
- last = (ENetListIterator) dataLast;
- first->previous->next = last->next;
- last->next->previous = first->previous;
- first->previous = position->previous;
- last->next = position;
- first->previous->next = first;
- position->previous = last;
- return first;
- }
- size_t enet_list_size(ENetList *list) {
- size_t size = 0;
- ENetListIterator position;
- for (position = enet_list_begin(list);
- position != enet_list_end(list);
- position = enet_list_next(position))
- ++size;
- return size;
- }
- /** @} */
- // @from_file: src/packet.c
- /**
- * @file packet.c
- * @brief ENet packet management functions
- */
- # define ENET_BUILDING_LIB 1
- /** @defgroup Packet ENet packet functions
- * @{
- */
- /** Creates a packet that may be sent to a peer.
- * @param data initial contents of the packet's data; the packet's data will remain uninitialized if data is NULL.
- * @param dataLength size of the data allocated for this packet
- * @param flags flags for this packet as described for the ENetPacket structure.
- * @returns the packet on success, NULL on failure
- */
- ENetPacket * enet_packet_create(const void *data, size_t dataLength, enet_uint32 flags) {
- ENetPacket *packet = (ENetPacket *) enet_malloc(sizeof(ENetPacket));
- if (packet == NULL) {
- return NULL;
- }
- if (flags & ENET_PACKET_FLAG_NO_ALLOCATE) {
- packet->data = (enet_uint8 *) data;
- } else if (dataLength <= 0) {
- packet->data = NULL;
- } else {
- packet->data = (enet_uint8 *) enet_malloc(dataLength);
- if (packet->data == NULL) {
- enet_free(packet);
- return NULL;
- }
- if (data != NULL) {
- memcpy(packet->data, data, dataLength);
- }
- }
- packet->referenceCount = 0;
- packet->flags = flags;
- packet->dataLength = dataLength;
- packet->freeCallback = NULL;
- packet->userData = NULL;
- return packet;
- }
- ENetPacket * enet_packet_create_offset(const void *data, size_t dataLength, size_t dataOffset, enet_uint32 flags) {
- ENetPacket *packet = (ENetPacket *) enet_malloc(sizeof(ENetPacket));
- if (packet == NULL) {
- return NULL;
- }
- if (flags & ENET_PACKET_FLAG_NO_ALLOCATE) {
- packet->data = (enet_uint8 *) data;
- } else if ((dataLength + dataOffset) <= 0) {
- packet->data = NULL;
- } else {
- packet->data = (enet_uint8 *) enet_malloc(dataLength + dataOffset);
- if (packet->data == NULL) {
- enet_free(packet);
- return NULL;
- }
- if (data != NULL) {
- memcpy(packet->data + dataOffset, data, dataLength);
- }
- }
- packet->referenceCount = 0;
- packet->flags = flags;
- packet->dataLength = dataLength + dataOffset;
- packet->freeCallback = NULL;
- packet->userData = NULL;
- return packet;
- }
- /** Destroys the packet and deallocates its data.
- * @param packet packet to be destroyed
- */
- void enet_packet_destroy(ENetPacket *packet) {
- if (packet == NULL) {
- return;
- }
- if (packet->freeCallback != NULL) {
- (*packet->freeCallback)(packet);
- }
- if (!(packet->flags & ENET_PACKET_FLAG_NO_ALLOCATE) &&
- packet->data != NULL)
- {
- enet_free(packet->data);
- }
- enet_free(packet);
- }
- /** Attempts to resize the data in the packet to length specified in the
- * dataLength parameter
- * @param packet packet to resize
- * @param dataLength new size for the packet data
- * @returns 0 on success, < 0 on failure
- */
- int enet_packet_resize(ENetPacket *packet, size_t dataLength) {
- enet_uint8 *newData;
- if (dataLength <= packet->dataLength || (packet->flags & ENET_PACKET_FLAG_NO_ALLOCATE)) {
- packet->dataLength = dataLength;
- return 0;
- }
- newData = (enet_uint8 *) enet_malloc(dataLength);
- if (newData == NULL) {
- return -1;
- }
- memcpy(newData, packet->data, packet->dataLength);
- enet_free(packet->data);
- packet->data = newData;
- packet->dataLength = dataLength;
- return 0;
- }
- static int initializedCRC32 = 0;
- static enet_uint32 crcTable [256];
- static enet_uint32 reflect_crc(int val, int bits) {
- int result = 0, bit;
- for (bit = 0; bit < bits; bit++) {
- if (val & 1) { result |= 1 << (bits - 1 - bit); }
- val >>= 1;
- }
- return result;
- }
- static void initialize_crc32(void) {
- int byte;
- for (byte = 0; byte < 256; ++byte) {
- enet_uint32 crc = reflect_crc(byte, 8) << 24;
- int offset;
- for (offset = 0; offset < 8; ++offset) {
- if (crc & 0x80000000) {
- crc = (crc << 1) ^ 0x04c11db7;
- } else {
- crc <<= 1;
- }
- }
- crcTable [byte] = reflect_crc(crc, 32);
- }
- initializedCRC32 = 1;
- }
- enet_uint32 enet_crc32(const ENetBuffer *buffers, size_t bufferCount) {
- enet_uint32 crc = 0xFFFFFFFF;
- if (!initializedCRC32) { initialize_crc32(); }
- while (bufferCount-- > 0) {
- const enet_uint8 *data = (const enet_uint8 *) buffers->data,
- *dataEnd = &data [buffers->dataLength];
- while (data < dataEnd) {
- crc = (crc >> 8) ^ crcTable [(crc & 0xFF) ^ *data++];
- }
- ++buffers;
- }
- return ENET_HOST_TO_NET_32(~crc);
- }
- /** @} */
- // @from_file: src/peer.c
- /**
- * @file peer.c
- * @brief ENet peer management functions
- */
- # define ENET_BUILDING_LIB 1
- /** @defgroup peer ENet peer functions
- * @{
- */
- /** Configures throttle parameter for a peer.
- *
- * Unreliable packets are dropped by ENet in response to the varying conditions
- * of the Internet connection to the peer. The throttle represents a probability
- * that an unreliable packet should not be dropped and thus sent by ENet to the peer.
- * The lowest mean round trip time from the sending of a reliable packet to the
- * receipt of its acknowledgement is measured over an amount of time specified by
- * the interval parameter in milliseconds. If a measured round trip time happens to
- * be significantly less than the mean round trip time measured over the interval,
- * then the throttle probability is increased to allow more traffic by an amount
- * specified in the acceleration parameter, which is a ratio to the ENET_PEER_PACKET_THROTTLE_SCALE
- * constant. If a measured round trip time happens to be significantly greater than
- * the mean round trip time measured over the interval, then the throttle probability
- * is decreased to limit traffic by an amount specified in the deceleration parameter, which
- * is a ratio to the ENET_PEER_PACKET_THROTTLE_SCALE constant. When the throttle has
- * a value of ENET_PEER_PACKET_THROTTLE_SCALE, no unreliable packets are dropped by
- * ENet, and so 100% of all unreliable packets will be sent. When the throttle has a
- * value of 0, all unreliable packets are dropped by ENet, and so 0% of all unreliable
- * packets will be sent. Intermediate values for the throttle represent intermediate
- * probabilities between 0% and 100% of unreliable packets being sent. The bandwidth
- * limits of the local and foreign hosts are taken into account to determine a
- * sensible limit for the throttle probability above which it should not raise even in
- * the best of conditions.
- *
- * @param peer peer to configure
- * @param interval interval, in milliseconds, over which to measure lowest mean RTT; the default value is ENET_PEER_PACKET_THROTTLE_INTERVAL.
- * @param acceleration rate at which to increase the throttle probability as mean RTT declines
- * @param deceleration rate at which to decrease the throttle probability as mean RTT increases
- */
- void enet_peer_throttle_configure(ENetPeer *peer, enet_uint32 interval, enet_uint32 acceleration,
- enet_uint32 deceleration) {
- ENetProtocol command;
- peer->packetThrottleInterval = interval;
- peer->packetThrottleAcceleration = acceleration;
- peer->packetThrottleDeceleration = deceleration;
- command.header.command = ENET_PROTOCOL_COMMAND_THROTTLE_CONFIGURE | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- command.header.channelID = 0xFF;
- command.throttleConfigure.packetThrottleInterval = ENET_HOST_TO_NET_32(interval);
- command.throttleConfigure.packetThrottleAcceleration = ENET_HOST_TO_NET_32(acceleration);
- command.throttleConfigure.packetThrottleDeceleration = ENET_HOST_TO_NET_32(deceleration);
- enet_peer_queue_outgoing_command(peer, &command, NULL, 0, 0);
- }
- int enet_peer_throttle(ENetPeer *peer, enet_uint32 rtt) {
- if (peer->lastRoundTripTime <= peer->lastRoundTripTimeVariance) {
- peer->packetThrottle = peer->packetThrottleLimit;
- } else if (rtt < peer->lastRoundTripTime) {
- peer->packetThrottle += peer->packetThrottleAcceleration;
- if (peer->packetThrottle > peer->packetThrottleLimit) {
- peer->packetThrottle = peer->packetThrottleLimit;
- }
- return 1;
- } else if (rtt > peer->lastRoundTripTime + 2 * peer->lastRoundTripTimeVariance) {
- if (peer->packetThrottle > peer->packetThrottleDeceleration) {
- peer->packetThrottle -= peer->packetThrottleDeceleration;
- } else {
- peer->packetThrottle = 0;
- }
- return -1;
- }
- return 0;
- }
- /** Queues a packet to be sent.
- * @param peer destination for the packet
- * @param channelID channel on which to send
- * @param packet packet to send
- * @retval 0 on success
- * @retval < 0 on failure
- */
- int enet_peer_send(ENetPeer *peer, enet_uint8 channelID, ENetPacket *packet) {
- ENetChannel *channel = &peer->channels [channelID];
- ENetProtocol command;
- size_t fragmentLength;
- if (peer->state != ENET_PEER_STATE_CONNECTED ||
- channelID >= peer->channelCount ||
- packet->dataLength > peer->host->maximumPacketSize)
- {
- return -1;
- }
- fragmentLength = peer->mtu - sizeof(ENetProtocolHeader) - sizeof(ENetProtocolSendFragment);
- if (peer->host->checksum != NULL) {
- fragmentLength -= sizeof(enet_uint32);
- }
- if (packet->dataLength > fragmentLength) {
- enet_uint32 fragmentCount = (packet->dataLength + fragmentLength - 1) / fragmentLength,
- fragmentNumber,
- fragmentOffset;
- enet_uint8 commandNumber;
- enet_uint16 startSequenceNumber;
- ENetList fragments;
- ENetOutgoingCommand *fragment;
- if (fragmentCount > ENET_PROTOCOL_MAXIMUM_FRAGMENT_COUNT) {
- return -1;
- }
- if ((packet->flags & (ENET_PACKET_FLAG_RELIABLE | ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT)) ==
- ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT &&
- channel->outgoingUnreliableSequenceNumber < 0xFFFF)
- {
- commandNumber = ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE_FRAGMENT;
- startSequenceNumber = ENET_HOST_TO_NET_16(channel->outgoingUnreliableSequenceNumber + 1);
- } else {
- commandNumber = ENET_PROTOCOL_COMMAND_SEND_FRAGMENT | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- startSequenceNumber = ENET_HOST_TO_NET_16(channel->outgoingReliableSequenceNumber + 1);
- }
- enet_list_clear(&fragments);
- for (fragmentNumber = 0,
- fragmentOffset = 0;
- fragmentOffset < packet->dataLength;
- ++fragmentNumber,
- fragmentOffset += fragmentLength)
- {
- if (packet->dataLength - fragmentOffset < fragmentLength) {
- fragmentLength = packet->dataLength - fragmentOffset;
- }
- fragment = (ENetOutgoingCommand *) enet_malloc(sizeof(ENetOutgoingCommand));
- if (fragment == NULL) {
- while (!enet_list_empty(&fragments)) {
- fragment = (ENetOutgoingCommand *) enet_list_remove(enet_list_begin(&fragments));
- enet_free(fragment);
- }
- return -1;
- }
- fragment->fragmentOffset = fragmentOffset;
- fragment->fragmentLength = fragmentLength;
- fragment->packet = packet;
- fragment->command.header.command = commandNumber;
- fragment->command.header.channelID = channelID;
- fragment->command.sendFragment.startSequenceNumber = startSequenceNumber;
- fragment->command.sendFragment.dataLength = ENET_HOST_TO_NET_16(fragmentLength);
- fragment->command.sendFragment.fragmentCount = ENET_HOST_TO_NET_32(fragmentCount);
- fragment->command.sendFragment.fragmentNumber = ENET_HOST_TO_NET_32(fragmentNumber);
- fragment->command.sendFragment.totalLength = ENET_HOST_TO_NET_32(packet->dataLength);
- fragment->command.sendFragment.fragmentOffset = ENET_NET_TO_HOST_32(fragmentOffset);
- enet_list_insert(enet_list_end(&fragments), fragment);
- }
- packet->referenceCount += fragmentNumber;
- while (!enet_list_empty(&fragments)) {
- fragment = (ENetOutgoingCommand *) enet_list_remove(enet_list_begin(&fragments));
- enet_peer_setup_outgoing_command(peer, fragment);
- }
- return 0;
- }
- command.header.channelID = channelID;
- if ((packet->flags & (ENET_PACKET_FLAG_RELIABLE | ENET_PACKET_FLAG_UNSEQUENCED)) == ENET_PACKET_FLAG_UNSEQUENCED) {
- command.header.command = ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED | ENET_PROTOCOL_COMMAND_FLAG_UNSEQUENCED;
- command.sendUnsequenced.dataLength = ENET_HOST_TO_NET_16(packet->dataLength);
- } else if (packet->flags & ENET_PACKET_FLAG_RELIABLE || channel->outgoingUnreliableSequenceNumber >= 0xFFFF) {
- command.header.command = ENET_PROTOCOL_COMMAND_SEND_RELIABLE | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- command.sendReliable.dataLength = ENET_HOST_TO_NET_16(packet->dataLength);
- } else {
- command.header.command = ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE;
- command.sendUnreliable.dataLength = ENET_HOST_TO_NET_16(packet->dataLength);
- }
- if (enet_peer_queue_outgoing_command(peer, &command, packet, 0, packet->dataLength) == NULL) {
- return -1;
- }
- return 0;
- } // enet_peer_send
- /** Attempts to dequeue any incoming queued packet.
- * @param peer peer to dequeue packets from
- * @param channelID holds the channel ID of the channel the packet was received on success
- * @returns a pointer to the packet, or NULL if there are no available incoming queued packets
- */
- ENetPacket * enet_peer_receive(ENetPeer *peer, enet_uint8 *channelID) {
- ENetIncomingCommand *incomingCommand;
- ENetPacket *packet;
- if (enet_list_empty(&peer->dispatchedCommands)) {
- return NULL;
- }
- incomingCommand = (ENetIncomingCommand *) enet_list_remove(enet_list_begin(&peer->dispatchedCommands));
- if (channelID != NULL) {
- *channelID = incomingCommand->command.header.channelID;
- }
- packet = incomingCommand->packet;
- --packet->referenceCount;
- if (incomingCommand->fragments != NULL) {
- enet_free(incomingCommand->fragments);
- }
- enet_free(incomingCommand);
- peer->totalWaitingData -= packet->dataLength;
- return packet;
- }
- static void enet_peer_reset_outgoing_commands(ENetList *queue) {
- ENetOutgoingCommand *outgoingCommand;
- while (!enet_list_empty(queue)) {
- outgoingCommand = (ENetOutgoingCommand *) enet_list_remove(enet_list_begin(queue));
- if (outgoingCommand->packet != NULL) {
- --outgoingCommand->packet->referenceCount;
- if (outgoingCommand->packet->referenceCount == 0) {
- enet_packet_destroy(outgoingCommand->packet);
- }
- }
- enet_free(outgoingCommand);
- }
- }
- static void enet_peer_remove_incoming_commands(ENetList *queue, ENetListIterator startCommand,
- ENetListIterator endCommand) {
- ENetListIterator currentCommand;
- for (currentCommand = startCommand; currentCommand != endCommand;) {
- ENetIncomingCommand *incomingCommand = (ENetIncomingCommand *) currentCommand;
- currentCommand = enet_list_next(currentCommand);
- enet_list_remove(&incomingCommand->incomingCommandList);
- if (incomingCommand->packet != NULL) {
- --incomingCommand->packet->referenceCount;
- if (incomingCommand->packet->referenceCount == 0) {
- enet_packet_destroy(incomingCommand->packet);
- }
- }
- if (incomingCommand->fragments != NULL) {
- enet_free(incomingCommand->fragments);
- }
- enet_free(incomingCommand);
- }
- }
- static void enet_peer_reset_incoming_commands(ENetList *queue) {
- enet_peer_remove_incoming_commands(queue, enet_list_begin(queue), enet_list_end(queue));
- }
- void enet_peer_reset_queues(ENetPeer *peer) {
- ENetChannel *channel;
- if (peer->needsDispatch) {
- enet_list_remove(&peer->dispatchList);
- peer->needsDispatch = 0;
- }
- while (!enet_list_empty(&peer->acknowledgements))
- enet_free(enet_list_remove(enet_list_begin(&peer->acknowledgements)));
- enet_peer_reset_outgoing_commands(&peer->sentReliableCommands);
- enet_peer_reset_outgoing_commands(&peer->sentUnreliableCommands);
- enet_peer_reset_outgoing_commands(&peer->outgoingReliableCommands);
- enet_peer_reset_outgoing_commands(&peer->outgoingUnreliableCommands);
- enet_peer_reset_incoming_commands(&peer->dispatchedCommands);
- if (peer->channels != NULL && peer->channelCount > 0) {
- for (channel = peer->channels;
- channel < &peer->channels [peer->channelCount];
- ++channel)
- {
- enet_peer_reset_incoming_commands(&channel->incomingReliableCommands);
- enet_peer_reset_incoming_commands(&channel->incomingUnreliableCommands);
- }
- enet_free(peer->channels);
- }
- peer->channels = NULL;
- peer->channelCount = 0;
- } // enet_peer_reset_queues
- void enet_peer_on_connect(ENetPeer *peer) {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- if (peer->incomingBandwidth != 0) {
- ++peer->host->bandwidthLimitedPeers;
- }
- ++peer->host->connectedPeers;
- }
- }
- void enet_peer_on_disconnect(ENetPeer *peer) {
- if (peer->state == ENET_PEER_STATE_CONNECTED || peer->state == ENET_PEER_STATE_DISCONNECT_LATER) {
- if (peer->incomingBandwidth != 0) {
- --peer->host->bandwidthLimitedPeers;
- }
- --peer->host->connectedPeers;
- }
- }
- /** Forcefully disconnects a peer.
- * @param peer peer to forcefully disconnect
- * @remarks The foreign host represented by the peer is not notified of the disconnection and will timeout
- * on its connection to the local host.
- */
- void enet_peer_reset(ENetPeer *peer) {
- enet_peer_on_disconnect(peer);
- peer->outgoingPeerID = ENET_PROTOCOL_MAXIMUM_PEER_ID;
- peer->connectID = 0;
- peer->state = ENET_PEER_STATE_DISCONNECTED;
- peer->incomingBandwidth = 0;
- peer->outgoingBandwidth = 0;
- peer->incomingBandwidthThrottleEpoch = 0;
- peer->outgoingBandwidthThrottleEpoch = 0;
- peer->incomingDataTotal = 0;
- peer->outgoingDataTotal = 0;
- peer->lastSendTime = 0;
- peer->lastReceiveTime = 0;
- peer->nextTimeout = 0;
- peer->earliestTimeout = 0;
- peer->packetLossEpoch = 0;
- peer->packetsSent = 0;
- peer->packetsLost = 0;
- peer->packetLoss = 0;
- peer->packetLossVariance = 0;
- peer->packetThrottle = ENET_PEER_DEFAULT_PACKET_THROTTLE;
- peer->packetThrottleLimit = ENET_PEER_PACKET_THROTTLE_SCALE;
- peer->packetThrottleCounter = 0;
- peer->packetThrottleEpoch = 0;
- peer->packetThrottleAcceleration = ENET_PEER_PACKET_THROTTLE_ACCELERATION;
- peer->packetThrottleDeceleration = ENET_PEER_PACKET_THROTTLE_DECELERATION;
- peer->packetThrottleInterval = ENET_PEER_PACKET_THROTTLE_INTERVAL;
- peer->pingInterval = ENET_PEER_PING_INTERVAL;
- peer->timeoutLimit = ENET_PEER_TIMEOUT_LIMIT;
- peer->timeoutMinimum = ENET_PEER_TIMEOUT_MINIMUM;
- peer->timeoutMaximum = ENET_PEER_TIMEOUT_MAXIMUM;
- peer->lastRoundTripTime = ENET_PEER_DEFAULT_ROUND_TRIP_TIME;
- peer->lowestRoundTripTime = ENET_PEER_DEFAULT_ROUND_TRIP_TIME;
- peer->lastRoundTripTimeVariance = 0;
- peer->highestRoundTripTimeVariance = 0;
- peer->roundTripTime = ENET_PEER_DEFAULT_ROUND_TRIP_TIME;
- peer->roundTripTimeVariance = 0;
- peer->mtu = peer->host->mtu;
- peer->reliableDataInTransit = 0;
- peer->outgoingReliableSequenceNumber = 0;
- peer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- peer->incomingUnsequencedGroup = 0;
- peer->outgoingUnsequencedGroup = 0;
- peer->eventData = 0;
- peer->totalWaitingData = 0;
- memset(peer->unsequencedWindow, 0, sizeof(peer->unsequencedWindow));
- enet_peer_reset_queues(peer);
- } // enet_peer_reset
- /** Sends a ping request to a peer.
- * @param peer destination for the ping request
- * @remarks ping requests factor into the mean round trip time as designated by the
- * roundTripTime field in the ENetPeer structure. ENet automatically pings all connected
- * peers at regular intervals, however, this function may be called to ensure more
- * frequent ping requests.
- */
- void enet_peer_ping(ENetPeer *peer) {
- ENetProtocol command;
- if (peer->state != ENET_PEER_STATE_CONNECTED) {
- return;
- }
- command.header.command = ENET_PROTOCOL_COMMAND_PING | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- command.header.channelID = 0xFF;
- enet_peer_queue_outgoing_command(peer, &command, NULL, 0, 0);
- }
- /** Sets the interval at which pings will be sent to a peer.
- *
- * Pings are used both to monitor the liveness of the connection and also to dynamically
- * adjust the throttle during periods of low traffic so that the throttle has reasonable
- * responsiveness during traffic spikes.
- *
- * @param peer the peer to adjust
- * @param pingInterval the interval at which to send pings; defaults to ENET_PEER_PING_INTERVAL if 0
- */
- void enet_peer_ping_interval(ENetPeer *peer, enet_uint32 pingInterval) {
- peer->pingInterval = pingInterval ? pingInterval : ENET_PEER_PING_INTERVAL;
- }
- /** Sets the timeout parameters for a peer.
- *
- * The timeout parameter control how and when a peer will timeout from a failure to acknowledge
- * reliable traffic. Timeout values use an exponential backoff mechanism, where if a reliable
- * packet is not acknowledge within some multiple of the average RTT plus a variance tolerance,
- * the timeout will be doubled until it reaches a set limit. If the timeout is thus at this
- * limit and reliable packets have been sent but not acknowledged within a certain minimum time
- * period, the peer will be disconnected. Alternatively, if reliable packets have been sent
- * but not acknowledged for a certain maximum time period, the peer will be disconnected regardless
- * of the current timeout limit value.
- *
- * @param peer the peer to adjust
- * @param timeoutLimit the timeout limit; defaults to ENET_PEER_TIMEOUT_LIMIT if 0
- * @param timeoutMinimum the timeout minimum; defaults to ENET_PEER_TIMEOUT_MINIMUM if 0
- * @param timeoutMaximum the timeout maximum; defaults to ENET_PEER_TIMEOUT_MAXIMUM if 0
- */
- void enet_peer_timeout(ENetPeer *peer, enet_uint32 timeoutLimit, enet_uint32 timeoutMinimum,
- enet_uint32 timeoutMaximum) {
- peer->timeoutLimit = timeoutLimit ? timeoutLimit : ENET_PEER_TIMEOUT_LIMIT;
- peer->timeoutMinimum = timeoutMinimum ? timeoutMinimum : ENET_PEER_TIMEOUT_MINIMUM;
- peer->timeoutMaximum = timeoutMaximum ? timeoutMaximum : ENET_PEER_TIMEOUT_MAXIMUM;
- }
- /** Force an immediate disconnection from a peer.
- * @param peer peer to disconnect
- * @param data data describing the disconnection
- * @remarks No ENET_EVENT_DISCONNECT event will be generated. The foreign peer is not
- * guaranteed to receive the disconnect notification, and is reset immediately upon
- * return from this function.
- */
- void enet_peer_disconnect_now(ENetPeer *peer, enet_uint32 data) {
- ENetProtocol command;
- if (peer->state == ENET_PEER_STATE_DISCONNECTED) {
- return;
- }
- if (peer->state != ENET_PEER_STATE_ZOMBIE &&
- peer->state != ENET_PEER_STATE_DISCONNECTING)
- {
- enet_peer_reset_queues(peer);
- command.header.command = ENET_PROTOCOL_COMMAND_DISCONNECT | ENET_PROTOCOL_COMMAND_FLAG_UNSEQUENCED;
- command.header.channelID = 0xFF;
- command.disconnect.data = ENET_HOST_TO_NET_32(data);
- enet_peer_queue_outgoing_command(peer, &command, NULL, 0, 0);
- enet_host_flush(peer->host);
- }
- enet_peer_reset(peer);
- }
- /** Request a disconnection from a peer.
- * @param peer peer to request a disconnection
- * @param data data describing the disconnection
- * @remarks An ENET_EVENT_DISCONNECT event will be generated by enet_host_service()
- * once the disconnection is complete.
- */
- void enet_peer_disconnect(ENetPeer *peer, enet_uint32 data) {
- ENetProtocol command;
- if (peer->state == ENET_PEER_STATE_DISCONNECTING ||
- peer->state == ENET_PEER_STATE_DISCONNECTED ||
- peer->state == ENET_PEER_STATE_ACKNOWLEDGING_DISCONNECT ||
- peer->state == ENET_PEER_STATE_ZOMBIE)
- {
- return;
- }
- enet_peer_reset_queues(peer);
- command.header.command = ENET_PROTOCOL_COMMAND_DISCONNECT;
- command.header.channelID = 0xFF;
- command.disconnect.data = ENET_HOST_TO_NET_32(data);
- if (peer->state == ENET_PEER_STATE_CONNECTED || peer->state == ENET_PEER_STATE_DISCONNECT_LATER) {
- command.header.command |= ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- } else {
- command.header.command |= ENET_PROTOCOL_COMMAND_FLAG_UNSEQUENCED;
- }
- enet_peer_queue_outgoing_command(peer, &command, NULL, 0, 0);
- if (peer->state == ENET_PEER_STATE_CONNECTED || peer->state == ENET_PEER_STATE_DISCONNECT_LATER) {
- enet_peer_on_disconnect(peer);
- peer->state = ENET_PEER_STATE_DISCONNECTING;
- } else {
- enet_host_flush(peer->host);
- enet_peer_reset(peer);
- }
- } // enet_peer_disconnect
- /** Request a disconnection from a peer, but only after all queued outgoing packets are sent.
- * @param peer peer to request a disconnection
- * @param data data describing the disconnection
- * @remarks An ENET_EVENT_DISCONNECT event will be generated by enet_host_service()
- * once the disconnection is complete.
- */
- void enet_peer_disconnect_later(ENetPeer *peer, enet_uint32 data) {
- if ((peer->state == ENET_PEER_STATE_CONNECTED || peer->state == ENET_PEER_STATE_DISCONNECT_LATER) &&
- !(enet_list_empty(&peer->outgoingReliableCommands) &&
- enet_list_empty(&peer->outgoingUnreliableCommands) &&
- enet_list_empty(&peer->sentReliableCommands)))
- {
- peer->state = ENET_PEER_STATE_DISCONNECT_LATER;
- peer->eventData = data;
- } else {
- enet_peer_disconnect(peer, data);
- }
- }
- ENetAcknowledgement * enet_peer_queue_acknowledgement(ENetPeer *peer, const ENetProtocol *command,
- enet_uint16 sentTime) {
- ENetAcknowledgement *acknowledgement;
- if (command->header.channelID < peer->channelCount) {
- ENetChannel *channel = &peer->channels [command->header.channelID];
- enet_uint16 reliableWindow = command->header.reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE,
- currentWindow = channel->incomingReliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (command->header.reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- reliableWindow += ENET_PEER_RELIABLE_WINDOWS;
- }
- if (reliableWindow >= currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS - 1 &&
- reliableWindow <= currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS)
- {
- return NULL;
- }
- }
- acknowledgement = (ENetAcknowledgement *) enet_malloc(sizeof(ENetAcknowledgement));
- if (acknowledgement == NULL) {
- return NULL;
- }
- peer->outgoingDataTotal += sizeof(ENetProtocolAcknowledge);
- acknowledgement->sentTime = sentTime;
- acknowledgement->command = *command;
- enet_list_insert(enet_list_end(&peer->acknowledgements), acknowledgement);
- return acknowledgement;
- }
- void enet_peer_setup_outgoing_command(ENetPeer *peer, ENetOutgoingCommand *outgoingCommand) {
- ENetChannel *channel = &peer->channels [outgoingCommand->command.header.channelID];
- peer->outgoingDataTotal += enet_protocol_command_size(outgoingCommand->command.header.command)
- + outgoingCommand->fragmentLength;
- if (outgoingCommand->command.header.channelID == 0xFF) {
- ++peer->outgoingReliableSequenceNumber;
- outgoingCommand->reliableSequenceNumber = peer->outgoingReliableSequenceNumber;
- outgoingCommand->unreliableSequenceNumber = 0;
- } else if (outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE) {
- ++channel->outgoingReliableSequenceNumber;
- channel->outgoingUnreliableSequenceNumber = 0;
- outgoingCommand->reliableSequenceNumber = channel->outgoingReliableSequenceNumber;
- outgoingCommand->unreliableSequenceNumber = 0;
- } else if (outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_FLAG_UNSEQUENCED) {
- ++peer->outgoingUnsequencedGroup;
- outgoingCommand->reliableSequenceNumber = 0;
- outgoingCommand->unreliableSequenceNumber = 0;
- } else {
- if (outgoingCommand->fragmentOffset == 0) {
- ++channel->outgoingUnreliableSequenceNumber;
- }
- outgoingCommand->reliableSequenceNumber = channel->outgoingReliableSequenceNumber;
- outgoingCommand->unreliableSequenceNumber = channel->outgoingUnreliableSequenceNumber;
- }
- outgoingCommand->sendAttempts = 0;
- outgoingCommand->sentTime = 0;
- outgoingCommand->roundTripTimeout = 0;
- outgoingCommand->roundTripTimeoutLimit = 0;
- outgoingCommand->command.header.reliableSequenceNumber =
- ENET_HOST_TO_NET_16(outgoingCommand->reliableSequenceNumber);
- switch (outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK) {
- case ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE:
- outgoingCommand->command.sendUnreliable.unreliableSequenceNumber = ENET_HOST_TO_NET_16(
- outgoingCommand->unreliableSequenceNumber);
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED:
- outgoingCommand->command.sendUnsequenced.unsequencedGroup = ENET_HOST_TO_NET_16(
- peer->outgoingUnsequencedGroup);
- break;
- default:
- break;
- }
- if (outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE) {
- enet_list_insert(enet_list_end(&peer->outgoingReliableCommands), outgoingCommand);
- } else {
- enet_list_insert(enet_list_end(&peer->outgoingUnreliableCommands), outgoingCommand);
- }
- } // enet_peer_setup_outgoing_command
- ENetOutgoingCommand * enet_peer_queue_outgoing_command(ENetPeer *peer, const ENetProtocol *command, ENetPacket *packet,
- enet_uint32 offset, enet_uint16 length) {
- ENetOutgoingCommand *outgoingCommand = (ENetOutgoingCommand *) enet_malloc(sizeof(ENetOutgoingCommand));
- if (outgoingCommand == NULL) {
- return NULL;
- }
- outgoingCommand->command = *command;
- outgoingCommand->fragmentOffset = offset;
- outgoingCommand->fragmentLength = length;
- outgoingCommand->packet = packet;
- if (packet != NULL) {
- ++packet->referenceCount;
- }
- enet_peer_setup_outgoing_command(peer, outgoingCommand);
- return outgoingCommand;
- }
- void enet_peer_dispatch_incoming_unreliable_commands(ENetPeer *peer, ENetChannel *channel) {
- ENetListIterator droppedCommand, startCommand, currentCommand;
- for (droppedCommand = startCommand = currentCommand = enet_list_begin(&channel->incomingUnreliableCommands);
- currentCommand != enet_list_end(&channel->incomingUnreliableCommands);
- currentCommand = enet_list_next(currentCommand))
- {
- ENetIncomingCommand *incomingCommand = (ENetIncomingCommand *) currentCommand;
- if ((incomingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK) ==
- ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED)
- {
- continue;
- }
- if (incomingCommand->reliableSequenceNumber == channel->incomingReliableSequenceNumber) {
- if (incomingCommand->fragmentsRemaining <= 0) {
- channel->incomingUnreliableSequenceNumber = incomingCommand->unreliableSequenceNumber;
- continue;
- }
- if (startCommand != currentCommand) {
- enet_list_move(enet_list_end(&peer->dispatchedCommands), startCommand, enet_list_previous(
- currentCommand));
- if (!peer->needsDispatch) {
- enet_list_insert(enet_list_end(&peer->host->dispatchQueue), &peer->dispatchList);
- peer->needsDispatch = 1;
- }
- droppedCommand = currentCommand;
- } else if (droppedCommand != currentCommand) {
- droppedCommand = enet_list_previous(currentCommand);
- }
- } else {
- enet_uint16 reliableWindow = incomingCommand->reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE,
- currentWindow = channel->incomingReliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (incomingCommand->reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- reliableWindow += ENET_PEER_RELIABLE_WINDOWS;
- }
- if (reliableWindow >= currentWindow &&
- reliableWindow < currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS - 1)
- {
- break;
- }
- droppedCommand = enet_list_next(currentCommand);
- if (startCommand != currentCommand) {
- enet_list_move(enet_list_end(&peer->dispatchedCommands), startCommand, enet_list_previous(
- currentCommand));
- if (!peer->needsDispatch) {
- enet_list_insert(enet_list_end(&peer->host->dispatchQueue), &peer->dispatchList);
- peer->needsDispatch = 1;
- }
- }
- }
- startCommand = enet_list_next(currentCommand);
- }
- if (startCommand != currentCommand) {
- enet_list_move(enet_list_end(&peer->dispatchedCommands), startCommand, enet_list_previous(currentCommand));
- if (!peer->needsDispatch) {
- enet_list_insert(enet_list_end(&peer->host->dispatchQueue), &peer->dispatchList);
- peer->needsDispatch = 1;
- }
- droppedCommand = currentCommand;
- }
- enet_peer_remove_incoming_commands(&channel->incomingUnreliableCommands,
- enet_list_begin(&channel->incomingUnreliableCommands), droppedCommand);
- } // enet_peer_dispatch_incoming_unreliable_commands
- void enet_peer_dispatch_incoming_reliable_commands(ENetPeer *peer, ENetChannel *channel) {
- ENetListIterator currentCommand;
- for (currentCommand = enet_list_begin(&channel->incomingReliableCommands);
- currentCommand != enet_list_end(&channel->incomingReliableCommands);
- currentCommand = enet_list_next(currentCommand))
- {
- ENetIncomingCommand *incomingCommand = (ENetIncomingCommand *) currentCommand;
- if (incomingCommand->fragmentsRemaining > 0 ||
- incomingCommand->reliableSequenceNumber != (enet_uint16) (channel->incomingReliableSequenceNumber + 1))
- {
- break;
- }
- channel->incomingReliableSequenceNumber = incomingCommand->reliableSequenceNumber;
- if (incomingCommand->fragmentCount > 0) {
- channel->incomingReliableSequenceNumber += incomingCommand->fragmentCount - 1;
- }
- }
- if (currentCommand == enet_list_begin(&channel->incomingReliableCommands)) {
- return;
- }
- channel->incomingUnreliableSequenceNumber = 0;
- enet_list_move(enet_list_end(&peer->dispatchedCommands), enet_list_begin(
- &channel->incomingReliableCommands), enet_list_previous(currentCommand));
- if (!peer->needsDispatch) {
- enet_list_insert(enet_list_end(&peer->host->dispatchQueue), &peer->dispatchList);
- peer->needsDispatch = 1;
- }
- if (!enet_list_empty(&channel->incomingUnreliableCommands)) {
- enet_peer_dispatch_incoming_unreliable_commands(peer, channel);
- }
- } // enet_peer_dispatch_incoming_reliable_commands
- ENetIncomingCommand * enet_peer_queue_incoming_command(ENetPeer *peer, const ENetProtocol *command, const void *data,
- size_t dataLength, enet_uint32 flags,
- enet_uint32 fragmentCount) {
- static ENetIncomingCommand dummyCommand;
- ENetChannel *channel = &peer->channels [command->header.channelID];
- enet_uint32 unreliableSequenceNumber = 0, reliableSequenceNumber = 0;
- enet_uint16 reliableWindow, currentWindow;
- ENetIncomingCommand *incomingCommand;
- ENetListIterator currentCommand;
- ENetPacket *packet = NULL;
- if (peer->state == ENET_PEER_STATE_DISCONNECT_LATER) {
- goto discardCommand;
- }
- if ((command->header.command & ENET_PROTOCOL_COMMAND_MASK) != ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED) {
- reliableSequenceNumber = command->header.reliableSequenceNumber;
- reliableWindow = reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- currentWindow = channel->incomingReliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- reliableWindow += ENET_PEER_RELIABLE_WINDOWS;
- }
- if (reliableWindow < currentWindow || reliableWindow >= currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS - 1) {
- goto discardCommand;
- }
- }
- switch (command->header.command & ENET_PROTOCOL_COMMAND_MASK) {
- case ENET_PROTOCOL_COMMAND_SEND_FRAGMENT:
- case ENET_PROTOCOL_COMMAND_SEND_RELIABLE:
- if (reliableSequenceNumber == channel->incomingReliableSequenceNumber) {
- goto discardCommand;
- }
- for (currentCommand = enet_list_previous(enet_list_end(&channel->incomingReliableCommands));
- currentCommand != enet_list_end(&channel->incomingReliableCommands);
- currentCommand = enet_list_previous(currentCommand))
- {
- incomingCommand = (ENetIncomingCommand *) currentCommand;
- if (reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- continue;
- }
- } else if (incomingCommand->reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber <= reliableSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < reliableSequenceNumber) {
- break;
- }
- goto discardCommand;
- }
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE:
- case ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE_FRAGMENT:
- unreliableSequenceNumber = ENET_NET_TO_HOST_16(command->sendUnreliable.unreliableSequenceNumber);
- if (reliableSequenceNumber == channel->incomingReliableSequenceNumber &&
- unreliableSequenceNumber <= channel->incomingUnreliableSequenceNumber)
- {
- goto discardCommand;
- }
- for (currentCommand = enet_list_previous(enet_list_end(&channel->incomingUnreliableCommands));
- currentCommand != enet_list_end(&channel->incomingUnreliableCommands);
- currentCommand = enet_list_previous(currentCommand))
- {
- incomingCommand = (ENetIncomingCommand *) currentCommand;
- if ((command->header.command & ENET_PROTOCOL_COMMAND_MASK) == ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED) {
- continue;
- }
- if (reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- continue;
- }
- } else if (incomingCommand->reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber < reliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber > reliableSequenceNumber) {
- continue;
- }
- if (incomingCommand->unreliableSequenceNumber <= unreliableSequenceNumber) {
- if (incomingCommand->unreliableSequenceNumber < unreliableSequenceNumber) {
- break;
- }
- goto discardCommand;
- }
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED:
- currentCommand = enet_list_end(&channel->incomingUnreliableCommands);
- break;
- default:
- goto discardCommand;
- }
- if (peer->totalWaitingData >= peer->host->maximumWaitingData) {
- goto notifyError;
- }
- packet = enet_packet_create(data, dataLength, flags);
- if (packet == NULL) {
- goto notifyError;
- }
- incomingCommand = (ENetIncomingCommand *) enet_malloc(sizeof(ENetIncomingCommand));
- if (incomingCommand == NULL) {
- goto notifyError;
- }
- incomingCommand->reliableSequenceNumber = command->header.reliableSequenceNumber;
- incomingCommand->unreliableSequenceNumber = unreliableSequenceNumber & 0xFFFF;
- incomingCommand->command = *command;
- incomingCommand->fragmentCount = fragmentCount;
- incomingCommand->fragmentsRemaining = fragmentCount;
- incomingCommand->packet = packet;
- incomingCommand->fragments = NULL;
- if (fragmentCount > 0) {
- if (fragmentCount <= ENET_PROTOCOL_MAXIMUM_FRAGMENT_COUNT) {
- incomingCommand->fragments = (enet_uint32 *) enet_malloc((fragmentCount + 31) / 32 * sizeof(enet_uint32));
- }
- if (incomingCommand->fragments == NULL) {
- enet_free(incomingCommand);
- goto notifyError;
- }
- memset(incomingCommand->fragments, 0, (fragmentCount + 31) / 32 * sizeof(enet_uint32));
- }
- if (packet != NULL) {
- ++packet->referenceCount;
- peer->totalWaitingData += packet->dataLength;
- }
- enet_list_insert(enet_list_next(currentCommand), incomingCommand);
- switch (command->header.command & ENET_PROTOCOL_COMMAND_MASK) {
- case ENET_PROTOCOL_COMMAND_SEND_FRAGMENT:
- case ENET_PROTOCOL_COMMAND_SEND_RELIABLE:
- enet_peer_dispatch_incoming_reliable_commands(peer, channel);
- break;
- default:
- enet_peer_dispatch_incoming_unreliable_commands(peer, channel);
- break;
- }
- return incomingCommand;
- discardCommand:
- if (fragmentCount > 0) {
- goto notifyError;
- }
- if (packet != NULL && packet->referenceCount == 0) {
- enet_packet_destroy(packet);
- }
- return &dummyCommand;
- notifyError:
- if (packet != NULL && packet->referenceCount == 0) {
- enet_packet_destroy(packet);
- }
- return NULL;
- } // enet_peer_queue_incoming_command
- /** @} */
- // @from_file: src/protocol.c
- /**
- * @file protocol.c
- * @brief ENet protocol functions
- */
- # include <stdio.h>
- # define ENET_BUILDING_LIB 1
- static size_t commandSizes [ENET_PROTOCOL_COMMAND_COUNT] =
- {
- 0,
- sizeof(ENetProtocolAcknowledge),
- sizeof(ENetProtocolConnect),
- sizeof(ENetProtocolVerifyConnect),
- sizeof(ENetProtocolDisconnect),
- sizeof(ENetProtocolPing),
- sizeof(ENetProtocolSendReliable),
- sizeof(ENetProtocolSendUnreliable),
- sizeof(ENetProtocolSendFragment),
- sizeof(ENetProtocolSendUnsequenced),
- sizeof(ENetProtocolBandwidthLimit),
- sizeof(ENetProtocolThrottleConfigure),
- sizeof(ENetProtocolSendFragment)
- };
- size_t enet_protocol_command_size(enet_uint8 commandNumber) {
- return commandSizes [commandNumber & ENET_PROTOCOL_COMMAND_MASK];
- }
- static void enet_protocol_change_state(ENetHost *host, ENetPeer *peer, ENetPeerState state) {
- if (state == ENET_PEER_STATE_CONNECTED || state == ENET_PEER_STATE_DISCONNECT_LATER) {
- enet_peer_on_connect(peer);
- } else {
- enet_peer_on_disconnect(peer);
- }
- peer->state = state;
- }
- static void enet_protocol_dispatch_state(ENetHost *host, ENetPeer *peer, ENetPeerState state) {
- enet_protocol_change_state(host, peer, state);
- if (!peer->needsDispatch) {
- enet_list_insert(enet_list_end(&host->dispatchQueue), &peer->dispatchList);
- peer->needsDispatch = 1;
- }
- }
- static int enet_protocol_dispatch_incoming_commands(ENetHost *host, ENetEvent *event) {
- while (!enet_list_empty(&host->dispatchQueue)) {
- ENetPeer *peer = (ENetPeer *) enet_list_remove(enet_list_begin(&host->dispatchQueue));
- peer->needsDispatch = 0;
- switch (peer->state) {
- case ENET_PEER_STATE_CONNECTION_PENDING:
- case ENET_PEER_STATE_CONNECTION_SUCCEEDED:
- enet_protocol_change_state(host, peer, ENET_PEER_STATE_CONNECTED);
- event->type = ENET_EVENT_TYPE_CONNECT;
- event->peer = peer;
- event->data = peer->eventData;
- return 1;
- case ENET_PEER_STATE_ZOMBIE:
- host->recalculateBandwidthLimits = 1;
- event->type = ENET_EVENT_TYPE_DISCONNECT;
- event->peer = peer;
- event->data = peer->eventData;
- enet_peer_reset(peer);
- return 1;
- case ENET_PEER_STATE_CONNECTED:
- if (enet_list_empty(&peer->dispatchedCommands)) {
- continue;
- }
- event->packet = enet_peer_receive(peer, &event->channelID);
- if (event->packet == NULL) {
- continue;
- }
- event->type = ENET_EVENT_TYPE_RECEIVE;
- event->peer = peer;
- if (!enet_list_empty(&peer->dispatchedCommands)) {
- peer->needsDispatch = 1;
- enet_list_insert(enet_list_end(&host->dispatchQueue), &peer->dispatchList);
- }
- return 1;
- default:
- break;
- }
- }
- return 0;
- } // enet_protocol_dispatch_incoming_commands
- static void enet_protocol_notify_connect(ENetHost *host, ENetPeer *peer, ENetEvent *event) {
- host->recalculateBandwidthLimits = 1;
- if (event != NULL) {
- enet_protocol_change_state(host, peer, ENET_PEER_STATE_CONNECTED);
- event->type = ENET_EVENT_TYPE_CONNECT;
- event->peer = peer;
- event->data = peer->eventData;
- } else {
- enet_protocol_dispatch_state(host, peer,
- peer->state ==
- ENET_PEER_STATE_CONNECTING ? ENET_PEER_STATE_CONNECTION_SUCCEEDED : ENET_PEER_STATE_CONNECTION_PENDING);
- }
- }
- static void enet_protocol_notify_disconnect(ENetHost *host, ENetPeer *peer, ENetEvent *event) {
- if (peer->state >= ENET_PEER_STATE_CONNECTION_PENDING) {
- host->recalculateBandwidthLimits = 1;
- }
- if (peer->state != ENET_PEER_STATE_CONNECTING && peer->state < ENET_PEER_STATE_CONNECTION_SUCCEEDED) {
- enet_peer_reset(peer);
- } else if (event != NULL) {
- event->type = ENET_EVENT_TYPE_DISCONNECT;
- event->peer = peer;
- event->data = 0;
- enet_peer_reset(peer);
- } else {
- peer->eventData = 0;
- enet_protocol_dispatch_state(host, peer, ENET_PEER_STATE_ZOMBIE);
- }
- }
- static void enet_protocol_remove_sent_unreliable_commands(ENetPeer *peer) {
- ENetOutgoingCommand *outgoingCommand;
- while (!enet_list_empty(&peer->sentUnreliableCommands)) {
- outgoingCommand = (ENetOutgoingCommand *) enet_list_front(&peer->sentUnreliableCommands);
- enet_list_remove(&outgoingCommand->outgoingCommandList);
- if (outgoingCommand->packet != NULL) {
- --outgoingCommand->packet->referenceCount;
- if (outgoingCommand->packet->referenceCount == 0) {
- outgoingCommand->packet->flags |= ENET_PACKET_FLAG_SENT;
- enet_packet_destroy(outgoingCommand->packet);
- }
- }
- enet_free(outgoingCommand);
- }
- }
- static ENetProtocolCommand enet_protocol_remove_sent_reliable_command(ENetPeer *peer,
- enet_uint16 reliableSequenceNumber,
- enet_uint8 channelID) {
- ENetOutgoingCommand *outgoingCommand = NULL;
- ENetListIterator currentCommand;
- ENetProtocolCommand commandNumber;
- int wasSent = 1;
- for (currentCommand = enet_list_begin(&peer->sentReliableCommands);
- currentCommand != enet_list_end(&peer->sentReliableCommands);
- currentCommand = enet_list_next(currentCommand))
- {
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- if (outgoingCommand->reliableSequenceNumber == reliableSequenceNumber &&
- outgoingCommand->command.header.channelID == channelID)
- {
- break;
- }
- }
- if (currentCommand == enet_list_end(&peer->sentReliableCommands)) {
- for (currentCommand = enet_list_begin(&peer->outgoingReliableCommands);
- currentCommand != enet_list_end(&peer->outgoingReliableCommands);
- currentCommand = enet_list_next(currentCommand))
- {
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- if (outgoingCommand->sendAttempts < 1) { return ENET_PROTOCOL_COMMAND_NONE; }
- if (outgoingCommand->reliableSequenceNumber == reliableSequenceNumber &&
- outgoingCommand->command.header.channelID == channelID)
- {
- break;
- }
- }
- if (currentCommand == enet_list_end(&peer->outgoingReliableCommands)) {
- return ENET_PROTOCOL_COMMAND_NONE;
- }
- wasSent = 0;
- }
- if (outgoingCommand == NULL) {
- return ENET_PROTOCOL_COMMAND_NONE;
- }
- if (channelID < peer->channelCount) {
- ENetChannel *channel = &peer->channels [channelID];
- enet_uint16 reliableWindow = reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (channel->reliableWindows [reliableWindow] > 0) {
- --channel->reliableWindows [reliableWindow];
- if (!channel->reliableWindows [reliableWindow]) {
- channel->usedReliableWindows &= ~(1 << reliableWindow);
- }
- }
- }
- commandNumber = (ENetProtocolCommand) (outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK);
- enet_list_remove(&outgoingCommand->outgoingCommandList);
- if (outgoingCommand->packet != NULL) {
- if (wasSent) {
- peer->reliableDataInTransit -= outgoingCommand->fragmentLength;
- }
- --outgoingCommand->packet->referenceCount;
- if (outgoingCommand->packet->referenceCount == 0) {
- outgoingCommand->packet->flags |= ENET_PACKET_FLAG_SENT;
- enet_packet_destroy(outgoingCommand->packet);
- }
- }
- enet_free(outgoingCommand);
- if (enet_list_empty(&peer->sentReliableCommands)) {
- return commandNumber;
- }
- outgoingCommand = (ENetOutgoingCommand *) enet_list_front(&peer->sentReliableCommands);
- peer->nextTimeout = outgoingCommand->sentTime + outgoingCommand->roundTripTimeout;
- return commandNumber;
- } // enet_protocol_remove_sent_reliable_command
- static ENetPeer * enet_protocol_handle_connect(ENetHost *host, ENetProtocolHeader *header, ENetProtocol *command) {
- enet_uint8 incomingSessionID, outgoingSessionID;
- enet_uint32 mtu, windowSize;
- ENetChannel *channel;
- size_t channelCount, duplicatePeers = 0;
- ENetPeer *currentPeer, *peer = NULL;
- ENetProtocol verifyCommand;
- channelCount = ENET_NET_TO_HOST_32(command->connect.channelCount);
- if (channelCount < ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT ||
- channelCount > ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT)
- {
- return NULL;
- }
- for (currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- if (currentPeer->state == ENET_PEER_STATE_DISCONNECTED) {
- if (peer == NULL) {
- peer = currentPeer;
- }
- } else if (currentPeer->state != ENET_PEER_STATE_CONNECTING &&
- currentPeer->address.host == host->receivedAddress.host)
- {
- if (currentPeer->address.port == host->receivedAddress.port &&
- currentPeer->connectID == command->connect.connectID)
- {
- return NULL;
- }
- ++duplicatePeers;
- }
- }
- if (peer == NULL || duplicatePeers >= host->duplicatePeers) {
- return NULL;
- }
- if (channelCount > host->channelLimit) {
- channelCount = host->channelLimit;
- }
- peer->channels = (ENetChannel *) enet_malloc(channelCount * sizeof(ENetChannel));
- if (peer->channels == NULL) {
- return NULL;
- }
- peer->channelCount = channelCount;
- peer->state = ENET_PEER_STATE_ACKNOWLEDGING_CONNECT;
- peer->connectID = command->connect.connectID;
- peer->address = host->receivedAddress;
- peer->outgoingPeerID = ENET_NET_TO_HOST_16(command->connect.outgoingPeerID);
- peer->incomingBandwidth = ENET_NET_TO_HOST_32(command->connect.incomingBandwidth);
- peer->outgoingBandwidth = ENET_NET_TO_HOST_32(command->connect.outgoingBandwidth);
- peer->packetThrottleInterval = ENET_NET_TO_HOST_32(command->connect.packetThrottleInterval);
- peer->packetThrottleAcceleration = ENET_NET_TO_HOST_32(command->connect.packetThrottleAcceleration);
- peer->packetThrottleDeceleration = ENET_NET_TO_HOST_32(command->connect.packetThrottleDeceleration);
- peer->eventData = ENET_NET_TO_HOST_32(command->connect.data);
- incomingSessionID = command->connect.incomingSessionID ==
- 0xFF ? peer->outgoingSessionID : command->connect.incomingSessionID;
- incomingSessionID = (incomingSessionID + 1)
- & (ENET_PROTOCOL_HEADER_SESSION_MASK >> ENET_PROTOCOL_HEADER_SESSION_SHIFT);
- if (incomingSessionID == peer->outgoingSessionID) {
- incomingSessionID = (incomingSessionID + 1)
- & (ENET_PROTOCOL_HEADER_SESSION_MASK >> ENET_PROTOCOL_HEADER_SESSION_SHIFT);
- }
- peer->outgoingSessionID = incomingSessionID;
- outgoingSessionID = command->connect.outgoingSessionID ==
- 0xFF ? peer->incomingSessionID : command->connect.outgoingSessionID;
- outgoingSessionID = (outgoingSessionID + 1)
- & (ENET_PROTOCOL_HEADER_SESSION_MASK >> ENET_PROTOCOL_HEADER_SESSION_SHIFT);
- if (outgoingSessionID == peer->incomingSessionID) {
- outgoingSessionID = (outgoingSessionID + 1)
- & (ENET_PROTOCOL_HEADER_SESSION_MASK >> ENET_PROTOCOL_HEADER_SESSION_SHIFT);
- }
- peer->incomingSessionID = outgoingSessionID;
- for (channel = peer->channels;
- channel < &peer->channels [channelCount];
- ++channel)
- {
- channel->outgoingReliableSequenceNumber = 0;
- channel->outgoingUnreliableSequenceNumber = 0;
- channel->incomingReliableSequenceNumber = 0;
- channel->incomingUnreliableSequenceNumber = 0;
- enet_list_clear(&channel->incomingReliableCommands);
- enet_list_clear(&channel->incomingUnreliableCommands);
- channel->usedReliableWindows = 0;
- memset(channel->reliableWindows, 0, sizeof(channel->reliableWindows));
- }
- mtu = ENET_NET_TO_HOST_32(command->connect.mtu);
- if (mtu < ENET_PROTOCOL_MINIMUM_MTU) {
- mtu = ENET_PROTOCOL_MINIMUM_MTU;
- } else if (mtu > ENET_PROTOCOL_MAXIMUM_MTU) {
- mtu = ENET_PROTOCOL_MAXIMUM_MTU;
- }
- peer->mtu = mtu;
- if (host->outgoingBandwidth == 0 &&
- peer->incomingBandwidth == 0)
- {
- peer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- } else if (host->outgoingBandwidth == 0 ||
- peer->incomingBandwidth == 0)
- {
- peer->windowSize = (ENET_MAX(host->outgoingBandwidth, peer->incomingBandwidth)
- / ENET_PEER_WINDOW_SIZE_SCALE)
- * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else {
- peer->windowSize = (ENET_MIN(host->outgoingBandwidth, peer->incomingBandwidth)
- / ENET_PEER_WINDOW_SIZE_SCALE)
- * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- }
- if (peer->windowSize < ENET_PROTOCOL_MINIMUM_WINDOW_SIZE) {
- peer->windowSize = ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else if (peer->windowSize > ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE) {
- peer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- }
- if (host->incomingBandwidth == 0) {
- windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- } else {
- windowSize = (host->incomingBandwidth / ENET_PEER_WINDOW_SIZE_SCALE)
- * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- }
- if (windowSize > ENET_NET_TO_HOST_32(command->connect.windowSize)) {
- windowSize = ENET_NET_TO_HOST_32(command->connect.windowSize);
- }
- if (windowSize < ENET_PROTOCOL_MINIMUM_WINDOW_SIZE) {
- windowSize = ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else if (windowSize > ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE) {
- windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- }
- verifyCommand.header.command = ENET_PROTOCOL_COMMAND_VERIFY_CONNECT | ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE;
- verifyCommand.header.channelID = 0xFF;
- verifyCommand.verifyConnect.outgoingPeerID = ENET_HOST_TO_NET_16(peer->incomingPeerID);
- verifyCommand.verifyConnect.incomingSessionID = incomingSessionID;
- verifyCommand.verifyConnect.outgoingSessionID = outgoingSessionID;
- verifyCommand.verifyConnect.mtu = ENET_HOST_TO_NET_32(peer->mtu);
- verifyCommand.verifyConnect.windowSize = ENET_HOST_TO_NET_32(windowSize);
- verifyCommand.verifyConnect.channelCount = ENET_HOST_TO_NET_32(channelCount);
- verifyCommand.verifyConnect.incomingBandwidth = ENET_HOST_TO_NET_32(host->incomingBandwidth);
- verifyCommand.verifyConnect.outgoingBandwidth = ENET_HOST_TO_NET_32(host->outgoingBandwidth);
- verifyCommand.verifyConnect.packetThrottleInterval = ENET_HOST_TO_NET_32(peer->packetThrottleInterval);
- verifyCommand.verifyConnect.packetThrottleAcceleration = ENET_HOST_TO_NET_32(peer->packetThrottleAcceleration);
- verifyCommand.verifyConnect.packetThrottleDeceleration = ENET_HOST_TO_NET_32(peer->packetThrottleDeceleration);
- verifyCommand.verifyConnect.connectID = peer->connectID;
- enet_peer_queue_outgoing_command(peer, &verifyCommand, NULL, 0, 0);
- return peer;
- } // enet_protocol_handle_connect
- static int enet_protocol_handle_send_reliable(ENetHost *host, ENetPeer *peer, const ENetProtocol *command,
- enet_uint8 **currentData) {
- size_t dataLength;
- if (command->header.channelID >= peer->channelCount ||
- (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER))
- {
- return -1;
- }
- dataLength = ENET_NET_TO_HOST_16(command->sendReliable.dataLength);
- *currentData += dataLength;
- if (dataLength > host->maximumPacketSize ||
- *currentData < host->receivedData ||
- *currentData > &host->receivedData [host->receivedDataLength])
- {
- return -1;
- }
- if (enet_peer_queue_incoming_command(peer, command, (const enet_uint8 *) command + sizeof(ENetProtocolSendReliable),
- dataLength, ENET_PACKET_FLAG_RELIABLE, 0) == NULL)
- {
- return -1;
- }
- return 0;
- }
- static int enet_protocol_handle_send_unsequenced(ENetHost *host, ENetPeer *peer, const ENetProtocol *command,
- enet_uint8 **currentData) {
- enet_uint32 unsequencedGroup, index;
- size_t dataLength;
- if (command->header.channelID >= peer->channelCount ||
- (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER))
- {
- return -1;
- }
- dataLength = ENET_NET_TO_HOST_16(command->sendUnsequenced.dataLength);
- *currentData += dataLength;
- if (dataLength > host->maximumPacketSize ||
- *currentData < host->receivedData ||
- *currentData > &host->receivedData [host->receivedDataLength])
- {
- return -1;
- }
- unsequencedGroup = ENET_NET_TO_HOST_16(command->sendUnsequenced.unsequencedGroup);
- index = unsequencedGroup % ENET_PEER_UNSEQUENCED_WINDOW_SIZE;
- if (unsequencedGroup < peer->incomingUnsequencedGroup) {
- unsequencedGroup += 0x10000;
- }
- if (unsequencedGroup >=
- (enet_uint32) peer->incomingUnsequencedGroup + ENET_PEER_FREE_UNSEQUENCED_WINDOWS
- * ENET_PEER_UNSEQUENCED_WINDOW_SIZE)
- {
- return 0;
- }
- unsequencedGroup &= 0xFFFF;
- if (unsequencedGroup - index != peer->incomingUnsequencedGroup) {
- peer->incomingUnsequencedGroup = unsequencedGroup - index;
- memset(peer->unsequencedWindow, 0, sizeof(peer->unsequencedWindow));
- } else if (peer->unsequencedWindow [index / 32] & (1 << (index % 32))) {
- return 0;
- }
- if (enet_peer_queue_incoming_command(peer, command,
- (const enet_uint8 *) command + sizeof(ENetProtocolSendUnsequenced), dataLength, ENET_PACKET_FLAG_UNSEQUENCED,
- 0) == NULL)
- {
- return -1;
- }
- peer->unsequencedWindow [index / 32] |= 1 << (index % 32);
- return 0;
- } // enet_protocol_handle_send_unsequenced
- static int enet_protocol_handle_send_unreliable(ENetHost *host, ENetPeer *peer, const ENetProtocol *command,
- enet_uint8 **currentData) {
- size_t dataLength;
- if (command->header.channelID >= peer->channelCount ||
- (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER))
- {
- return -1;
- }
- dataLength = ENET_NET_TO_HOST_16(command->sendUnreliable.dataLength);
- *currentData += dataLength;
- if (dataLength > host->maximumPacketSize ||
- *currentData < host->receivedData ||
- *currentData > &host->receivedData [host->receivedDataLength])
- {
- return -1;
- }
- if (enet_peer_queue_incoming_command(peer, command,
- (const enet_uint8 *) command + sizeof(ENetProtocolSendUnreliable), dataLength, 0, 0) == NULL)
- {
- return -1;
- }
- return 0;
- }
- static int enet_protocol_handle_send_fragment(ENetHost *host, ENetPeer *peer, const ENetProtocol *command,
- enet_uint8 **currentData) {
- enet_uint32 fragmentNumber,
- fragmentCount,
- fragmentOffset,
- fragmentLength,
- startSequenceNumber,
- totalLength;
- ENetChannel *channel;
- enet_uint16 startWindow, currentWindow;
- ENetListIterator currentCommand;
- ENetIncomingCommand *startCommand = NULL;
- if (command->header.channelID >= peer->channelCount ||
- (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER))
- {
- return -1;
- }
- fragmentLength = ENET_NET_TO_HOST_16(command->sendFragment.dataLength);
- *currentData += fragmentLength;
- if (fragmentLength > host->maximumPacketSize ||
- *currentData < host->receivedData ||
- *currentData > &host->receivedData [host->receivedDataLength])
- {
- return -1;
- }
- channel = &peer->channels [command->header.channelID];
- startSequenceNumber = ENET_NET_TO_HOST_16(command->sendFragment.startSequenceNumber);
- startWindow = startSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- currentWindow = channel->incomingReliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (startSequenceNumber < channel->incomingReliableSequenceNumber) {
- startWindow += ENET_PEER_RELIABLE_WINDOWS;
- }
- if (startWindow < currentWindow || startWindow >= currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS - 1) {
- return 0;
- }
- fragmentNumber = ENET_NET_TO_HOST_32(command->sendFragment.fragmentNumber);
- fragmentCount = ENET_NET_TO_HOST_32(command->sendFragment.fragmentCount);
- fragmentOffset = ENET_NET_TO_HOST_32(command->sendFragment.fragmentOffset);
- totalLength = ENET_NET_TO_HOST_32(command->sendFragment.totalLength);
- if (fragmentCount > ENET_PROTOCOL_MAXIMUM_FRAGMENT_COUNT ||
- fragmentNumber >= fragmentCount ||
- totalLength > host->maximumPacketSize ||
- fragmentOffset >= totalLength ||
- fragmentLength > totalLength - fragmentOffset)
- {
- return -1;
- }
- for (currentCommand = enet_list_previous(enet_list_end(&channel->incomingReliableCommands));
- currentCommand != enet_list_end(&channel->incomingReliableCommands);
- currentCommand = enet_list_previous(currentCommand))
- {
- ENetIncomingCommand *incomingCommand = (ENetIncomingCommand *) currentCommand;
- if (startSequenceNumber >= channel->incomingReliableSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- continue;
- }
- } else if (incomingCommand->reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber <= startSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < startSequenceNumber) {
- break;
- }
- if ((incomingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK) !=
- ENET_PROTOCOL_COMMAND_SEND_FRAGMENT ||
- totalLength != incomingCommand->packet->dataLength ||
- fragmentCount != incomingCommand->fragmentCount)
- {
- return -1;
- }
- startCommand = incomingCommand;
- break;
- }
- }
- if (startCommand == NULL) {
- ENetProtocol hostCommand = *command;
- hostCommand.header.reliableSequenceNumber = startSequenceNumber;
- startCommand = enet_peer_queue_incoming_command(peer, &hostCommand, NULL, totalLength,
- ENET_PACKET_FLAG_RELIABLE, fragmentCount);
- if (startCommand == NULL) {
- return -1;
- }
- }
- if ((startCommand->fragments [fragmentNumber / 32] & (1 << (fragmentNumber % 32))) == 0) {
- --startCommand->fragmentsRemaining;
- startCommand->fragments [fragmentNumber / 32] |= (1 << (fragmentNumber % 32));
- if (fragmentOffset + fragmentLength > startCommand->packet->dataLength) {
- fragmentLength = startCommand->packet->dataLength - fragmentOffset;
- }
- memcpy(startCommand->packet->data + fragmentOffset,
- (enet_uint8 *) command + sizeof(ENetProtocolSendFragment),
- fragmentLength);
- if (startCommand->fragmentsRemaining <= 0) {
- enet_peer_dispatch_incoming_reliable_commands(peer, channel);
- }
- }
- return 0;
- } // enet_protocol_handle_send_fragment
- static int enet_protocol_handle_send_unreliable_fragment(ENetHost *host, ENetPeer *peer, const ENetProtocol *command,
- enet_uint8 **currentData) {
- enet_uint32 fragmentNumber,
- fragmentCount,
- fragmentOffset,
- fragmentLength,
- reliableSequenceNumber,
- startSequenceNumber,
- totalLength;
- enet_uint16 reliableWindow, currentWindow;
- ENetChannel *channel;
- ENetListIterator currentCommand;
- ENetIncomingCommand *startCommand = NULL;
- if (command->header.channelID >= peer->channelCount ||
- (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER))
- {
- return -1;
- }
- fragmentLength = ENET_NET_TO_HOST_16(command->sendFragment.dataLength);
- *currentData += fragmentLength;
- if (fragmentLength > host->maximumPacketSize ||
- *currentData < host->receivedData ||
- *currentData > &host->receivedData [host->receivedDataLength])
- {
- return -1;
- }
- channel = &peer->channels [command->header.channelID];
- reliableSequenceNumber = command->header.reliableSequenceNumber;
- startSequenceNumber = ENET_NET_TO_HOST_16(command->sendFragment.startSequenceNumber);
- reliableWindow = reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- currentWindow = channel->incomingReliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- reliableWindow += ENET_PEER_RELIABLE_WINDOWS;
- }
- if (reliableWindow < currentWindow || reliableWindow >= currentWindow + ENET_PEER_FREE_RELIABLE_WINDOWS - 1) {
- return 0;
- }
- if (reliableSequenceNumber == channel->incomingReliableSequenceNumber &&
- startSequenceNumber <= channel->incomingUnreliableSequenceNumber)
- {
- return 0;
- }
- fragmentNumber = ENET_NET_TO_HOST_32(command->sendFragment.fragmentNumber);
- fragmentCount = ENET_NET_TO_HOST_32(command->sendFragment.fragmentCount);
- fragmentOffset = ENET_NET_TO_HOST_32(command->sendFragment.fragmentOffset);
- totalLength = ENET_NET_TO_HOST_32(command->sendFragment.totalLength);
- if (fragmentCount > ENET_PROTOCOL_MAXIMUM_FRAGMENT_COUNT ||
- fragmentNumber >= fragmentCount ||
- totalLength > host->maximumPacketSize ||
- fragmentOffset >= totalLength ||
- fragmentLength > totalLength - fragmentOffset)
- {
- return -1;
- }
- for (currentCommand = enet_list_previous(enet_list_end(&channel->incomingUnreliableCommands));
- currentCommand != enet_list_end(&channel->incomingUnreliableCommands);
- currentCommand = enet_list_previous(currentCommand))
- {
- ENetIncomingCommand *incomingCommand = (ENetIncomingCommand *) currentCommand;
- if (reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- if (incomingCommand->reliableSequenceNumber < channel->incomingReliableSequenceNumber) {
- continue;
- }
- } else if (incomingCommand->reliableSequenceNumber >= channel->incomingReliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber < reliableSequenceNumber) {
- break;
- }
- if (incomingCommand->reliableSequenceNumber > reliableSequenceNumber) {
- continue;
- }
- if (incomingCommand->unreliableSequenceNumber <= startSequenceNumber) {
- if (incomingCommand->unreliableSequenceNumber < startSequenceNumber) {
- break;
- }
- if ((incomingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK) !=
- ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE_FRAGMENT ||
- totalLength != incomingCommand->packet->dataLength ||
- fragmentCount != incomingCommand->fragmentCount)
- {
- return -1;
- }
- startCommand = incomingCommand;
- break;
- }
- }
- if (startCommand == NULL) {
- startCommand = enet_peer_queue_incoming_command(peer, command, NULL, totalLength,
- ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT, fragmentCount);
- if (startCommand == NULL) {
- return -1;
- }
- }
- if ((startCommand->fragments [fragmentNumber / 32] & (1 << (fragmentNumber % 32))) == 0) {
- --startCommand->fragmentsRemaining;
- startCommand->fragments [fragmentNumber / 32] |= (1 << (fragmentNumber % 32));
- if (fragmentOffset + fragmentLength > startCommand->packet->dataLength) {
- fragmentLength = startCommand->packet->dataLength - fragmentOffset;
- }
- memcpy(startCommand->packet->data + fragmentOffset,
- (enet_uint8 *) command + sizeof(ENetProtocolSendFragment),
- fragmentLength);
- if (startCommand->fragmentsRemaining <= 0) {
- enet_peer_dispatch_incoming_unreliable_commands(peer, channel);
- }
- }
- return 0;
- } // enet_protocol_handle_send_unreliable_fragment
- static int enet_protocol_handle_ping(ENetHost *host, ENetPeer *peer, const ENetProtocol *command) {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- return -1;
- }
- return 0;
- }
- static int enet_protocol_handle_bandwidth_limit(ENetHost *host, ENetPeer *peer, const ENetProtocol *command) {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- return -1;
- }
- if (peer->incomingBandwidth != 0) {
- --host->bandwidthLimitedPeers;
- }
- peer->incomingBandwidth = ENET_NET_TO_HOST_32(command->bandwidthLimit.incomingBandwidth);
- peer->outgoingBandwidth = ENET_NET_TO_HOST_32(command->bandwidthLimit.outgoingBandwidth);
- if (peer->incomingBandwidth != 0) {
- ++host->bandwidthLimitedPeers;
- }
- if (peer->incomingBandwidth == 0 && host->outgoingBandwidth == 0) {
- peer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- } else if (peer->incomingBandwidth == 0 || host->outgoingBandwidth == 0) {
- peer->windowSize = (ENET_MAX(peer->incomingBandwidth, host->outgoingBandwidth)
- / ENET_PEER_WINDOW_SIZE_SCALE) * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else {
- peer->windowSize = (ENET_MIN(peer->incomingBandwidth, host->outgoingBandwidth)
- / ENET_PEER_WINDOW_SIZE_SCALE) * ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- }
- if (peer->windowSize < ENET_PROTOCOL_MINIMUM_WINDOW_SIZE) {
- peer->windowSize = ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- } else if (peer->windowSize > ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE) {
- peer->windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- }
- return 0;
- } // enet_protocol_handle_bandwidth_limit
- static int enet_protocol_handle_throttle_configure(ENetHost *host, ENetPeer *peer, const ENetProtocol *command) {
- if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- return -1;
- }
- peer->packetThrottleInterval = ENET_NET_TO_HOST_32(command->throttleConfigure.packetThrottleInterval);
- peer->packetThrottleAcceleration = ENET_NET_TO_HOST_32(command->throttleConfigure.packetThrottleAcceleration);
- peer->packetThrottleDeceleration = ENET_NET_TO_HOST_32(command->throttleConfigure.packetThrottleDeceleration);
- return 0;
- }
- static int enet_protocol_handle_disconnect(ENetHost *host, ENetPeer *peer, const ENetProtocol *command) {
- if (peer->state == ENET_PEER_STATE_DISCONNECTED || peer->state == ENET_PEER_STATE_ZOMBIE ||
- peer->state == ENET_PEER_STATE_ACKNOWLEDGING_DISCONNECT)
- {
- return 0;
- }
- enet_peer_reset_queues(peer);
- if (peer->state == ENET_PEER_STATE_CONNECTION_SUCCEEDED || peer->state == ENET_PEER_STATE_DISCONNECTING ||
- peer->state == ENET_PEER_STATE_CONNECTING)
- {
- enet_protocol_dispatch_state(host, peer, ENET_PEER_STATE_ZOMBIE);
- } else if (peer->state != ENET_PEER_STATE_CONNECTED && peer->state != ENET_PEER_STATE_DISCONNECT_LATER) {
- if (peer->state == ENET_PEER_STATE_CONNECTION_PENDING) { host->recalculateBandwidthLimits = 1; }
- enet_peer_reset(peer);
- } else if (command->header.command & ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE) {
- enet_protocol_change_state(host, peer, ENET_PEER_STATE_ACKNOWLEDGING_DISCONNECT);
- } else {
- enet_protocol_dispatch_state(host, peer, ENET_PEER_STATE_ZOMBIE);
- }
- if (peer->state != ENET_PEER_STATE_DISCONNECTED) {
- peer->eventData = ENET_NET_TO_HOST_32(command->disconnect.data);
- }
- return 0;
- }
- static int enet_protocol_handle_acknowledge(ENetHost *host, ENetEvent *event, ENetPeer *peer,
- const ENetProtocol *command) {
- enet_uint32 roundTripTime,
- receivedSentTime,
- receivedReliableSequenceNumber;
- ENetProtocolCommand commandNumber;
- if (peer->state == ENET_PEER_STATE_DISCONNECTED || peer->state == ENET_PEER_STATE_ZOMBIE) {
- return 0;
- }
- receivedSentTime = ENET_NET_TO_HOST_16(command->acknowledge.receivedSentTime);
- receivedSentTime |= host->serviceTime & 0xFFFF0000;
- if ((receivedSentTime & 0x8000) > (host->serviceTime & 0x8000)) {
- receivedSentTime -= 0x10000;
- }
- if (ENET_TIME_LESS(host->serviceTime, receivedSentTime)) {
- return 0;
- }
- peer->lastReceiveTime = host->serviceTime;
- peer->earliestTimeout = 0;
- roundTripTime = ENET_TIME_DIFFERENCE(host->serviceTime, receivedSentTime);
- enet_peer_throttle(peer, roundTripTime);
- peer->roundTripTimeVariance -= peer->roundTripTimeVariance / 4;
- if (roundTripTime >= peer->roundTripTime) {
- peer->roundTripTime += (roundTripTime - peer->roundTripTime) / 8;
- peer->roundTripTimeVariance += (roundTripTime - peer->roundTripTime) / 4;
- } else {
- peer->roundTripTime -= (peer->roundTripTime - roundTripTime) / 8;
- peer->roundTripTimeVariance += (peer->roundTripTime - roundTripTime) / 4;
- }
- if (peer->roundTripTime < peer->lowestRoundTripTime) {
- peer->lowestRoundTripTime = peer->roundTripTime;
- }
- if (peer->roundTripTimeVariance > peer->highestRoundTripTimeVariance) {
- peer->highestRoundTripTimeVariance = peer->roundTripTimeVariance;
- }
- if (peer->packetThrottleEpoch == 0 ||
- ENET_TIME_DIFFERENCE(host->serviceTime, peer->packetThrottleEpoch) >= peer->packetThrottleInterval)
- {
- peer->lastRoundTripTime = peer->lowestRoundTripTime;
- peer->lastRoundTripTimeVariance = peer->highestRoundTripTimeVariance;
- peer->lowestRoundTripTime = peer->roundTripTime;
- peer->highestRoundTripTimeVariance = peer->roundTripTimeVariance;
- peer->packetThrottleEpoch = host->serviceTime;
- }
- receivedReliableSequenceNumber = ENET_NET_TO_HOST_16(command->acknowledge.receivedReliableSequenceNumber);
- commandNumber = enet_protocol_remove_sent_reliable_command(peer, receivedReliableSequenceNumber,
- command->header.channelID);
- switch (peer->state) {
- case ENET_PEER_STATE_ACKNOWLEDGING_CONNECT:
- if (commandNumber != ENET_PROTOCOL_COMMAND_VERIFY_CONNECT) {
- return -1;
- }
- enet_protocol_notify_connect(host, peer, event);
- break;
- case ENET_PEER_STATE_DISCONNECTING:
- if (commandNumber != ENET_PROTOCOL_COMMAND_DISCONNECT) {
- return -1;
- }
- enet_protocol_notify_disconnect(host, peer, event);
- break;
- case ENET_PEER_STATE_DISCONNECT_LATER:
- if (enet_list_empty(&peer->outgoingReliableCommands) &&
- enet_list_empty(&peer->outgoingUnreliableCommands) &&
- enet_list_empty(&peer->sentReliableCommands))
- {
- enet_peer_disconnect(peer, peer->eventData);
- }
- break;
- default:
- break;
- }
- return 0;
- } // enet_protocol_handle_acknowledge
- static int enet_protocol_handle_verify_connect(ENetHost *host, ENetEvent *event, ENetPeer *peer,
- const ENetProtocol *command) {
- enet_uint32 mtu, windowSize;
- size_t channelCount;
- if (peer->state != ENET_PEER_STATE_CONNECTING) {
- return 0;
- }
- channelCount = ENET_NET_TO_HOST_32(command->verifyConnect.channelCount);
- if (channelCount < ENET_PROTOCOL_MINIMUM_CHANNEL_COUNT || channelCount > ENET_PROTOCOL_MAXIMUM_CHANNEL_COUNT ||
- ENET_NET_TO_HOST_32(command->verifyConnect.packetThrottleInterval) != peer->packetThrottleInterval ||
- ENET_NET_TO_HOST_32(command->verifyConnect.packetThrottleAcceleration) != peer->packetThrottleAcceleration ||
- ENET_NET_TO_HOST_32(command->verifyConnect.packetThrottleDeceleration) != peer->packetThrottleDeceleration ||
- command->verifyConnect.connectID != peer->connectID)
- {
- peer->eventData = 0;
- enet_protocol_dispatch_state(host, peer, ENET_PEER_STATE_ZOMBIE);
- return -1;
- }
- enet_protocol_remove_sent_reliable_command(peer, 1, 0xFF);
- if (channelCount < peer->channelCount) {
- peer->channelCount = channelCount;
- }
- peer->outgoingPeerID = ENET_NET_TO_HOST_16(command->verifyConnect.outgoingPeerID);
- peer->incomingSessionID = command->verifyConnect.incomingSessionID;
- peer->outgoingSessionID = command->verifyConnect.outgoingSessionID;
- mtu = ENET_NET_TO_HOST_32(command->verifyConnect.mtu);
- if (mtu < ENET_PROTOCOL_MINIMUM_MTU) {
- mtu = ENET_PROTOCOL_MINIMUM_MTU;
- } else if (mtu > ENET_PROTOCOL_MAXIMUM_MTU) {
- mtu = ENET_PROTOCOL_MAXIMUM_MTU;
- }
- if (mtu < peer->mtu) {
- peer->mtu = mtu;
- }
- windowSize = ENET_NET_TO_HOST_32(command->verifyConnect.windowSize);
- if (windowSize < ENET_PROTOCOL_MINIMUM_WINDOW_SIZE) {
- windowSize = ENET_PROTOCOL_MINIMUM_WINDOW_SIZE;
- }
- if (windowSize > ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE) {
- windowSize = ENET_PROTOCOL_MAXIMUM_WINDOW_SIZE;
- }
- if (windowSize < peer->windowSize) {
- peer->windowSize = windowSize;
- }
- peer->incomingBandwidth = ENET_NET_TO_HOST_32(command->verifyConnect.incomingBandwidth);
- peer->outgoingBandwidth = ENET_NET_TO_HOST_32(command->verifyConnect.outgoingBandwidth);
- enet_protocol_notify_connect(host, peer, event);
- return 0;
- } // enet_protocol_handle_verify_connect
- static int enet_protocol_handle_incoming_commands(ENetHost *host, ENetEvent *event) {
- ENetProtocolHeader *header;
- ENetProtocol *command;
- ENetPeer *peer;
- enet_uint8 *currentData;
- size_t headerSize;
- enet_uint16 peerID, flags;
- enet_uint8 sessionID;
- if (host->receivedDataLength < (size_t) &((ENetProtocolHeader *) 0)->sentTime) {
- return 0;
- }
- header = (ENetProtocolHeader *) host->receivedData;
- peerID = ENET_NET_TO_HOST_16(header->peerID);
- sessionID = (peerID & ENET_PROTOCOL_HEADER_SESSION_MASK) >> ENET_PROTOCOL_HEADER_SESSION_SHIFT;
- flags = peerID & ENET_PROTOCOL_HEADER_FLAG_MASK;
- peerID &= ~(ENET_PROTOCOL_HEADER_FLAG_MASK | ENET_PROTOCOL_HEADER_SESSION_MASK);
- headerSize =
- (flags
- & ENET_PROTOCOL_HEADER_FLAG_SENT_TIME ? sizeof(ENetProtocolHeader) : (size_t) &((ENetProtocolHeader *) 0)->
- sentTime);
- if (host->checksum != NULL) {
- headerSize += sizeof(enet_uint32);
- }
- if (peerID == ENET_PROTOCOL_MAXIMUM_PEER_ID) {
- peer = NULL;
- } else if (peerID >= host->peerCount) {
- return 0;
- } else {
- peer = &host->peers [peerID];
- if (peer->state == ENET_PEER_STATE_DISCONNECTED ||
- peer->state == ENET_PEER_STATE_ZOMBIE ||
- ((host->receivedAddress.host != peer->address.host ||
- host->receivedAddress.port != peer->address.port) &&
- peer->address.host != ENET_HOST_BROADCAST) ||
- (peer->outgoingPeerID < ENET_PROTOCOL_MAXIMUM_PEER_ID &&
- sessionID != peer->incomingSessionID))
- {
- return 0;
- }
- }
- if (flags & ENET_PROTOCOL_HEADER_FLAG_COMPRESSED) {
- size_t originalSize;
- if (host->compressor.context == NULL || host->compressor.decompress == NULL) {
- return 0;
- }
- originalSize = host->compressor.decompress(host->compressor.context,
- host->receivedData + headerSize,
- host->receivedDataLength - headerSize,
- host->packetData [1] + headerSize,
- sizeof(host->packetData [1]) - headerSize);
- if (originalSize <= 0 || originalSize > sizeof(host->packetData [1]) - headerSize) {
- return 0;
- }
- memcpy(host->packetData [1], header, headerSize);
- host->receivedData = host->packetData [1];
- host->receivedDataLength = headerSize + originalSize;
- }
- if (host->checksum != NULL) {
- enet_uint32 *checksum = (enet_uint32 *) &host->receivedData [headerSize - sizeof(enet_uint32)],
- desiredChecksum = *checksum;
- ENetBuffer buffer;
- *checksum = peer != NULL ? peer->connectID : 0;
- buffer.data = host->receivedData;
- buffer.dataLength = host->receivedDataLength;
- if (host->checksum(&buffer, 1) != desiredChecksum) {
- return 0;
- }
- }
- if (peer != NULL) {
- peer->address.host = host->receivedAddress.host;
- peer->address.port = host->receivedAddress.port;
- peer->incomingDataTotal += host->receivedDataLength;
- }
- currentData = host->receivedData + headerSize;
- while (currentData < &host->receivedData [host->receivedDataLength]) {
- enet_uint8 commandNumber;
- size_t commandSize;
- command = (ENetProtocol *) currentData;
- if (currentData + sizeof(ENetProtocolCommandHeader) > &host->receivedData [host->receivedDataLength]) {
- break;
- }
- commandNumber = command->header.command & ENET_PROTOCOL_COMMAND_MASK;
- if (commandNumber >= ENET_PROTOCOL_COMMAND_COUNT) {
- break;
- }
- commandSize = commandSizes [commandNumber];
- if (commandSize == 0 || currentData + commandSize > &host->receivedData [host->receivedDataLength]) {
- break;
- }
- currentData += commandSize;
- if (peer == NULL && commandNumber != ENET_PROTOCOL_COMMAND_CONNECT) {
- break;
- }
- command->header.reliableSequenceNumber = ENET_NET_TO_HOST_16(command->header.reliableSequenceNumber);
- switch (commandNumber) {
- case ENET_PROTOCOL_COMMAND_ACKNOWLEDGE:
- if (enet_protocol_handle_acknowledge(host, event, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_CONNECT:
- if (peer != NULL) {
- goto commandError;
- }
- peer = enet_protocol_handle_connect(host, header, command);
- if (peer == NULL) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_VERIFY_CONNECT:
- if (enet_protocol_handle_verify_connect(host, event, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_DISCONNECT:
- if (enet_protocol_handle_disconnect(host, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_PING:
- if (enet_protocol_handle_ping(host, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_RELIABLE:
- if (enet_protocol_handle_send_reliable(host, peer, command, ¤tData)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE:
- if (enet_protocol_handle_send_unreliable(host, peer, command, ¤tData)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNSEQUENCED:
- if (enet_protocol_handle_send_unsequenced(host, peer, command, ¤tData)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_FRAGMENT:
- if (enet_protocol_handle_send_fragment(host, peer, command, ¤tData)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_BANDWIDTH_LIMIT:
- if (enet_protocol_handle_bandwidth_limit(host, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_THROTTLE_CONFIGURE:
- if (enet_protocol_handle_throttle_configure(host, peer, command)) {
- goto commandError;
- }
- break;
- case ENET_PROTOCOL_COMMAND_SEND_UNRELIABLE_FRAGMENT:
- if (enet_protocol_handle_send_unreliable_fragment(host, peer, command, ¤tData)) {
- goto commandError;
- }
- break;
- default:
- goto commandError;
- }
- if (peer != NULL &&
- (command->header.command & ENET_PROTOCOL_COMMAND_FLAG_ACKNOWLEDGE) != 0)
- {
- enet_uint16 sentTime;
- if (!(flags & ENET_PROTOCOL_HEADER_FLAG_SENT_TIME)) {
- break;
- }
- sentTime = ENET_NET_TO_HOST_16(header->sentTime);
- switch (peer->state) {
- case ENET_PEER_STATE_DISCONNECTING:
- case ENET_PEER_STATE_ACKNOWLEDGING_CONNECT:
- case ENET_PEER_STATE_DISCONNECTED:
- case ENET_PEER_STATE_ZOMBIE:
- break;
- case ENET_PEER_STATE_ACKNOWLEDGING_DISCONNECT:
- if ((command->header.command & ENET_PROTOCOL_COMMAND_MASK) == ENET_PROTOCOL_COMMAND_DISCONNECT) {
- enet_peer_queue_acknowledgement(peer, command, sentTime);
- }
- break;
- default:
- enet_peer_queue_acknowledgement(peer, command, sentTime);
- break;
- }
- }
- }
- commandError:
- if (event != NULL && event->type != ENET_EVENT_TYPE_NONE) {
- return 1;
- }
- return 0;
- } // enet_protocol_handle_incoming_commands
- static int enet_protocol_receive_incoming_commands(ENetHost *host, ENetEvent *event) {
- int packets;
- for (packets = 0; packets < 256; ++packets) {
- int receivedLength;
- ENetBuffer buffer;
- buffer.data = host->packetData [0];
- buffer.dataLength = sizeof(host->packetData [0]);
- receivedLength = enet_socket_receive(host->socket,
- &host->receivedAddress,
- &buffer,
- 1);
- if (receivedLength < 0) {
- return -1;
- }
- if (receivedLength == 0) {
- return 0;
- }
- host->receivedData = host->packetData [0];
- host->receivedDataLength = receivedLength;
- host->totalReceivedData += receivedLength;
- host->totalReceivedPackets++;
- if (host->intercept != NULL) {
- switch (host->intercept(host, event)) {
- case 1:
- if (event != NULL && event->type != ENET_EVENT_TYPE_NONE) {
- return 1;
- }
- continue;
- case -1:
- return -1;
- default:
- break;
- }
- }
- switch (enet_protocol_handle_incoming_commands(host, event)) {
- case 1:
- return 1;
- case -1:
- return -1;
- default:
- break;
- }
- }
- return -1;
- } // enet_protocol_receive_incoming_commands
- static void enet_protocol_send_acknowledgements(ENetHost *host, ENetPeer *peer) {
- ENetProtocol *command = &host->commands [host->commandCount];
- ENetBuffer *buffer = &host->buffers [host->bufferCount];
- ENetAcknowledgement *acknowledgement;
- ENetListIterator currentAcknowledgement;
- enet_uint16 reliableSequenceNumber;
- currentAcknowledgement = enet_list_begin(&peer->acknowledgements);
- while (currentAcknowledgement != enet_list_end(&peer->acknowledgements)) {
- if (command >= &host->commands [sizeof(host->commands) / sizeof(ENetProtocol)] ||
- buffer >= &host->buffers [sizeof(host->buffers) / sizeof(ENetBuffer)] ||
- peer->mtu - host->packetSize < sizeof(ENetProtocolAcknowledge))
- {
- host->continueSending = 1;
- break;
- }
- acknowledgement = (ENetAcknowledgement *) currentAcknowledgement;
- currentAcknowledgement = enet_list_next(currentAcknowledgement);
- buffer->data = command;
- buffer->dataLength = sizeof(ENetProtocolAcknowledge);
- host->packetSize += buffer->dataLength;
- reliableSequenceNumber = ENET_HOST_TO_NET_16(acknowledgement->command.header.reliableSequenceNumber);
- command->header.command = ENET_PROTOCOL_COMMAND_ACKNOWLEDGE;
- command->header.channelID = acknowledgement->command.header.channelID;
- command->header.reliableSequenceNumber = reliableSequenceNumber;
- command->acknowledge.receivedReliableSequenceNumber = reliableSequenceNumber;
- command->acknowledge.receivedSentTime = ENET_HOST_TO_NET_16(acknowledgement->sentTime);
- if ((acknowledgement->command.header.command & ENET_PROTOCOL_COMMAND_MASK) ==
- ENET_PROTOCOL_COMMAND_DISCONNECT)
- {
- enet_protocol_dispatch_state(host, peer, ENET_PEER_STATE_ZOMBIE);
- }
- enet_list_remove(&acknowledgement->acknowledgementList);
- enet_free(acknowledgement);
- ++command;
- ++buffer;
- }
- host->commandCount = command - host->commands;
- host->bufferCount = buffer - host->buffers;
- } // enet_protocol_send_acknowledgements
- static void enet_protocol_send_unreliable_outgoing_commands(ENetHost *host, ENetPeer *peer) {
- ENetProtocol *command = &host->commands [host->commandCount];
- ENetBuffer *buffer = &host->buffers [host->bufferCount];
- ENetOutgoingCommand *outgoingCommand;
- ENetListIterator currentCommand;
- currentCommand = enet_list_begin(&peer->outgoingUnreliableCommands);
- while (currentCommand != enet_list_end(&peer->outgoingUnreliableCommands)) {
- size_t commandSize;
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- commandSize = commandSizes [outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK];
- if (command >= &host->commands [sizeof(host->commands) / sizeof(ENetProtocol)] ||
- buffer + 1 >= &host->buffers [sizeof(host->buffers) / sizeof(ENetBuffer)] ||
- peer->mtu - host->packetSize < commandSize ||
- (outgoingCommand->packet != NULL &&
- peer->mtu - host->packetSize < commandSize + outgoingCommand->fragmentLength))
- {
- host->continueSending = 1;
- break;
- }
- currentCommand = enet_list_next(currentCommand);
- if (outgoingCommand->packet != NULL && outgoingCommand->fragmentOffset == 0) {
- peer->packetThrottleCounter += ENET_PEER_PACKET_THROTTLE_COUNTER;
- peer->packetThrottleCounter %= ENET_PEER_PACKET_THROTTLE_SCALE;
- if (peer->packetThrottleCounter > peer->packetThrottle) {
- enet_uint16 reliableSequenceNumber = outgoingCommand->reliableSequenceNumber,
- unreliableSequenceNumber = outgoingCommand->unreliableSequenceNumber;
- for (;;) {
- --outgoingCommand->packet->referenceCount;
- if (outgoingCommand->packet->referenceCount == 0) {
- enet_packet_destroy(outgoingCommand->packet);
- }
- enet_list_remove(&outgoingCommand->outgoingCommandList);
- enet_free(outgoingCommand);
- if (currentCommand == enet_list_end(&peer->outgoingUnreliableCommands)) {
- break;
- }
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- if (outgoingCommand->reliableSequenceNumber != reliableSequenceNumber ||
- outgoingCommand->unreliableSequenceNumber != unreliableSequenceNumber)
- {
- break;
- }
- currentCommand = enet_list_next(currentCommand);
- }
- continue;
- }
- }
- buffer->data = command;
- buffer->dataLength = commandSize;
- host->packetSize += buffer->dataLength;
- *command = outgoingCommand->command;
- enet_list_remove(&outgoingCommand->outgoingCommandList);
- if (outgoingCommand->packet != NULL) {
- ++buffer;
- buffer->data = outgoingCommand->packet->data + outgoingCommand->fragmentOffset;
- buffer->dataLength = outgoingCommand->fragmentLength;
- host->packetSize += buffer->dataLength;
- enet_list_insert(enet_list_end(&peer->sentUnreliableCommands), outgoingCommand);
- } else {
- enet_free(outgoingCommand);
- }
- ++command;
- ++buffer;
- }
- host->commandCount = command - host->commands;
- host->bufferCount = buffer - host->buffers;
- if (peer->state == ENET_PEER_STATE_DISCONNECT_LATER &&
- enet_list_empty(&peer->outgoingReliableCommands) &&
- enet_list_empty(&peer->outgoingUnreliableCommands) &&
- enet_list_empty(&peer->sentReliableCommands))
- {
- enet_peer_disconnect(peer, peer->eventData);
- }
- } // enet_protocol_send_unreliable_outgoing_commands
- static int enet_protocol_check_timeouts(ENetHost *host, ENetPeer *peer, ENetEvent *event) {
- ENetOutgoingCommand *outgoingCommand;
- ENetListIterator currentCommand, insertPosition;
- currentCommand = enet_list_begin(&peer->sentReliableCommands);
- insertPosition = enet_list_begin(&peer->outgoingReliableCommands);
- while (currentCommand != enet_list_end(&peer->sentReliableCommands)) {
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- currentCommand = enet_list_next(currentCommand);
- if (ENET_TIME_DIFFERENCE(host->serviceTime, outgoingCommand->sentTime) < outgoingCommand->roundTripTimeout) {
- continue;
- }
- if (peer->earliestTimeout == 0 ||
- ENET_TIME_LESS(outgoingCommand->sentTime, peer->earliestTimeout))
- {
- peer->earliestTimeout = outgoingCommand->sentTime;
- }
- if (peer->earliestTimeout != 0 &&
- (ENET_TIME_DIFFERENCE(host->serviceTime, peer->earliestTimeout) >= peer->timeoutMaximum ||
- (outgoingCommand->roundTripTimeout >= outgoingCommand->roundTripTimeoutLimit &&
- ENET_TIME_DIFFERENCE(host->serviceTime, peer->earliestTimeout) >= peer->timeoutMinimum)))
- {
- enet_protocol_notify_disconnect(host, peer, event);
- return 1;
- }
- if (outgoingCommand->packet != NULL) {
- peer->reliableDataInTransit -= outgoingCommand->fragmentLength;
- }
- ++peer->packetsLost;
- outgoingCommand->roundTripTimeout *= 2;
- enet_list_insert(insertPosition, enet_list_remove(&outgoingCommand->outgoingCommandList));
- if (currentCommand == enet_list_begin(&peer->sentReliableCommands) &&
- !enet_list_empty(&peer->sentReliableCommands))
- {
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- peer->nextTimeout = outgoingCommand->sentTime + outgoingCommand->roundTripTimeout;
- }
- }
- return 0;
- } // enet_protocol_check_timeouts
- static int enet_protocol_send_reliable_outgoing_commands(ENetHost *host, ENetPeer *peer) {
- ENetProtocol *command = &host->commands [host->commandCount];
- ENetBuffer *buffer = &host->buffers [host->bufferCount];
- ENetOutgoingCommand *outgoingCommand;
- ENetListIterator currentCommand;
- ENetChannel *channel;
- enet_uint16 reliableWindow;
- size_t commandSize;
- int windowExceeded = 0, windowWrap = 0, canPing = 1;
- currentCommand = enet_list_begin(&peer->outgoingReliableCommands);
- while (currentCommand != enet_list_end(&peer->outgoingReliableCommands)) {
- outgoingCommand = (ENetOutgoingCommand *) currentCommand;
- channel = outgoingCommand->command.header.channelID <
- peer->channelCount ? &peer->channels [outgoingCommand->command.header.channelID] : NULL;
- reliableWindow = outgoingCommand->reliableSequenceNumber / ENET_PEER_RELIABLE_WINDOW_SIZE;
- if (channel != NULL) {
- if (!windowWrap &&
- outgoingCommand->sendAttempts < 1 &&
- !(outgoingCommand->reliableSequenceNumber % ENET_PEER_RELIABLE_WINDOW_SIZE) &&
- (channel->reliableWindows [(reliableWindow + ENET_PEER_RELIABLE_WINDOWS - 1)
- % ENET_PEER_RELIABLE_WINDOWS] >= ENET_PEER_RELIABLE_WINDOW_SIZE ||
- channel->usedReliableWindows & ((((1 << ENET_PEER_FREE_RELIABLE_WINDOWS) - 1) << reliableWindow)
- | (((1 << ENET_PEER_FREE_RELIABLE_WINDOWS) - 1) >> (ENET_PEER_RELIABLE_WINDOWS - reliableWindow)))))
- {
- windowWrap = 1;
- }
- if (windowWrap) {
- currentCommand = enet_list_next(currentCommand);
- continue;
- }
- }
- if (outgoingCommand->packet != NULL) {
- if (!windowExceeded) {
- enet_uint32 windowSize = (peer->packetThrottle * peer->windowSize) / ENET_PEER_PACKET_THROTTLE_SCALE;
- if (peer->reliableDataInTransit + outgoingCommand->fragmentLength > ENET_MAX(windowSize, peer->mtu)) {
- windowExceeded = 1;
- }
- }
- if (windowExceeded) {
- currentCommand = enet_list_next(currentCommand);
- continue;
- }
- }
- canPing = 0;
- commandSize = commandSizes [outgoingCommand->command.header.command & ENET_PROTOCOL_COMMAND_MASK];
- if (command >= &host->commands [sizeof(host->commands) / sizeof(ENetProtocol)] ||
- buffer + 1 >= &host->buffers [sizeof(host->buffers) / sizeof(ENetBuffer)] ||
- peer->mtu - host->packetSize < commandSize ||
- (outgoingCommand->packet != NULL &&
- (enet_uint16) (peer->mtu - host->packetSize) < (enet_uint16) (commandSize + outgoingCommand->fragmentLength)))
- {
- host->continueSending = 1;
- break;
- }
- currentCommand = enet_list_next(currentCommand);
- if (channel != NULL && outgoingCommand->sendAttempts < 1) {
- channel->usedReliableWindows |= 1 << reliableWindow;
- ++channel->reliableWindows [reliableWindow];
- }
- ++outgoingCommand->sendAttempts;
- if (outgoingCommand->roundTripTimeout == 0) {
- outgoingCommand->roundTripTimeout = peer->roundTripTime + 4 * peer->roundTripTimeVariance;
- outgoingCommand->roundTripTimeoutLimit = peer->timeoutLimit * outgoingCommand->roundTripTimeout;
- }
- if (enet_list_empty(&peer->sentReliableCommands)) {
- peer->nextTimeout = host->serviceTime + outgoingCommand->roundTripTimeout;
- }
- enet_list_insert(enet_list_end(&peer->sentReliableCommands),
- enet_list_remove(&outgoingCommand->outgoingCommandList));
- outgoingCommand->sentTime = host->serviceTime;
- buffer->data = command;
- buffer->dataLength = commandSize;
- host->packetSize += buffer->dataLength;
- host->headerFlags |= ENET_PROTOCOL_HEADER_FLAG_SENT_TIME;
- *command = outgoingCommand->command;
- if (outgoingCommand->packet != NULL) {
- ++buffer;
- buffer->data = outgoingCommand->packet->data + outgoingCommand->fragmentOffset;
- buffer->dataLength = outgoingCommand->fragmentLength;
- host->packetSize += outgoingCommand->fragmentLength;
- peer->reliableDataInTransit += outgoingCommand->fragmentLength;
- }
- ++peer->packetsSent;
- ++command;
- ++buffer;
- }
- host->commandCount = command - host->commands;
- host->bufferCount = buffer - host->buffers;
- return canPing;
- } // enet_protocol_send_reliable_outgoing_commands
- static int enet_protocol_send_outgoing_commands(ENetHost *host, ENetEvent *event, int checkForTimeouts) {
- enet_uint8 headerData [sizeof(ENetProtocolHeader) + sizeof(enet_uint32)];
- ENetProtocolHeader *header = (ENetProtocolHeader *) headerData;
- ENetPeer *currentPeer;
- int sentLength;
- size_t shouldCompress = 0;
- host->continueSending = 1;
- while (host->continueSending)
- for (host->continueSending = 0,
- currentPeer = host->peers;
- currentPeer < &host->peers [host->peerCount];
- ++currentPeer)
- {
- if (currentPeer->state == ENET_PEER_STATE_DISCONNECTED ||
- currentPeer->state == ENET_PEER_STATE_ZOMBIE)
- {
- continue;
- }
- host->headerFlags = 0;
- host->commandCount = 0;
- host->bufferCount = 1;
- host->packetSize = sizeof(ENetProtocolHeader);
- if (!enet_list_empty(¤tPeer->acknowledgements)) {
- enet_protocol_send_acknowledgements(host, currentPeer);
- }
- if (checkForTimeouts != 0 &&
- !enet_list_empty(¤tPeer->sentReliableCommands) &&
- ENET_TIME_GREATER_EQUAL(host->serviceTime, currentPeer->nextTimeout) &&
- enet_protocol_check_timeouts(host, currentPeer, event) == 1)
- {
- if (event != NULL && event->type != ENET_EVENT_TYPE_NONE) {
- return 1;
- } else {
- continue;
- }
- }
- if ((enet_list_empty(¤tPeer->outgoingReliableCommands) ||
- enet_protocol_send_reliable_outgoing_commands(host, currentPeer)) &&
- enet_list_empty(¤tPeer->sentReliableCommands) &&
- ENET_TIME_DIFFERENCE(host->serviceTime, currentPeer->lastReceiveTime) >= currentPeer->pingInterval &&
- currentPeer->mtu - host->packetSize >= sizeof(ENetProtocolPing))
- {
- enet_peer_ping(currentPeer);
- enet_protocol_send_reliable_outgoing_commands(host, currentPeer);
- }
- if (!enet_list_empty(¤tPeer->outgoingUnreliableCommands)) {
- enet_protocol_send_unreliable_outgoing_commands(host, currentPeer);
- }
- if (host->commandCount == 0) {
- continue;
- }
- if (currentPeer->packetLossEpoch == 0) {
- currentPeer->packetLossEpoch = host->serviceTime;
- } else if (ENET_TIME_DIFFERENCE(host->serviceTime,
- currentPeer->packetLossEpoch) >= ENET_PEER_PACKET_LOSS_INTERVAL &&
- currentPeer->packetsSent > 0)
- {
- enet_uint32 packetLoss = currentPeer->packetsLost * ENET_PEER_PACKET_LOSS_SCALE
- / currentPeer->packetsSent;
- # ifdef ENET_DEBUG
- printf(
- "peer %u: %f%%+-%f%% packet loss, %u+-%u ms round trip time, %f%% throttle, %u/%u outgoing, %u/%u incoming\n", currentPeer->incomingPeerID,
- currentPeer->packetLoss / (float) ENET_PEER_PACKET_LOSS_SCALE,
- currentPeer->packetLossVariance / (float) ENET_PEER_PACKET_LOSS_SCALE, currentPeer->roundTripTime, currentPeer->roundTripTimeVariance,
- currentPeer->packetThrottle / (float) ENET_PEER_PACKET_THROTTLE_SCALE,
- enet_list_size(¤tPeer->outgoingReliableCommands),
- enet_list_size(
- ¤tPeer->outgoingUnreliableCommands),
- currentPeer->channels != NULL ? enet_list_size(
- ¤tPeer->channels->incomingReliableCommands) : 0,
- currentPeer->channels !=
- NULL ? enet_list_size(¤tPeer->channels->incomingUnreliableCommands) : 0);
- # endif
- currentPeer->packetLossVariance -= currentPeer->packetLossVariance / 4;
- if (packetLoss >= currentPeer->packetLoss) {
- currentPeer->packetLoss += (packetLoss - currentPeer->packetLoss) / 8;
- currentPeer->packetLossVariance += (packetLoss - currentPeer->packetLoss) / 4;
- } else {
- currentPeer->packetLoss -= (currentPeer->packetLoss - packetLoss) / 8;
- currentPeer->packetLossVariance += (currentPeer->packetLoss - packetLoss) / 4;
- }
- currentPeer->packetLossEpoch = host->serviceTime;
- currentPeer->packetsSent = 0;
- currentPeer->packetsLost = 0;
- }
- host->buffers->data = headerData;
- if (host->headerFlags & ENET_PROTOCOL_HEADER_FLAG_SENT_TIME) {
- header->sentTime = ENET_HOST_TO_NET_16(host->serviceTime & 0xFFFF);
- host->buffers->dataLength = sizeof(ENetProtocolHeader);
- } else {
- host->buffers->dataLength = (size_t) &((ENetProtocolHeader *) 0)->sentTime;
- }
- shouldCompress = 0;
- if (host->compressor.context != NULL && host->compressor.compress != NULL) {
- size_t originalSize = host->packetSize - sizeof(ENetProtocolHeader),
- compressedSize = host->compressor.compress(host->compressor.context,
- &host->buffers [1], host->bufferCount - 1,
- originalSize,
- host->packetData [1],
- originalSize);
- if (compressedSize > 0 && compressedSize < originalSize) {
- host->headerFlags |= ENET_PROTOCOL_HEADER_FLAG_COMPRESSED;
- shouldCompress = compressedSize;
- # ifdef ENET_DEBUG_COMPRESS
- printf("peer %u: compressed %u -> %u (%u%%)\n", currentPeer->incomingPeerID, originalSize,
- compressedSize, (compressedSize * 100) / originalSize);
- # endif
- }
- }
- if (currentPeer->outgoingPeerID < ENET_PROTOCOL_MAXIMUM_PEER_ID) {
- host->headerFlags |= currentPeer->outgoingSessionID << ENET_PROTOCOL_HEADER_SESSION_SHIFT;
- }
- header->peerID = ENET_HOST_TO_NET_16(currentPeer->outgoingPeerID | host->headerFlags);
- if (host->checksum != NULL) {
- enet_uint32 *checksum = (enet_uint32 *) &headerData [host->buffers->dataLength];
- *checksum = currentPeer->outgoingPeerID < ENET_PROTOCOL_MAXIMUM_PEER_ID ? currentPeer->connectID : 0;
- host->buffers->dataLength += sizeof(enet_uint32);
- *checksum = host->checksum(host->buffers, host->bufferCount);
- }
- if (shouldCompress > 0) {
- host->buffers [1].data = host->packetData [1];
- host->buffers [1].dataLength = shouldCompress;
- host->bufferCount = 2;
- }
- currentPeer->lastSendTime = host->serviceTime;
- sentLength = enet_socket_send(host->socket, ¤tPeer->address, host->buffers, host->bufferCount);
- enet_protocol_remove_sent_unreliable_commands(currentPeer);
- if (sentLength < 0) {
- return -1;
- }
- host->totalSentData += sentLength;
- host->totalSentPackets++;
- }
- return 0;
- } // enet_protocol_send_outgoing_commands
- /** Sends any queued packets on the host specified to its designated peers.
- *
- * @param host host to flush
- * @remarks this function need only be used in circumstances where one wishes to send queued packets earlier than in a call to enet_host_service().
- * @ingroup host
- */
- void enet_host_flush(ENetHost *host) {
- host->serviceTime = enet_time_get();
- enet_protocol_send_outgoing_commands(host, NULL, 0);
- }
- /** Checks for any queued events on the host and dispatches one if available.
- *
- * @param host host to check for events
- * @param event an event structure where event details will be placed if available
- * @retval > 0 if an event was dispatched
- * @retval 0 if no events are available
- * @retval < 0 on failure
- * @ingroup host
- */
- int enet_host_check_events(ENetHost *host, ENetEvent *event) {
- if (event == NULL) { return -1; }
- event->type = ENET_EVENT_TYPE_NONE;
- event->peer = NULL;
- event->packet = NULL;
- return enet_protocol_dispatch_incoming_commands(host, event);
- }
- /** Waits for events on the host specified and shuttles packets between
- * the host and its peers.
- *
- * @param host host to service
- * @param event an event structure where event details will be placed if one occurs
- * if event == NULL then no events will be delivered
- * @param timeout number of milliseconds that ENet should wait for events
- * @retval > 0 if an event occurred within the specified time limit
- * @retval 0 if no event occurred
- * @retval < 0 on failure
- * @remarks enet_host_service should be called fairly regularly for adequate performance
- * @ingroup host
- */
- int enet_host_service(ENetHost *host, ENetEvent *event, enet_uint32 timeout) {
- enet_uint32 waitCondition;
- if (event != NULL) {
- event->type = ENET_EVENT_TYPE_NONE;
- event->peer = NULL;
- event->packet = NULL;
- switch (enet_protocol_dispatch_incoming_commands(host, event)) {
- case 1:
- return 1;
- case -1:
- # ifdef ENET_DEBUG
- perror("Error dispatching incoming packets");
- # endif
- return -1;
- default:
- break;
- }
- }
- host->serviceTime = enet_time_get();
- timeout += host->serviceTime;
- do {
- if (ENET_TIME_DIFFERENCE(host->serviceTime,
- host->bandwidthThrottleEpoch) >= ENET_HOST_BANDWIDTH_THROTTLE_INTERVAL)
- {
- enet_host_bandwidth_throttle(host);
- }
- switch (enet_protocol_send_outgoing_commands(host, event, 1)) {
- case 1:
- return 1;
- case -1:
- # ifdef ENET_DEBUG
- perror("Error sending outgoing packets");
- # endif
- return -1;
- default:
- break;
- }
- switch (enet_protocol_receive_incoming_commands(host, event)) {
- case 1:
- return 1;
- case -1:
- # ifdef ENET_DEBUG
- perror("Error receiving incoming packets");
- # endif
- return -1;
- default:
- break;
- }
- switch (enet_protocol_send_outgoing_commands(host, event, 1)) {
- case 1:
- return 1;
- case -1:
- # ifdef ENET_DEBUG
- perror("Error sending outgoing packets");
- # endif
- return -1;
- default:
- break;
- }
- if (event != NULL) {
- switch (enet_protocol_dispatch_incoming_commands(host, event)) {
- case 1:
- return 1;
- case -1:
- # ifdef ENET_DEBUG
- perror("Error dispatching incoming packets");
- # endif
- return -1;
- default:
- break;
- }
- }
- if (ENET_TIME_GREATER_EQUAL(host->serviceTime, timeout)) {
- return 0;
- }
- do {
- host->serviceTime = enet_time_get();
- if (ENET_TIME_GREATER_EQUAL(host->serviceTime, timeout)) {
- return 0;
- }
- waitCondition = ENET_SOCKET_WAIT_RECEIVE | ENET_SOCKET_WAIT_INTERRUPT;
- if (enet_socket_wait(host->socket, &waitCondition, ENET_TIME_DIFFERENCE(timeout, host->serviceTime)) != 0) {
- return -1;
- }
- } while (waitCondition & ENET_SOCKET_WAIT_INTERRUPT);
- host->serviceTime = enet_time_get();
- } while (waitCondition & ENET_SOCKET_WAIT_RECEIVE);
- return 0;
- } // enet_host_service
- // @from_file: src/unix.c
- /**
- * @file unix.c
- * @brief ENet Unix system specific functions
- */
- # ifndef _WIN32
- # include <sys/ioctl.h>
- # include <arpa/inet.h>
- # include <netinet/tcp.h>
- # include <netdb.h>
- # include <errno.h>
- # include <time.h>
- # define ENET_BUILDING_LIB 1
- # ifdef __APPLE__
- # ifdef HAS_POLL
- # undef HAS_POLL
- # endif
- # ifndef HAS_FCNTL
- # define HAS_FCNTL 1
- # endif
- # ifndef HAS_INET_PTON
- # define HAS_INET_PTON 1
- # endif
- # ifndef HAS_INET_NTOP
- # define HAS_INET_NTOP 1
- # endif
- # ifndef HAS_MSGHDR_FLAGS
- # define HAS_MSGHDR_FLAGS 1
- # endif
- # ifndef HAS_SOCKLEN_T
- # define HAS_SOCKLEN_T 1
- # endif
- # ifndef HAS_GETADDRINFO
- # define HAS_GETADDRINFO 1
- # endif
- # ifndef HAS_GETNAMEINFO
- # define HAS_GETNAMEINFO 1
- # endif
- # endif // ifdef __APPLE__
- # ifdef HAS_FCNTL
- # include <fcntl.h>
- # endif
- # ifdef HAS_POLL
- # include <sys/poll.h>
- # endif
- # ifndef HAS_SOCKLEN_T
- typedef int socklen_t;
- # endif
- # ifndef MSG_NOSIGNAL
- # define MSG_NOSIGNAL 0
- # endif
- static enet_uint64 timeBase = 0;
- int enet_initialize(void) {
- return 0;
- }
- void enet_deinitialize(void)
- { }
- enet_uint64 enet_host_random_seed(void) {
- return (enet_uint64) time(NULL);
- }
- enet_uint64 enet_time_get(void) {
- struct timeval timeVal;
- gettimeofday(&timeVal, NULL);
- return timeVal.tv_sec * 1000 + timeVal.tv_usec / 1000 - timeBase;
- }
- void enet_time_set(enet_uint64 newTimeBase) {
- struct timeval timeVal;
- gettimeofday(&timeVal, NULL);
- timeBase = timeVal.tv_sec * 1000 + timeVal.tv_usec / 1000 - newTimeBase;
- }
- int enet_address_set_host_ip(ENetAddress *address, const char *name) {
- # ifdef HAS_INET_PTON
- if (!inet_pton(AF_INET, name, &address->host))
- # else
- if (!inet_aton(name, (struct in_addr *) &address->host))
- # endif
- { return -1; }
- return 0;
- }
- int enet_address_set_host(ENetAddress *address, const char *name) {
- # ifdef HAS_GETADDRINFO
- struct addrinfo hints, *resultList = NULL, *result = NULL;
- memset(&hints, 0, sizeof(hints));
- hints.ai_family = AF_INET;
- if (getaddrinfo(name, NULL, NULL, &resultList) != 0) {
- return -1;
- }
- for (result = resultList; result != NULL; result = result->ai_next) {
- if (result->ai_family == AF_INET && result->ai_addr != NULL &&
- result->ai_addrlen >= sizeof(struct sockaddr_in))
- {
- struct sockaddr_in *sin = (struct sockaddr_in *) result->ai_addr;
- address->host = sin->sin_addr.s_addr;
- freeaddrinfo(resultList);
- return 0;
- }
- }
- if (resultList != NULL) {
- freeaddrinfo(resultList);
- }
- # else // ifdef HAS_GETADDRINFO
- struct hostent *hostEntry = NULL;
- # ifdef HAS_GETHOSTBYNAME_R
- struct hostent hostData;
- char buffer [2048];
- int errnum;
- # if defined(linux) || defined(__linux) || defined(__linux__) || defined(__FreeBSD__) || \
- defined(__FreeBSD_kernel__) || defined(__DragonFly__) || defined(__EMSCRIPTEN__)
- gethostbyname_r(name, &hostData, buffer, sizeof(buffer), &hostEntry, &errnum);
- # else
- hostEntry = gethostbyname_r(name, &hostData, buffer, sizeof(buffer), &errnum);
- # endif
- # else
- hostEntry = gethostbyname(name);
- # endif // ifdef HAS_GETHOSTBYNAME_R
- if (hostEntry != NULL && hostEntry->h_addrtype == AF_INET) {
- address->host = *(enet_uint32 *) hostEntry->h_addr_list [0];
- return 0;
- }
- # endif // ifdef HAS_GETADDRINFO
- return enet_address_set_host_ip(address, name);
- } // enet_address_set_host
- int enet_address_get_host_ip(const ENetAddress *address, char *name, size_t nameLength) {
- # ifdef HAS_INET_NTOP
- if (inet_ntop(AF_INET, &address->host, name, nameLength) == NULL)
- # else
- char *addr = inet_ntoa(*(struct in_addr *) &address->host);
- if (addr != NULL) {
- size_t addrLen = strlen(addr);
- if (addrLen >= nameLength) {
- return -1;
- }
- memcpy(name, addr, addrLen + 1);
- } else
- # endif
- { return -1; }
- return 0;
- }
- int enet_address_get_host(const ENetAddress *address, char *name, size_t nameLength) {
- # ifdef HAS_GETNAMEINFO
- struct sockaddr_in sin;
- int err;
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- err = getnameinfo((struct sockaddr *) &sin, sizeof(sin), name, nameLength, NULL, 0, NI_NAMEREQD);
- if (!err) {
- if (name != NULL && nameLength > 0 && !memchr(name, '\0', nameLength)) {
- return -1;
- }
- return 0;
- }
- if (err != EAI_NONAME) {
- return -1;
- }
- # else // ifdef HAS_GETNAMEINFO
- struct in_addr in;
- struct hostent *hostEntry = NULL;
- # ifdef HAS_GETHOSTBYADDR_R
- struct hostent hostData;
- char buffer [2048];
- int errnum;
- in.s_addr = address->host;
- # if defined(linux) || defined(__linux) || defined(__linux__) || defined(__FreeBSD__) || \
- defined(__FreeBSD_kernel__) || defined(__DragonFly__) || defined(__EMSCRIPTEN__)
- gethostbyaddr_r((char *) &in, sizeof(struct in_addr), AF_INET, &hostData, buffer, sizeof(buffer), &hostEntry,
- &errnum);
- # else
- hostEntry = gethostbyaddr_r((char *) &in, sizeof(struct in_addr), AF_INET, &hostData, buffer, sizeof(buffer),
- &errnum);
- # endif
- # else // ifdef HAS_GETHOSTBYADDR_R
- in.s_addr = address->host;
- hostEntry = gethostbyaddr((char *) &in, sizeof(struct in_addr), AF_INET);
- # endif // ifdef HAS_GETHOSTBYADDR_R
- if (hostEntry != NULL) {
- size_t hostLen = strlen(hostEntry->h_name);
- if (hostLen >= nameLength) {
- return -1;
- }
- memcpy(name, hostEntry->h_name, hostLen + 1);
- return 0;
- }
- # endif // ifdef HAS_GETNAMEINFO
- return enet_address_get_host_ip(address, name, nameLength);
- } // enet_address_get_host
- int enet_socket_bind(ENetSocket socket, const ENetAddress *address) {
- struct sockaddr_in sin;
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- if (address != NULL) {
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- } else {
- sin.sin_port = 0;
- sin.sin_addr.s_addr = INADDR_ANY;
- }
- return bind(socket,
- (struct sockaddr *) &sin,
- sizeof(struct sockaddr_in));
- }
- int enet_socket_get_address(ENetSocket socket, ENetAddress *address) {
- struct sockaddr_in sin;
- socklen_t sinLength = sizeof(struct sockaddr_in);
- if (getsockname(socket, (struct sockaddr *) &sin, &sinLength) == -1) {
- return -1;
- }
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- return 0;
- }
- int enet_socket_listen(ENetSocket socket, int backlog) {
- return listen(socket, backlog < 0 ? SOMAXCONN : backlog);
- }
- ENetSocket enet_socket_create(ENetSocketType type) {
- return socket(PF_INET, type == ENET_SOCKET_TYPE_DATAGRAM ? SOCK_DGRAM : SOCK_STREAM, 0);
- }
- int enet_socket_set_option(ENetSocket socket, ENetSocketOption option, int value) {
- int result = -1;
- switch (option) {
- case ENET_SOCKOPT_NONBLOCK:
- # ifdef HAS_FCNTL
- result = fcntl(socket, F_SETFL, (value ? O_NONBLOCK : 0) | (fcntl(socket, F_GETFL) & ~O_NONBLOCK));
- # else
- result = ioctl(socket, FIONBIO, &value);
- # endif
- break;
- case ENET_SOCKOPT_BROADCAST:
- result = setsockopt(socket, SOL_SOCKET, SO_BROADCAST, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_REUSEADDR:
- result = setsockopt(socket, SOL_SOCKET, SO_REUSEADDR, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_RCVBUF:
- result = setsockopt(socket, SOL_SOCKET, SO_RCVBUF, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_SNDBUF:
- result = setsockopt(socket, SOL_SOCKET, SO_SNDBUF, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_RCVTIMEO: {
- struct timeval timeVal;
- timeVal.tv_sec = value / 1000;
- timeVal.tv_usec = (value % 1000) * 1000;
- result = setsockopt(socket, SOL_SOCKET, SO_RCVTIMEO, (char *) &timeVal, sizeof(struct timeval));
- break;
- }
- case ENET_SOCKOPT_SNDTIMEO: {
- struct timeval timeVal;
- timeVal.tv_sec = value / 1000;
- timeVal.tv_usec = (value % 1000) * 1000;
- result = setsockopt(socket, SOL_SOCKET, SO_SNDTIMEO, (char *) &timeVal, sizeof(struct timeval));
- break;
- }
- case ENET_SOCKOPT_NODELAY:
- result = setsockopt(socket, IPPROTO_TCP, TCP_NODELAY, (char *) &value, sizeof(int));
- break;
- default:
- break;
- }
- return result == -1 ? -1 : 0;
- } // enet_socket_set_option
- int enet_socket_get_option(ENetSocket socket, ENetSocketOption option, int *value) {
- int result = -1;
- socklen_t len;
- switch (option) {
- case ENET_SOCKOPT_ERROR:
- len = sizeof(int);
- result = getsockopt(socket, SOL_SOCKET, SO_ERROR, value, &len);
- break;
- default:
- break;
- }
- return result == -1 ? -1 : 0;
- }
- int enet_socket_connect(ENetSocket socket, const ENetAddress *address) {
- struct sockaddr_in sin;
- int result;
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- result = connect(socket, (struct sockaddr *) &sin, sizeof(struct sockaddr_in));
- if (result == -1 && errno == EINPROGRESS) {
- return 0;
- }
- return result;
- }
- ENetSocket enet_socket_accept(ENetSocket socket, ENetAddress *address) {
- int result;
- struct sockaddr_in sin;
- socklen_t sinLength = sizeof(struct sockaddr_in);
- result = accept(socket,
- address != NULL ? (struct sockaddr *) &sin : NULL,
- address != NULL ? &sinLength : NULL);
- if (result == -1) {
- return ENET_SOCKET_NULL;
- }
- if (address != NULL) {
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- }
- return result;
- }
- int enet_socket_shutdown(ENetSocket socket, ENetSocketShutdown how) {
- return shutdown(socket, (int) how);
- }
- void enet_socket_destroy(ENetSocket socket) {
- if (socket != -1) {
- close(socket);
- }
- }
- int enet_socket_send(ENetSocket socket,
- const ENetAddress * address,
- const ENetBuffer * buffers,
- size_t bufferCount) {
- struct msghdr msgHdr;
- struct sockaddr_in sin;
- int sentLength;
- memset(&msgHdr, 0, sizeof(struct msghdr));
- if (address != NULL) {
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- msgHdr.msg_name = &sin;
- msgHdr.msg_namelen = sizeof(struct sockaddr_in);
- }
- msgHdr.msg_iov = (struct iovec *) buffers;
- msgHdr.msg_iovlen = bufferCount;
- sentLength = sendmsg(socket, &msgHdr, MSG_NOSIGNAL);
- if (sentLength == -1) {
- if (errno == EWOULDBLOCK) {
- return 0;
- }
- return -1;
- }
- return sentLength;
- } // enet_socket_send
- int enet_socket_receive(ENetSocket socket,
- ENetAddress * address,
- ENetBuffer * buffers,
- size_t bufferCount) {
- struct msghdr msgHdr;
- struct sockaddr_in sin;
- int recvLength;
- memset(&msgHdr, 0, sizeof(struct msghdr));
- if (address != NULL) {
- msgHdr.msg_name = &sin;
- msgHdr.msg_namelen = sizeof(struct sockaddr_in);
- }
- msgHdr.msg_iov = (struct iovec *) buffers;
- msgHdr.msg_iovlen = bufferCount;
- recvLength = recvmsg(socket, &msgHdr, MSG_NOSIGNAL);
- if (recvLength == -1) {
- if (errno == EWOULDBLOCK) {
- return 0;
- }
- return -1;
- }
- # ifdef HAS_MSGHDR_FLAGS
- if (msgHdr.msg_flags & MSG_TRUNC) {
- return -1;
- }
- # endif
- if (address != NULL) {
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- }
- return recvLength;
- } // enet_socket_receive
- int enet_socketset_select(ENetSocket maxSocket, ENetSocketSet *readSet, ENetSocketSet *writeSet, enet_uint32 timeout) {
- struct timeval timeVal;
- timeVal.tv_sec = timeout / 1000;
- timeVal.tv_usec = (timeout % 1000) * 1000;
- return select(maxSocket + 1, readSet, writeSet, NULL, &timeVal);
- }
- int enet_socket_wait(ENetSocket socket, enet_uint32 *condition, enet_uint64 timeout) {
- # ifdef HAS_POLL
- struct pollfd pollSocket;
- int pollCount;
- pollSocket.fd = socket;
- pollSocket.events = 0;
- if (*condition & ENET_SOCKET_WAIT_SEND) {
- pollSocket.events |= POLLOUT;
- }
- if (*condition & ENET_SOCKET_WAIT_RECEIVE) {
- pollSocket.events |= POLLIN;
- }
- pollCount = poll(&pollSocket, 1, timeout);
- if (pollCount < 0) {
- if (errno == EINTR && *condition & ENET_SOCKET_WAIT_INTERRUPT) {
- *condition = ENET_SOCKET_WAIT_INTERRUPT;
- return 0;
- }
- return -1;
- }
- *condition = ENET_SOCKET_WAIT_NONE;
- if (pollCount == 0) {
- return 0;
- }
- if (pollSocket.revents & POLLOUT) {
- *condition |= ENET_SOCKET_WAIT_SEND;
- }
- if (pollSocket.revents & POLLIN) {
- *condition |= ENET_SOCKET_WAIT_RECEIVE;
- }
- return 0;
- # else // ifdef HAS_POLL
- fd_set readSet, writeSet;
- struct timeval timeVal;
- int selectCount;
- timeVal.tv_sec = timeout / 1000;
- timeVal.tv_usec = (timeout % 1000) * 1000;
- FD_ZERO(&readSet);
- FD_ZERO(&writeSet);
- if (*condition & ENET_SOCKET_WAIT_SEND) {
- FD_SET(socket, &writeSet);
- }
- if (*condition & ENET_SOCKET_WAIT_RECEIVE) {
- FD_SET(socket, &readSet);
- }
- selectCount = select(socket + 1, &readSet, &writeSet, NULL, &timeVal);
- if (selectCount < 0) {
- if (errno == EINTR && *condition & ENET_SOCKET_WAIT_INTERRUPT) {
- *condition = ENET_SOCKET_WAIT_INTERRUPT;
- return 0;
- }
- return -1;
- }
- *condition = ENET_SOCKET_WAIT_NONE;
- if (selectCount == 0) {
- return 0;
- }
- if (FD_ISSET(socket, &writeSet)) {
- *condition |= ENET_SOCKET_WAIT_SEND;
- }
- if (FD_ISSET(socket, &readSet)) {
- *condition |= ENET_SOCKET_WAIT_RECEIVE;
- }
- return 0;
- # endif // ifdef HAS_POLL
- } // enet_socket_wait
- # endif // ifndef _WIN32
- // @from_file: src/win32.c
- /**
- * @file win32.c
- * @brief ENet Win32 system specific functions
- */
- # ifdef _WIN32
- # define ENET_BUILDING_LIB 1
- # include <windows.h>
- # include <mmsystem.h>
- static enet_uint64 timeBase = 0;
- int enet_initialize(void) {
- WORD versionRequested = MAKEWORD(1, 1);
- WSADATA wsaData;
- if (WSAStartup(versionRequested, &wsaData)) {
- return -1;
- }
- if (LOBYTE(wsaData.wVersion) != 1 ||
- HIBYTE(wsaData.wVersion) != 1)
- {
- WSACleanup();
- return -1;
- }
- timeBeginPeriod(1);
- return 0;
- }
- void enet_deinitialize(void) {
- timeEndPeriod(1);
- WSACleanup();
- }
- enet_uint64 enet_host_random_seed(void) {
- return (enet_uint64) timeGetTime();
- }
- enet_uint64 enet_time_get(void) {
- return (enet_uint64) timeGetTime() - timeBase;
- }
- void enet_time_set(enet_uint64 newTimeBase) {
- timeBase = (enet_uint64) timeGetTime() - newTimeBase;
- }
- int enet_address_set_host_ip(ENetAddress *address, const char *name) {
- enet_uint8 vals [4] = { 0, 0, 0, 0 };
- int i;
- for (i = 0; i < 4; ++i) {
- const char *next = name + 1;
- if (*name != '0') {
- long val = strtol(name, (char **) &next, 10);
- if (val < 0 || val > 255 || next == name || next - name > 3) {
- return -1;
- }
- vals [i] = (enet_uint8) val;
- }
- if (*next != (i < 3 ? '.' : '\0')) {
- return -1;
- }
- name = next + 1;
- }
- memcpy(&address->host, vals, sizeof(enet_uint32));
- return 0;
- }
- int enet_address_set_host(ENetAddress *address, const char *name) {
- struct hostent *hostEntry;
- hostEntry = gethostbyname(name);
- if (hostEntry == NULL ||
- hostEntry->h_addrtype != AF_INET)
- {
- return enet_address_set_host_ip(address, name);
- }
- address->host = *(enet_uint32 *) hostEntry->h_addr_list [0];
- return 0;
- }
- int enet_address_get_host_ip(const ENetAddress *address, char *name, size_t nameLength) {
- char *addr = inet_ntoa(*(struct in_addr *) &address->host);
- if (addr == NULL) {
- return -1;
- } else {
- size_t addrLen = strlen(addr);
- if (addrLen >= nameLength) {
- return -1;
- }
- memcpy(name, addr, addrLen + 1);
- }
- return 0;
- }
- int enet_address_get_host(const ENetAddress *address, char *name, size_t nameLength) {
- struct in_addr in;
- struct hostent *hostEntry;
- in.s_addr = address->host;
- hostEntry = gethostbyaddr((char *) &in, sizeof(struct in_addr), AF_INET);
- if (hostEntry == NULL) {
- return enet_address_get_host_ip(address, name, nameLength);
- } else {
- size_t hostLen = strlen(hostEntry->h_name);
- if (hostLen >= nameLength) {
- return -1;
- }
- memcpy(name, hostEntry->h_name, hostLen + 1);
- }
- return 0;
- }
- int enet_socket_bind(ENetSocket socket, const ENetAddress *address) {
- struct sockaddr_in sin;
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- if (address != NULL) {
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- } else {
- sin.sin_port = 0;
- sin.sin_addr.s_addr = INADDR_ANY;
- }
- return bind(socket,
- (struct sockaddr *) &sin,
- sizeof(struct sockaddr_in)) == SOCKET_ERROR ? -1 : 0;
- }
- int enet_socket_get_address(ENetSocket socket, ENetAddress *address) {
- struct sockaddr_in sin;
- int sinLength = sizeof(struct sockaddr_in);
- if (getsockname(socket, (struct sockaddr *) &sin, &sinLength) == -1) {
- return -1;
- }
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- return 0;
- }
- int enet_socket_listen(ENetSocket socket, int backlog) {
- return listen(socket, backlog < 0 ? SOMAXCONN : backlog) == SOCKET_ERROR ? -1 : 0;
- }
- ENetSocket enet_socket_create(ENetSocketType type) {
- return socket(PF_INET, type == ENET_SOCKET_TYPE_DATAGRAM ? SOCK_DGRAM : SOCK_STREAM, 0);
- }
- int enet_socket_set_option(ENetSocket socket, ENetSocketOption option, int value) {
- int result = SOCKET_ERROR;
- switch (option) {
- case ENET_SOCKOPT_NONBLOCK: {
- u_long nonBlocking = (u_long) value;
- result = ioctlsocket(socket, FIONBIO, &nonBlocking);
- break;
- }
- case ENET_SOCKOPT_BROADCAST:
- result = setsockopt(socket, SOL_SOCKET, SO_BROADCAST, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_REUSEADDR:
- result = setsockopt(socket, SOL_SOCKET, SO_REUSEADDR, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_RCVBUF:
- result = setsockopt(socket, SOL_SOCKET, SO_RCVBUF, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_SNDBUF:
- result = setsockopt(socket, SOL_SOCKET, SO_SNDBUF, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_RCVTIMEO:
- result = setsockopt(socket, SOL_SOCKET, SO_RCVTIMEO, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_SNDTIMEO:
- result = setsockopt(socket, SOL_SOCKET, SO_SNDTIMEO, (char *) &value, sizeof(int));
- break;
- case ENET_SOCKOPT_NODELAY:
- result = setsockopt(socket, IPPROTO_TCP, TCP_NODELAY, (char *) &value, sizeof(int));
- break;
- default:
- break;
- }
- return result == SOCKET_ERROR ? -1 : 0;
- } // enet_socket_set_option
- int enet_socket_get_option(ENetSocket socket, ENetSocketOption option, int *value) {
- int result = SOCKET_ERROR, len;
- switch (option) {
- case ENET_SOCKOPT_ERROR:
- len = sizeof(int);
- result = getsockopt(socket, SOL_SOCKET, SO_ERROR, (char *) value, &len);
- break;
- default:
- break;
- }
- return result == SOCKET_ERROR ? -1 : 0;
- }
- int enet_socket_connect(ENetSocket socket, const ENetAddress *address) {
- struct sockaddr_in sin;
- int result;
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- result = connect(socket, (struct sockaddr *) &sin, sizeof(struct sockaddr_in));
- if (result == SOCKET_ERROR && WSAGetLastError() != WSAEWOULDBLOCK) {
- return -1;
- }
- return 0;
- }
- ENetSocket enet_socket_accept(ENetSocket socket, ENetAddress *address) {
- SOCKET result;
- struct sockaddr_in sin;
- int sinLength = sizeof(struct sockaddr_in);
- result = accept(socket,
- address != NULL ? (struct sockaddr *) &sin : NULL,
- address != NULL ? &sinLength : NULL);
- if (result == INVALID_SOCKET) {
- return ENET_SOCKET_NULL;
- }
- if (address != NULL) {
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- }
- return result;
- }
- int enet_socket_shutdown(ENetSocket socket, ENetSocketShutdown how) {
- return shutdown(socket, (int) how) == SOCKET_ERROR ? -1 : 0;
- }
- void enet_socket_destroy(ENetSocket socket) {
- if (socket != INVALID_SOCKET) {
- closesocket(socket);
- }
- }
- int enet_socket_send(ENetSocket socket,
- const ENetAddress * address,
- const ENetBuffer * buffers,
- size_t bufferCount) {
- struct sockaddr_in sin;
- DWORD sentLength;
- if (address != NULL) {
- memset(&sin, 0, sizeof(struct sockaddr_in));
- sin.sin_family = AF_INET;
- sin.sin_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin_addr.s_addr = address->host;
- }
- if (WSASendTo(socket,
- (LPWSABUF) buffers,
- (DWORD) bufferCount,
- &sentLength,
- 0,
- address != NULL ? (struct sockaddr *) &sin : NULL,
- address != NULL ? sizeof(struct sockaddr_in) : 0,
- NULL,
- NULL) == SOCKET_ERROR)
- {
- if (WSAGetLastError() == WSAEWOULDBLOCK) {
- return 0;
- }
- return -1;
- }
- return (int) sentLength;
- }
- int enet_socket_receive(ENetSocket socket,
- ENetAddress * address,
- ENetBuffer * buffers,
- size_t bufferCount) {
- INT sinLength = sizeof(struct sockaddr_in);
- DWORD flags = 0,
- recvLength;
- struct sockaddr_in sin;
- if (WSARecvFrom(socket,
- (LPWSABUF) buffers,
- (DWORD) bufferCount,
- &recvLength,
- &flags,
- address != NULL ? (struct sockaddr *) &sin : NULL,
- address != NULL ? &sinLength : NULL,
- NULL,
- NULL) == SOCKET_ERROR)
- {
- switch (WSAGetLastError()) {
- case WSAEWOULDBLOCK:
- case WSAECONNRESET:
- return 0;
- }
- return -1;
- }
- if (flags & MSG_PARTIAL) {
- return -1;
- }
- if (address != NULL) {
- address->host = (enet_uint32) sin.sin_addr.s_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin_port);
- }
- return (int) recvLength;
- } // enet_socket_receive
- int enet_socketset_select(ENetSocket maxSocket, ENetSocketSet *readSet, ENetSocketSet *writeSet, enet_uint32 timeout) {
- struct timeval timeVal;
- timeVal.tv_sec = timeout / 1000;
- timeVal.tv_usec = (timeout % 1000) * 1000;
- return select(maxSocket + 1, readSet, writeSet, NULL, &timeVal);
- }
- int enet_socket_wait(ENetSocket socket, enet_uint32 *condition, enet_uint64 timeout) {
- fd_set readSet, writeSet;
- struct timeval timeVal;
- int selectCount;
- timeVal.tv_sec = timeout / 1000;
- timeVal.tv_usec = (timeout % 1000) * 1000;
- FD_ZERO(&readSet);
- FD_ZERO(&writeSet);
- if (*condition & ENET_SOCKET_WAIT_SEND) {
- FD_SET(socket, &writeSet);
- }
- if (*condition & ENET_SOCKET_WAIT_RECEIVE) {
- FD_SET(socket, &readSet);
- }
- selectCount = select(socket + 1, &readSet, &writeSet, NULL, &timeVal);
- if (selectCount < 0) {
- return -1;
- }
- *condition = ENET_SOCKET_WAIT_NONE;
- if (selectCount == 0) {
- return 0;
- }
- if (FD_ISSET(socket, &writeSet)) {
- *condition |= ENET_SOCKET_WAIT_SEND;
- }
- if (FD_ISSET(socket, &readSet)) {
- *condition |= ENET_SOCKET_WAIT_RECEIVE;
- }
- return 0;
- } // enet_socket_wait
- # endif // ifdef _WIN32
- # ifdef __cplusplus
- }
- # endif
- # endif // if defined(ENET_IMPLEMENTATION) && !defined(ENET_IMPLEMENTATION_DONE)
- #endif /* __ENET_ENET_H__ */
|