use smartjump on nq servers.
try to work around a Chrome webaudio bug, to stop an exception preventing the webgl port from starting any maps. don't open so many ipv6 client sockets (hybrid sockets don't need it). fix fteqcc to support qccx a bit better. crmod should now compile when using -fqccx, seems to work in proquake too (ignoring the string replacer hack/tool). git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@5105 fc73d0e0-1445-4013-8a0c-d673dee63da5
This commit is contained in:
parent
445e7f476b
commit
f64cb13cab
7 changed files with 230 additions and 165 deletions
|
@ -284,7 +284,8 @@ void IN_JumpDown (void)
|
||||||
#endif
|
#endif
|
||||||
#ifdef QUAKESTATS
|
#ifdef QUAKESTATS
|
||||||
if (condition && cl.playerview[pnum].stats[STAT_HEALTH] > 0 && !cls.demoplayback && !cl.spectator &&
|
if (condition && cl.playerview[pnum].stats[STAT_HEALTH] > 0 && !cls.demoplayback && !cl.spectator &&
|
||||||
cl.inframes[cl.validsequence&UPDATE_MASK].playerstate[cl.playerview[pnum].playernum].messagenum == cl.validsequence && cl.playerview[pnum].waterlevel >= 2 && (!cl.teamfortress || !(in_forward.state[pnum] & 1))
|
(cls.protocol==CP_NETQUAKE || cl.inframes[cl.validsequence&UPDATE_MASK].playerstate[cl.playerview[pnum].playernum].messagenum == cl.validsequence)
|
||||||
|
&& cl.playerview[pnum].waterlevel >= 2 && (!cl.teamfortress || !(in_forward.state[pnum] & 1))
|
||||||
)
|
)
|
||||||
KeyDown(&in_up);
|
KeyDown(&in_up);
|
||||||
else
|
else
|
||||||
|
|
|
@ -499,7 +499,6 @@ void OpenAL_CvarInit(void)
|
||||||
Cvar_Register(&s_al_speedofsound, SOUNDVARS);
|
Cvar_Register(&s_al_speedofsound, SOUNDVARS);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern float voicevolumemod;
|
|
||||||
static void OpenAL_ListenerUpdate(soundcardinfo_t *sc, int entnum, vec3_t origin, vec3_t forward, vec3_t right, vec3_t up, vec3_t velocity)
|
static void OpenAL_ListenerUpdate(soundcardinfo_t *sc, int entnum, vec3_t origin, vec3_t forward, vec3_t right, vec3_t up, vec3_t velocity)
|
||||||
{
|
{
|
||||||
oalinfo_t *oali = sc->handle;
|
oalinfo_t *oali = sc->handle;
|
||||||
|
@ -913,7 +912,18 @@ static void OpenAL_ChannelUpdate(soundcardinfo_t *sc, channel_t *chan, unsigned
|
||||||
case 2: //linear, mimic quake.
|
case 2: //linear, mimic quake.
|
||||||
case 3: //linear clamped to further than ref distance
|
case 3: //linear clamped to further than ref distance
|
||||||
palSourcef(src, AL_ROLLOFF_FACTOR, 1);
|
palSourcef(src, AL_ROLLOFF_FACTOR, 1);
|
||||||
|
#ifdef FTE_TARGET_WEB
|
||||||
|
//chrome complains about 0.
|
||||||
|
//with the expontential model, 0 results in division by zero, but we're not using that model and the maths for the linear model is fine with it.
|
||||||
|
//the web audio spec says 'The default value is 1. A RangeError exception must be thrown if this is set to a non-negative value.'
|
||||||
|
//which of course means that the PannerNode's constructor must throw an exception, which kinda prevents you ever creating one.
|
||||||
|
//it also says elsewhere 'If dref = 0, the value of the [exponential|inverse] model is taken to be 0, ...'
|
||||||
|
//which shows that the spec should read 'negative values' for rangeerrors (rather than non-positive). so chrome is being shit.
|
||||||
|
//unfortutely due to the nature of javascript and exceptions, this is fucking everything else up. thanks chrome!
|
||||||
|
palSourcef(src, AL_REFERENCE_DISTANCE, 0.01);
|
||||||
|
#else
|
||||||
palSourcef(src, AL_REFERENCE_DISTANCE, 0);
|
palSourcef(src, AL_REFERENCE_DISTANCE, 0);
|
||||||
|
#endif
|
||||||
palSourcef(src, AL_MAX_DISTANCE, 1/chan->dist_mult);
|
palSourcef(src, AL_MAX_DISTANCE, 1/chan->dist_mult);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2958,9 +2958,26 @@ qboolean NET_PortToAdr (netadrtype_t adrfamily, netproto_t adrprot, const char *
|
||||||
/*just here to prevent the client from spamming new sockets, which can be a problem with certain q2 servers*/
|
/*just here to prevent the client from spamming new sockets, which can be a problem with certain q2 servers*/
|
||||||
qboolean FTENET_Datagram_ChangeLocalAddress(struct ftenet_generic_connection_s *con, netadr_t *adr)
|
qboolean FTENET_Datagram_ChangeLocalAddress(struct ftenet_generic_connection_s *con, netadr_t *adr)
|
||||||
{
|
{
|
||||||
if (adr->type == con->addrtype[0] || adr->type == con->addrtype[1])
|
struct sockaddr_qstorage address;
|
||||||
if (adr->port == 0)
|
netadr_t current;
|
||||||
return true; //they want to use addr_any. it doesn't matter one jot which port we're currently listening on then.
|
int namelen;
|
||||||
|
if (getsockname (con->thesocket, (struct sockaddr *)&address, &namelen) == 0)
|
||||||
|
{
|
||||||
|
SockadrToNetadr(&address, ¤t);
|
||||||
|
|
||||||
|
//make sure the types match (special check for ipv6 hybrid sockets that accept ipv4 too)
|
||||||
|
if (adr->type == current.type
|
||||||
|
#if defined(HAVE_IPV4) && defined(HAVE_IPV6)
|
||||||
|
|| (net_hybriddualstack.ival && adr->type == NA_IP && current.type == NA_IPV6)
|
||||||
|
#endif
|
||||||
|
)
|
||||||
|
{ //make sure the port is currect (or they don't care which port)
|
||||||
|
if (adr->port == current.port || !adr->port)
|
||||||
|
return true; //then pretend we changed it, because needed to change in the first place.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//doesn't match how its currently bound, so I guess we need to rebind then.
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7271,12 +7288,21 @@ void NET_InitClient(qboolean loopbackonly)
|
||||||
#endif
|
#endif
|
||||||
if (loopbackonly)
|
if (loopbackonly)
|
||||||
port = "";
|
port = "";
|
||||||
#ifdef HAVE_IPV4
|
#if defined(HAVE_IPV4) && defined(HAVE_IPV6)
|
||||||
FTENET_AddToCollection(cls.sockets, "CLUDP4", port, NA_IP, NP_DGRAM, true);
|
if (net_hybriddualstack.ival)
|
||||||
#endif
|
{
|
||||||
#ifdef IPPROTO_IPV6
|
FTENET_AddToCollection(cls.sockets, "CLUDP", port, NA_IP, NP_DGRAM, true);
|
||||||
FTENET_AddToCollection(cls.sockets, "CLUDP6", port, NA_IPV6, NP_DGRAM, true);
|
}
|
||||||
|
else
|
||||||
#endif
|
#endif
|
||||||
|
{
|
||||||
|
#ifdef HAVE_IPV4
|
||||||
|
FTENET_AddToCollection(cls.sockets, "CLUDP4", port, NA_IP, NP_DGRAM, true);
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_IPV6
|
||||||
|
FTENET_AddToCollection(cls.sockets, "CLUDP6", port, NA_IPV6, NP_DGRAM, true);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
#ifdef USEIPX
|
#ifdef USEIPX
|
||||||
FTENET_AddToCollection(cls.sockets, "CLIPX", port, NA_IPX, NP_DGRAM, true);
|
FTENET_AddToCollection(cls.sockets, "CLIPX", port, NA_IPX, NP_DGRAM, true);
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -757,7 +757,8 @@ enum {
|
||||||
WARN_CONSTANTCOMPARISON,
|
WARN_CONSTANTCOMPARISON,
|
||||||
WARN_UNSAFEFUNCTIONRETURNTYPE,
|
WARN_UNSAFEFUNCTIONRETURNTYPE,
|
||||||
WARN_MISSINGOPTIONAL,
|
WARN_MISSINGOPTIONAL,
|
||||||
WARN_SYSTEMCRC,
|
WARN_SYSTEMCRC, //unknown system crc
|
||||||
|
WARN_SYSTEMCRC2, //legacy/dp system crc
|
||||||
WARN_CONDITIONALTYPEMISMATCH,
|
WARN_CONDITIONALTYPEMISMATCH,
|
||||||
WARN_MISSINGMEMBERQUALIFIER,//virtual/static/nonvirtual qualifier is missing
|
WARN_MISSINGMEMBERQUALIFIER,//virtual/static/nonvirtual qualifier is missing
|
||||||
WARN_SELFNOTTHIS, //warned for because 'self' does not have the right type. we convert such references to 'this' instead, which is more usable.
|
WARN_SELFNOTTHIS, //warned for because 'self' does not have the right type. we convert such references to 'this' instead, which is more usable.
|
||||||
|
|
|
@ -2287,7 +2287,7 @@ QCC_sref_t QCC_PR_StatementFlags ( QCC_opcode_t *op, QCC_sref_t var_a, QCC_sref_
|
||||||
|
|
||||||
if (var_a.sym)
|
if (var_a.sym)
|
||||||
{
|
{
|
||||||
var_a.sym->referenced = true;
|
var_a.sym->referenced = true;
|
||||||
if (flags&STFL_PRESERVEA)
|
if (flags&STFL_PRESERVEA)
|
||||||
QCC_UnFreeTemp(var_a);
|
QCC_UnFreeTemp(var_a);
|
||||||
}
|
}
|
||||||
|
@ -5173,7 +5173,6 @@ QCC_sref_t QCC_PR_GenerateFunctionCallRef (QCC_sref_t newself, QCC_sref_t func,
|
||||||
if (copyop_idx == 0)
|
if (copyop_idx == 0)
|
||||||
src.ofs += copyop_v?3:1;
|
src.ofs += copyop_v?3:1;
|
||||||
}
|
}
|
||||||
QCC_FreeTemp(fparm);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{ //small and simple. yay.
|
{ //small and simple. yay.
|
||||||
|
@ -5224,6 +5223,8 @@ QCC_sref_t QCC_PR_GenerateFunctionCallRef (QCC_sref_t newself, QCC_sref_t func,
|
||||||
else
|
else
|
||||||
QCC_FreeTemp(QCC_PR_StatementFlags (&pr_opcodes[OP_STORE_F], args[i].ref, d, NULL, 0));
|
QCC_FreeTemp(QCC_PR_StatementFlags (&pr_opcodes[OP_STORE_F], args[i].ref, d, NULL, 0));
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
QCC_FreeTemp(args[i].ref);
|
||||||
}
|
}
|
||||||
args[i].ref = d;
|
args[i].ref = d;
|
||||||
}
|
}
|
||||||
|
@ -5938,9 +5939,13 @@ QCC_sref_t QCC_PR_ParseFunctionCall (QCC_ref_t *funcref) //warning, the func cou
|
||||||
QCC_PR_ParsePrintSRef(WARN_TOOMANYPARAMETERSFORFUNC, func);
|
QCC_PR_ParsePrintSRef(WARN_TOOMANYPARAMETERSFORFUNC, func);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flag_qccx && QCC_PR_CheckToken("#"))
|
if (QCC_PR_CheckToken("#"))
|
||||||
{
|
{
|
||||||
e = QCC_PR_BuildRef(¶mbuf[arg], REF_GLOBAL, QCC_MakeSRef(&def_parms[arg], 0, p?p:type_variant), nullsref, p?p:type_variant, true);
|
QCC_sref_t sr = QCC_MakeSRefForce(&def_parms[arg], 0, p?p:type_variant);
|
||||||
|
// sr.sym = &def_parms[arg];
|
||||||
|
// sr.ofs = 0;
|
||||||
|
// sr.cast = p?p:type_variant;
|
||||||
|
e = QCC_PR_BuildRef(¶mbuf[arg], REF_GLOBAL, sr, nullsref, p?p:type_variant, true);
|
||||||
}
|
}
|
||||||
else if (arg < t->num_parms && (QCC_PR_PeekToken (",") || QCC_PR_PeekToken (")")))
|
else if (arg < t->num_parms && (QCC_PR_PeekToken (",") || QCC_PR_PeekToken (")")))
|
||||||
{
|
{
|
||||||
|
@ -6734,7 +6739,8 @@ static QCC_ref_t *QCC_PR_ParseField(QCC_ref_t *refbuf, QCC_ref_t *lhs)
|
||||||
lhs = QCC_PR_ParseField(refbuf, lhs);
|
lhs = QCC_PR_ParseField(refbuf, lhs);
|
||||||
}
|
}
|
||||||
else if (flag_qccx && t->type == ev_entity && QCC_PR_CheckToken("["))
|
else if (flag_qccx && t->type == ev_entity && QCC_PR_CheckToken("["))
|
||||||
{
|
{ //p[%0] gives a regular array reference. except that p is probably a float, and we're expecting OP_LOAD_F
|
||||||
|
//might also be assigned to, so just create a regular field ref and figure that stuff out later.
|
||||||
QCC_ref_t *field;
|
QCC_ref_t *field;
|
||||||
QCC_ref_t fieldbuf;
|
QCC_ref_t fieldbuf;
|
||||||
field = QCC_PR_RefExpression(&fieldbuf, TOP_PRIORITY, 0);
|
field = QCC_PR_RefExpression(&fieldbuf, TOP_PRIORITY, 0);
|
||||||
|
@ -6744,6 +6750,21 @@ static QCC_ref_t *QCC_PR_ParseField(QCC_ref_t *refbuf, QCC_ref_t *lhs)
|
||||||
lhs = QCC_PR_BuildRef(refbuf, REF_FIELD, QCC_RefToDef(lhs, true), QCC_RefToDef(field, true), type_float, false);
|
lhs = QCC_PR_BuildRef(refbuf, REF_FIELD, QCC_RefToDef(lhs, true), QCC_RefToDef(field, true), type_float, false);
|
||||||
|
|
||||||
|
|
||||||
|
lhs = QCC_PR_ParseField(refbuf, lhs);
|
||||||
|
lhs = QCC_PR_ParseRefArrayPointer (refbuf, lhs, false, false);
|
||||||
|
}
|
||||||
|
else if (flag_qccx && t->type == ev_entity && QCC_PR_CheckToken("^"))
|
||||||
|
{ //p^[%0] is evaluated as an OP_LOAD_V (or OP_ADDRESS+OP_STOREP_V)
|
||||||
|
QCC_ref_t *field;
|
||||||
|
QCC_ref_t fieldbuf;
|
||||||
|
QCC_PR_Expect("[");
|
||||||
|
field = QCC_PR_RefExpression(&fieldbuf, TOP_PRIORITY, 0);
|
||||||
|
field->cast = type_floatfield;
|
||||||
|
QCC_PR_Expect("]");
|
||||||
|
|
||||||
|
lhs = QCC_PR_BuildRef(refbuf, REF_FIELD, QCC_RefToDef(lhs, true), QCC_RefToDef(field, true), type_vector, false);
|
||||||
|
|
||||||
|
|
||||||
lhs = QCC_PR_ParseField(refbuf, lhs);
|
lhs = QCC_PR_ParseField(refbuf, lhs);
|
||||||
lhs = QCC_PR_ParseRefArrayPointer (refbuf, lhs, false, false);
|
lhs = QCC_PR_ParseRefArrayPointer (refbuf, lhs, false, false);
|
||||||
}
|
}
|
||||||
|
@ -7186,11 +7207,10 @@ QCC_ref_t *QCC_PR_ParseRefValue (QCC_ref_t *refbuf, QCC_type_t *assumeclass, pbo
|
||||||
if (pr_token_type == tt_immediate)
|
if (pr_token_type == tt_immediate)
|
||||||
{
|
{
|
||||||
d = QCC_PR_ParseImmediate ();
|
d = QCC_PR_ParseImmediate ();
|
||||||
d.sym->referenced = true;
|
// d.sym->referenced = true;
|
||||||
return QCC_DefToRef(refbuf, d);
|
// return QCC_DefToRef(refbuf, d);
|
||||||
}
|
}
|
||||||
|
else if (QCC_PR_CheckToken("["))
|
||||||
if (QCC_PR_CheckToken("["))
|
|
||||||
{
|
{
|
||||||
//originally used for reacc - taking the form of [5 84 2]
|
//originally used for reacc - taking the form of [5 84 2]
|
||||||
//we redefine it to include statements - [a+b, c, 3+(d*2)]
|
//we redefine it to include statements - [a+b, c, 3+(d*2)]
|
||||||
|
@ -7224,153 +7244,155 @@ QCC_ref_t *QCC_PR_ParseRefValue (QCC_ref_t *refbuf, QCC_type_t *assumeclass, pbo
|
||||||
QCC_PR_Expect("]");
|
QCC_PR_Expect("]");
|
||||||
|
|
||||||
d = QCC_PR_GenerateVector(x,y,z);
|
d = QCC_PR_GenerateVector(x,y,z);
|
||||||
d.sym->referenced = true;
|
// d.sym->referenced = true;
|
||||||
return QCC_DefToRef(refbuf, d);
|
// return QCC_DefToRef(refbuf, d);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
if (QCC_PR_CheckToken("::"))
|
|
||||||
{
|
{
|
||||||
assumeclass = NULL;
|
if (QCC_PR_CheckToken("::"))
|
||||||
expandmemberfields = false; //::classname is always usable for eg: the find builtin.
|
|
||||||
}
|
|
||||||
name = QCC_PR_ParseName ();
|
|
||||||
|
|
||||||
//fixme: namespaces should be relative
|
|
||||||
if (QCC_PR_CheckToken("::"))
|
|
||||||
{
|
|
||||||
expandmemberfields = false; //this::classname should also be available to the find builtin, etc. this won't affect self.classname::member nor classname::staticfunc
|
|
||||||
|
|
||||||
if (assumeclass && !strcmp(name, "super"))
|
|
||||||
t = assumeclass->parentclass;
|
|
||||||
else if (assumeclass && !strcmp(name, "this"))
|
|
||||||
t = assumeclass;
|
|
||||||
else
|
|
||||||
t = QCC_TypeForName(name);
|
|
||||||
if (!t || t->type != ev_entity)
|
|
||||||
{
|
{
|
||||||
QCC_PR_ParseError (ERR_NOTATYPE, "Not a class \"%s\"", name);
|
assumeclass = NULL;
|
||||||
d = nullsref;
|
expandmemberfields = false; //::classname is always usable for eg: the find builtin.
|
||||||
|
}
|
||||||
|
name = QCC_PR_ParseName ();
|
||||||
|
|
||||||
|
//fixme: namespaces should be relative
|
||||||
|
if (QCC_PR_CheckToken("::"))
|
||||||
|
{
|
||||||
|
expandmemberfields = false; //this::classname should also be available to the find builtin, etc. this won't affect self.classname::member nor classname::staticfunc
|
||||||
|
|
||||||
|
if (assumeclass && !strcmp(name, "super"))
|
||||||
|
t = assumeclass->parentclass;
|
||||||
|
else if (assumeclass && !strcmp(name, "this"))
|
||||||
|
t = assumeclass;
|
||||||
|
else
|
||||||
|
t = QCC_TypeForName(name);
|
||||||
|
if (!t || t->type != ev_entity)
|
||||||
|
{
|
||||||
|
QCC_PR_ParseError (ERR_NOTATYPE, "Not a class \"%s\"", name);
|
||||||
|
d = nullsref;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QCC_type_t *p;
|
||||||
|
char membername[1024];
|
||||||
|
name = QCC_PR_ParseName ();
|
||||||
|
//walk up the parents if needed, to find one that has that field
|
||||||
|
for(d = nullsref, p = t; !d.cast && p; p = p->parentclass)
|
||||||
|
{
|
||||||
|
//use static functions in preference to virtual functions. kinda needed so you can use super::func...
|
||||||
|
QC_snprintfz(membername, sizeof(membername), "%s::%s", p->name, name);
|
||||||
|
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
||||||
|
if (!d.cast)
|
||||||
|
{
|
||||||
|
QC_snprintfz(membername, sizeof(membername), "%s::"MEMBERFIELDNAME, p->name, name);
|
||||||
|
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!d.cast)
|
||||||
|
{
|
||||||
|
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s::%s\"", t->name, name);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
QCC_type_t *p;
|
d = nullsref;
|
||||||
char membername[1024];
|
// 'testvar' becomes 'this::testvar'
|
||||||
name = QCC_PR_ParseName ();
|
if (assumeclass && assumeclass->parentclass)
|
||||||
//walk up the parents if needed, to find one that has that field
|
{ //try getting a member.
|
||||||
for(d = nullsref, p = t; !d.cast && p; p = p->parentclass)
|
QCC_type_t *type;
|
||||||
{
|
for(type = assumeclass; type && !d.cast; type = type->parentclass)
|
||||||
//use static functions in preference to virtual functions. kinda needed so you can use super::func...
|
|
||||||
QC_snprintfz(membername, sizeof(membername), "%s::%s", p->name, name);
|
|
||||||
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
|
||||||
if (!d.cast)
|
|
||||||
{
|
{
|
||||||
QC_snprintfz(membername, sizeof(membername), "%s::"MEMBERFIELDNAME, p->name, name);
|
//look for virtual things
|
||||||
|
QC_snprintfz(membername, sizeof(membername), "%s::"MEMBERFIELDNAME, type->name, name);
|
||||||
|
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
||||||
|
}
|
||||||
|
for(type = assumeclass; type && !d.cast; type = type->parentclass)
|
||||||
|
{
|
||||||
|
//look for non-virtual things (functions: after virtual stuff, because this will find the actual function def too)
|
||||||
|
QC_snprintfz(membername, sizeof(membername), "%s::%s", type->name, name);
|
||||||
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!d.cast)
|
if (!d.cast)
|
||||||
{
|
{
|
||||||
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s::%s\"", t->name, name);
|
// look through the defs
|
||||||
}
|
d = QCC_PR_GetSRef (NULL, name, pr_scope, false, 0, false);
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
d = nullsref;
|
|
||||||
// 'testvar' becomes 'this::testvar'
|
|
||||||
if (assumeclass && assumeclass->parentclass)
|
|
||||||
{ //try getting a member.
|
|
||||||
QCC_type_t *type;
|
|
||||||
for(type = assumeclass; type && !d.cast; type = type->parentclass)
|
|
||||||
{
|
|
||||||
//look for virtual things
|
|
||||||
QC_snprintfz(membername, sizeof(membername), "%s::"MEMBERFIELDNAME, type->name, name);
|
|
||||||
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
|
||||||
}
|
|
||||||
for(type = assumeclass; type && !d.cast; type = type->parentclass)
|
|
||||||
{
|
|
||||||
//look for non-virtual things (functions: after virtual stuff, because this will find the actual function def too)
|
|
||||||
QC_snprintfz(membername, sizeof(membername), "%s::%s", type->name, name);
|
|
||||||
d = QCC_PR_GetSRef (NULL, membername, pr_scope, false, 0, false);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!d.cast)
|
if (!d.cast)
|
||||||
{
|
{
|
||||||
// look through the defs
|
if (!strcmp(name, "nil"))
|
||||||
d = QCC_PR_GetSRef (NULL, name, pr_scope, false, 0, false);
|
d = QCC_MakeIntConst(0);
|
||||||
}
|
else if ( (!strcmp(name, "randomv")) ||
|
||||||
}
|
(!strcmp(name, "sizeof")) ||
|
||||||
|
(!strcmp(name, "alloca")) ||
|
||||||
if (!d.cast)
|
(!strcmp(name, "entnum")) ||
|
||||||
{
|
(!strcmp(name, "autocvar")) ||
|
||||||
if (!strcmp(name, "nil"))
|
(!strcmp(name, "used_model")) ||
|
||||||
d = QCC_MakeIntConst(0);
|
(!strcmp(name, "used_sound")) ||
|
||||||
else if ( (!strcmp(name, "randomv")) ||
|
(!strcmp(name, "va_arg")) ||
|
||||||
(!strcmp(name, "sizeof")) ||
|
(!strcmp(name, "...")) || //for compat. otherwise wtf?
|
||||||
(!strcmp(name, "alloca")) ||
|
(!strcmp(name, "_")) ) //intrinsics, any old function with no args will do.
|
||||||
(!strcmp(name, "entnum")) ||
|
|
||||||
(!strcmp(name, "autocvar")) ||
|
|
||||||
(!strcmp(name, "used_model")) ||
|
|
||||||
(!strcmp(name, "used_sound")) ||
|
|
||||||
(!strcmp(name, "va_arg")) ||
|
|
||||||
(!strcmp(name, "...")) || //for compat. otherwise wtf?
|
|
||||||
(!strcmp(name, "_")) ) //intrinsics, any old function with no args will do.
|
|
||||||
{
|
|
||||||
d = QCC_PR_GetSRef (type_function, name, NULL, true, 0, false);
|
|
||||||
// d->initialized = 0;
|
|
||||||
}
|
|
||||||
else if ( (!strcmp(name, "random" )) ) //intrinsics, any old function with no args will do. returning a float just in case people declare things in the wrong order
|
|
||||||
{
|
|
||||||
d = QCC_PR_GetSRef (type_floatfunction, name, NULL, true, 0, false);
|
|
||||||
// d.sym->initialized = 0;
|
|
||||||
}
|
|
||||||
else if (keyword_class && !strcmp(name, "this"))
|
|
||||||
{
|
|
||||||
if (!pr_classtype)
|
|
||||||
QCC_PR_ParseError(ERR_NOTANAME, "Cannot use 'this' outside of an OO function\n");
|
|
||||||
d = QCC_PR_GetSRef(type_entity, "self", NULL, true, 0, false);
|
|
||||||
d.cast = pr_classtype;
|
|
||||||
}
|
|
||||||
else if (keyword_class && !strcmp(name, "super"))
|
|
||||||
{
|
|
||||||
if (!assumeclass)
|
|
||||||
QCC_PR_ParseError(ERR_NOTANAME, "Cannot use 'super' outside of an OO function\n");
|
|
||||||
if (!assumeclass->parentclass)
|
|
||||||
QCC_PR_ParseError(ERR_NOTANAME, "class %s has no super\n", pr_classtype->name);
|
|
||||||
d = QCC_PR_GetSRef(NULL, "self", NULL, true, 0, false);
|
|
||||||
d.cast = assumeclass->parentclass;
|
|
||||||
}
|
|
||||||
else if (pr_assumetermtype)
|
|
||||||
{
|
|
||||||
d = QCC_PR_GetSRef (pr_assumetermtype, name, pr_assumetermscope, true, 0, pr_assumetermflags);
|
|
||||||
if (!d.cast)
|
|
||||||
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s\"", name);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
d = QCC_PR_GetSRef (type_variant, name, pr_scope, true, 0, false);
|
|
||||||
if (!expandmemberfields && assumeclass)
|
|
||||||
{
|
{
|
||||||
|
d = QCC_PR_GetSRef (type_function, name, NULL, true, 0, false);
|
||||||
|
// d->initialized = 0;
|
||||||
|
}
|
||||||
|
else if ( (!strcmp(name, "random" )) ) //intrinsics, any old function with no args will do. returning a float just in case people declare things in the wrong order
|
||||||
|
{
|
||||||
|
d = QCC_PR_GetSRef (type_floatfunction, name, NULL, true, 0, false);
|
||||||
|
// d.sym->initialized = 0;
|
||||||
|
}
|
||||||
|
else if (keyword_class && !strcmp(name, "this"))
|
||||||
|
{
|
||||||
|
if (!pr_classtype)
|
||||||
|
QCC_PR_ParseError(ERR_NOTANAME, "Cannot use 'this' outside of an OO function\n");
|
||||||
|
d = QCC_PR_GetSRef(type_entity, "self", NULL, true, 0, false);
|
||||||
|
d.cast = pr_classtype;
|
||||||
|
}
|
||||||
|
else if (keyword_class && !strcmp(name, "super"))
|
||||||
|
{
|
||||||
|
if (!assumeclass)
|
||||||
|
QCC_PR_ParseError(ERR_NOTANAME, "Cannot use 'super' outside of an OO function\n");
|
||||||
|
if (!assumeclass->parentclass)
|
||||||
|
QCC_PR_ParseError(ERR_NOTANAME, "class %s has no super\n", pr_classtype->name);
|
||||||
|
d = QCC_PR_GetSRef(NULL, "self", NULL, true, 0, false);
|
||||||
|
d.cast = assumeclass->parentclass;
|
||||||
|
}
|
||||||
|
else if (pr_assumetermtype)
|
||||||
|
{
|
||||||
|
d = QCC_PR_GetSRef (pr_assumetermtype, name, pr_assumetermscope, true, 0, pr_assumetermflags);
|
||||||
if (!d.cast)
|
if (!d.cast)
|
||||||
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown field \"%s\" in class \"%s\"", name, assumeclass->name);
|
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s\"", name);
|
||||||
else if (!assumeclass->parentclass && assumeclass != type_entity)
|
|
||||||
{
|
|
||||||
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Class \"%s\" is not defined, cannot access memeber \"%s\"", assumeclass->name, name);
|
|
||||||
if (!autoprototype && !autoprototyped)
|
|
||||||
QCC_PR_Note(ERR_UNKNOWNVALUE, s_filen, pr_source_line, "Consider using #pragma autoproto");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Unknown field \"%s\" in class \"%s\"", name, assumeclass->name);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (!d.cast)
|
d = QCC_PR_GetSRef (type_variant, name, pr_scope, true, 0, false);
|
||||||
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s\"", name);
|
if (!expandmemberfields && assumeclass)
|
||||||
|
{
|
||||||
|
if (!d.cast)
|
||||||
|
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown field \"%s\" in class \"%s\"", name, assumeclass->name);
|
||||||
|
else if (!assumeclass->parentclass && assumeclass != type_entity)
|
||||||
|
{
|
||||||
|
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Class \"%s\" is not defined, cannot access memeber \"%s\"", assumeclass->name, name);
|
||||||
|
if (!autoprototype && !autoprototyped)
|
||||||
|
QCC_PR_Note(ERR_UNKNOWNVALUE, s_filen, pr_source_line, "Consider using #pragma autoproto");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Unknown field \"%s\" in class \"%s\"", name, assumeclass->name);
|
||||||
|
}
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Unknown value \"%s\".", name);
|
if (!d.cast)
|
||||||
|
QCC_PR_ParseError (ERR_UNKNOWNVALUE, "Unknown value \"%s\"", name);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QCC_PR_ParseWarning (ERR_UNKNOWNVALUE, "Unknown value \"%s\".", name);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7697,12 +7719,13 @@ QCC_ref_t *QCC_PR_RefTerm (QCC_ref_t *retbuf, unsigned int exprflags)
|
||||||
{
|
{
|
||||||
r = QCC_PR_RefExpression (retbuf, UNARY_PRIORITY, EXPR_DISALLOW_COMMA);
|
r = QCC_PR_RefExpression (retbuf, UNARY_PRIORITY, EXPR_DISALLOW_COMMA);
|
||||||
if (flag_qccx && r->cast->type == ev_float)
|
if (flag_qccx && r->cast->type == ev_float)
|
||||||
{
|
{ //&%342 casts it to a (pre-dereferenced) pointer.
|
||||||
r->readonly = false;
|
r = QCC_PR_BuildRef(retbuf, REF_POINTER, QCC_RefToDef(r, true), nullsref, type_float, false);
|
||||||
r->cast = QCC_PR_PointerType(r->cast);
|
|
||||||
}
|
}
|
||||||
else if (flag_qccx && (r->cast->type == ev_string || r->cast->type == ev_field || r->cast->type == ev_entity || r->cast->type == ev_function))
|
else if (flag_qccx && (r->cast->type == ev_string || r->cast->type == ev_field || r->cast->type == ev_entity || r->cast->type == ev_function))
|
||||||
|
{ //&string casts it to a float. does not dereference it
|
||||||
r->cast = type_float;
|
r->cast = type_float;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
r = QCC_PR_GenerateAddressOf(retbuf, r);
|
r = QCC_PR_GenerateAddressOf(retbuf, r);
|
||||||
return r;
|
return r;
|
||||||
|
@ -7711,7 +7734,9 @@ QCC_ref_t *QCC_PR_RefTerm (QCC_ref_t *retbuf, unsigned int exprflags)
|
||||||
{
|
{
|
||||||
e = QCC_PR_Expression (UNARY_PRIORITY, EXPR_DISALLOW_COMMA);
|
e = QCC_PR_Expression (UNARY_PRIORITY, EXPR_DISALLOW_COMMA);
|
||||||
if (flag_qccx && (e.cast->type == ev_float || e.cast->type == ev_integer))
|
if (flag_qccx && (e.cast->type == ev_float || e.cast->type == ev_integer))
|
||||||
|
{ //just an evil cast. note that qccx assumes offsets rather than indexes, so these are often quite large and typically refer to some index into the world entity.
|
||||||
return QCC_PR_BuildRef(retbuf, REF_GLOBAL, e, nullsref, type_entity, false);
|
return QCC_PR_BuildRef(retbuf, REF_GLOBAL, e, nullsref, type_entity, false);
|
||||||
|
}
|
||||||
else if (e.cast->type == ev_pointer) //FIXME: arrays
|
else if (e.cast->type == ev_pointer) //FIXME: arrays
|
||||||
return QCC_PR_BuildRef(retbuf, REF_POINTER, e, nullsref, e.cast->aux_type, false);
|
return QCC_PR_BuildRef(retbuf, REF_POINTER, e, nullsref, e.cast->aux_type, false);
|
||||||
else if (e.cast->accessors)
|
else if (e.cast->accessors)
|
||||||
|
@ -9292,17 +9317,14 @@ QCC_ref_t *QCC_PR_RefExpression (QCC_ref_t *retbuf, int priority, int exprflags)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (flag_qccx && lhsr->cast->type == ev_pointer && rhsd.cast->type == ev_float)
|
/*if (flag_qccx && lhsr->cast->type == ev_pointer && rhsd.cast->type == ev_float)
|
||||||
{
|
{ //&%555 = 4.0;
|
||||||
char totypename[256], fromtypename[256], destname[256];
|
char destname[256];
|
||||||
TypeName(lhsr->cast, totypename, sizeof(totypename));
|
QCC_PR_ParseWarning(WARN_LAXCAST, "Implicit pointer dereference on assignment to %s", QCC_GetRefName(lhsr, destname, sizeof(destname)));
|
||||||
TypeName(rhsd.cast, fromtypename, sizeof(fromtypename));
|
|
||||||
QCC_PR_ParseWarning(WARN_LAXCAST, "Implicit type mismatch on assignment to %s. Needed %s, got %s.", QCC_GetRefName(lhsr, destname, sizeof(destname)), totypename, fromtypename);
|
|
||||||
|
|
||||||
lhsd = QCC_RefToDef(lhsr, true);
|
lhsd = QCC_RefToDef(lhsr, true);
|
||||||
lhsr = QCC_PR_BuildRef(retbuf, REF_POINTER, lhsd, nullsref, lhsd.cast->aux_type, false);
|
lhsr = QCC_PR_BuildRef(retbuf, REF_POINTER, lhsd, nullsref, lhsd.cast->aux_type, false);
|
||||||
}
|
}
|
||||||
else if (QCC_SRef_IsNull(rhsd))
|
else */if (QCC_SRef_IsNull(rhsd))
|
||||||
{
|
{
|
||||||
QCC_FreeTemp(rhsd);
|
QCC_FreeTemp(rhsd);
|
||||||
if (lhsr->cast->type == ev_vector)
|
if (lhsr->cast->type == ev_vector)
|
||||||
|
@ -9865,9 +9887,13 @@ void QCC_PR_ParseStatement (void)
|
||||||
}
|
}
|
||||||
else if (pr_scope->type->aux_type->type != e.cast->type)
|
else if (pr_scope->type->aux_type->type != e.cast->type)
|
||||||
{
|
{
|
||||||
e = QCC_EvaluateCast(e, pr_scope->type->aux_type, true);
|
if (pr_scope->type->aux_type->type == ev_void)
|
||||||
//e = QCC_SupplyConversion(e, pr_scope->type->aux_type->type, true);
|
{ //returning a value inside a function defined to return void is bad dude.
|
||||||
// QCC_PR_ParseWarning(WARN_WRONGRETURNTYPE, "\'%s\' returned %s, expected %s", pr_scope->name, e->type->name, pr_scope->type->aux_type->name);
|
QCC_PR_ParseWarning(WARN_WRONGRETURNTYPE, "\'%s\' returned %s, expected %s", pr_scope->name, e.sym->type->name, pr_scope->type->aux_type->name);
|
||||||
|
e = QCC_EvaluateCast(e, type_variant, true);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
e = QCC_EvaluateCast(e, pr_scope->type->aux_type, true);
|
||||||
}
|
}
|
||||||
PR_GenerateReturnOuts();
|
PR_GenerateReturnOuts();
|
||||||
QCC_FreeTemp(QCC_PR_Statement (&pr_opcodes[OP_RETURN], e, nullsref, NULL));
|
QCC_FreeTemp(QCC_PR_Statement (&pr_opcodes[OP_RETURN], e, nullsref, NULL));
|
||||||
|
@ -13608,7 +13634,7 @@ void QCC_PR_ParseEnum(pbool flags)
|
||||||
name = QCC_PR_ParseName();
|
name = QCC_PR_ParseName();
|
||||||
if (QCC_PR_CheckToken("="))
|
if (QCC_PR_CheckToken("="))
|
||||||
{
|
{
|
||||||
if (pr_token_type == tt_immediate && pr_immediate_type->type == ev_float)
|
/*if (pr_token_type == tt_immediate && pr_immediate_type->type == ev_float)
|
||||||
{
|
{
|
||||||
iv = fv = pr_immediate._float;
|
iv = fv = pr_immediate._float;
|
||||||
QCC_PR_Lex();
|
QCC_PR_Lex();
|
||||||
|
@ -13618,10 +13644,10 @@ void QCC_PR_ParseEnum(pbool flags)
|
||||||
fv = iv = pr_immediate._int;
|
fv = iv = pr_immediate._int;
|
||||||
QCC_PR_Lex();
|
QCC_PR_Lex();
|
||||||
}
|
}
|
||||||
else
|
else*/
|
||||||
{
|
{
|
||||||
const QCC_eval_t *eval;
|
const QCC_eval_t *eval;
|
||||||
sref = QCC_PR_GetSRef(NULL, QCC_PR_ParseName(), NULL, false, 0, GDF_STRIP);
|
sref = QCC_PR_Expression(TOP_PRIORITY, EXPR_DISALLOW_COMMA);
|
||||||
eval = QCC_SRef_EvalConst(sref);
|
eval = QCC_SRef_EvalConst(sref);
|
||||||
if (eval)
|
if (eval)
|
||||||
{
|
{
|
||||||
|
|
|
@ -3445,7 +3445,7 @@ void QCC_PR_Lex (void)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!flag_qccx && c == '#' && !(pr_file_p[1]=='\"' || pr_file_p[1]=='-' || (pr_file_p[1]>='0' && pr_file_p[1] <='9'))) //hash and not number
|
if (!flag_qccx && c == '#' && !(pr_file_p[1]==')' || pr_file_p[1]==',' || pr_file_p[1]=='\"' || pr_file_p[1]=='-' || (pr_file_p[1]>='0' && pr_file_p[1] <='9'))) //hash and not number
|
||||||
{
|
{
|
||||||
pr_file_p++;
|
pr_file_p++;
|
||||||
if (!QCC_PR_CheckCompConst())
|
if (!QCC_PR_CheckCompConst())
|
||||||
|
|
|
@ -215,6 +215,7 @@ struct {
|
||||||
{" F208", WARN_NOTREFERENCEDCONST},
|
{" F208", WARN_NOTREFERENCEDCONST},
|
||||||
{" F209", WARN_EXTRAPRECACHE},
|
{" F209", WARN_EXTRAPRECACHE},
|
||||||
{" F210", WARN_NOTPRECACHED},
|
{" F210", WARN_NOTPRECACHED},
|
||||||
|
{" F211", WARN_SYSTEMCRC2},
|
||||||
|
|
||||||
//frikqcc errors
|
//frikqcc errors
|
||||||
//Q608: PrecacheSound: numsounds
|
//Q608: PrecacheSound: numsounds
|
||||||
|
@ -2321,7 +2322,7 @@ strofs = (strofs+3)&~3;
|
||||||
size = (size+15)&(~15); //and will allocate it on the hunk with 16-byte alignment
|
size = (size+15)&(~15); //and will allocate it on the hunk with 16-byte alignment
|
||||||
|
|
||||||
//this global receives the offset from world to the start of the progs def _IN VANILLA QUAKE_.
|
//this global receives the offset from world to the start of the progs def _IN VANILLA QUAKE_.
|
||||||
//this is a negative index due to allocation ordering
|
//this is a negative index due to allocation ordering with the assumption that the progs.dat was loaded on the heap directly followed by the entities.
|
||||||
//this will NOT work in FTE, DP, QuakeForge due to entity indexes. Various other engines will likely mess up too, if they change the allocation order or sizes etc. 64bit is screwed.
|
//this will NOT work in FTE, DP, QuakeForge due to entity indexes. Various other engines will likely mess up too, if they change the allocation order or sizes etc. 64bit is screwed.
|
||||||
if (progs.blockscompressed&32)
|
if (progs.blockscompressed&32)
|
||||||
printf("unable to write value for 'entity progs'\n"); //would not work anyway
|
printf("unable to write value for 'entity progs'\n"); //would not work anyway
|
||||||
|
@ -2332,7 +2333,7 @@ strofs = (strofs+3)&~3;
|
||||||
if (def->initialized)
|
if (def->initialized)
|
||||||
i = PRLittleLong(qcc_pr_globals[def->ofs]._int);
|
i = PRLittleLong(qcc_pr_globals[def->ofs]._int);
|
||||||
else
|
else
|
||||||
{
|
{ //entsize(=96)+hunk header size(=32)
|
||||||
if (verbose)
|
if (verbose)
|
||||||
printf("qccx hack - 'entity progs' uninitialised. Assuming 112.\n");
|
printf("qccx hack - 'entity progs' uninitialised. Assuming 112.\n");
|
||||||
i = 112; //match qccx.
|
i = 112; //match qccx.
|
||||||
|
@ -3434,7 +3435,7 @@ unsigned short QCC_PR_WriteProgdefs (char *filename)
|
||||||
break;
|
break;
|
||||||
case 17105:
|
case 17105:
|
||||||
case 32199: //outdated ext_csqc
|
case 32199: //outdated ext_csqc
|
||||||
QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "Recognised progs as outdated CSQC module\n");
|
QCC_PR_Warning(WARN_SYSTEMCRC2, NULL, 0, "Recognised progs as outdated CSQC module\n");
|
||||||
break;
|
break;
|
||||||
case 52195: //this is what DP requires. don't print it as the warning that it is as that would royally piss off xonotic and their use of -Werror.
|
case 52195: //this is what DP requires. don't print it as the warning that it is as that would royally piss off xonotic and their use of -Werror.
|
||||||
printf("Recognised progs as DP-specific CSQC module\n");
|
printf("Recognised progs as DP-specific CSQC module\n");
|
||||||
|
|
Loading…
Reference in a new issue