1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057 |
- // implementation of generic tools
- #include "cube.h"
- #ifdef NO_POSIX_R
- char *strtok_r(char *s, const char *delim, char **b)
- {
- if(s) *b = s;
- *b += strspn(*b, delim);
- if(!**b) return NULL;
- s = *b;
- *b += strcspn(s, delim);
- if(**b) *(*b)++ = '\0';
- return s;
- }
- #endif
- string _timestringbuffer;
- const char *timestring(time_t t, bool local, const char *fmt, char *buf)
- {
- #ifdef NO_POSIX_R
- struct tm *timeinfo;
- timeinfo = local ? localtime(&t) : gmtime (&t);
- #else
- struct tm *timeinfo, b;
- timeinfo = local ? localtime_r(&t, &b) : gmtime_r(&t, &b);
- #endif
- strftime(buf, sizeof(string) - 1, fmt && *fmt ? fmt : "%Y%m%d_%H.%M.%S", timeinfo); // sortable time for filenames
- return buf;
- }
- const char *asctimestr()
- {
- return timestring(true, "%c");
- }
- const char *numtime()
- {
- static string numt;
- time_t t = time(NULL);
- int t1 = t / 1000000, t2 = t % 1000000;
- formatstring(numt)("%d%06d", t1, t2);
- return numt + strspn(numt, "0");
- }
- static const uchar transformenttab[] = {
- /* mapformat 1..5 6..7 */
- /* 0 */ /* NOTUSED */ NOTUSED, /* NOTUSED */ NOTUSED,
- /* 1 */ /* LIGHT */ LIGHT, /* LIGHT */ LIGHT,
- /* 2 */ /* PLAYERSTART */ PLAYERSTART, /* PLAYERSTART */ PLAYERSTART,
- /* 3 */ /* I_SHELLS */ I_AMMO, /* I_CLIPS */ I_CLIPS,
- /* 4 */ /* I_BULLETS */ I_AMMO, /* I_AMMO */ I_AMMO,
- /* 5 */ /* I_ROCKETS */ I_AMMO, /* I_GRENADE */ I_GRENADE,
- /* 6 */ /* I_ROUNDS */ I_AMMO, /* I_HEALTH */ I_HEALTH,
- /* 7 */ /* I_HEALTH */ I_HEALTH, /* I_ARMOUR */ I_ARMOUR,
- /* 8 */ /* I_BOOST */ I_HEALTH, /* I_AKIMBO */ I_AKIMBO,
- /* 9 */ /* I_GREENARMOUR */ I_HELMET, /* MAPMODEL */ MAPMODEL,
- /* 10 */ /* I_YELLOWARMOUR */ I_ARMOUR, /* CARROT */ CARROT,
- /* 11 */ /* I_QUAD */ I_AKIMBO, /* LADDER */ LADDER,
- /* 12 */ /* TELEPORT */ NOTUSED, /* CTF_FLAG */ CTF_FLAG,
- /* 13 */ /* TELEDEST */ NOTUSED, /* SOUND */ SOUND,
- /* 14 */ /* MAPMODEL */ MAPMODEL, /* CLIP */ CLIP,
- /* 15 */ /* MONSTER */ NOTUSED, /* PLCLIP */ PLCLIP,
- /* 16 */ /* CARROT */ NOTUSED, 16,
- /* 17 */ /* JUMPPAD */ NOTUSED, 17 };
- void transformoldentitytypes(int mapversion, uchar &enttype)
- {
- const uchar *usetab = transformenttab + (mapversion > 5 ? 1 : 0);
- if(mapversion < 8 && enttype < 18) enttype = usetab[enttype * 2];
- }
- int fixmapheadersize(int version, int headersize) // we can't trust hdr.headersize for file versions < 10 (thx flow)
- {
- if(version < 4) return sizeof(header) - sizeof(int) * 16;
- else if(version == 7 || version == 8) return sizeof(header) + sizeof(char) * 128; // mediareq
- else if(version < 10 || headersize < int(sizeof(header))) return sizeof(header);
- return headersize;
- }
- // map geometry statistics
- servsqr *createservworld(const sqr *s, int _cubicsize) // create a server-style floorplan on the client, to use same statistics functions on client and server
- {
- servsqr *res = new servsqr[_cubicsize], *d = res;
- loopirev(_cubicsize)
- {
- d->type = s->type == SOLID ? SOLID : ((s->type & TAGTRIGGERMASK) | (s->tag & ~TAGTRIGGERMASK)); // guaranteed to not have tagclips on SOLID cubes - so we can check for SOLID without masks
- d->ceil = s->ceil;
- d->floor = s->floor;
- d->vdelta = s->vdelta;
- d++;
- s++;
- }
- return res;
- }
- int calcmapdims(mapdim_s &md, const servsqr *s, int _ssize)
- {
- int res = 0;
- md.x1 = md.y1 = _ssize;
- md.x2 = md.y2 = 0;
- md.minfloor = 127; md.maxceil = -128;
- for(int y = 0; y < _ssize; y++) for(int x = 0; x < _ssize; x++)
- {
- if((s->type & TAGTRIGGERMASK) != SOLID)
- {
- if(x < md.x1) md.x1 = x;
- if(x > md.x2) md.x2 = x;
- if(y < md.y1) md.y1 = y;
- md.y2 = y;
- if(s->floor < md.minfloor) md.minfloor = s->floor;
- if(s->ceil > md.maxceil) md.maxceil = s->ceil;
- }
- s++;
- }
- if(md.x2 < md.x1 || md.y2 < md.y1)
- { // map is completely solid -> default to empty map values
- md.x1 = md.y1 = 2;
- md.x2 = md.y2 = _ssize - 3;
- md.minfloor = 0;
- md.maxceil = 16;
- res = -1; // reject map on servers
- }
- if(md.x1 < MINBORD || md.y1 < MINBORD || md.x2 >= _ssize - MINBORD || md.y2 >= _ssize - MINBORD) res = -2; // reject map because of world border violation
- md.xspan = md.x2 - md.x1 + 1;
- md.yspan = md.y2 - md.y1 + 1;
- md.xm = md.x1 + md.xspan / 2.0f;
- md.ym = md.y1 + md.yspan / 2.0f;
- return res;
- }
- int calcmapareastats(mapareastats_s &ms, servsqr *_servworld, int _ssize, const mapdim_s &md)
- {
- memset(&ms, 0, sizeof(ms));
- // count steep FHF and CHF
- int linegap = _ssize - md.xspan;
- #ifndef STANDALONE
- int totalmax = 0;
- #endif
- servsqr *bb = _servworld + _ssize * md.y1 + md.x1, *ss = bb;
- for(int j = md.yspan; j > 0; j--, ss += linegap) loopirev(md.xspan)
- {
- int type = ss->type & TAGTRIGGERMASK;
- if(type == FHF || type == CHF) // only CHF and FHF use vdelta
- {
- servsqr *r[3] = { ss + 1, ss + _ssize, ss + _ssize + 1 };
- int min = ss->vdelta, max = min;
- loopi(3)
- {
- if(r[i]->vdelta < min) min = r[i]->vdelta;
- else if(r[i]->vdelta > max) max = r[i]->vdelta;
- }
- max -= min;
- #ifndef STANDALONE
- if(max > totalmax)
- {
- ms.steepest = int(ss - _servworld);
- totalmax = max;
- }
- #endif
- int d = max / MAS_VDELTA_QUANT;
- ASSERT(d >= 0);
- if(d >= MAS_VDELTA_TABSIZE) d = MAS_VDELTA_TABSIZE - 1;
- ms.vdd[d]++;
- }
- ss++;
- }
- loopirev(MAS_VDELTA_TABSIZE) ms.vdds = (ms.vdds << 2) | ((ms.vdd[i] > MAS_VDELTA_THRES) << 1) | (ms.vdd[i] > 0);
- // check occlusion by SOLIDs (destroys vdelta values)
- ss = bb;
- for(int j = md.yspan; j > 0; j--, ss += linegap) loopirev(md.xspan) (ss++)->vdelta = 0; // reset all vdelta values
- int xc = (md.xspan + MAS_GRID / 2) / (MAS_GRID + 1), yc = (md.yspan + MAS_GRID / 2) / (MAS_GRID + 1);
- if(md.x1 + MAS_GRID * xc >= _ssize || md.y1 + MAS_GRID * yc >= _ssize) return -1; // malformed map
- int xb = min(md.x1 + xc, _ssize - MAS_GRID * xc - md.x1 - 1) / 2, yb = min(md.y1 + yc, _ssize - MAS_GRID * yc - md.y1 - 1) / 2; // make sure, all tp points are inside the map area
- int tgx = min(xb, max(3, xc / 3)), tgy = min(yb, max(3, yc / 3)) * _ssize, tp[8] = { tgx, -tgx, tgy, -tgy, 2 * tgx + 2 * tgy, -2 * tgx - 2 * tgy, -2 * tgx + 2 * tgy, 2 * tgx - 2 * tgy }; // 8 positions to try, if the stariing point is solid
- uchar epoch = 1;
- servsqr *s = bb + xc + yc * _ssize; ss = s;
- vector<threeint> tab;
- threeint pp;
- for(int y = 0; y < MAS_GRID; y++, ss = s += _ssize * yc) for(int x = 0; x < MAS_GRID; x++, ss += xc) // measure visible area in MAS_GRID x MAS_GRID probe points
- { // calculate MAS_GRID^2 probe points (with a low-res version of computeraytable()
- servsqr *r = ss, *rr;
- if(SOLID(r)) loopi(8)
- { // if initial position is SOLID, we try 8 points around that spot
- if(!SOLID(r + tp[i]))
- {
- r += tp[i];
- break;
- }
- }
- pp.val2 = int(r - _servworld);
- int area = 0, volume = 0, frac;
- loopk(MAS_RESOLUTION)
- {
- #define RAYS(da, db) \
- rr = r; frac = 0; \
- for(;;) \
- { \
- if((frac += k) >= MAS_RESOLUTION) rr += da, frac -= MAS_RESOLUTION; \
- rr += db; \
- if(SOLID(rr)) break; \
- if(rr->vdelta != epoch) { area += 1; volume += rr->ceil - rr->floor; } \
- rr->vdelta = epoch; \
- }
- RAYS(_ssize, 1);
- RAYS(_ssize, -1);
- RAYS(-_ssize, 1);
- RAYS(-_ssize, -1);
- RAYS(1, _ssize);
- RAYS(1, -_ssize);
- RAYS(-1, _ssize);
- RAYS(-1, -_ssize);
- #undef RAYS
- }
- pp.val1 = volume;
- pp.key = area;
- tab.add(pp);
- if(area) epoch++;
- if(!epoch) epoch++;
- }
- ss = bb;
- for(int j = md.yspan; j > 0; j--, ss += linegap) loopirev(md.xspan)
- {
- if(!SOLID(ss))
- {
- ms.total++;
- if(!ss->vdelta) ms.rest++; // count all cubes not in view of one of the probe points
- }
- ss++;
- }
- ASSERT(tab.length() == MAS_GRID2);
- tab.sort(cmpintdesc);
- loopv(tab)
- { // sort probe poiints in descending order of area
- ms.ppv[i] = tab[i].val1;
- ms.ppa[i] = tab[i].key;
- #ifndef STANDALONE
- ms.ppp[i] = tab[i].val2;
- #endif
- }
- return 0;
- }
- void calcentitystats(entitystats_s &es, const persistent_entity *pents, int pentsize)
- {
- #ifndef STANDALONE
- vector<persistent_entity> _pents;
- if(!pents)
- { // use regular ents list
- loopv(ents) _pents.add() = ents[i];
- pentsize = _pents.length();
- if(pentsize) pents = &_pents[0];
- }
- #endif
- memset(&es, 0, sizeof(es));
- loopi(MAXENTTYPES) es.first[i] = pentsize;
- vector<int> picks;
- loopi(pentsize)
- {
- const persistent_entity &e = pents[i];
- if(e.type < MAXENTTYPES)
- {
- es.entcnt[e.type]++;
- if(es.first[e.type] == pentsize) es.first[e.type] = i;
- es.last[e.type] = i;
- if(e.type == PLAYERSTART) switch(e.attr2)
- {
- case 0: es.spawns[0]++; break;
- case 1: es.spawns[1]++; break;
- case 100: es.spawns[2]++; break;
- default: es.unknownspawns++; break;
- }
- if(e.type == CTF_FLAG) switch(e.attr2)
- {
- case 0:
- case 1:
- es.flags[e.attr2]++;
- es.flagents[e.attr2] = i;
- break;
- default:
- es.unknownflags++;
- break;
- }
- if(isitem(e.type)) picks.add(i);
- }
- }
- es.hasffaspawns = es.spawns[2] >= MINSPAWNS;
- es.hasteamspawns = es.spawns[0] >= MINSPAWNS && es.spawns[1] >= MINSPAWNS;
- if(es.flags[0] == 1 && es.flags[1] == 1)
- {
- vec fd(pents[es.flagents[0]].x - pents[es.flagents[1]].x, pents[es.flagents[0]].y - pents[es.flagents[1]].y, 0);
- es.flagentdistance = fd.magnitude();
- es.hasflags = es.flagentdistance >= MINFLAGDISTANCE;
- }
- int r = 0;
- loopvjrev(picks) loopirev(j)
- { // calculate the distances between all pickups (takes n^2/2 calculations - which should be fine for any sane map)
- double d = pow2(pents[picks[i]].x - pents[picks[j]].x) + pow2(pents[picks[i]].y - pents[picks[j]].y); // deltaX^2 + deltaY^2
- frexp(d, &r);
- r /= 2; // sqrt ;)
- if(r > LARGEST_FACTOR) r = LARGEST_FACTOR;
- es.pickupdistance[r]++;
- }
- es.pickups = picks.length();
- }
- #if 0
- const char *rateentitystats(entitystats_s &es)
- {
- static char res[LARGEST_FACTOR + 1];
- }
- #endif
- int cmpintasc(const int *a, const int *b) { return *a - *b; } // leads to ascending sort order
- int cmpintdesc(const int *a, const int *b) { return *b - *a; } // leads to descending sort order
- int stringsort(const char **a, const char **b) { return strcmp(*a, *b); }
- int stringsortrev(const char **a, const char **b) { return strcmp(*b, *a); }
- int stringsortignorecase(const char **a, const char **b) { return strcasecmp(*a, *b); }
- int stringsortignorecaserev(const char **a, const char **b) { return strcasecmp(*b, *a); }
- extern char *maplayout, *testlayout;
- extern int maplayout_factor, testlayout_factor, Mvolume, Marea, Mopen, SHhits;
- extern float Mheight;
- extern int checkarea(int, char *);
- mapstats *loadmapstats(const char *filename, bool getlayout)
- {
- const int sizeof_header = sizeof(header), sizeof_baseheader = sizeof_header - sizeof(int) * 16;
- static mapstats s;
- static uchar *enttypes = NULL;
- static short *entposs = NULL;
- DELETEA(enttypes);
- loopi(MAXENTTYPES) s.entcnt[i] = 0;
- loopi(3) s.spawns[i] = 0;
- loopi(2) s.flags[i] = 0;
- stream *f = opengzfile(filename, "rb");
- if(!f) return NULL;
- memset(&s.hdr, 0, sizeof_header);
- if(f->read(&s.hdr, sizeof_baseheader) != sizeof_baseheader || (strncmp(s.hdr.head, "CUBE", 4) && strncmp(s.hdr.head, "ACMP",4))) { delete f; return NULL; }
- lilswap(&s.hdr.version, 4);
- s.hdr.headersize = fixmapheadersize(s.hdr.version, s.hdr.headersize);
- int restofhead = min(s.hdr.headersize, sizeof_header) - sizeof_baseheader;
- if(s.hdr.version > MAPVERSION || s.hdr.numents > MAXENTITIES ||
- f->read(&s.hdr.waterlevel, restofhead) != restofhead ||
- !f->seek(clamp(s.hdr.headersize - sizeof_header, 0, MAXHEADEREXTRA), SEEK_CUR)) { delete f; return NULL; }
- if(s.hdr.version>=4)
- {
- lilswap(&s.hdr.waterlevel, 1);
- lilswap(&s.hdr.maprevision, 2);
- }
- else s.hdr.waterlevel = -100000;
- entity e;
- enttypes = new uchar[s.hdr.numents];
- entposs = new short[s.hdr.numents * 3];
- loopi(s.hdr.numents)
- {
- f->read(&e, s.hdr.version < 10 ? 12 : sizeof(persistent_entity));
- lilswap((short *)&e, 4);
- transformoldentitytypes(s.hdr.version, e.type);
- if(e.type == PLAYERSTART && (e.attr2 == 0 || e.attr2 == 1 || e.attr2 == 100)) s.spawns[e.attr2 == 100 ? 2 : e.attr2]++;
- if(e.type == CTF_FLAG && (e.attr2 == 0 || e.attr2 == 1)) { s.flags[e.attr2]++; s.flagents[e.attr2] = i; }
- s.entcnt[e.type]++;
- enttypes[i] = e.type;
- entposs[i * 3] = e.x; entposs[i * 3 + 1] = e.y; entposs[i * 3 + 2] = e.z + e.attr1;
- }
- DELETEA(testlayout);
- int minfloor = 0;
- int maxceil = 0;
- if(s.hdr.sfactor <= LARGEST_FACTOR && s.hdr.sfactor >= SMALLEST_FACTOR)
- {
- testlayout_factor = s.hdr.sfactor;
- int layoutsize = 1 << (testlayout_factor * 2);
- bool fail = false;
- testlayout = new char[layoutsize + 256];
- memset(testlayout, 0, layoutsize * sizeof(char));
- char *t = NULL;
- char floor = 0, ceil;
- int diff = 0;
- Mvolume = Marea = SHhits = 0;
- loopk(layoutsize)
- {
- char *c = testlayout + k;
- int type = f->getchar();
- int n = 1;
- switch(type)
- {
- case 255:
- {
- if(!t || (n = f->getchar()) < 0) { fail = true; break; }
- memset(c, *t, n);
- k += n - 1;
- break;
- }
- case 254: // only in MAPVERSION<=2
- if(!t) { fail = true; break; }
- *c = *t;
- f->getchar(); f->getchar();
- break;
- default:
- if(type<0 || type>=MAXTYPE) { fail = true; break; }
- floor = f->getchar();
- ceil = f->getchar();
- if(floor >= ceil && ceil > -128) floor = ceil - 1; // for pre 12_13
- diff = ceil - floor;
- if(type == FHF) floor = -128;
- if(floor!=-128 && floor<minfloor) minfloor = floor;
- if(ceil>maxceil) maxceil = ceil;
- f->getchar(); f->getchar();
- if(s.hdr.version>=2) f->getchar();
- if(s.hdr.version>=5) f->getchar();
- case SOLID:
- *c = type == SOLID ? 127 : floor;
- f->getchar(); f->getchar();
- if(s.hdr.version<=2) { f->getchar(); f->getchar(); }
- break;
- }
- if ( type != SOLID && diff > 6 )
- {
- // Lucas (10mar2013): Removed "pow2" because it was too strict
- if (diff > MAXMHEIGHT) SHhits += /*pow2*/(diff-MAXMHEIGHT)*n;
- Marea += n;
- Mvolume += diff * n;
- }
- if(fail) break;
- t = c;
- }
- if(fail) { DELETEA(testlayout); }
- else
- {
- Mheight = Marea ? (float)Mvolume/Marea : 0;
- Mopen = checkarea(testlayout_factor, testlayout);
- }
- }
- if(getlayout)
- {
- DELETEA(maplayout);
- if (testlayout)
- {
- maplayout_factor = testlayout_factor;
- extern int maplayoutssize;
- maplayoutssize = 1 << testlayout_factor;
- int layoutsize = 1 << (testlayout_factor * 2);
- maplayout = new char[layoutsize + 256];
- memcpy(maplayout, testlayout, layoutsize * sizeof(char));
- /* memset(&mapdims, 0, sizeof(struct mapdim_s));
- mapdims.x1 = mapdims.y1 = maplayoutssize;
- loopk(layoutsize) if (testlayout[k] != 127)
- {
- int cwx = k%maplayoutssize,
- cwy = k/maplayoutssize;
- if(cwx < mapdims.x1) mapdims.x1 = cwx;
- if(cwy < mapdims.y1) mapdims.y1 = cwy;
- if(cwx > mapdims.x2) mapdims.x2 = cwx;
- if(cwy > mapdims.y2) mapdims.y2 = cwy;
- }
- mapdims.xspan = mapdims.x2 - mapdims.x1;
- mapdims.yspan = mapdims.y2 - mapdims.y1;
- mapdims.minfloor = minfloor;
- mapdims.maxceil = maxceil;
- */ }
- }
- delete f;
- s.hasffaspawns = s.spawns[2] > 0;
- s.hasteamspawns = s.spawns[0] > 0 && s.spawns[1] > 0;
- s.hasflags = s.flags[0] > 0 && s.flags[1] > 0;
- s.enttypes = enttypes;
- s.entposs = entposs;
- s.cgzsize = getfilesize(filename);
- return &s;
- }
- ///////////////////////// debugging ///////////////////////
- #if defined(WIN32) && !defined(_DEBUG) && !defined(__GNUC__)
- void stackdumper(unsigned int type, EXCEPTION_POINTERS *ep)
- {
- if(!ep) fatal("unknown type");
- EXCEPTION_RECORD *er = ep->ExceptionRecord;
- CONTEXT *context = ep->ContextRecord;
- string out, t;
- formatstring(out)("Win32 Exception: 0x%x [0x%x]\n\n", er->ExceptionCode, er->ExceptionCode==EXCEPTION_ACCESS_VIOLATION ? er->ExceptionInformation[1] : -1);
- STACKFRAME sf = {{context->Eip, 0, AddrModeFlat}, {}, {context->Ebp, 0, AddrModeFlat}, {context->Esp, 0, AddrModeFlat}, 0};
- SymInitialize(GetCurrentProcess(), NULL, TRUE);
- while(::StackWalk(IMAGE_FILE_MACHINE_I386, GetCurrentProcess(), GetCurrentThread(), &sf, context, NULL, ::SymFunctionTableAccess, ::SymGetModuleBase, NULL))
- {
- struct { IMAGEHLP_SYMBOL sym; string n; } si = { { sizeof( IMAGEHLP_SYMBOL ), 0, 0, 0, sizeof(string) } };
- IMAGEHLP_LINE li = { sizeof( IMAGEHLP_LINE ) };
- DWORD off;
- if(SymGetSymFromAddr(GetCurrentProcess(), (DWORD)sf.AddrPC.Offset, &off, &si.sym) && SymGetLineFromAddr(GetCurrentProcess(), (DWORD)sf.AddrPC.Offset, &off, &li))
- {
- char *del = strrchr(li.FileName, '\\');
- formatstring(t)("%s - %s [%d]\n", si.sym.Name, del ? del + 1 : li.FileName, li.LineNumber);
- concatstring(out, t);
- }
- }
- #if !defined(STANDALONE)
- if(clientlogfile) clientlogfile->printf("%s\n", out);
- #endif
- fatal("%s", out);
- }
- #elif defined(__ANDROID__)
- // fixme ah
- #elif defined(linux) || defined(__linux) || defined(__linux__)
- #include <execinfo.h>
- // stack dumping on linux, inspired by Sachin Agrawal's sample code
- struct signalbinder
- {
- static void stackdumper(int sig)
- {
- printf("stacktrace:\n");
- #if !defined(STANDALONE)
- if(clientlogfile) clientlogfile->printf("stacktrace\n");
- #endif
- const int BTSIZE = 25;
- void *array[BTSIZE];
- int n = backtrace(array, BTSIZE);
- char **symbols = backtrace_symbols(array, n);
- for(int i = 0; i < n; i++)
- {
- printf("%s\n", symbols[i]);
- #if !defined(STANDALONE)
- if(clientlogfile) clientlogfile->printf("%s\n", symbols[i]);
- #endif
- }
- free(symbols);
- fatal("AssaultCube error (%d)", sig);
- }
- signalbinder()
- {
- // register signals to dump the stack if they are raised,
- // use constructor for early registering
- signal(SIGSEGV, stackdumper);
- signal(SIGFPE, stackdumper);
- signal(SIGILL, stackdumper);
- signal(SIGBUS, stackdumper);
- signal(SIGSYS, stackdumper);
- signal(SIGABRT, stackdumper);
- }
- };
- signalbinder sigbinder;
- #endif
- ///////////////////////// misc tools ///////////////////////
- bool cmpb(void *b, int n, enet_uint32 c)
- {
- ENetBuffer buf;
- buf.data = b;
- buf.dataLength = n;
- return enet_crc32(&buf, 1)==c;
- }
- bool cmpf(char *fn, enet_uint32 c)
- {
- int n = 0;
- char *b = loadfile(fn, &n);
- bool r = cmpb(b, n, c);
- delete[] b;
- return r;
- }
- enet_uint32 adler(unsigned char *data, size_t len)
- {
- enet_uint32 a = 1, b = 0;
- while (len--)
- {
- a += *data++;
- b += a;
- }
- return b;
- }
- bool isbigendian()
- {
- return !*(const uchar *)&islittleendian;
- }
- void strtoupper(char *t, const char *s)
- {
- if(!s) s = t;
- while(*s)
- {
- *t = toupper(*s);
- t++; s++;
- }
- *t = '\0';
- }
- const char *atoip(const char *s, enet_uint32 *ip)
- {
- unsigned int d[4];
- int n;
- if(!s || sscanf(s, "%u.%u.%u.%u%n", d, d + 1, d + 2, d + 3, &n) != 4) return NULL;
- *ip = 0;
- loopi(4)
- {
- if(d[i] > 255) return NULL;
- *ip = (*ip << 8) + d[i];
- }
- return s + n;
- }
- const char *atoipr(const char *s, iprange *ir)
- {
- if((s = atoip(s, &ir->lr)) == NULL) return NULL;
- ir->ur = ir->lr;
- s += strspn(s, " \t");
- if(*s == '-')
- {
- if(!(s = atoip(s + 1, &ir->ur)) || ir->lr > ir->ur) return NULL;
- }
- else if(*s == '/')
- {
- int m, n;
- if(sscanf(s + 1, "%d%n", &m, &n) != 1 || m < 0 || m > 32) return NULL;
- unsigned long bm = (1 << (32 - m)) - 1;
- ir->lr &= ~bm;
- ir->ur |= bm;
- s += 1 + n;
- }
- return s;
- }
- const char *iptoa(const enet_uint32 ip)
- {
- static string s[2];
- static int buf = 0;
- buf = (buf + 1) % 2;
- formatstring(s[buf])("%d.%d.%d.%d", (ip >> 24) & 255, (ip >> 16) & 255, (ip >> 8) & 255, ip & 255);
- return s[buf];
- }
- const char *iprtoa(const struct iprange &ipr)
- {
- static string s[2];
- static int buf = 0;
- buf = (buf + 1) % 2;
- if(ipr.lr == ipr.ur) copystring(s[buf], iptoa(ipr.lr));
- else formatstring(s[buf])("%s-%s", iptoa(ipr.lr), iptoa(ipr.ur));
- return s[buf];
- }
- int cmpiprange(const struct iprange *a, const struct iprange *b)
- {
- if(a->lr < b->lr) return -1;
- if(a->lr > b->lr) return 1;
- return 0;
- }
- int cmpipmatch(const struct iprange *a, const struct iprange *b)
- {
- return - (a->lr < b->lr) + (a->lr > b->ur);
- }
- char *concatformatstring(char *d, const char *s, ...)
- {
- static defvformatstring(temp, s, s);
- return concatstring(d, temp);
- }
- int cvecprintf(vector<char> &v, const char *s, ...)
- {
- defvformatstring(temp, s, s);
- int len = strlen(temp);
- if(len) v.put(temp, len);
- return len;
- }
- const char *hiddenpwd(const char *pwd, int showchars)
- {
- static int sc = 3;
- static string text;
- copystring(text, pwd);
- if(showchars > 0) sc = showchars;
- for(int i = (int)strlen(text) - 1; i >= sc; i--) text[i] = '*';
- return text;
- }
- int getlistindex(const char *key, const char *list[], bool acceptnumeric, int deflt)
- {
- int max = 0;
- while(list[max][0]) if(!strcasecmp(key, list[max])) return max; else max++;
- if(acceptnumeric && isdigit(key[0]))
- {
- int i = (int)strtol(key, NULL, 0);
- if(i >= 0 && i < max) return i;
- }
- #if !defined(STANDALONE) && defined(_DEBUG)
- char *opts = conc(list, -1, true);
- if(*key) clientlogf("warning: unknown token \"%s\" (not in list [%s])", key, opts);
- delstring(opts);
- #endif
- return deflt;
- }
- //////////////// geometry utils ////////////////
- static inline float det2x2(float a, float b, float c, float d) { return a*d - b*c; }
- static inline float det3x3(float a1, float a2, float a3,
- float b1, float b2, float b3,
- float c1, float c2, float c3)
- {
- return a1 * det2x2(b2, b3, c2, c3)
- - b1 * det2x2(a2, a3, c2, c3)
- + c1 * det2x2(a2, a3, b2, b3);
- }
- float glmatrixf::determinant() const
- {
- float a1 = v[0], a2 = v[1], a3 = v[2], a4 = v[3],
- b1 = v[4], b2 = v[5], b3 = v[6], b4 = v[7],
- c1 = v[8], c2 = v[9], c3 = v[10], c4 = v[11],
- d1 = v[12], d2 = v[13], d3 = v[14], d4 = v[15];
- return a1 * det3x3(b2, b3, b4, c2, c3, c4, d2, d3, d4)
- - b1 * det3x3(a2, a3, a4, c2, c3, c4, d2, d3, d4)
- + c1 * det3x3(a2, a3, a4, b2, b3, b4, d2, d3, d4)
- - d1 * det3x3(a2, a3, a4, b2, b3, b4, c2, c3, c4);
- }
- void glmatrixf::adjoint(const glmatrixf &m)
- {
- float a1 = m.v[0], a2 = m.v[1], a3 = m.v[2], a4 = m.v[3],
- b1 = m.v[4], b2 = m.v[5], b3 = m.v[6], b4 = m.v[7],
- c1 = m.v[8], c2 = m.v[9], c3 = m.v[10], c4 = m.v[11],
- d1 = m.v[12], d2 = m.v[13], d3 = m.v[14], d4 = m.v[15];
- v[0] = det3x3(b2, b3, b4, c2, c3, c4, d2, d3, d4);
- v[1] = -det3x3(a2, a3, a4, c2, c3, c4, d2, d3, d4);
- v[2] = det3x3(a2, a3, a4, b2, b3, b4, d2, d3, d4);
- v[3] = -det3x3(a2, a3, a4, b2, b3, b4, c2, c3, c4);
- v[4] = -det3x3(b1, b3, b4, c1, c3, c4, d1, d3, d4);
- v[5] = det3x3(a1, a3, a4, c1, c3, c4, d1, d3, d4);
- v[6] = -det3x3(a1, a3, a4, b1, b3, b4, d1, d3, d4);
- v[7] = det3x3(a1, a3, a4, b1, b3, b4, c1, c3, c4);
- v[8] = det3x3(b1, b2, b4, c1, c2, c4, d1, d2, d4);
- v[9] = -det3x3(a1, a2, a4, c1, c2, c4, d1, d2, d4);
- v[10] = det3x3(a1, a2, a4, b1, b2, b4, d1, d2, d4);
- v[11] = -det3x3(a1, a2, a4, b1, b2, b4, c1, c2, c4);
- v[12] = -det3x3(b1, b2, b3, c1, c2, c3, d1, d2, d3);
- v[13] = det3x3(a1, a2, a3, c1, c2, c3, d1, d2, d3);
- v[14] = -det3x3(a1, a2, a3, b1, b2, b3, d1, d2, d3);
- v[15] = det3x3(a1, a2, a3, b1, b2, b3, c1, c2, c3);
- }
- bool glmatrixf::invert(const glmatrixf &m, float mindet)
- {
- float a1 = m.v[0], b1 = m.v[4], c1 = m.v[8], d1 = m.v[12];
- adjoint(m);
- float det = a1*v[0] + b1*v[1] + c1*v[2] + d1*v[3]; // float det = m.determinant();
- if(fabs(det) < mindet) return false;
- float invdet = 1/det;
- loopi(16) v[i] *= invdet;
- return true;
- }
- // multithreading and ipc tools wrapper for the server
- // all embedded servers and all standalone servers except on linux use SDL
- // the standalone linux version uses native linux libraries - and also makes use of shared memory
- #ifdef AC_USE_SDL_THREADS
- #include "SDL_timer.h"
- #include "SDL_thread.h" // also fetches SDL_mutex.h
- #else
- #include <pthread.h>
- #include <semaphore.h>
- #include <sys/shm.h>
- #endif
- static int sl_sem_errorcountdummy = 0;
- #ifdef AC_USE_SDL_THREADS
- sl_semaphore::sl_semaphore(int init, int *ecnt)
- {
- data = (void *)SDL_CreateSemaphore(init);
- errorcount = ecnt ? ecnt : &sl_sem_errorcountdummy;
- if(data == NULL) (*errorcount)++;
- }
- sl_semaphore::~sl_semaphore()
- {
- if(data) SDL_DestroySemaphore((SDL_sem *) data);
- }
- void sl_semaphore::wait()
- {
- if(SDL_SemWait((SDL_sem *) data) != 0) (*errorcount)++;
- }
- int sl_semaphore::trywait()
- {
- return SDL_SemTryWait((SDL_sem *) data);
- }
- int sl_semaphore::timedwait(int howlongmillis)
- {
- return SDL_SemWaitTimeout((SDL_sem *) data, howlongmillis);
- }
- int sl_semaphore::getvalue()
- {
- return SDL_SemValue((SDL_sem *) data);
- }
- void sl_semaphore::post()
- {
- if(SDL_SemPost((SDL_sem *) data) != 0) (*errorcount)++;
- }
- #else
- sl_semaphore::sl_semaphore(int init, int *ecnt)
- {
- errorcount = ecnt ? ecnt : &sl_sem_errorcountdummy;
- data = (void *) new sem_t;
- if(data == NULL || sem_init((sem_t *) data, 0, init) != 0) (*errorcount)++;
- }
- sl_semaphore::~sl_semaphore()
- {
- if(data)
- {
- if(sem_destroy((sem_t *) data) != 0) (*errorcount)++;
- delete (sem_t *)data;
- }
- }
- void sl_semaphore::wait()
- {
- if(sem_wait((sem_t *) data) != 0) (*errorcount)++;
- }
- int sl_semaphore::trywait()
- {
- return sem_trywait((sem_t *) data);
- }
- int sl_semaphore::timedwait(int howlongmillis)
- {
- struct timespec t;
- if(clock_gettime(CLOCK_REALTIME, &t))
- {
- (*errorcount)++;
- return sem_trywait((sem_t *) data);
- }
- howlongmillis += t.tv_nsec / 1000000;
- t.tv_nsec = (howlongmillis % 1000) * 1000000;
- t.tv_sec += howlongmillis / 1000;
- return sem_timedwait((sem_t *) data, &t);
- }
- int sl_semaphore::getvalue()
- {
- int ret;
- if(sem_getvalue((sem_t *) data, &ret) != 0) (*errorcount)++;
- return ret;
- }
- void sl_semaphore::post()
- {
- if(sem_post((sem_t *) data) != 0) (*errorcount)++;
- }
- #endif
- // (wrapping threads is slightly ugly, since SDL threads use a different return value (int) than pthreads (void *) - and that can't be solved with a typecast)
- #ifdef AC_USE_SDL_THREADS
- struct sl_threadinfo { int (*fn)(void *); void *data; SDL_Thread *handle; volatile char done; };
- static int sl_thread_indir(void *info)
- {
- int res = (*((sl_threadinfo*)info)->fn)(((sl_threadinfo*)info)->data);
- ((sl_threadinfo*)info)->done = 1;
- return res;
- }
- void *sl_createthread(int (*fn)(void *), void *data)
- {
- sl_threadinfo *ti = new sl_threadinfo;
- ti->data = data;
- ti->fn = fn;
- ti->done = 0;
- ti->handle = SDL_CreateThread(sl_thread_indir, NULL, ti);
- return (void *) ti;
- }
- int sl_waitthread(void *ti)
- {
- int res;
- SDL_WaitThread(((sl_threadinfo *)ti)->handle, &res);
- delete (sl_threadinfo *) ti;
- return res;
- }
- static vector<sl_threadinfo *> oldthreads;
- static sl_semaphore sem_oldthreads(1, NULL);
- void sl_detachthread(void *ti) // SDL can't actually detach threads, so this is manual cleanup
- {
- if(ti && sl_pollthread(ti))
- {
- SDL_WaitThread(((sl_threadinfo *)ti)->handle, NULL);
- delete (sl_threadinfo *) ti;
- ti = NULL;
- }
- if(ti || sem_oldthreads.getvalue() > 0)
- {
- sem_oldthreads.wait();
- if(ti) oldthreads.add((sl_threadinfo *)ti);
- loopvrev(oldthreads)
- {
- if(oldthreads[i]->done)
- {
- SDL_WaitThread(oldthreads[i]->handle, NULL);
- delete oldthreads.remove(i);
- }
- }
- sem_oldthreads.post();
- }
- }
- static uint32_t mainthreadid = SDL_ThreadID();
- bool ismainthread() { return mainthreadid == SDL_ThreadID(); }
- #else
- struct sl_threadinfo { int (*fn)(void *); void *data; pthread_t handle; int res; volatile char done; };
- static void *sl_thread_indir(void *info)
- {
- sl_threadinfo *ti = (sl_threadinfo*) info;
- ti->res = (ti->fn)(ti->data);
- ti->done = 1;
- return &ti->res;
- }
- void *sl_createthread(int (*fn)(void *), void *data)
- {
- sl_threadinfo *ti = new sl_threadinfo;
- ti->data = data;
- ti->fn = fn;
- ti->done = 0;
- pthread_create(&(ti->handle), NULL, sl_thread_indir, ti);
- return (void *) ti;
- }
- int sl_waitthread(void *ti)
- {
- void *res;
- pthread_join(((sl_threadinfo *)ti)->handle, &res);
- int ires = *((int *)res);
- delete (sl_threadinfo *) ti;
- return ires;
- }
- static vector<sl_threadinfo *> oldthreads;
- static sl_semaphore sem_oldthreads(1, NULL);
- void sl_detachthread(void *ti)
- {
- if(ti) pthread_detach(((sl_threadinfo *)ti)->handle);
- if(ti || sem_oldthreads.getvalue() > 0)
- {
- sem_oldthreads.wait();
- if(ti) oldthreads.add((sl_threadinfo *)ti);
- loopvrev(oldthreads) if(oldthreads[i]->done) delete oldthreads.remove(i);
- sem_oldthreads.post();
- }
- }
- static pthread_t mainthreadid = pthread_self();
- bool ismainthread() { return pthread_equal(mainthreadid, pthread_self()) != 0; }
- #endif
- bool sl_pollthread(void *ti)
- {
- return ((sl_threadinfo*)ti)->done != 0;
- }
- // platform dependent stuff not covered by enet (use POSIX or, if possible, SDL)
- #ifdef AC_USE_SDL_THREADS
- void sl_sleep(int duration)
- {
- SDL_Delay(duration);
- }
- #else
- void sl_sleep(int duration)
- {
- struct timespec t = { duration / 1000, (duration % 1000) * 1000000 };
- nanosleep(&t, NULL);
- }
- #endif
- void parseupdatelist(hashtable<const char *, int> &ht, char *buf, const char *prefix, const char *suffix)
- {
- for(char *d = buf; *d; d++) if(!isalnum(*d) && !strchr("._-() /\n", *d)) *d = ' '; // allowed chars in media path strings (except ' ')
- char *p, *l = buf, *m;
- int rev, *revp, plen = prefix ? (int)strlen(prefix) : 0, slen = suffix ? (int)strlen(suffix) : 0;
- do
- {
- if((p = strchr(l, '\n'))) *p = '\0'; // break into single lines
- l += strspn(l, " "); // skip leading spaces
- if((m = strchr(l, ' ')) && (rev = atoi(m + 1))) // string has a space and a number != 0 after it
- {
- *m = '\0';
- if((!plen || !strncmp(l, prefix, plen)) && // prefix is either not required or present
- (!slen || (m - l > slen && !strcmp(m - slen, suffix)))) // suffix is either not required or present
- {
- l += plen; // skip prefix
- m[-slen] = '\0'; // cut suffix
- revp = ht.access(l);
- if(revp) *revp = rev;
- else ht.access(newstring(l), rev);
- }
- }
- l = p + 1;
- }
- while(p);
- }
|