summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author/dev/humancontroller <devhc@example.com>2015-03-23 20:16:11 -0500
committerTim Angus <tim@ngus.net>2015-06-07 12:03:36 +0100
commitc3456c9e5eca93fbdcce6494aa9759c481e6a1ea (patch)
tree51678310dc6ac475adf2e6246a369786a026f133
parenta6507173e482bf939fec0cb766c2fa4ae2df0ba5 (diff)
Fix return values in nested system calls from QVMs
When the engine is compiled with Clang it appears that the return value is being written to the WRONG address, either due to the vm_ variables being changed (unexpectedly) elsewhere, or as a result of bad assembly assumptions; having a stack variable pointing to where to write the return value seems to do the trick. This fixes the case where, for a trap_Register()-like call, weird numbers are being returned when, during the process, an error message is printed (which in Tremulous results in a QVM call and (nested) system call).
-rw-r--r--src/qcommon/vm_x86.c7
1 files changed, 4 insertions, 3 deletions
diff --git a/src/qcommon/vm_x86.c b/src/qcommon/vm_x86.c
index b6fc6a63..2056785d 100644
--- a/src/qcommon/vm_x86.c
+++ b/src/qcommon/vm_x86.c
@@ -413,23 +413,24 @@ static void DoSyscall(void)
if(vm_syscallNum < 0)
{
- int *data;
+ int *data, *ret;
#if idx64
int index;
intptr_t args[MAX_VMSYSCALL_ARGS];
#endif
data = (int *) (savedVM->dataBase + vm_programStack + 4);
+ ret = &vm_opStackBase[vm_opStackOfs + 1];
#if idx64
args[0] = ~vm_syscallNum;
for(index = 1; index < ARRAY_LEN(args); index++)
args[index] = data[index];
- vm_opStackBase[vm_opStackOfs + 1] = savedVM->systemCall(args);
+ *ret = savedVM->systemCall(args);
#else
data[0] = ~vm_syscallNum;
- vm_opStackBase[vm_opStackOfs + 1] = savedVM->systemCall((intptr_t *) data);
+ *ret = savedVM->systemCall((intptr_t *) data);
#endif
}
else