mirror of https://gitlab.com/nakst/essence
				
				
				
			scripting engine cleanup to make porting easier
This commit is contained in:
		
							parent
							
								
									dfb502f95a
								
							
						
					
					
						commit
						c751659fc7
					
				
							
								
								
									
										441
									
								
								util/script.c
								
								
								
								
							
							
						
						
									
										441
									
								
								util/script.c
								
								
								
								
							|  | @ -174,6 +174,17 @@ | ||||||
| 	STACK_READ_STRING(textVariable1, bytesVariable1, 1); \ | 	STACK_READ_STRING(textVariable1, bytesVariable1, 1); \ | ||||||
| 	STACK_READ_STRING(textVariable2, bytesVariable2, 2); \ | 	STACK_READ_STRING(textVariable2, bytesVariable2, 2); \ | ||||||
| 	context->c->stackPointer -= 2; | 	context->c->stackPointer -= 2; | ||||||
|  | #define RETURN_STRING_COPY(_text, _bytes) \ | ||||||
|  | 	returnValue->i = HeapAllocate(context); \ | ||||||
|  | 	context->heap[returnValue->i].type = T_STR; \ | ||||||
|  | 	context->heap[returnValue->i].bytes = _bytes; \ | ||||||
|  | 	context->heap[returnValue->i].text = (char *) AllocateResize(NULL, context->heap[returnValue->i].bytes); \ | ||||||
|  | 	MemoryCopy(context->heap[returnValue->i].text, _text, context->heap[returnValue->i].bytes); | ||||||
|  | #define RETURN_STRING_NO_COPY(_text, _bytes) \ | ||||||
|  | 	returnValue->i = HeapAllocate(context); \ | ||||||
|  | 	context->heap[returnValue->i].type = T_STR; \ | ||||||
|  | 	context->heap[returnValue->i].bytes = _bytes; \ | ||||||
|  | 	context->heap[returnValue->i].text = _text; | ||||||
| 
 | 
 | ||||||
