| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169 |
- /*
- * ENet reliable UDP networking library
- * Copyright (c) 2018 Lee Salzman, Vladyslav Hrytsenko, Dominik Madarász, Stanislav Denisov
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in all
- * copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- * SOFTWARE.
- */
- #ifndef ENET_H
- #define ENET_H
- #include <stdarg.h>
- #include <stdbool.h>
- #include <stdint.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <time.h>
- #define ENET_VERSION_MAJOR 2
- #define ENET_VERSION_MINOR 3
- #define ENET_VERSION_PATCH 0
- #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)
- #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))
- #define ENET_MAX(x, y) ((x) > (y) ? (x) : (y))
- #define ENET_MIN(x, y) ((x) < (y) ? (x) : (y))
- #define ENET_SRTT_INITIAL 1.0
- #define ENET_SRTT_PARA_G 0.125
- static FILE *enet_log_fp = NULL;
- enum enet_log_type
- {
- ENET_LOG_TYPE_TRACE,
- ENET_LOG_TYPE_ERROR,
- };
- static const char *const enet_log_type_names[] = {
- [ENET_LOG_TYPE_TRACE] = "TRACE",
- [ENET_LOG_TYPE_ERROR] = "ERROR",
- };
- #if ENET_DEBUG
- #define ENET_LOG_TRACE(...) enet_log(ENET_LOG_TYPE_TRACE, __FUNCTION__, __LINE__, __VA_ARGS__)
- #define ENET_LOG_ERROR(...) g_enet_fp = fopen("enet_log.txt", "a")
- #else
- #define ENET_LOG_TRACE(...) ((void)0)
- #define ENET_LOG_ERROR(...) ((void)0)
- #endif
- #define ENET_LOG_FILE "enet_log.txt"
- static inline void enet_log(enum enet_log_type type, const char *func, int line, const char *fmt, ...)
- {
- if (!enet_log_fp) enet_log_fp = fopen(ENET_LOG_FILE, "a");
- if (!enet_log_fp) return;
- va_list args;
- time_t tstamp = time(NULL);
- struct tm *local_time = localtime(&tstamp);
- char time_buf[64];
- time_buf[strftime(time_buf, sizeof(time_buf), "%Y-%m-%d %H:%M:%S", local_time)] = '\0';
- fprintf(enet_log_fp, "%s [%s] [%s:%d] ", time_buf, enet_log_type_names[type], func, line);
- va_start(args, fmt);
- vfprintf(enet_log_fp, fmt, args);
- va_end(args);
- fflush(enet_log_fp);
- }
- /*
- =======================================================================
- System differences
- =======================================================================
- */
- #ifdef _WIN32
- #if defined(_MSC_VER) && defined(ENET_IMPLEMENTATION)
- #pragma warning(disable: 4244) /* 64-bit to 32-bit integer conversion */
- #pragma warning(disable: 4267) /* size_t to integer conversion */
- #endif
- #ifndef ENET_NO_PRAGMA_LINK
- #pragma comment(lib, "ws2_32.lib")
- #pragma comment(lib, "winmm.lib")
- #endif
- #if _MSC_VER >= 1910
- /* It looks like there were changes as of Visual Studio 2017 and there are no 32/64 bit
- versions of _InterlockedExchange[operation], only InterlockedExchange[operation]
- (without leading underscore), so we have to distinguish between compiler versions */
- #define NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- #endif
- #ifdef __GNUC__
- #if (_WIN32_WINNT < 0x0501)
- #undef _WIN32_WINNT
- #define _WIN32_WINNT 0x0501
- #endif
- #endif
- #include <winsock2.h>
- #include <ws2tcpip.h>
- #include <mmsystem.h>
- #include <intrin.h>
- #if defined(_WIN32) && defined(_MSC_VER)
- #if _MSC_VER < 1900
- typedef struct timespec {
- long tv_sec;
- long tv_nsec;
- };
- #endif
- #define CLOCK_MONOTONIC 0
- #endif
- typedef SOCKET ENetSocket;
- #define ENET_SOCKET_NULL INVALID_SOCKET
- typedef struct {
- size_t dataLength;
- void* data;
- } ENetBuffer;
- #define ENET_CALLBACK __cdecl
- #ifdef ENET_DLL
- #ifdef ENET_IMPLEMENTATION
- #define ENET_API __declspec(dllexport)
- #else
- #define ENET_API __declspec(dllimport)
- #endif
- #else
- #define ENET_API extern
- #endif
- #else
- #include <sys/types.h>
- #include <sys/ioctl.h>
- #include <sys/time.h>
- #include <sys/socket.h>
- #include <poll.h>
- #include <arpa/inet.h>
- #include <netinet/in.h>
- #include <netinet/tcp.h>
- #include <netdb.h>
- #include <unistd.h>
- #include <string.h>
- #include <errno.h>
- #include <fcntl.h>
- #ifdef __APPLE__
- #include <mach/clock.h>
- #include <mach/mach.h>
- #include <Availability.h>
- #endif
- #ifndef MSG_NOSIGNAL
- #define MSG_NOSIGNAL 0
- #endif
- #ifdef MSG_MAXIOVLEN
- #define ENET_BUFFER_MAXIMUM MSG_MAXIOVLEN
- #endif
- typedef int ENetSocket;
- #define ENET_SOCKET_NULL -1
- typedef struct {
- void* data;
- size_t dataLength;
- } ENetBuffer;
- #define ENET_CALLBACK
- #define ENET_API extern
- #endif
- #ifndef ENET_BUFFER_MAXIMUM
- #define ENET_BUFFER_MAXIMUM (1 + 2 * ENET_PROTOCOL_MAXIMUM_PACKET_COMMANDS)
- #endif
- #define ENET_HOST_ANY in6addr_any
- #define ENET_PORT_ANY 0
- #define ENET_HOST_SIZE 1025
- #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))
- #ifdef __cplusplus
- extern "C" {
- #endif
- /*
- =======================================================================
- Internals
- =======================================================================
- */
- typedef uint8_t enet_uint8;
- typedef uint16_t enet_uint16;
- typedef uint32_t enet_uint32;
- typedef uint64_t enet_uint64;
- typedef enet_uint32 ENetVersion;
- typedef fd_set ENetSocketSet;
- typedef struct _ENetCallbacks {
- void* (ENET_CALLBACK *malloc)(size_t size);
- void (ENET_CALLBACK *free)(void* memory);
- void (ENET_CALLBACK *noMemory)(void);
- } ENetCallbacks;
- extern void* enet_malloc(size_t);
- extern void enet_free(void*);
- typedef struct _ENetListNode {
- struct _ENetListNode* next;
- struct _ENetListNode* previous;
- } ENetListNode;
- typedef ENetListNode* ENetListIterator;
- typedef struct _ENetList {
- ENetListNode sentinel;
- } ENetList;
- extern ENetListIterator enet_list_insert(ENetListIterator, void*);
- extern ENetListIterator enet_list_move(ENetListIterator, void*, void*);
- extern void* enet_list_remove(ENetListIterator);
- extern void enet_list_clear(ENetList*);
- 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)
- #ifndef IN4ADDR
- #define IN4ADDR
- struct in4_addr {
- uint8_t zeros[10];
- uint16_t ffff;
- struct in_addr ip;
- };
- #endif
- /*
- =======================================================================
- Protocol
- =======================================================================
- */
- 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_SENT_TIME = (1 << 14),
- ENET_PROTOCOL_HEADER_FLAG_COMPRESSED = (1 << 15),
- ENET_PROTOCOL_HEADER_FLAG_MASK = ENET_PROTOCOL_HEADER_FLAG_SENT_TIME | ENET_PROTOCOL_HEADER_FLAG_COMPRESSED,
- 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
- /*
- =======================================================================
- General structs/enums
- =======================================================================
- */
- 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,
- ENET_SOCKOPT_IPV6_V6ONLY = 10
- } ENetSocketOption;
- typedef enum _ENetSocketShutdown {
- ENET_SOCKET_SHUTDOWN_READ = 0,
- ENET_SOCKET_SHUTDOWN_WRITE = 1,
- ENET_SOCKET_SHUTDOWN_READ_WRITE = 2
- } ENetSocketShutdown;
- typedef struct _ENetAddress {
- union {
- struct in6_addr ipv6;
- struct in4_addr ipv4;
- };
- uint16_t port;
- } ENetAddress;
- #define in6_equal(a, b) (memcmp(&a, &b, sizeof(struct in6_addr)) == 0)
- typedef enum _ENetPacketFlag {
- ENET_PACKET_FLAG_NONE = 0,
- ENET_PACKET_FLAG_RELIABLE = (1 << 0),
- ENET_PACKET_FLAG_UNSEQUENCED = (1 << 1),
- ENET_PACKET_FLAG_NO_ALLOCATE = (1 << 2),
- ENET_PACKET_FLAG_UNRELIABLE_FRAGMENTED = (1 << 3),
- ENET_PACKET_FLAG_INSTANT = (1 << 4),
- ENET_PACKET_FLAG_SENT = (1 << 8)
- } ENetPacketFlag;
- typedef void (ENET_CALLBACK *ENetPacketFreeCallback)(void*);
- typedef struct _ENetPacket {
- enet_uint32 flags;
- enet_uint32 dataLength;
- enet_uint8* data;
- ENetPacketFreeCallback freeCallback;
- enet_uint32 referenceCount;
- } 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;
- enum {
- ENET_HOST_BUFFER_SIZE_MIN = 256 * 1024,
- ENET_HOST_BUFFER_SIZE_MAX = 1024 * 1024,
- ENET_HOST_BANDWIDTH_THROTTLE_INTERVAL = 1000,
- ENET_HOST_DEFAULT_MTU = 1280,
- 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;
- 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;
- void* data;
- ENetPeerState state;
- ENetChannel* channels;
- size_t channelCount;
- enet_uint32 incomingBandwidth;
- enet_uint32 outgoingBandwidth;
- enet_uint32 incomingBandwidthThrottleEpoch;
- enet_uint32 outgoingBandwidthThrottleEpoch;
- enet_uint32 incomingDataTotal;
- enet_uint64 totalDataReceived;
- enet_uint32 outgoingDataTotal;
- enet_uint64 totalDataSent;
- enet_uint32 lastSendTime;
- enet_uint32 lastReceiveTime;
- enet_uint32 nextTimeout;
- enet_uint32 earliestTimeout;
- enet_uint32 packetLossEpoch;
- enet_uint32 packetsSent;
- enet_uint64 totalPacketsSent;
- enet_uint32 packetsLost;
- enet_uint64 totalPacketsLost;
- enet_uint32 packetLoss;
- enet_uint32 packetLossVariance;
- enet_uint32 packetThrottle;
- enet_uint32 packetThrottleThreshold;
- 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 smoothedRoundTripTime;
- enet_uint32 lastRoundTripTime;
- enet_uint32 lowestRoundTripTime;
- enet_uint32 lastRoundTripTimeVariance;
- enet_uint32 highestRoundTripTimeVariance;
- enet_uint32 roundTripTime;
- 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;
- typedef enet_uint32 (ENET_CALLBACK *ENetChecksumCallback)(const ENetBuffer* buffers, size_t bufferCount);
- typedef int (ENET_CALLBACK *ENetInterceptCallback)(struct _ENetHost* host, void* event);
- typedef struct _ENetHost {
- ENetSocket socket;
- ENetAddress address;
- enet_uint32 incomingBandwidth;
- enet_uint32 outgoingBandwidth;
- enet_uint32 bandwidthThrottleEpoch;
- enet_uint32 mtu;
- enet_uint32 randomSeed;
- int recalculateBandwidthLimits;
- enet_uint8 preventConnections;
- ENetPeer* peers;
- size_t peerCount;
- size_t channelLimit;
- enet_uint32 serviceTime;
- ENetList dispatchQueue;
- int continueSending;
- size_t packetSize;
- enet_uint16 headerFlags;
- enet_uint32 totalSentData;
- enet_uint32 totalSentPackets;
- enet_uint32 totalReceivedData;
- enet_uint32 totalReceivedPackets;
- ENetProtocol commands[ENET_PROTOCOL_MAXIMUM_PACKET_COMMANDS];
- size_t commandCount;
- ENetBuffer buffers[ENET_BUFFER_MAXIMUM];
- size_t bufferCount;
- enet_uint8 compression;
- char* compressionBuffer;
- size_t compressionBufferSize;
- ENetChecksumCallback checksumCallback;
- enet_uint8 packetData[2][ENET_PROTOCOL_MAXIMUM_MTU];
- ENetAddress receivedAddress;
- enet_uint8* receivedData;
- size_t receivedDataLength;
- ENetInterceptCallback interceptCallback;
- size_t connectedPeers;
- size_t bandwidthLimitedPeers;
- size_t duplicatePeers;
- size_t maximumPacketSize;
- size_t maximumWaitingData;
- } ENetHost;
- typedef enum _ENetEventType {
- ENET_EVENT_TYPE_NONE = 0,
- ENET_EVENT_TYPE_CONNECT = 1,
- ENET_EVENT_TYPE_DISCONNECT = 2,
- ENET_EVENT_TYPE_RECEIVE = 3,
- ENET_EVENT_TYPE_DISCONNECT_TIMEOUT = 4
- } ENetEventType;
- typedef struct _ENetEvent {
- ENetEventType type;
- ENetPeer* peer;
- enet_uint8 channelID;
- enet_uint32 data;
- ENetPacket* packet;
- } ENetEvent;
- /*
- =======================================================================
- Public API
- =======================================================================
- */
- ENET_API int enet_initialize(void);
- ENET_API int enet_initialize_with_callbacks(ENetVersion, const ENetCallbacks*);
- ENET_API void enet_deinitialize(void);
- ENET_API ENetVersion enet_linked_version(void);
- ENET_API int enet_array_is_zeroed(const uint8_t*, int);
- ENET_API size_t enet_string_copy(char*, const char*, size_t);
- ENET_API enet_uint32 enet_time_get(void);
- ENET_API enet_uint32 enet_crc32(const ENetBuffer*, size_t);
- 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_peer_send(ENetPeer*, enet_uint8, ENetPacket*);
- ENET_API ENetPacket* enet_peer_receive(ENetPeer*, enet_uint8*);
- 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, enet_uint32);
- ENET_API ENetHost* enet_host_create(const ENetAddress*, size_t, size_t, enet_uint32, enet_uint32, int);
- ENET_API void enet_host_destroy(ENetHost*);
- ENET_API void enet_host_enable_compression(ENetHost*);
- ENET_API void enet_host_prevent_connections(ENetHost*, enet_uint8);
- 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_broadcast_exclude(ENetHost*, enet_uint8, ENetPacket*, ENetPeer*);
- ENET_API void enet_host_broadcast_selective(ENetHost*, enet_uint8, ENetPacket*, ENetPeer**, size_t);
- ENET_API void enet_host_channel_limit(ENetHost*, size_t);
- ENET_API void enet_host_bandwidth_limit(ENetHost*, enet_uint32, enet_uint32);
- ENET_API int enet_address_set_host_ip(ENetAddress*, const char*);
- ENET_API int enet_address_set_host(ENetAddress*, const char*);
- ENET_API int enet_address_get_host_ip(const ENetAddress*, char*, size_t);
- ENET_API int enet_address_get_host(const ENetAddress*, char*, size_t);
- 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_socket_set_select(ENetSocket, ENetSocketSet*, ENetSocketSet*, enet_uint32);
- /* Extended API for easier binding in other programming languages */
- ENET_API void* enet_packet_get_data(const ENetPacket*);
- ENET_API int enet_packet_get_length(const ENetPacket*);
- ENET_API void enet_packet_set_free_callback(ENetPacket*, const void*);
- ENET_API int enet_packet_check_references(const ENetPacket*);
- ENET_API void enet_packet_dispose(ENetPacket*);
- ENET_API enet_uint32 enet_host_get_peers_count(const ENetHost*);
- ENET_API enet_uint32 enet_host_get_packets_sent(const ENetHost*);
- ENET_API enet_uint32 enet_host_get_packets_received(const ENetHost*);
- ENET_API enet_uint32 enet_host_get_bytes_sent(const ENetHost*);
- ENET_API enet_uint32 enet_host_get_bytes_received(const ENetHost*);
- ENET_API enet_uint32 enet_peer_get_id(const ENetPeer*);
- ENET_API int enet_peer_get_ip(const ENetPeer*, char*, size_t);
- ENET_API enet_uint16 enet_peer_get_port(const ENetPeer*);
- ENET_API enet_uint32 enet_peer_get_mtu(const ENetPeer*);
- ENET_API ENetPeerState enet_peer_get_state(const ENetPeer*);
- ENET_API enet_uint32 enet_peer_get_rtt(const ENetPeer*);
- ENET_API enet_uint32 enet_peer_get_lastsendtime(const ENetPeer*);
- ENET_API enet_uint32 enet_peer_get_lastreceivetime(const ENetPeer*);
- ENET_API enet_uint64 enet_peer_get_packets_sent(const ENetPeer*);
- ENET_API enet_uint64 enet_peer_get_packets_lost(const ENetPeer*);
- ENET_API enet_uint64 enet_peer_get_bytes_sent(const ENetPeer*);
- ENET_API enet_uint64 enet_peer_get_bytes_received(const ENetPeer*);
- ENET_API void* enet_peer_get_data(const ENetPeer*);
- ENET_API void enet_peer_set_data(ENetPeer*, const void*);
- /*
- =======================================================================
- Private API
- =======================================================================
- */
- extern void enet_host_bandwidth_throttle(ENetHost*);
- extern enet_uint64 enet_host_random_seed(void);
- 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*);
- extern size_t enet_protocol_command_size(enet_uint8);
- #ifdef __cplusplus
- }
- #endif
- #if defined(ENET_IMPLEMENTATION) && !defined(ENET_IMPLEMENTATION_DONE)
- #define ENET_IMPLEMENTATION_DONE 1
- #ifdef __cplusplus
- extern "C" {
- #endif
- #ifdef ENET_LZ4
- #include "lz4/lz4.h"
- #endif
- #ifdef __MINGW32__
- #include "mingw/inet_ntop.c"
- #include "mingw/inet_pton.c"
- #endif
- /*
- =======================================================================
- Atomics
- =======================================================================
- */
- #ifdef _MSC_VER
- #define ENET_AT_CASSERT_PRED(predicate) sizeof(char[2 * !!(predicate) - 1])
- #define ENET_IS_SUPPORTED_ATOMIC(size) ENET_AT_CASSERT_PRED(size == 1 || size == 2 || size == 4 || size == 8)
- #define ENET_ATOMIC_SIZEOF(variable) (ENET_IS_SUPPORTED_ATOMIC(sizeof(*(variable))), sizeof(*(variable)))
- __inline int64_t enet_at_atomic_read(char* ptr, size_t size) {
- switch (size) {
- case 1:
- return _InterlockedExchangeAdd8((volatile char*)ptr, 0);
- case 2:
- return _InterlockedExchangeAdd16((volatile SHORT*)ptr, 0);
- case 4:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchangeAdd((volatile LONG*)ptr, 0);
- #else
- return _InterlockedExchangeAdd((volatile LONG*)ptr, 0);
- #endif
- case 8:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchangeAdd64((volatile LONGLONG*)ptr, 0);
- #else
- return _InterlockedExchangeAdd64((volatile LONGLONG*)ptr, 0);
- #endif
- default:
- return 0x0;
- }
- }
- __inline int64_t enet_at_atomic_write(char* ptr, int64_t value, size_t size) {
- switch (size) {
- case 1:
- return _InterlockedExchange8((volatile char*)ptr, (char)value);
- case 2:
- return _InterlockedExchange16((volatile SHORT*)ptr, (SHORT)value);
- case 4:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchange((volatile LONG*)ptr, (LONG)value);
- #else
- return _InterlockedExchange((volatile LONG*)ptr, (LONG)value);
- #endif
- case 8:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchange64((volatile LONGLONG*)ptr, (LONGLONG)value);
- #else
- return _InterlockedExchange64((volatile LONGLONG*)ptr, (LONGLONG)value);
- #endif
- default:
- return 0x0;
- }
- }
- __inline int64_t enet_at_atomic_cas(char* ptr, int64_t new_val, int64_t old_val, size_t size) {
- switch (size) {
- case 1:
- return _InterlockedCompareExchange8((volatile char*)ptr, (char)new_val, (char)old_val);
- case 2:
- return _InterlockedCompareExchange16((volatile SHORT*)ptr, (SHORT)new_val, (SHORT)old_val);
- case 4:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedCompareExchange((volatile LONG*)ptr, (LONG)new_val, (LONG)old_val);
- #else
- return _InterlockedCompareExchange((volatile LONG*)ptr, (LONG)new_val, (LONG)old_val);
- #endif
- case 8:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedCompareExchange64((volatile LONGLONG*)ptr, (LONGLONG)new_val, (LONGLONG)old_val);
- #else
- return _InterlockedCompareExchange64((volatile LONGLONG*)ptr, (LONGLONG)new_val, (LONGLONG)old_val);
- #endif
- default:
- return 0x0;
- }
- }
- __inline int64_t enet_at_atomic_inc(char* ptr, int64_t delta, size_t data_size) {
- switch (data_size) {
- case 1:
- return _InterlockedExchangeAdd8((volatile char*)ptr, (char)delta);
- case 2:
- return _InterlockedExchangeAdd16((volatile SHORT*)ptr, (SHORT)delta);
- case 4:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchangeAdd((volatile LONG*)ptr, (LONG)delta);
- #else
- return _InterlockedExchangeAdd((volatile LONG*)ptr, (LONG)delta);
- #endif
- case 8:
- #ifdef NOT_UNDERSCORED_INTERLOCKED_EXCHANGE
- return InterlockedExchangeAdd64((volatile LONGLONG*)ptr, (LONGLONG)delta);
- #else
- return _InterlockedExchangeAdd64((volatile LONGLONG*)ptr, (LONGLONG)delta);
- #endif
- default:
- return 0x0;
- }
- }
- #define ENET_ATOMIC_READ(variable) enet_at_atomic_read((char*)(variable), ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_WRITE(variable, new_val) enet_at_atomic_write((char*)(variable), (int64_t)(new_val), ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_CAS(variable, old_value, new_val) enet_at_atomic_cas((char*)(variable), (int64_t)(new_val), (int64_t)(old_value), ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_INC(variable) enet_at_atomic_inc((char*)(variable), 1, ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_DEC(variable) enet_at_atomic_inc((char*)(variable), -1, ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_INC_BY(variable, delta) enet_at_atomic_inc((char*)(variable), (delta), ENET_ATOMIC_SIZEOF(variable))
- #define ENET_ATOMIC_DEC_BY(variable, delta) enet_at_atomic_inc((char*)(variable), -(delta), ENET_ATOMIC_SIZEOF(variable))
- #elif defined(__GNUC__) || defined(__clang__)
- #if defined(__clang__) || (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 7))
- #define AT_HAVE_ATOMICS
- #endif
- /* We want to use __atomic built-ins if possible because the __sync primitives are
- deprecated, because the __atomic build-ins allow us to use ENET_ATOMIC_WRITE on
- uninitialized memory without running into undefined behavior, and because the
- __atomic versions generate more efficient code since we don't need to rely on
- CAS when we don't actually want it.
- Note that we use acquire-release memory order (like mutexes do). We could use
- sequentially consistent memory order but that has lower performance and is
- almost always unneeded. */
- #ifdef AT_HAVE_ATOMICS
- #define ENET_ATOMIC_READ(ptr) __atomic_load_n((ptr), __ATOMIC_ACQUIRE)
- #define ENET_ATOMIC_WRITE(ptr, value) __atomic_store_n((ptr), (value), __ATOMIC_RELEASE)
- #ifndef typeof
- #define typeof __typeof__
- #endif
- /* clang_analyzer doesn't know that CAS writes to memory so it complains about
- potentially lost data. Replace the code with the equivalent non-sync code. */
- #ifdef __clang_analyzer__
- #define ENET_ATOMIC_CAS(ptr, old_value, new_value) \
- ({ \
- typeof(*(ptr)) ENET_ATOMIC_CAS_old_actual_ = (*(ptr)); \
- if (ATOMIC_CAS_old_actual_ == (old_value)) \
- *(ptr) = new_value; \
- ENET_ATOMIC_CAS_old_actual_; \
- })
- #else
- /* Could use __auto_type instead of typeof but that shouldn't work in C++.
- The ({ }) syntax is a GCC extension called statement expression. It lets
- us return a value out of the macro.
- TODO We should return bool here instead of the old value to avoid the ABA
- problem. */
- #define ENET_ATOMIC_CAS(ptr, old_value, new_value) \
- ({ \
- typeof(*(ptr)) ENET_ATOMIC_CAS_expected_ = (old_value); \
- __atomic_compare_exchange_n((ptr), &ENET_ATOMIC_CAS_expected_, (new_value), false, \
- __ATOMIC_ACQ_REL, __ATOMIC_ACQUIRE); \
- ENET_ATOMIC_CAS_expected_; \
- })
- #endif
- #define ENET_ATOMIC_INC(ptr) __atomic_fetch_add((ptr), 1, __ATOMIC_ACQ_REL)
- #define ENET_ATOMIC_DEC(ptr) __atomic_fetch_sub((ptr), 1, __ATOMIC_ACQ_REL)
- #define ENET_ATOMIC_INC_BY(ptr, delta) __atomic_fetch_add((ptr), (delta), __ATOMIC_ACQ_REL)
- #define ENET_ATOMIC_DEC_BY(ptr, delta) __atomic_fetch_sub((ptr), (delta), __ATOMIC_ACQ_REL)
- #else
- #define ENET_ATOMIC_READ(variable) __sync_fetch_and_add(variable, 0)
- #define ENET_ATOMIC_WRITE(variable, new_val) (void)__sync_val_compare_and_swap((variable), *(variable), (new_val))
- #define ENET_ATOMIC_CAS(variable, old_value, new_val) __sync_val_compare_and_swap((variable), (old_value), (new_val))
- #define ENET_ATOMIC_INC(variable) __sync_fetch_and_add((variable), 1)
- #define ENET_ATOMIC_DEC(variable) __sync_fetch_and_sub((variable), 1)
- #define ENET_ATOMIC_INC_BY(variable, delta) __sync_fetch_and_add((variable), (delta), 1)
- #define ENET_ATOMIC_DEC_BY(variable, delta) __sync_fetch_and_sub((variable), (delta), 1)
- #endif
- #undef AT_HAVE_ATOMICS
- #endif
- /*
- =======================================================================
- Callbacks
- =======================================================================
- */
- 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->noMemory != NULL)
- callbacks.noMemory = inits->noMemory;
- return enet_initialize();
- }
- void* enet_malloc(size_t size) {
- void* memory = callbacks.malloc(size);
- if (memory == NULL)
- callbacks.noMemory();
- return memory;
- }
- void enet_free(void* memory) {
- callbacks.free(memory);
- }
- /*
- =======================================================================
- List
- =======================================================================
- */
- 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;
- ENetListIterator 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;
- }
- /*
- =======================================================================
- Utilities
- =======================================================================
- */
- ENetVersion enet_linked_version(void) {
- return ENET_VERSION;
- }
- int enet_array_is_zeroed(const uint8_t* array, int length) {
- size_t i;
- for (i = 0; i < length; i++) {
- if (array[i] != 0)
- return -1;
- }
- return 0;
- }
- size_t enet_string_copy(char* destination, const char* source, size_t length) {
- char *d = destination;
- const char *s = source;
- size_t n = length;
- if (n != 0 && --n != 0) {
- do {
- if ((*d++ = *s++) == 0)
- break;
- }
- while (--n != 0);
- }
- if (n == 0) {
- if (length != 0)
- *d = '\0';
- while (*s++);
- }
- return (s - source - 1);
- }
- /*
- =======================================================================
- Time
- =======================================================================
- */
- #ifdef _WIN32
- static LARGE_INTEGER gettime_offset(void) {
- SYSTEMTIME s;
- FILETIME f;
- LARGE_INTEGER t;
- s.wYear = 1970;
- s.wMonth = 1;
- s.wDay = 1;
- s.wHour = 0;
- s.wMinute = 0;
- s.wSecond = 0;
- s.wMilliseconds = 0;
- SystemTimeToFileTime(&s, &f);
- t.QuadPart = f.dwHighDateTime;
- t.QuadPart <<= 32;
- t.QuadPart |= f.dwLowDateTime;
- return t;
- }
- int clock_gettime(int X, struct timespec* tv) {
- LARGE_INTEGER t;
- FILETIME f;
- double microseconds;
- static LARGE_INTEGER offset;
- static double frequencyToMicroseconds;
- static int initialized = 0;
- static BOOL usePerformanceCounter = 0;
- if (!initialized) {
- LARGE_INTEGER performanceFrequency;
- initialized = 1;
- usePerformanceCounter = QueryPerformanceFrequency(&performanceFrequency);
- if (usePerformanceCounter) {
- QueryPerformanceCounter(&offset);
- frequencyToMicroseconds = (double)performanceFrequency.QuadPart / 1000000.;
- } else {
- offset = gettime_offset();
- frequencyToMicroseconds = 10.;
- }
- }
- if (usePerformanceCounter) {
- QueryPerformanceCounter(&t);
- } else {
- GetSystemTimeAsFileTime(&f);
- t.QuadPart = f.dwHighDateTime;
- t.QuadPart <<= 32;
- t.QuadPart |= f.dwLowDateTime;
- }
- t.QuadPart -= offset.QuadPart;
- microseconds = (double)t.QuadPart / frequencyToMicroseconds;
- t.QuadPart = (LONGLONG)microseconds;
- tv->tv_sec = (long)(t.QuadPart / 1000000);
- tv->tv_nsec = t.QuadPart % 1000000 * 1000;
- return 0;
- }
- #elif __APPLE__ && __MAC_OS_X_VERSION_MIN_REQUIRED < 101200 && !defined(CLOCK_MONOTONIC)
- #define CLOCK_MONOTONIC 0
- int clock_gettime(int X, struct timespec* ts) {
- clock_serv_t cclock;
- mach_timespec_t mts;
- host_get_clock_service(mach_host_self(), SYSTEM_CLOCK, &cclock);
- clock_get_time(cclock, &mts);
- mach_port_deallocate(mach_task_self(), cclock);
- ts->tv_sec = mts.tv_sec;
- ts->tv_nsec = mts.tv_nsec;
- return 0;
- }
- #endif
- enet_uint32 enet_time_get(void) {
- static enet_uint64 start_time_ns = 0;
- struct timespec ts;
- #ifdef CLOCK_MONOTONIC_RAW
- clock_gettime(CLOCK_MONOTONIC_RAW, &ts);
- #else
- clock_gettime(CLOCK_MONOTONIC, &ts);
- #endif
- static const enet_uint64 ns_in_s = 1000 * 1000 * 1000;
- static const enet_uint64 ns_in_ms = 1000 * 1000;
- enet_uint64 current_time_ns = ts.tv_nsec + (enet_uint64)ts.tv_sec * ns_in_s;
- enet_uint64 offset_ns = ENET_ATOMIC_READ(&start_time_ns);
- if (offset_ns == 0) {
- enet_uint64 want_value = current_time_ns - 1 * ns_in_ms;
- enet_uint64 old_value = ENET_ATOMIC_CAS(&start_time_ns, 0, want_value);
- offset_ns = old_value == 0 ? want_value : old_value;
- }
- enet_uint64 result_in_ns = current_time_ns - offset_ns;
- return (enet_uint32)(result_in_ns / ns_in_ms);
- }
- /*
- =======================================================================
- Checksum
- =======================================================================
- */
- 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;
- const enet_uint8* dataEnd = &data[buffers->dataLength];
- while (data < dataEnd) {
- crc = (crc >> 8) ^ crcTable[(crc & 0xFF)^* data++];
- }
- ++buffers;
- }
- return ENET_HOST_TO_NET_32(~crc);
- }
- /*
- =======================================================================
- Packet
- =======================================================================
- */
- ENetPacket* enet_packet_create(const void* data, size_t dataLength, enet_uint32 flags) {
- ENetPacket* packet;
- if (flags & ENET_PACKET_FLAG_NO_ALLOCATE) {
- packet = (ENetPacket*)enet_malloc(sizeof(ENetPacket));
- if (packet == NULL)
- return NULL;
- packet->data = (enet_uint8*)data;
- } else {
- packet = (ENetPacket*)enet_malloc(sizeof(ENetPacket) + dataLength);
- if (packet == NULL)
- return NULL;
- packet->data = (enet_uint8*)packet + sizeof(ENetPacket);
- if (data != NULL)
- memcpy(packet->data, data, dataLength);
- }
- packet->referenceCount = 0;
- packet->flags = flags;
- packet->dataLength = dataLength;
- packet->freeCallback = NULL;
- return packet;
- }
- ENetPacket* enet_packet_create_offset(const void* data, size_t dataLength, size_t dataOffset, enet_uint32 flags) {
- ENetPacket* packet;
- if (flags & ENET_PACKET_FLAG_NO_ALLOCATE) {
- packet = (ENetPacket*)enet_malloc(sizeof(ENetPacket));
- if (packet == NULL)
- return NULL;
- packet->data = (enet_uint8*)data;
- } else {
- packet = (ENetPacket*)enet_malloc(sizeof(ENetPacket) + dataLength - dataOffset);
- if (packet == NULL)
- return NULL;
- packet->data = (enet_uint8*)packet + sizeof(ENetPacket);
- if (data != NULL)
- memcpy(packet->data, (char*)data + dataOffset, dataLength - dataOffset);
- }
- packet->referenceCount = 0;
- packet->flags = flags;
- packet->dataLength = dataLength - dataOffset;
- packet->freeCallback = NULL;
- return packet;
- }
- void enet_packet_destroy(ENetPacket* packet) {
- if (packet == NULL)
- return;
- if (packet->freeCallback != NULL)
- (*packet->freeCallback)((void*)packet);
- enet_free(packet);
- }
- /*
- =======================================================================
- Protocol
- =======================================================================
- */
- 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;
- }
- 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);
- peer->totalDataSent = 0;
- peer->totalDataReceived = 0;
- peer->totalPacketsSent = 0;
- peer->totalPacketsLost = 0;
- 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_notify_disconnect_timeout(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_TIMEOUT;
- 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;
- if (enet_list_empty(&peer->sentUnreliableCommands))
- return;
- do {
- 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);
- }
- while (!enet_list_empty(&peer->sentUnreliableCommands));
- 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);
- }
- 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;
- }
- 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 && in6_equal(currentPeer->address.ipv6, host->receivedAddress.ipv6)) {
- 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;
- }
- 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;
- }
- 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;
- }
- 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_FRAGMENTED, 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;
- }
- 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;
- }
- 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);
- if (roundTripTime == 0)
- roundTripTime = 1;
- if (peer->smoothedRoundTripTime == 0)
- peer->smoothedRoundTripTime = (enet_uint32)((1 - ENET_SRTT_PARA_G) * ENET_SRTT_INITIAL + ENET_SRTT_PARA_G * roundTripTime);
- else
- peer->smoothedRoundTripTime = (enet_uint32)((1 - ENET_SRTT_PARA_G) * peer->smoothedRoundTripTime + ENET_SRTT_PARA_G * roundTripTime);
- enet_peer_throttle(peer, peer->smoothedRoundTripTime);
- peer->roundTripTimeVariance -= peer->roundTripTimeVariance / 4;
- if (peer->smoothedRoundTripTime >= peer->roundTripTime) {
- peer->roundTripTime += (peer->smoothedRoundTripTime - peer->roundTripTime) / 8;
- peer->roundTripTimeVariance += (peer->smoothedRoundTripTime - peer->roundTripTime) / 4;
- } else {
- peer->roundTripTime -= (peer->roundTripTime - peer->smoothedRoundTripTime) / 8;
- peer->roundTripTimeVariance += (peer->roundTripTime - peer->smoothedRoundTripTime) / 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;
- }
- 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;
- }
- 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->checksumCallback != 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 || ((!in6_equal(host->receivedAddress.ipv6, peer->address.ipv6) || host->receivedAddress.port != peer->address.port) && peer->address.ipv4.ip.s_addr != INADDR_BROADCAST) || (peer->outgoingPeerID < ENET_PROTOCOL_MAXIMUM_PEER_ID && sessionID != peer->incomingSessionID))
- return 0;
- }
- #ifdef ENET_LZ4
- if (flags & ENET_PROTOCOL_HEADER_FLAG_COMPRESSED) {
- size_t originalSize = LZ4_decompress_safe((const char*)host->receivedData + headerSize, (char*)host->packetData[1] + headerSize, host->receivedDataLength - 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;
- }
- #endif
- if (host->checksumCallback != NULL) {
- enet_uint32* checksum = (enet_uint32*)&host->receivedData[headerSize - sizeof(enet_uint32)];
- enet_uint32 desiredChecksum = *checksum;
- ENetBuffer buffer;
- *checksum = peer != NULL ? peer->connectID : 0;
- buffer.data = host->receivedData;
- buffer.dataLength = host->receivedDataLength;
- if (host->checksumCallback(&buffer, 1) != desiredChecksum)
- return 0;
- }
- if (peer != NULL) {
- peer->address.ipv6 = host->receivedAddress.ipv6;
- peer->address.port = host->receivedAddress.port;
- peer->incomingDataTotal += host->receivedDataLength;
- peer->totalDataReceived += 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 || currentData < &host->receivedData[host->receivedDataLength]))
- 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;
- if (host->preventConnections == 0) {
- 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;
- }
- 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 = host->mtu;
- receivedLength = enet_socket_receive(host->socket, &host->receivedAddress, &buffer, 1);
- if (receivedLength == -2)
- continue;
- 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->interceptCallback != NULL) {
- switch (host->interceptCallback(host, (void*)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;
- }
- 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;
- }
- 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;
- enet_uint16 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_list_empty(&peer->sentUnreliableCommands))
- enet_peer_disconnect(peer, peer->eventData);
- }
- 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_timeout(host, peer, event);
- return 1;
- }
- if (outgoingCommand->packet != NULL)
- peer->reliableDataInTransit -= outgoingCommand->fragmentLength;
- ++peer->packetsLost;
- ++peer->totalPacketsLost;
- outgoingCommand->roundTripTimeout = peer->roundTripTime + 4 * peer->roundTripTimeVariance;
- outgoingCommand->roundTripTimeoutLimit = peer->timeoutLimit * outgoingCommand->roundTripTimeout;
- 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;
- }
- 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;
- ++peer->totalPacketsSent;
- ++command;
- ++buffer;
- }
- host->commandCount = command - host->commands;
- host->bufferCount = buffer - host->buffers;
- return canPing;
- }
- 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;
- host->continueSending = 1;
- #ifdef ENET_LZ4
- size_t shouldCompress = 0;
- #endif
- 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;
- ENET_LOG_TRACE("happy times");
- #ifdef ENET_DEBUG
- ENET_LOG_TRACE(
- "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;
- }
- #ifdef ENET_LZ4
- if (host->compression == 1 && host->packetSize > 64) {
- size_t originalSize = host->packetSize - sizeof(ENetProtocolHeader), compressedSize = 0;
- const ENetBuffer* buffers = &host->buffers[1];
- if (host->compressionBufferSize < originalSize) {
- enet_free(host->compressionBuffer);
- host->compressionBuffer = (char*)enet_malloc(originalSize);
- host->compressionBufferSize = originalSize;
- }
- size_t totalSize = originalSize, dataSize = 0;
- while (totalSize) {
- size_t i;
- for (i = 0; i < host->bufferCount - 1; i++) {
- size_t copySize = ENET_MIN(totalSize, buffers[i].dataLength);
- memcpy(host->compressionBuffer + dataSize, buffers[i].data, copySize);
- totalSize -= copySize;
- dataSize += copySize;
- }
- }
- compressedSize = LZ4_compress_default((const char*)host->compressionBuffer, (char*)host->packetData[1], dataSize, originalSize);
- if (compressedSize > 0 && compressedSize < originalSize) {
- host->headerFlags |= ENET_PROTOCOL_HEADER_FLAG_COMPRESSED;
- shouldCompress = compressedSize;
- #ifdef ENET_DEBUG
- ENET_LOG_TRACE("peer %u: compressed %u->%u (%u%%)\n", currentPeer->incomingPeerID, originalSize, compressedSize, (compressedSize * 100) / originalSize);
- #endif
- }
- }
- #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->checksumCallback != 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->checksumCallback(host->buffers, host->bufferCount);
- }
- #ifdef ENET_LZ4
- if (shouldCompress > 0) {
- host->buffers[1].data = host->packetData[1];
- host->buffers[1].dataLength = shouldCompress;
- host->bufferCount = 2;
- }
- #endif
- 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;
- currentPeer->totalDataSent += sentLength;
- host->totalSentPackets++;
- }
- }
- return 0;
- }
- void enet_host_flush(ENetHost* host) {
- host->serviceTime = enet_time_get();
- enet_protocol_send_outgoing_commands(host, NULL, 0);
- }
- 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);
- }
- 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;
- }
- /*
- =======================================================================
- Peer
- =======================================================================
- */
- void enet_peer_throttle_configure(ENetPeer* peer, enet_uint32 interval, enet_uint32 acceleration, enet_uint32 deceleration, enet_uint32 threshold) {
- ENetProtocol command;
- peer->packetThrottleThreshold = threshold;
- 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 + peer->packetThrottleThreshold + 2 * peer->lastRoundTripTimeVariance) {
- if (peer->packetThrottle > peer->packetThrottleDeceleration)
- peer->packetThrottle -= peer->packetThrottleDeceleration;
- else
- peer->packetThrottle = 0;
- return -1;
- }
- return 0;
- }
- 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->checksumCallback != 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_FRAGMENTED)) == ENET_PACKET_FLAG_UNRELIABLE_FRAGMENTED && 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;
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- enet_host_flush(peer->host);
- return 0;
- }
- 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;
- }
- 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;
- }
- }
- void enet_peer_reset(ENetPeer* peer) {
- enet_peer_on_disconnect(peer);
- peer->outgoingPeerID = ENET_PROTOCOL_MAXIMUM_PEER_ID;
- peer->state = ENET_PEER_STATE_DISCONNECTED;
- peer->incomingBandwidth = 0;
- peer->outgoingBandwidth = 0;
- peer->incomingBandwidthThrottleEpoch = 0;
- peer->outgoingBandwidthThrottleEpoch = 0;
- peer->incomingDataTotal = 0;
- peer->totalDataReceived = 0;
- peer->outgoingDataTotal = 0;
- peer->totalDataSent = 0;
- peer->lastSendTime = 0;
- peer->lastReceiveTime = 0;
- peer->nextTimeout = 0;
- peer->earliestTimeout = 0;
- peer->packetLossEpoch = 0;
- peer->packetsSent = 0;
- peer->totalPacketsSent = 0;
- peer->packetsLost = 0;
- peer->totalPacketsLost = 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->smoothedRoundTripTime = 0;
- 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);
- }
- 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);
- }
- void enet_peer_ping_interval(ENetPeer* peer, enet_uint32 pingInterval) {
- peer->pingInterval = pingInterval ? pingInterval : ENET_PEER_PING_INTERVAL;
- }
- 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;
- }
- 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);
- }
- 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);
- }
- }
- 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;
- enet_uint16 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);
- }
- 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;
- enet_uint16 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);
- }
- 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);
- }
- 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;
- }
- /*
- =======================================================================
- Host
- =======================================================================
- */
- ENetHost* enet_host_create(const ENetAddress* address, size_t peerCount, size_t channelLimit, enet_uint32 incomingBandwidth, enet_uint32 outgoingBandwidth, int bufferSize) {
- 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)
- enet_socket_set_option(host->socket, ENET_SOCKOPT_IPV6_V6ONLY, 0);
- 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;
- }
- if (bufferSize > ENET_HOST_BUFFER_SIZE_MAX)
- bufferSize = ENET_HOST_BUFFER_SIZE_MAX;
- else if (bufferSize < ENET_HOST_BUFFER_SIZE_MIN)
- bufferSize = ENET_HOST_BUFFER_SIZE_MIN;
- 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, bufferSize);
- enet_socket_set_option(host->socket, ENET_SOCKOPT_SNDBUF, bufferSize);
- 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->preventConnections = 0;
- host->mtu = ENET_HOST_DEFAULT_MTU;
- host->peerCount = peerCount;
- host->commandCount = 0;
- host->bufferCount = 0;
- host->compression = 0;
- host->compressionBufferSize = 0;
- host->checksumCallback = NULL;
- host->receivedAddress.ipv6 = 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->interceptCallback = 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;
- }
- 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);
- }
- enet_free(host->peers);
- enet_free(host->compressionBuffer);
- enet_free(host);
- }
- void enet_host_enable_compression(ENetHost* host) {
- if (host == NULL)
- return;
- host->compression = 1;
- }
- void enet_host_prevent_connections(ENetHost* host, enet_uint8 state) {
- if (host == NULL)
- return;
- host->preventConnections = state;
- }
- 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;
- }
- void enet_host_broadcast(ENetHost* host, enet_uint8 channelID, ENetPacket* packet) {
- ENetPeer* currentPeer;
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- ++packet->referenceCount;
- 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->flags & ENET_PACKET_FLAG_INSTANT)
- --packet->referenceCount;
- if (packet->referenceCount == 0)
- enet_packet_destroy(packet);
- }
- void enet_host_broadcast_exclude(ENetHost* host, enet_uint8 channelID, ENetPacket* packet, ENetPeer* excludedPeer) {
- ENetPeer* currentPeer;
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- ++packet->referenceCount;
- for (currentPeer = host->peers; currentPeer < &host->peers[host->peerCount]; ++currentPeer) {
- if (currentPeer->state != ENET_PEER_STATE_CONNECTED || currentPeer == excludedPeer)
- continue;
- enet_peer_send(currentPeer, channelID, packet);
- }
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- --packet->referenceCount;
- if (packet->referenceCount == 0)
- enet_packet_destroy(packet);
- }
- void enet_host_broadcast_selective(ENetHost* host, enet_uint8 channelID, ENetPacket* packet, ENetPeer** peers, size_t length) {
- ENetPeer* currentPeer;
- size_t i;
- if (host == NULL)
- return;
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- ++packet->referenceCount;
- for (i = 0; i < length; i++) {
- currentPeer = peers[i];
- if (currentPeer == NULL || currentPeer->state != ENET_PEER_STATE_CONNECTED)
- continue;
- enet_peer_send(currentPeer, channelID, packet);
- }
- if (packet->flags & ENET_PACKET_FLAG_INSTANT)
- --packet->referenceCount;
- if (packet->referenceCount == 0)
- enet_packet_destroy(packet);
- }
- 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;
- }
- 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();
- enet_uint32 elapsedTime = timeCurrent - host->bandwidthThrottleEpoch;
- enet_uint32 peersRemaining = (enet_uint32)host->connectedPeers;
- enet_uint32 dataTotal = ~0;
- enet_uint32 bandwidth = ~0;
- enet_uint32 throttle = 0;
- enet_uint32 bandwidthLimit = 0;
- int needsAdjustment = host->bandwidthLimitedPeers > 0 ? 1 : 0;
- ENetPeer* peer;
- ENetProtocol command;
- if (elapsedTime < ENET_HOST_BANDWIDTH_THROTTLE_INTERVAL)
- return;
- if (host->outgoingBandwidth == 0 && host->incomingBandwidth == 0)
- 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);
- }
- }
- }
- /*
- =======================================================================
- Address
- =======================================================================
- */
- int enet_address_set_host_ip(ENetAddress* address, const char* ip) {
- int type = AF_INET6;
- void* destination = &address->ipv6;
- if (strchr(ip, ':') == NULL) {
- type = AF_INET;
- address->ipv4.ffff = 0xFFFF;
- destination = &address->ipv4.ip;
- }
- if (!inet_pton(type, ip, destination))
- return -1;
- return 0;
- }
- int enet_address_set_host(ENetAddress* address, const char* name) {
- struct addrinfo hints, *resultList = NULL, *result = NULL;
- memset(&hints, 0, sizeof(hints));
- hints.ai_family = AF_UNSPEC;
- if (getaddrinfo(name, NULL, &hints, &resultList) != 0)
- return -1;
- for (result = resultList; result != NULL; result = result->ai_next) {
- if (result->ai_addr != NULL && result->ai_addrlen >= sizeof(struct sockaddr_in)) {
- if (result->ai_family == AF_INET) {
- struct sockaddr_in* sin = (struct sockaddr_in*)result->ai_addr;
- memset(address, 0, sizeof(address->ipv4.zeros));
- address->ipv4.ffff = 0xFFFF;
- address->ipv4.ip.s_addr = sin->sin_addr.s_addr;
- freeaddrinfo(resultList);
- return 0;
- } else if (result->ai_family == AF_INET6) {
- struct sockaddr_in6* sin = (struct sockaddr_in6*)result->ai_addr;
- address->ipv6 = sin->sin6_addr;
- freeaddrinfo(resultList);
- return 0;
- }
- }
- }
- if (resultList != NULL)
- freeaddrinfo(resultList);
- return enet_address_set_host_ip(address, name);
- }
- int enet_address_get_host_ip(const ENetAddress* address, char* ip, size_t ipLength) {
- if (inet_ntop(AF_INET6, &address->ipv6, ip, ipLength) == NULL)
- return -1;
- if (enet_array_is_zeroed(address->ipv4.zeros, sizeof(address->ipv4.zeros)) == 0 && address->ipv4.ffff == 0xFFFF)
- enet_string_copy(ip, ip + 7, ipLength);
- return 0;
- }
- int enet_address_get_host(const ENetAddress* address, char* name, size_t nameLength) {
- struct sockaddr_in6 sin;
- int err;
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- 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;
- return enet_address_get_host_ip(address, name, nameLength);
- }
- /*
- =======================================================================
- Platform-specific (Unix)
- =======================================================================
- */
- #ifndef _WIN32
- int enet_initialize(void) {
- return 0;
- }
- void enet_deinitialize(void) { }
- enet_uint64 enet_host_random_seed(void) {
- struct timeval timeVal;
- gettimeofday(&timeVal, NULL);
- return (timeVal.tv_sec * 1000) ^ (timeVal.tv_usec / 1000);
- }
- int enet_socket_bind(ENetSocket socket, const ENetAddress* address) {
- struct sockaddr_in6 sin;
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- if (address != NULL) {
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- } else {
- sin.sin6_port = 0;
- sin.sin6_addr = ENET_HOST_ANY;
- }
- return bind(socket, (struct sockaddr*)&sin, sizeof(struct sockaddr_in6));
- }
- int enet_socket_get_address(ENetSocket socket, ENetAddress* address) {
- struct sockaddr_in6 sin;
- socklen_t sinLength = sizeof(struct sockaddr_in6);
- if (getsockname(socket, (struct sockaddr*)&sin, &sinLength) == -1)
- return -1;
- address->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_port);
- return 0;
- }
- int enet_socket_listen(ENetSocket socket, int backlog) {
- return listen(socket, backlog < 0 ? SOMAXCONN : backlog);
- }
- ENetSocket enet_socket_create(ENetSocketType type) {
- int socketType = (type == ENET_SOCKET_TYPE_DATAGRAM ? SOCK_DGRAM : SOCK_STREAM);
- #ifdef SOCK_CLOEXEC
- socketType |= SOCK_CLOEXEC;
- #endif
- return socket(PF_INET6, socketType, 0);
- }
- int enet_socket_set_option(ENetSocket socket, ENetSocketOption option, int value) {
- int result = -1;
- switch (option) {
- case ENET_SOCKOPT_NONBLOCK:
- result = fcntl(socket, F_SETFL, (value ? O_NONBLOCK : 0) | (fcntl(socket, F_GETFL) & ~O_NONBLOCK));
- 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;
- case ENET_SOCKOPT_IPV6_V6ONLY:
- result = setsockopt(socket, IPPROTO_IPV6, IPV6_V6ONLY, (char*)&value, sizeof(int));
- break;
- default:
- break;
- }
- return result == -1 ? -1 : 0;
- }
- 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) {
- int result = -1;
- struct sockaddr_in6 sin;
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- result = connect(socket, (struct sockaddr*)&sin, sizeof(struct sockaddr_in6));
- if (result == -1 && errno == EINPROGRESS)
- return 0;
- return result;
- }
- ENetSocket enet_socket_accept(ENetSocket socket, ENetAddress* address) {
- int result = -1;
- struct sockaddr_in6 sin;
- socklen_t sinLength = sizeof(struct sockaddr_in6);
- result = accept(socket, address != NULL ? (struct sockaddr*)&sin : NULL, address != NULL ? &sinLength : NULL);
- if (result == -1)
- return ENET_SOCKET_NULL;
- if (address != NULL) {
- address->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_port);
- }
- return result;
- }
- int enet_socket_shutdown(ENetSocket socket, ENetSocketShutdown how) {
- return shutdown(socket, (int)how);
- }
- void enet_socket_destroy(ENetSocket socket) {
- if (socket != ENET_SOCKET_NULL)
- close(socket);
- }
- int enet_socket_send(ENetSocket socket, const ENetAddress* address, const ENetBuffer* buffers, size_t bufferCount) {
- struct msghdr msgHdr;
- struct sockaddr_in6 sin;
- int sentLength;
- memset(&msgHdr, 0, sizeof(struct msghdr));
- if (address != NULL) {
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- msgHdr.msg_name = &sin;
- msgHdr.msg_namelen = sizeof(struct sockaddr_in6);
- }
- 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;
- }
- int enet_socket_receive(ENetSocket socket, ENetAddress* address, ENetBuffer* buffers, size_t bufferCount) {
- struct msghdr msgHdr;
- struct sockaddr_in6 sin;
- int recvLength;
- memset(&msgHdr, 0, sizeof(struct msghdr));
- if (address != NULL) {
- msgHdr.msg_name = &sin;
- msgHdr.msg_namelen = sizeof(struct sockaddr_in6);
- }
- 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;
- }
- if (msgHdr.msg_flags & MSG_TRUNC)
- return -1;
- if (address != NULL) {
- address->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_port);
- }
- return recvLength;
- }
- int enet_socket_set_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) {
- 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;
- }
- #endif
- /*
- =======================================================================
- Platform-specific (Windows)
- =======================================================================
- */
- #ifdef _WIN32
- int enet_initialize(void) {
- WORD versionRequested = MAKEWORD(2, 2);
- WSADATA wsaData;
- if (WSAStartup(versionRequested, &wsaData))
- return -1;
- if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2) {
- 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();
- }
- int enet_socket_bind(ENetSocket socket, const ENetAddress* address) {
- struct sockaddr_in6 sin;
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- if (address != NULL) {
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- } else {
- sin.sin6_port = 0;
- sin.sin6_addr = in6addr_any;
- }
- return bind(socket, (struct sockaddr*)&sin, sizeof(struct sockaddr_in6)) == SOCKET_ERROR ? -1 : 0;
- }
- int enet_socket_get_address(ENetSocket socket, ENetAddress* address) {
- struct sockaddr_in6 sin;
- int sinLength = sizeof(struct sockaddr_in6);
- if (getsockname(socket, (struct sockaddr*)&sin, &sinLength) == -1)
- return -1;
- address->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_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_INET6, 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;
- case ENET_SOCKOPT_IPV6_V6ONLY:
- result = setsockopt(socket, IPPROTO_IPV6, IPV6_V6ONLY, (char*)&value, sizeof(int));
- break;
- default:
- break;
- }
- return result == SOCKET_ERROR ? -1 : 0;
- }
- 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) {
- int result = -1;
- struct sockaddr_in6 sin;
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- result = connect(socket, (struct sockaddr*)&sin, sizeof(struct sockaddr_in6));
- if (result == SOCKET_ERROR && WSAGetLastError() != WSAEWOULDBLOCK)
- return -1;
- return 0;
- }
- ENetSocket enet_socket_accept(ENetSocket socket, ENetAddress* address) {
- SOCKET result;
- struct sockaddr_in6 sin;
- int sinLength = sizeof(struct sockaddr_in6);
- 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->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_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_in6 sin;
- DWORD sentLength;
- if (address != NULL) {
- memset(&sin, 0, sizeof(struct sockaddr_in6));
- sin.sin6_family = AF_INET6;
- sin.sin6_port = ENET_HOST_TO_NET_16(address->port);
- sin.sin6_addr = address->ipv6;
- }
- if (WSASendTo(socket, (LPWSABUF)buffers, (DWORD)bufferCount, &sentLength, 0, address != NULL ? (struct sockaddr*)&sin : NULL, address != NULL ? sizeof(struct sockaddr_in6) : 0, NULL, NULL) == SOCKET_ERROR)
- return (WSAGetLastError() == WSAEWOULDBLOCK) ? 0 : -1;
- return (int)sentLength;
- }
- int enet_socket_receive(ENetSocket socket, ENetAddress* address, ENetBuffer* buffers, size_t bufferCount) {
- INT sinLength = sizeof(struct sockaddr_in6);
- DWORD flags = 0, recvLength;
- struct sockaddr_in6 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->ipv6 = sin.sin6_addr;
- address->port = ENET_NET_TO_HOST_16(sin.sin6_port);
- }
- return (int)recvLength;
- }
- int enet_socket_set_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;
- }
- #endif
- /*
- =======================================================================
- Extended functionality
- =======================================================================
- */
- void* enet_packet_get_data(const ENetPacket* packet) {
- return (void*)packet->data;
- }
- int enet_packet_get_length(const ENetPacket* packet) {
- return packet->dataLength;
- }
- void enet_packet_set_free_callback(ENetPacket* packet, const void* callback) {
- packet->freeCallback = (ENetPacketFreeCallback)callback;
- }
- int enet_packet_check_references(const ENetPacket* packet) {
- return (int)packet->referenceCount;
- }
- void enet_packet_dispose(ENetPacket* packet) {
- if (packet->referenceCount == 0)
- enet_packet_destroy(packet);
- }
- enet_uint32 enet_host_get_peers_count(const ENetHost* host) {
- return host->connectedPeers;
- }
- enet_uint32 enet_host_get_packets_sent(const ENetHost* host) {
- return host->totalSentPackets;
- }
- enet_uint32 enet_host_get_packets_received(const ENetHost* host) {
- return host->totalReceivedPackets;
- }
- enet_uint32 enet_host_get_bytes_sent(const ENetHost* host) {
- return host->totalSentData;
- }
- enet_uint32 enet_host_get_bytes_received(const ENetHost* host) {
- return host->totalReceivedData;
- }
- enet_uint32 enet_peer_get_id(const ENetPeer* peer) {
- return peer->incomingPeerID;
- }
- int enet_peer_get_ip(const ENetPeer* peer, char* ip, size_t ipLength) {
- return enet_address_get_host_ip(&peer->address, ip, ipLength);
- }
- enet_uint16 enet_peer_get_port(const ENetPeer* peer) {
- return peer->address.port;
- }
- enet_uint32 enet_peer_get_mtu(const ENetPeer* peer) {
- return peer->mtu;
- }
- ENetPeerState enet_peer_get_state(const ENetPeer* peer) {
- return peer->state;
- }
- enet_uint32 enet_peer_get_rtt(const ENetPeer* peer) {
- return peer->smoothedRoundTripTime;
- }
- enet_uint32 enet_peer_get_lastsendtime(const ENetPeer* peer) {
- return peer->lastSendTime;
- }
- enet_uint32 enet_peer_get_lastreceivetime(const ENetPeer* peer) {
- return peer->lastReceiveTime;
- }
- enet_uint64 enet_peer_get_packets_sent(const ENetPeer* peer) {
- return peer->totalPacketsSent;
- }
- enet_uint64 enet_peer_get_packets_lost(const ENetPeer* peer) {
- return peer->totalPacketsLost;
- }
- enet_uint64 enet_peer_get_bytes_sent(const ENetPeer* peer) {
- return peer->totalDataSent;
- }
- enet_uint64 enet_peer_get_bytes_received(const ENetPeer* peer) {
- return peer->totalDataReceived;
- }
- void* enet_peer_get_data(const ENetPeer* peer) {
- return (void*)peer->data;
- }
- void enet_peer_set_data(ENetPeer* peer, const void* data) {
- peer->data = (enet_uint32*)data;
- }
- #ifdef __cplusplus
- }
- #endif
- #endif
- #endif
|