| typedef struct Token { | typedef struct Token { | ||||||
| 	struct ImportData *module; | 	struct ImportData *module; | ||||||
|  | @ -361,8 +372,7 @@ Node globalExpressionTypeStr = { .type = T_STR }; | ||||||
| Node globalExpressionTypeIntList = { .type = T_LIST, .firstChild = &globalExpressionTypeInt }; | Node globalExpressionTypeIntList = { .type = T_LIST, .firstChild = &globalExpressionTypeInt }; | ||||||
| 
 | 
 | ||||||
| // Global variables:
 | // Global variables:
 | ||||||
| char *scriptSourceDirectory; | const char *startFunction = "Start"; | ||||||
| char *startFunction = "Start"; |  | ||||||
| size_t startFunctionBytes = 5; | size_t startFunctionBytes = 5; | ||||||
| char **options; | char **options; | ||||||
| bool *optionsMatched; | bool *optionsMatched; | ||||||
|  | @ -380,8 +390,10 @@ uintptr_t HeapAllocate(ExecutionContext *context); | ||||||
| 
 | 
 | ||||||
| // --------------------------------- Platform layer definitions.
 | // --------------------------------- Platform layer definitions.
 | ||||||
| 
 | 
 | ||||||
|  | #if defined(_WIN32) || defined(__linux__) || defined(__APPLE__) | ||||||
| #include <assert.h> | #include <assert.h> | ||||||
| #define Assert assert | #define Assert assert | ||||||
|  | #endif | ||||||
| 
 | 
 | ||||||
| void *AllocateFixed(size_t bytes); | void *AllocateFixed(size_t bytes); | ||||||
| void *AllocateResize(void *old, size_t bytes); | void *AllocateResize(void *old, size_t bytes); | ||||||
|  | @ -409,7 +421,6 @@ char baseModuleSource[] = { | ||||||
| 
 | 
 | ||||||
| 	// String operations:
 | 	// String operations:
 | ||||||
| 
 | 
 | ||||||
| 	"str StringTrim(str x) #extcall;" |  | ||||||
| 	"str StringSlice(str x, int start, int end) #extcall;" | 	"str StringSlice(str x, int start, int end) #extcall;" | ||||||
| 	"int CharacterToByte(str x) #extcall;" | 	"int CharacterToByte(str x) #extcall;" | ||||||
| 	"bool StringContains(str haystack, str needle) {" | 	"bool StringContains(str haystack, str needle) {" | ||||||
|  | @ -420,6 +431,13 @@ char baseModuleSource[] = { | ||||||
| 	"	}" | 	"	}" | ||||||
| 	"	return false;" | 	"	return false;" | ||||||
| 	"}" | 	"}" | ||||||
|  | 	"str StringTrim(str string) {" | ||||||
|  | 	"	int start = 0;" | ||||||
|  | 	"	int end = string:len();" | ||||||
|  | 	"	while start != end && (string[start] == \" \" || string[start] == \"\\t\" || string[start] == \"\\r\" || string[start] == \"\\n\") { start += 1; }" | ||||||
|  | 	"	while start != end && (string[end - 1] == \" \" || string[end - 1] == \"\\t\" || string[end - 1] == \"\\r\" || string[end - 1] == \"\\n\") { end -= 1; }" | ||||||
|  | 	"	return StringSlice(string, start, end);" | ||||||
|  | 	"}" | ||||||
| 	"str[] StringSplitByCharacter(str string, str character, bool includeEmptyString) {" | 	"str[] StringSplitByCharacter(str string, str character, bool includeEmptyString) {" | ||||||
| 	"	str[] list = new str[];" | 	"	str[] list = new str[];" | ||||||
| 	"	int x = 0;" | 	"	int x = 0;" | ||||||
|  | @ -475,7 +493,6 @@ char baseModuleSource[] = { | ||||||
| 	"bool PathIsDirectory(str source) #extcall;" | 	"bool PathIsDirectory(str source) #extcall;" | ||||||
| 	"bool PathIsLink(str source) #extcall;" | 	"bool PathIsLink(str source) #extcall;" | ||||||
| 	"bool PathCreateDirectory(str x) #extcall;" // TODO Replace the return value with a enum.
 | 	"bool PathCreateDirectory(str x) #extcall;" // TODO Replace the return value with a enum.
 | ||||||
| 	"bool PathCreateLeadingDirectories(str x) #extcall;" |  | ||||||
| 	"bool PathDelete(str x) #extcall;" // TODO Replace the return value with a enum.
 | 	"bool PathDelete(str x) #extcall;" // TODO Replace the return value with a enum.
 | ||||||
| 	"bool PathMove(str source, str destination) #extcall;" | 	"bool PathMove(str source, str destination) #extcall;" | ||||||
| 	"str PathGetDefaultPrefix() #extcall;" | 	"str PathGetDefaultPrefix() #extcall;" | ||||||
|  | @ -624,6 +641,14 @@ char baseModuleSource[] = { | ||||||
| 	"	}" | 	"	}" | ||||||
| 	"	return \"\";" | 	"	return \"\";" | ||||||
| 	"}" | 	"}" | ||||||
|  | 	"bool PathCreateLeadingDirectories(str path) {" | ||||||
|  | 	"	for int i = 0; i < path:len(); i += 1 {" | ||||||
|  | 	"		if path[i] == \"/\" {" | ||||||
|  | 	"			PathCreateDirectory(StringSlice(path, 0, i));" | ||||||
|  | 	"		}" | ||||||
|  | 	"	}" | ||||||
|  | 	"	return PathCreateDirectory(path);" | ||||||
|  | 	"}" | ||||||
| 
 | 
 | ||||||
| 	// Persistent variables:
 | 	// Persistent variables:
 | ||||||
| 
 | 
 | ||||||
|  | @ -646,7 +671,6 @@ int ExternalPrintStdErr(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPrintStdErrWarning(ExecutionContext *context, Value *returnValue); | int ExternalPrintStdErrWarning(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPrintStdErrHighlight(ExecutionContext *context, Value *returnValue); | int ExternalPrintStdErrHighlight(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalConsoleGetLine(ExecutionContext *context, Value *returnValue); | int ExternalConsoleGetLine(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalStringTrim(ExecutionContext *context, Value *returnValue); |  | ||||||
| int ExternalStringSlice(ExecutionContext *context, Value *returnValue); | int ExternalStringSlice(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalCharacterToByte(ExecutionContext *context, Value *returnValue); | int ExternalCharacterToByte(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalSystemShellExecute(ExecutionContext *context, Value *returnValue); | int ExternalSystemShellExecute(ExecutionContext *context, Value *returnValue); | ||||||
|  | @ -659,7 +683,6 @@ int ExternalSystemSetEnvironmentVariable(ExecutionContext *context, Value *retur | ||||||
| int ExternalSystemRunningAsAdministrator(ExecutionContext *context, Value *returnValue); | int ExternalSystemRunningAsAdministrator(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalSystemGetHostName(ExecutionContext *context, Value *returnValue); | int ExternalSystemGetHostName(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPathCreateDirectory(ExecutionContext *context, Value *returnValue); | int ExternalPathCreateDirectory(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPathCreateLeadingDirectories(ExecutionContext *context, Value *returnValue); |  | ||||||
| int ExternalPathDelete(ExecutionContext *context, Value *returnValue); | int ExternalPathDelete(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPathExists(ExecutionContext *context, Value *returnValue); | int ExternalPathExists(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPathIsFile(ExecutionContext *context, Value *returnValue); | int ExternalPathIsFile(ExecutionContext *context, Value *returnValue); | ||||||
|  | @ -674,7 +697,7 @@ int ExternalFileCopy(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalFileGetSize(ExecutionContext *context, Value *returnValue); | int ExternalFileGetSize(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPersistRead(ExecutionContext *context, Value *returnValue); | int ExternalPersistRead(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalPersistWrite(ExecutionContext *context, Value *returnValue); | int ExternalPersistWrite(ExecutionContext *context, Value *returnValue); | ||||||
| int ExternalRandomInt(ExecutionContext *context, Value *returnValue); | int ExternalRandomInt(ExecutionContext *context, Value *returnValue); // TODO This shouldn't be in the platform layer.
 | ||||||
| int External_DirectoryInternalStartIteration(ExecutionContext *context, Value *returnValue); | int External_DirectoryInternalStartIteration(ExecutionContext *context, Value *returnValue); | ||||||
| int External_DirectoryInternalNextIteration(ExecutionContext *context, Value *returnValue); | int External_DirectoryInternalNextIteration(ExecutionContext *context, Value *returnValue); | ||||||
| int External_DirectoryInternalEndIteration(ExecutionContext *context, Value *returnValue); | int External_DirectoryInternalEndIteration(ExecutionContext *context, Value *returnValue); | ||||||
|  | @ -684,7 +707,6 @@ ExternalFunction externalFunctions[] = { | ||||||
| 	{ .cName = "PrintStdErrWarning", .callback = ExternalPrintStdErrWarning }, | 	{ .cName = "PrintStdErrWarning", .callback = ExternalPrintStdErrWarning }, | ||||||
| 	{ .cName = "PrintStdErrHighlight", .callback = ExternalPrintStdErrHighlight }, | 	{ .cName = "PrintStdErrHighlight", .callback = ExternalPrintStdErrHighlight }, | ||||||
| 	{ .cName = "ConsoleGetLine", .callback = ExternalConsoleGetLine }, | 	{ .cName = "ConsoleGetLine", .callback = ExternalConsoleGetLine }, | ||||||
| 	{ .cName = "StringTrim", .callback = ExternalStringTrim }, |  | ||||||
| 	{ .cName = "StringSlice", .callback = ExternalStringSlice }, | 	{ .cName = "StringSlice", .callback = ExternalStringSlice }, | ||||||
| 	{ .cName = "CharacterToByte", .callback = ExternalCharacterToByte }, | 	{ .cName = "CharacterToByte", .callback = ExternalCharacterToByte }, | ||||||
| 	{ .cName = "SystemShellExecute", .callback = ExternalSystemShellExecute }, | 	{ .cName = "SystemShellExecute", .callback = ExternalSystemShellExecute }, | ||||||
|  | @ -701,7 +723,6 @@ ExternalFunction externalFunctions[] = { | ||||||
| 	{ .cName = "PathIsDirectory", .callback = ExternalPathIsDirectory }, | 	{ .cName = "PathIsDirectory", .callback = ExternalPathIsDirectory }, | ||||||
| 	{ .cName = "PathIsLink", .callback = ExternalPathIsLink }, | 	{ .cName = "PathIsLink", .callback = ExternalPathIsLink }, | ||||||
| 	{ .cName = "PathCreateDirectory", .callback = ExternalPathCreateDirectory }, | 	{ .cName = "PathCreateDirectory", .callback = ExternalPathCreateDirectory }, | ||||||
| 	{ .cName = "PathCreateLeadingDirectories", .callback = ExternalPathCreateLeadingDirectories }, |  | ||||||
| 	{ .cName = "PathDelete", .callback = ExternalPathDelete }, | 	{ .cName = "PathDelete", .callback = ExternalPathDelete }, | ||||||
| 	{ .cName = "PathMove", .callback = ExternalPathMove }, | 	{ .cName = "PathMove", .callback = ExternalPathMove }, | ||||||
| 	{ .cName = "PathGetDefaultPrefix", .callback = ExternalPathGetDefaultPrefix }, | 	{ .cName = "PathGetDefaultPrefix", .callback = ExternalPathGetDefaultPrefix }, | ||||||
|  | @ -904,10 +925,11 @@ Token TokenNext(Tokenizer *tokenizer) { | ||||||
| 				} else if (tokenizer->input[i] == '\\') { | 				} else if (tokenizer->input[i] == '\\') { | ||||||
| 					if (i + 1 == tokenizer->inputBytes  | 					if (i + 1 == tokenizer->inputBytes  | ||||||
| 							|| (tokenizer->input[i + 1] != 'n' && tokenizer->input[i + 1] != 't'  | 							|| (tokenizer->input[i + 1] != 'n' && tokenizer->input[i + 1] != 't'  | ||||||
|  | 								&& tokenizer->input[i + 1] != 'r'  | ||||||
| 								&& tokenizer->input[i + 1] != '%'  | 								&& tokenizer->input[i + 1] != '%'  | ||||||
| 								&& tokenizer->input[i + 1] != '"' && tokenizer->input[i + 1] != '\\')) { | 								&& tokenizer->input[i + 1] != '"' && tokenizer->input[i + 1] != '\\')) { | ||||||
| 						PrintError(tokenizer, "String contains unrecognized escape sequence '\\%c'. " | 						PrintError(tokenizer, "String contains unrecognized escape sequence '\\%c'. " | ||||||
| 								"Possibilities are: '\\\\', '\\%%', '\\n', '\\t' and '\\\"'\n", tokenizer->input[i + 1]); | 								"Possibilities are: '\\\\', '\\%%', '\\n', '\\r', '\\t' and '\\\"'\n", tokenizer->input[i + 1]); | ||||||
| 						tokenizer->error = true; | 						tokenizer->error = true; | ||||||
| 						break; | 						break; | ||||||
| 					} else { | 					} else { | ||||||
|  | @ -1063,7 +1085,7 @@ Node *ParseExpression(Tokenizer *tokenizer, bool allowAssignment, uint8_t preced | ||||||
| 		size_t rawBytes = string->token.textBytes; | 		size_t rawBytes = string->token.textBytes; | ||||||
| 
 | 
 | ||||||
| 		// It's impossible for size of the string to increase.
 | 		// It's impossible for size of the string to increase.
 | ||||||
| 		char *output = AllocateFixed(rawBytes); | 		char *output = (char *) AllocateFixed(rawBytes); | ||||||
| 		size_t outputPosition = 0; | 		size_t outputPosition = 0; | ||||||
| 
 | 
 | ||||||
| 		string->token.text = output; | 		string->token.text = output; | ||||||
|  | @ -1078,6 +1100,7 @@ Node *ParseExpression(Tokenizer *tokenizer, bool allowAssignment, uint8_t preced | ||||||
| 				Assert(outputPosition != rawBytes); | 				Assert(outputPosition != rawBytes); | ||||||
| 				if (c == '\\') c = '\\'; | 				if (c == '\\') c = '\\'; | ||||||
| 				else if (c == 'n') c = '\n'; | 				else if (c == 'n') c = '\n'; | ||||||
|  | 				else if (c == 'r') c = '\r'; | ||||||
| 				else if (c == 't') c = '\t'; | 				else if (c == 't') c = '\t'; | ||||||
| 				else if (c == '%') c = '%'; | 				else if (c == '%') c = '%'; | ||||||
| 				else if (c == '"') c = '"'; | 				else if (c == '"') c = '"'; | ||||||
|  | @ -1222,7 +1245,7 @@ Node *ParseExpression(Tokenizer *tokenizer, bool allowAssignment, uint8_t preced | ||||||
| 			operation->firstChild = node; | 			operation->firstChild = node; | ||||||
| 			node->sibling = ParseExpression(tokenizer, false, TokenLookupPrecedence(token.type)); | 			node->sibling = ParseExpression(tokenizer, false, TokenLookupPrecedence(token.type)); | ||||||
| 			if (!node->sibling) return NULL; | 			if (!node->sibling) return NULL; | ||||||
| 			Node *nodeCopy = AllocateFixed(sizeof(Node)); | 			Node *nodeCopy = (Node *) AllocateFixed(sizeof(Node)); | ||||||
| 			*nodeCopy = *node; | 			*nodeCopy = *node; | ||||||
| 			node = operation; | 			node = operation; | ||||||
| 			operation = (Node *) AllocateFixed(sizeof(Node)); | 			operation = (Node *) AllocateFixed(sizeof(Node)); | ||||||
|  | @ -1875,7 +1898,7 @@ bool ScopeAddEntry(Tokenizer *tokenizer, Scope *scope, Node *node) { | ||||||
| 
 | 
 | ||||||
| void ASTFreeScopes(Node *node) { | void ASTFreeScopes(Node *node) { | ||||||
| 	if (node && node->scope) { | 	if (node && node->scope) { | ||||||
| 		node->scope->entries = AllocateResize(node->scope->entries, 0); | 		node->scope->entries = (Node **) AllocateResize(node->scope->entries, 0); | ||||||
| 
 | 
 | ||||||
| 		Node *child = node->firstChild; | 		Node *child = node->firstChild; | ||||||
| 
 | 
 | ||||||
|  | @ -1984,7 +2007,7 @@ bool ASTSetScopes(Tokenizer *tokenizer, ExecutionContext *context, Node *node, S | ||||||
| 				return false; | 				return false; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			node->importData = AllocateFixed(sizeof(ImportData)); | 			node->importData = (ImportData *) AllocateFixed(sizeof(ImportData)); | ||||||
| 			node->importData->fileDataBytes = t.inputBytes; | 			node->importData->fileDataBytes = t.inputBytes; | ||||||
| 			node->importData->fileData = fileData; | 			node->importData->fileData = fileData; | ||||||
| 			node->importData->path = path; | 			node->importData->path = path; | ||||||
|  | @ -2011,7 +2034,7 @@ bool ASTSetScopes(Tokenizer *tokenizer, ExecutionContext *context, Node *node, S | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			t.module = node->importData; | 			t.module = node->importData; | ||||||
| 			t.input = fileData; | 			t.input = (const char *) fileData; | ||||||
| 			t.line = 1; | 			t.line = 1; | ||||||
| 
 | 
 | ||||||
| 			if (!ScriptLoad(t, context, node->importData)) { | 			if (!ScriptLoad(t, context, node->importData)) { | ||||||
|  | @ -3138,8 +3161,8 @@ bool ASTGenerate(Tokenizer *tokenizer, Node *root, ExecutionContext *context) { | ||||||
| 
 | 
 | ||||||
| 	context->functionData->globalVariableOffset = context->globalVariableCount; | 	context->functionData->globalVariableOffset = context->globalVariableCount; | ||||||
| 	context->globalVariableCount += root->scope->variableEntryCount; | 	context->globalVariableCount += root->scope->variableEntryCount; | ||||||
| 	context->globalVariables = AllocateResize(context->globalVariables, sizeof(Value) * context->globalVariableCount); | 	context->globalVariables = (Value *) AllocateResize(context->globalVariables, sizeof(Value) * context->globalVariableCount); | ||||||
| 	context->globalVariableIsManaged = AllocateResize(context->globalVariableIsManaged, sizeof(Value) * context->globalVariableCount); | 	context->globalVariableIsManaged = (bool *) AllocateResize(context->globalVariableIsManaged, sizeof(Value) * context->globalVariableCount); | ||||||
| 
 | 
 | ||||||
| 	for (uintptr_t i = 0; i < root->scope->variableEntryCount; i++) { | 	for (uintptr_t i = 0; i < root->scope->variableEntryCount; i++) { | ||||||
| 		context->globalVariables[context->functionData->globalVariableOffset + i].i = 0; | 		context->globalVariables[context->functionData->globalVariableOffset + i].i = 0; | ||||||
|  | @ -3363,6 +3386,7 @@ size_t ScriptHeapEntryGetStringBytes(HeapEntry *entry) { | ||||||
| 		return entry->concatBytes; | 		return entry->concatBytes; | ||||||
| 	} else { | 	} else { | ||||||
| 		Assert(false); | 		Assert(false); | ||||||
|  | 		return 0; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -3407,7 +3431,7 @@ void ScriptHeapEntryConcatConvertToString(ExecutionContext *context, HeapEntry * | ||||||
| 	Assert(entry->concatBytes == part1Bytes + part2Bytes); | 	Assert(entry->concatBytes == part1Bytes + part2Bytes); | ||||||
| 	entry->type = T_STR; | 	entry->type = T_STR; | ||||||
| 	entry->bytes = part1Bytes + part2Bytes; | 	entry->bytes = part1Bytes + part2Bytes; | ||||||
| 	entry->text = AllocateResize(NULL, entry->bytes); | 	entry->text = (char *) AllocateResize(NULL, entry->bytes); | ||||||
| 	ScriptHeapEntryConcatConvertToStringWrite(context, part1, entry->text); | 	ScriptHeapEntryConcatConvertToStringWrite(context, part1, entry->text); | ||||||
| 	ScriptHeapEntryConcatConvertToStringWrite(context, part2, entry->text + part1Bytes); | 	ScriptHeapEntryConcatConvertToStringWrite(context, part2, entry->text + part1Bytes); | ||||||
| } | } | ||||||
|  | @ -3447,8 +3471,8 @@ int ScriptExecuteFunction(uintptr_t instructionPointer, ExecutionContext *contex | ||||||
| 			if (context->c->localVariableCount + newVariableCount > context->c->localVariablesAllocated) { | 			if (context->c->localVariableCount + newVariableCount > context->c->localVariablesAllocated) { | ||||||
| 				// TODO Handling memory errors here.
 | 				// TODO Handling memory errors here.
 | ||||||
| 				context->c->localVariablesAllocated = context->c->localVariableCount + newVariableCount; | 				context->c->localVariablesAllocated = context->c->localVariableCount + newVariableCount; | ||||||
| 				context->c->localVariables = AllocateResize(context->c->localVariables, context->c->localVariablesAllocated * sizeof(Value));  | 				context->c->localVariables = (Value *) AllocateResize(context->c->localVariables, context->c->localVariablesAllocated * sizeof(Value));  | ||||||
| 				context->c->localVariableIsManaged = AllocateResize(context->c->localVariableIsManaged, context->c->localVariablesAllocated * sizeof(bool));  | 				context->c->localVariableIsManaged = (bool *) AllocateResize(context->c->localVariableIsManaged, context->c->localVariablesAllocated * sizeof(bool));  | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			MemoryCopy(context->c->localVariableIsManaged + context->c->localVariableCount, functionData + instructionPointer, newVariableCount); | 			MemoryCopy(context->c->localVariableIsManaged + context->c->localVariableCount, functionData + instructionPointer, newVariableCount); | ||||||
|  | @ -4083,7 +4107,7 @@ int ScriptExecuteFunction(uintptr_t instructionPointer, ExecutionContext *contex | ||||||
| 			context->heap[index].allocated = newLength; | 			context->heap[index].allocated = newLength; | ||||||
| 
 | 
 | ||||||
| 			// TODO Handling out of memory errors.
 | 			// TODO Handling out of memory errors.
 | ||||||
| 			context->heap[index].list = AllocateResize(context->heap[index].list, newLength * sizeof(Value)); | 			context->heap[index].list = (Value *) AllocateResize(context->heap[index].list, newLength * sizeof(Value)); | ||||||
| 
 | 
 | ||||||
| 			for (uintptr_t i = oldLength; i < (size_t) newLength; i++) { | 			for (uintptr_t i = oldLength; i < (size_t) newLength; i++) { | ||||||
| 				context->heap[index].list[i].i = 0; | 				context->heap[index].list[i].i = 0; | ||||||
|  | @ -4118,7 +4142,7 @@ int ScriptExecuteFunction(uintptr_t instructionPointer, ExecutionContext *contex | ||||||
| 			if (entry->length > entry->allocated) { | 			if (entry->length > entry->allocated) { | ||||||
| 				// TODO Handling out of memory errors.
 | 				// TODO Handling out of memory errors.
 | ||||||
| 				entry->allocated = entry->allocated ? entry->allocated * 2 : 4; | 				entry->allocated = entry->allocated ? entry->allocated * 2 : 4; | ||||||
| 				entry->list = AllocateResize(entry->list, entry->allocated * sizeof(Value)); | 				entry->list = (Value *) AllocateResize(entry->list, entry->allocated * sizeof(Value)); | ||||||
| 				Assert(entry->length <= entry->allocated); | 				Assert(entry->length <= entry->allocated); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  | @ -4154,7 +4178,7 @@ int ScriptExecuteFunction(uintptr_t instructionPointer, ExecutionContext *contex | ||||||
| 			if (entry->length > entry->allocated) { | 			if (entry->length > entry->allocated) { | ||||||
| 				// TODO Handling out of memory errors.
 | 				// TODO Handling out of memory errors.
 | ||||||
| 				entry->allocated = entry->allocated ? entry->allocated * 2 : 4; | 				entry->allocated = entry->allocated ? entry->allocated * 2 : 4; | ||||||
| 				entry->list = AllocateResize(entry->list, entry->allocated * sizeof(Value)); | 				entry->list = (Value *) AllocateResize(entry->list, entry->allocated * sizeof(Value)); | ||||||
| 				Assert(entry->length <= entry->allocated); | 				Assert(entry->length <= entry->allocated); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  | @ -4191,7 +4215,7 @@ int ScriptExecuteFunction(uintptr_t instructionPointer, ExecutionContext *contex | ||||||
| 			if (entry->type != T_LIST) return -1; | 			if (entry->type != T_LIST) return -1; | ||||||
| 
 | 
 | ||||||
| 			context->heap[index].length = context->heap[index].allocated = 0; | 			context->heap[index].length = context->heap[index].allocated = 0; | ||||||
| 			context->heap[index].list = AllocateResize(context->heap[index].list, 0); | 			context->heap[index].list = (Value *) AllocateResize(context->heap[index].list, 0); | ||||||
| 			context->c->stackPointer--; | 			context->c->stackPointer--; | ||||||
| 		} else if (command == T_OP_FIND_AND_DELETE) { | 		} else if (command == T_OP_FIND_AND_DELETE) { | ||||||
| 			if (context->c->stackPointer < 2) return -1; | 			if (context->c->stackPointer < 2) return -1; | ||||||
|  | @ -4545,7 +4569,7 @@ bool ScriptParseOptions(ExecutionContext *context) { | ||||||
| 			uintptr_t heapIndex = HeapAllocate(context); | 			uintptr_t heapIndex = HeapAllocate(context); | ||||||
| 			context->heap[heapIndex].type = T_STR; | 			context->heap[heapIndex].type = T_STR; | ||||||
| 			context->heap[heapIndex].bytes = optionLength - equalsPosition - 1; | 			context->heap[heapIndex].bytes = optionLength - equalsPosition - 1; | ||||||
| 			context->heap[heapIndex].text = AllocateResize(NULL, context->heap[heapIndex].bytes); | 			context->heap[heapIndex].text = (char *) AllocateResize(NULL, context->heap[heapIndex].bytes); | ||||||
| 			context->globalVariables[index].i = heapIndex; | 			context->globalVariables[index].i = heapIndex; | ||||||
| 			MemoryCopy(context->heap[heapIndex].text, options[i] + equalsPosition + 1, context->heap[heapIndex].bytes); | 			MemoryCopy(context->heap[heapIndex].text, options[i] + equalsPosition + 1, context->heap[heapIndex].bytes); | ||||||
| 		} else if (node->expressionType->type == T_INT) { | 		} else if (node->expressionType->type == T_INT) { | ||||||
|  | @ -4746,8 +4770,137 @@ void ScriptFree(ExecutionContext *context) { | ||||||
| 	AllocateResize(context->scriptPersistFile, 0); | 	AllocateResize(context->scriptPersistFile, 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // --------------------------------- Helpers.
 | ||||||
|  | 
 | ||||||
|  | void LineNumberLookup(ExecutionContext *context, uint32_t instructionPointer, LineNumber *output) { | ||||||
|  | 	for (uintptr_t i = 0; i < context->functionData->lineNumberCount; i++) { | ||||||
|  | 		if (context->functionData->lineNumbers[i].instructionPointer == instructionPointer) { | ||||||
|  | 			*output = context->functionData->lineNumbers[i]; | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void PrintBackTrace(ExecutionContext *context, uint32_t instructionPointer, CoroutineState *c, const char *prefix) { | ||||||
|  | 	LineNumber lineNumber = { 0 }; | ||||||
|  | 	LineNumberLookup(context, instructionPointer, &lineNumber); | ||||||
|  | 
 | ||||||
|  | 	if (lineNumber.importData) { | ||||||
|  | 		PrintDebug("%s\t%s:%d %s %.*s\n", prefix, lineNumber.importData->path, lineNumber.lineNumber, lineNumber.function ? "in" : "", | ||||||
|  | 				lineNumber.function ? (int) lineNumber.function->textBytes : 0, lineNumber.function ? lineNumber.function->text : ""); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	uintptr_t btp = c->backTracePointer; | ||||||
|  | 	uintptr_t minimum = c->startedByAsync ? 1 : 0; | ||||||
|  | 
 | ||||||
|  | 	while (btp > minimum) { | ||||||
|  | 		BackTraceItem *link = &c->backTrace[--btp]; | ||||||
|  | 		LineNumberLookup(context, link->instructionPointer - 1, &lineNumber); | ||||||
|  | 		PrintDebug("%s\t%s:%d %s %.*s\n", prefix, lineNumber.importData->path ? lineNumber.importData->path : "??",  | ||||||
|  | 				lineNumber.lineNumber, lineNumber.function ? "in" : "", | ||||||
|  | 				lineNumber.function ? (int) lineNumber.function->textBytes : 0, lineNumber.function ? lineNumber.function->text : ""); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void PrintLine(ImportData *importData, uintptr_t line) { | ||||||
|  | 	if (!importData) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	uintptr_t position = 0; | ||||||
|  | 
 | ||||||
|  | 	for (uintptr_t i = 1; i < line; i++) { | ||||||
|  | 		while (position < importData->fileDataBytes) { | ||||||
|  | 			if (((char *) importData->fileData)[position] == '\n') { | ||||||
|  | 				position++; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			position++; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	uintptr_t length = 0; | ||||||
|  | 
 | ||||||
|  | 	for (uintptr_t i = position; i < importData->fileDataBytes; i++) { | ||||||
|  | 		if (((char *) importData->fileData)[i] == '\n') { | ||||||
|  | 			length = i - position; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	PrintDebug(">> %.*s\n", (int) length, &((char *) importData->fileData)[position]); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int ScriptExecuteFromPath(char *scriptPath, size_t scriptPathBytes) { | ||||||
|  | 	Tokenizer tokenizer = { 0 }; | ||||||
|  | 	ImportData importData = { 0 }; | ||||||
|  | 	importData.path = scriptPath; | ||||||
|  | 	importData.pathBytes = scriptPathBytes; | ||||||
|  | 	importData.fileData = FileLoad(scriptPath, &tokenizer.inputBytes); | ||||||
|  | 	importData.fileDataBytes = tokenizer.inputBytes; | ||||||
|  | 	tokenizer.module = &importData; | ||||||
|  | 	tokenizer.line = 1; | ||||||
|  | 	tokenizer.input = (const char *) importData.fileData; | ||||||
|  | 
 | ||||||
|  | 	if (!tokenizer.input) { | ||||||
|  | 		PrintDebug("Error: Could not load the input file '%s'.\n", scriptPath); | ||||||
|  | 		return 1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	FunctionBuilder builder = { 0 }; | ||||||
|  | 	ExecutionContext context = { 0 }; | ||||||
|  | 	context.functionData = &builder; | ||||||
|  | 	context.mainModule = &importData; | ||||||
|  | 
 | ||||||
|  | 	context.heapEntriesAllocated = 2; | ||||||
|  | 	context.heap = (HeapEntry *) AllocateResize(NULL, sizeof(HeapEntry) * context.heapEntriesAllocated); | ||||||
|  | 	context.heap[0].type = T_EOF; | ||||||
|  | 	context.heap[1].type = T_ERROR; | ||||||
|  | 	context.heap[1].nextUnusedEntry = 0; | ||||||
|  | 	context.heapFirstUnusedEntry = 1; | ||||||
|  | 	context.c = (CoroutineState *) AllocateResize(0, sizeof(CoroutineState)); | ||||||
|  | 	CoroutineState empty = { 0 }; | ||||||
|  | 	*context.c = empty; | ||||||
|  | 	context.c->stackEntriesAllocated = sizeof(context.c->stack) / sizeof(context.c->stack[0]); | ||||||
|  | 	context.c->previousCoroutineLink = &context.allCoroutines; | ||||||
|  | 	context.allCoroutines = context.c; | ||||||
|  | 
 | ||||||
|  | 	int result = ScriptLoad(tokenizer, &context, &importData) ? ScriptExecute(&context, &importData) : 1; | ||||||
|  | 	ScriptFree(&context); | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int ExternalStringSlice(ExecutionContext *context, Value *returnValue) { | ||||||
|  | 	(void) returnValue; | ||||||
|  | 	if (context->c->stackPointer < 3) return -1; | ||||||
|  | 	STACK_POP_STRING(string, bytes); | ||||||
|  | 	uint64_t start = context->c->stack[--context->c->stackPointer].i; | ||||||
|  | 	if (context->c->stackIsManaged[context->c->stackPointer]) return -1; | ||||||
|  | 	uint64_t end = context->c->stack[--context->c->stackPointer].i; | ||||||
|  | 	if (context->c->stackIsManaged[context->c->stackPointer]) return -1; | ||||||
|  | 
 | ||||||
|  | 	if (start > bytes || end > bytes || end < start) { | ||||||
|  | 		PrintError4(context, 0, "The slice range (%ld..%ld) is invalid for the string of length %ld.\n", | ||||||
|  | 				start, end, bytes); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	RETURN_STRING_COPY(string + start, end - start); | ||||||
|  | 	return 3; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int ExternalCharacterToByte(ExecutionContext *context, Value *returnValue) { | ||||||
|  | 	(void) returnValue; | ||||||
|  | 	STACK_POP_STRING(entryText, entryBytes); | ||||||
|  | 	returnValue->i = entryBytes ? entryText[0] : -1; | ||||||
|  | 	return 2; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // --------------------------------- Platform layer.
 | // --------------------------------- Platform layer.
 | ||||||
| 
 | 
 | ||||||
|  | #if defined(_WIN32) || defined(__linux__) || defined(__APPLE__) | ||||||
|  | 
 | ||||||
| #ifdef _WIN32 | #ifdef _WIN32 | ||||||
| #include <direct.h> | #include <direct.h> | ||||||
| #include <windows.h> | #include <windows.h> | ||||||
|  | @ -4781,6 +4934,8 @@ CoroutineState *externalCoroutineUnblockedList; | ||||||
| 
 | 
 | ||||||
| bool systemShellLoggingEnabled = true; | bool systemShellLoggingEnabled = true; | ||||||
| 
 | 
 | ||||||
|  | char *scriptSourceDirectory; | ||||||
|  | 
 | ||||||
| DIR *directoryIterator; | DIR *directoryIterator; | ||||||
| 
 | 
 | ||||||
| char *StringZeroTerminate(const char *text, size_t bytes) { | char *StringZeroTerminate(const char *text, size_t bytes) { | ||||||
|  | @ -4791,77 +4946,6 @@ char *StringZeroTerminate(const char *text, size_t bytes) { | ||||||
| 	return buffer; | 	return buffer; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int ExternalStringTrim(ExecutionContext *context, Value *returnValue) { |  | ||||||
| 	(void) returnValue; |  | ||||||
| 	STACK_POP_STRING(entryText, entryBytes); |  | ||||||
| 	if (entryBytes == 0) { returnValue->i = 0; return 3; } |  | ||||||
| 
 |  | ||||||
| 	uintptr_t start = 0, end = entryBytes; |  | ||||||
| 
 |  | ||||||
| 	while (start != end) { |  | ||||||
| 		if (entryText[start] == ' ' || entryText[start] == '\t' || entryText[start] == '\r' || entryText[start] == '\n') { |  | ||||||
| 			start++; |  | ||||||
| 		} else { |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	while (start != end) { |  | ||||||
| 		if (entryText[end - 1] == ' ' || entryText[end - 1] == '\t' || entryText[end - 1] == '\r' || entryText[end - 1] == '\n') { |  | ||||||
| 			end--; |  | ||||||
| 		} else { |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	char *buffer = AllocateResize(NULL, end - start); |  | ||||||
| 	MemoryCopy(buffer, entryText + start, end - start); |  | ||||||
| 
 |  | ||||||
| 	// TODO Handling allocation failures.
 |  | ||||||
| 	uintptr_t index = HeapAllocate(context); |  | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = end - start; |  | ||||||
| 	context->heap[index].text = buffer; |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 
 |  | ||||||
| 	return 3; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int ExternalStringSlice(ExecutionContext *context, Value *returnValue) { |  | ||||||
| 	(void) returnValue; |  | ||||||
| 	if (context->c->stackPointer < 3) return -1; |  | ||||||
| 	STACK_POP_STRING(string, bytes); |  | ||||||
| 	uint64_t start = context->c->stack[--context->c->stackPointer].i; |  | ||||||
| 	if (context->c->stackIsManaged[context->c->stackPointer]) return -1; |  | ||||||
| 	uint64_t end = context->c->stack[--context->c->stackPointer].i; |  | ||||||
| 	if (context->c->stackIsManaged[context->c->stackPointer]) return -1; |  | ||||||
| 
 |  | ||||||
| 	if (start > bytes || end > bytes || end < start) { |  | ||||||
| 		PrintError4(context, 0, "The slice range (%ld..%ld) is invalid for the string of length %ld.\n", |  | ||||||
| 				start, end, bytes); |  | ||||||
| 		return 0; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	char *buffer = AllocateResize(NULL, end - start); |  | ||||||
| 	MemoryCopy(buffer, string + start, end - start); |  | ||||||
| 
 |  | ||||||
| 	// TODO Handling allocation failures.
 |  | ||||||
| 	uintptr_t index = HeapAllocate(context); |  | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = end - start; |  | ||||||
| 	context->heap[index].text = buffer; |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 
 |  | ||||||
| 	return 3; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int ExternalCharacterToByte(ExecutionContext *context, Value *returnValue) { |  | ||||||
| 	(void) returnValue; |  | ||||||
| 	STACK_POP_STRING(entryText, entryBytes); |  | ||||||
| 	returnValue->i = entryBytes ? entryText[0] : -1; |  | ||||||
| 	return 2; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void ExternalCoroutineDone(CoroutineState *coroutine) { | void ExternalCoroutineDone(CoroutineState *coroutine) { | ||||||
| #ifdef __linux__ | #ifdef __linux__ | ||||||
| 	pthread_mutex_lock(&externalCoroutineMutex); | 	pthread_mutex_lock(&externalCoroutineMutex); | ||||||
|  | @ -5016,11 +5100,7 @@ int ExternalSystemShellEvaluate(ExecutionContext *context, Value *returnValue) { | ||||||
| 			buffer = (char *) realloc(buffer, position); // Shrink to match the size exactly.
 | 			buffer = (char *) realloc(buffer, position); // Shrink to match the size exactly.
 | ||||||
| 			pclose(f); | 			pclose(f); | ||||||
| 
 | 
 | ||||||
| 			uintptr_t index = HeapAllocate(context); | 			RETURN_STRING_NO_COPY(buffer, position); | ||||||
| 			context->heap[index].type = T_STR; |  | ||||||
| 			context->heap[index].bytes = position; |  | ||||||
| 			context->heap[index].text = buffer; |  | ||||||
| 			returnValue->i = index; |  | ||||||
| 		} else { | 		} else { | ||||||
| 			returnValue->i = 0; | 			returnValue->i = 0; | ||||||
| 		} | 		} | ||||||
|  | @ -5088,32 +5168,6 @@ int ExternalPathCreateDirectory(ExecutionContext *context, Value *returnValue) { | ||||||
| 	return 2; | 	return 2; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int ExternalPathCreateLeadingDirectories(ExecutionContext *context, Value *returnValue) { |  | ||||||
| 	(void) returnValue; |  | ||||||
| 	STACK_POP_STRING(entryText, entryBytes); |  | ||||||
| 	returnValue->i = 0; |  | ||||||
| 	if (entryBytes == 0) return 2; |  | ||||||
| 	char *temporary = StringZeroTerminate(entryText, entryBytes); |  | ||||||
| 	if (!temporary) return 2; |  | ||||||
| #ifdef _WIN32 |  | ||||||
| #pragma message ("ExternalPathCreateLeadingDirectories unimplemented") |  | ||||||
| #else |  | ||||||
| 	returnValue->i = 1; |  | ||||||
| 
 |  | ||||||
| 	for (uintptr_t i = 1; i < entryBytes; i++) { |  | ||||||
| 		if (temporary[i] == '/') { |  | ||||||
| 			temporary[i] = 0; |  | ||||||
| 			mkdir(temporary, S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH); |  | ||||||
| 			temporary[i] = '/'; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (mkdir(temporary, S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH)) returnValue->i = errno == EEXIST; |  | ||||||
| #endif |  | ||||||
| 	free(temporary); |  | ||||||
| 	return 2; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int ExternalPathDelete(ExecutionContext *context, Value *returnValue) { | int ExternalPathDelete(ExecutionContext *context, Value *returnValue) { | ||||||
| 	(void) returnValue; | 	(void) returnValue; | ||||||
| 	STACK_POP_STRING(entryText, entryBytes); | 	STACK_POP_STRING(entryText, entryBytes); | ||||||
|  | @ -5235,15 +5289,7 @@ int ExternalSystemGetEnvironmentVariable(ExecutionContext *context, Value *retur | ||||||
| 	char *temporary = StringZeroTerminate(entryText, entryBytes); | 	char *temporary = StringZeroTerminate(entryText, entryBytes); | ||||||
| 	if (!temporary) return 3; | 	if (!temporary) return 3; | ||||||
| 	char *data = getenv(temporary); | 	char *data = getenv(temporary); | ||||||
| 	size_t length = data ? strlen(data) : 0; | 	RETURN_STRING_COPY(data, data ? strlen(data) : 0); | ||||||
| 	char *copy = (char *) malloc(length + 1); |  | ||||||
| 	if (length) strcpy(copy, data); |  | ||||||
| 	else *copy = 0; |  | ||||||
| 	uintptr_t index = HeapAllocate(context); |  | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = length; |  | ||||||
| 	context->heap[index].text = copy; |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 	free(temporary); | 	free(temporary); | ||||||
| 	return 3; | 	return 3; | ||||||
| } | } | ||||||
|  | @ -5290,12 +5336,7 @@ int External_DirectoryInternalNextIteration(ExecutionContext *context, Value *re | ||||||
| 	struct dirent *entry = readdir(directoryIterator); | 	struct dirent *entry = readdir(directoryIterator); | ||||||
| 	while (entry && (0 == strcmp(entry->d_name, ".") || 0 == strcmp(entry->d_name, ".."))) entry = readdir(directoryIterator); | 	while (entry && (0 == strcmp(entry->d_name, ".") || 0 == strcmp(entry->d_name, ".."))) entry = readdir(directoryIterator); | ||||||
| 	if (!entry) { returnValue->i = 0; return 3; } | 	if (!entry) { returnValue->i = 0; return 3; } | ||||||
| 	uintptr_t index = HeapAllocate(context); | 	RETURN_STRING_COPY(entry->d_name, strlen(entry->d_name)); | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = strlen(entry->d_name); |  | ||||||
| 	context->heap[index].text = malloc(context->heap[index].bytes + 1); |  | ||||||
| 	strcpy(context->heap[index].text, entry->d_name); |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 	return 3; | 	return 3; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -5307,11 +5348,7 @@ int ExternalFileReadAll(ExecutionContext *context, Value *returnValue) { | ||||||
| 	if (!temporary) return 3; | 	if (!temporary) return 3; | ||||||
| 	size_t length = 0; | 	size_t length = 0; | ||||||
| 	void *data = FileLoad(temporary, &length); | 	void *data = FileLoad(temporary, &length); | ||||||
| 	uintptr_t index = HeapAllocate(context); | 	RETURN_STRING_NO_COPY(data, length); | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = length; |  | ||||||
| 	context->heap[index].text = data; |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 	free(temporary); | 	free(temporary); | ||||||
| 	return 3; | 	return 3; | ||||||
| } | } | ||||||
|  | @ -5360,11 +5397,7 @@ int ExternalPathGetDefaultPrefix(ExecutionContext *context, Value *returnValue) | ||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	uint64_t index = HeapAllocate(context); | 	RETURN_STRING_NO_COPY(realloc(data, strlen(data) + 1), strlen(data)); | ||||||
| 	context->heap[index].type = T_STR; |  | ||||||
| 	context->heap[index].bytes = strlen(data); |  | ||||||
| 	context->heap[index].text = realloc(data, strlen(data) + 1); |  | ||||||
| 	returnValue->i = index; |  | ||||||
| 	return 3; | 	return 3; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -5558,11 +5591,7 @@ int ExternalSystemGetHostName(ExecutionContext *context, Value *returnValue) { | ||||||
| 	uname(&buffer); | 	uname(&buffer); | ||||||
| 	name = buffer.sysname; | 	name = buffer.sysname; | ||||||
| #endif | #endif | ||||||
| 	returnValue->i = HeapAllocate(context); | 	RETURN_STRING_COPY(name, strlen(name)); | ||||||
| 	context->heap[returnValue->i].type = T_STR; |  | ||||||
| 	context->heap[returnValue->i].bytes = strlen(name); |  | ||||||
| 	context->heap[returnValue->i].text = AllocateResize(NULL, context->heap[returnValue->i].bytes); |  | ||||||
| 	memcpy(context->heap[returnValue->i].text, name, context->heap[returnValue->i].bytes); |  | ||||||
| 	return 3; | 	return 3; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -5661,36 +5690,6 @@ void PrintDebug(const char *format, ...) { | ||||||
| 	va_end(arguments); | 	va_end(arguments); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void PrintLine(ImportData *importData, uintptr_t line) { |  | ||||||
| 	if (!importData) { |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	uintptr_t position = 0; |  | ||||||
| 
 |  | ||||||
| 	for (uintptr_t i = 1; i < line; i++) { |  | ||||||
| 		while (position < importData->fileDataBytes) { |  | ||||||
| 			if (((char *) importData->fileData)[position] == '\n') { |  | ||||||
| 				position++; |  | ||||||
| 				break; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			position++; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	uintptr_t length = 0; |  | ||||||
| 
 |  | ||||||
| 	for (uintptr_t i = position; i < importData->fileDataBytes; i++) { |  | ||||||
| 		if (((char *) importData->fileData)[i] == '\n') { |  | ||||||
| 			length = i - position; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fprintf(stderr, ">> %.*s\n", (int) length, &((char *) importData->fileData)[position]); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void PrintError(Tokenizer *tokenizer, const char *format, ...) { | void PrintError(Tokenizer *tokenizer, const char *format, ...) { | ||||||
| 	fprintf(stderr, "\033[0;33mError on line %d of '%s':\033[0m\n", (int) tokenizer->line, tokenizer->module->path); | 	fprintf(stderr, "\033[0;33mError on line %d of '%s':\033[0m\n", (int) tokenizer->line, tokenizer->module->path); | ||||||
| 	va_list arguments; | 	va_list arguments; | ||||||
|  | @ -5717,36 +5716,6 @@ void PrintError3(const char *format, ...) { | ||||||
| 	va_end(arguments); | 	va_end(arguments); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void LineNumberLookup(ExecutionContext *context, uint32_t instructionPointer, LineNumber *output) { |  | ||||||
| 	for (uintptr_t i = 0; i < context->functionData->lineNumberCount; i++) { |  | ||||||
| 		if (context->functionData->lineNumbers[i].instructionPointer == instructionPointer) { |  | ||||||
| 			*output = context->functionData->lineNumbers[i]; |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void PrintBackTrace(ExecutionContext *context, uint32_t instructionPointer, CoroutineState *c, const char *prefix) { |  | ||||||
| 	LineNumber lineNumber = { 0 }; |  | ||||||
| 	LineNumberLookup(context, instructionPointer, &lineNumber); |  | ||||||
| 
 |  | ||||||
| 	if (lineNumber.importData) { |  | ||||||
| 		fprintf(stderr, "%s\t%s:%d %s %.*s\n", prefix, lineNumber.importData->path, lineNumber.lineNumber, lineNumber.function ? "in" : "", |  | ||||||
| 				lineNumber.function ? (int) lineNumber.function->textBytes : 0, lineNumber.function ? lineNumber.function->text : ""); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	uintptr_t btp = c->backTracePointer; |  | ||||||
| 	uintptr_t minimum = c->startedByAsync ? 1 : 0; |  | ||||||
| 
 |  | ||||||
| 	while (btp > minimum) { |  | ||||||
| 		BackTraceItem *link = &c->backTrace[--btp]; |  | ||||||
| 		LineNumberLookup(context, link->instructionPointer - 1, &lineNumber); |  | ||||||
| 		fprintf(stderr, "%s\t%s:%d %s %.*s\n", prefix, lineNumber.importData->path ? lineNumber.importData->path : "??",  |  | ||||||
| 				lineNumber.lineNumber, lineNumber.function ? "in" : "", |  | ||||||
| 				lineNumber.function ? (int) lineNumber.function->textBytes : 0, lineNumber.function ? lineNumber.function->text : ""); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void PrintError4(ExecutionContext *context, uint32_t instructionPointer, const char *format, ...) { | void PrintError4(ExecutionContext *context, uint32_t instructionPointer, const char *format, ...) { | ||||||
| 	LineNumber lineNumber = { 0 }; | 	LineNumber lineNumber = { 0 }; | ||||||
| 	LineNumberLookup(context, instructionPointer, &lineNumber); | 	LineNumberLookup(context, instructionPointer, &lineNumber); | ||||||
|  | @ -5807,41 +5776,7 @@ int main(int argc, char **argv) { | ||||||
| 	if (lastSlash) *lastSlash = 0; | 	if (lastSlash) *lastSlash = 0; | ||||||
| 	else strcpy(scriptSourceDirectory, "."); | 	else strcpy(scriptSourceDirectory, "."); | ||||||
| 
 | 
 | ||||||
| 	Tokenizer tokenizer = { 0 }; | 	int result = ScriptExecuteFromPath(scriptPath, strlen(scriptPath)); | ||||||
| 	ImportData importData = { 0 }; |  | ||||||
| 	importData.path = scriptPath; |  | ||||||
| 	importData.pathBytes = strlen(scriptPath); |  | ||||||
| 	importData.fileData = FileLoad(scriptPath, &tokenizer.inputBytes); |  | ||||||
| 	importData.fileDataBytes = tokenizer.inputBytes; |  | ||||||
| 	tokenizer.module = &importData; |  | ||||||
| 	tokenizer.line = 1; |  | ||||||
| 	tokenizer.input = importData.fileData; |  | ||||||
| 
 |  | ||||||
| 	if (!tokenizer.input) { |  | ||||||
| 		fprintf(stderr, "Error: Could not load the input file '%s'.\n", scriptPath); |  | ||||||
| 		return 1; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	FunctionBuilder builder = { 0 }; |  | ||||||
| 	ExecutionContext context = { 0 }; |  | ||||||
| 	context.functionData = &builder; |  | ||||||
| 	context.mainModule = &importData; |  | ||||||
| 
 |  | ||||||
| 	context.heapEntriesAllocated = 2; |  | ||||||
| 	context.heap = (HeapEntry *) AllocateResize(NULL, sizeof(HeapEntry) * context.heapEntriesAllocated); |  | ||||||
| 	context.heap[0].type = T_EOF; |  | ||||||
| 	context.heap[1].type = T_ERROR; |  | ||||||
| 	context.heap[1].nextUnusedEntry = 0; |  | ||||||
| 	context.heapFirstUnusedEntry = 1; |  | ||||||
| 	context.c = (CoroutineState *) AllocateResize(0, sizeof(CoroutineState)); |  | ||||||
| 	CoroutineState empty = { 0 }; |  | ||||||
| 	*context.c = empty; |  | ||||||
| 	context.c->stackEntriesAllocated = sizeof(context.c->stack) / sizeof(context.c->stack[0]); |  | ||||||
| 	context.c->previousCoroutineLink = &context.allCoroutines; |  | ||||||
| 	context.allCoroutines = context.c; |  | ||||||
| 
 |  | ||||||
| 	int result = ScriptLoad(tokenizer, &context, &importData) ? ScriptExecute(&context, &importData) : 1; |  | ||||||
| 	ScriptFree(&context); |  | ||||||
| 
 | 
 | ||||||
| 	while (fixedAllocationBlocks) { | 	while (fixedAllocationBlocks) { | ||||||
| 		void *block = fixedAllocationBlocks; | 		void *block = fixedAllocationBlocks; | ||||||
|  | @ -5854,3 +5789,5 @@ int main(int argc, char **argv) { | ||||||
| 
 | 
 | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 nakst
						nakst