diff --git a/util/automation/genovf.py b/util/automation/genovf.py deleted file mode 100644 index 6916dd9..0000000 --- a/util/automation/genovf.py +++ /dev/null @@ -1,14 +0,0 @@ -import os -import uuid -template = open('util/automation/template.ovf','r').read().split('$') -uuid1 = uuid.uuid4() -uuid2 = uuid.uuid4() -print(template[0], end='') -print(os.path.getsize('bin/drive'), end='') -print(template[1], end='') -print(uuid1, end='') -print(template[2], end='') -print(uuid2, end='') -print(template[3], end='') -print(uuid1, end='') -print(template[4], end='') diff --git a/util/script.c b/util/script.c index 29cdd2d..f9546aa 100644 --- a/util/script.c +++ b/util/script.c @@ -450,6 +450,22 @@ char baseModuleSource[] = { "int SystemGetProcessorCount() #extcall;" "bool SystemRunningAsAdministrator() #extcall;" "str SystemGetHostName() #extcall;" + "int RandomInt(int min, int max) #extcall;" + + "str UUIDGenerate() {" + " str hexChars = \"0123456789abcdef\";" + " str result;" + " for int i = 0; i < 8; i += 1 { result += hexChars[RandomInt(0, 15)]; }" + " result += \"-\";" + " for int i = 0; i < 4; i += 1 { result += hexChars[RandomInt(0, 15)]; }" + " result += \"-4\";" + " for int i = 0; i < 3; i += 1 { result += hexChars[RandomInt(0, 15)]; }" + " result += \"-\" + hexChars[RandomInt(8, 11)];" + " for int i = 0; i < 3; i += 1 { result += hexChars[RandomInt(0, 15)]; }" + " result += \"-\";" + " for int i = 0; i < 12; i += 1 { result += hexChars[RandomInt(0, 15)]; }" + " return result;" + "}" // File system access: @@ -464,6 +480,7 @@ char baseModuleSource[] = { "str FileReadAll(str path) #extcall;" // TODO Returning an error? "bool FileWriteAll(str path, str x) #extcall;" // TODO Returning an error? "bool FileCopy(str source, str destination) #extcall;" + "int FileGetSize(str path) #extcall;" // Returns -1 on error. TODO Returning an error code. // Persistent variables: @@ -509,8 +526,10 @@ int ExternalPathSetDefaultPrefixToScriptSourceDirectory(ExecutionContext *contex int ExternalFileReadAll(ExecutionContext *context, Value *returnValue); int ExternalFileWriteAll(ExecutionContext *context, Value *returnValue); int ExternalFileCopy(ExecutionContext *context, Value *returnValue); +int ExternalFileGetSize(ExecutionContext *context, Value *returnValue); int ExternalPersistRead(ExecutionContext *context, Value *returnValue); int ExternalPersistWrite(ExecutionContext *context, Value *returnValue); +int ExternalRandomInt(ExecutionContext *context, Value *returnValue); ExternalFunction externalFunctions[] = { { .cName = "PrintStdErr", .callback = ExternalPrintStdErr }, @@ -540,8 +559,10 @@ ExternalFunction externalFunctions[] = { { .cName = "FileReadAll", .callback = ExternalFileReadAll }, { .cName = "FileWriteAll", .callback = ExternalFileWriteAll }, { .cName = "FileCopy", .callback = ExternalFileCopy }, + { .cName = "FileGetSize", .callback = ExternalFileGetSize }, { .cName = "PersistRead", .callback = ExternalPersistRead }, { .cName = "PersistWrite", .callback = ExternalPersistWrite }, + { .cName = "RandomInt", .callback = ExternalRandomInt }, }; // --------------------------------- Tokenization and parsing. @@ -5080,6 +5101,23 @@ int ExternalFileReadAll(ExecutionContext *context, Value *returnValue) { return 3; } +int ExternalFileGetSize(ExecutionContext *context, Value *returnValue) { + STACK_POP_STRING(entryText, entryBytes); + returnValue->i = -1; + char *temporary = StringZeroTerminate(entryText, entryBytes); + if (!temporary) return 2; + FILE *file = fopen(temporary, "rb"); + + if (file) { + fseek(file, 0, SEEK_END); + returnValue->i = ftell(file); + fclose(file); + } + + free(temporary); + return 2; +} + int ExternalFileWriteAll(ExecutionContext *context, Value *returnValue) { STACK_POP_STRING_2(entryText, entryBytes, entry2Text, entry2Bytes); returnValue->i = 0; @@ -5313,6 +5351,16 @@ int ExternalSystemGetHostName(ExecutionContext *context, Value *returnValue) { return 3; } +int ExternalRandomInt(ExecutionContext *context, Value *returnValue) { + if (context->c->stackPointer < 2) return -1; + int64_t min = context->c->stack[context->c->stackPointer - 1].i; + int64_t max = context->c->stack[context->c->stackPointer - 2].i; + if (max < min) { PrintError4(context, 0, "RandomInt() called with maximum limit (%ld) less than the minimum limit (%ld).\n", max, min); return 0; } + returnValue->i = rand() % (max - min + 1) + min; + context->c->stackPointer -= 2; + return 2; +} + CoroutineState *ExternalCoroutineWaitAny(ExecutionContext *context) { (void) context; while (sem_wait(&externalCoroutineSemaphore) == -1); @@ -5519,6 +5567,8 @@ int main(int argc, char **argv) { return 1; } + srand(time(NULL)); + sem_init(&externalCoroutineSemaphore, 0, 0); char *scriptPath = NULL; diff --git a/util/start.script b/util/start.script index 295517d..2654fba 100644 --- a/util/start.script +++ b/util/start.script @@ -56,6 +56,17 @@ void Start() { PrintStdErrHighlight("\n"); } +void GenerateOVF() { + str[] template = StringSplitByCharacter(FileReadAll("util/automation/template.ovf"), "$", true); + assert template:len() == 5; + str uuid1 = UUIDGenerate(); + str uuid2 = UUIDGenerate(); + int driveSize = FileGetSize("bin/drive"); + assert driveSize > 0; + str result = template[0] + "%driveSize%" + template[1] + uuid1 + template[2] + uuid2 + template[3] + uuid1 + template[4]; + assert FileWriteAll("bin/ova/Essence.ovf", result); +} + void AutomationBuild() { // TODO: // Copy the source onto the drive for self hosting. @@ -104,7 +115,7 @@ void AutomationBuild() { // Create a virtual machine file. assert SystemShellExecute("qemu-img convert -f raw bin/drive -O vmdk -o adapter_type=lsilogic,subformat=streamOptimized,compat6 " + "bin/ova/Essence-disk001.vmdk"); - assert SystemShellExecute("python util/automation/genovf.py > bin/ova/Essence.ovf"); // TODO Replace. + GenerateOVF(); assert SystemShellExecuteWithWorkingDirectory("bin/ova", "tar -cf Essence.ova Essence.ovf Essence-disk001.vmdk"); // Copy licenses. @@ -115,7 +126,7 @@ void AutomationBuild() { assert FileCopy("util/stb_truetype.h", "Essence/Licenses/stb_truetype.h"); assert FileCopy("res/Fonts/Hack License.md", "Essence/Licenses/Hack License.md"); assert FileCopy("res/Fonts/Inter License.txt", "Essence/Licenses/Inter License.txt"); - assert FileCopy("res/Fonts/Atkinson Hyperlegible License.txt", "License.txt Essence/Licenses/Atkinson Hyperlegible License.txt"); + assert FileCopy("res/Fonts/Atkinson Hyperlegible License.txt", "Essence/Licenses/Atkinson Hyperlegible License.txt"); assert FileCopy("res/Fonts/OpenDyslexic License.txt", "Essence/Licenses/OpenDyslexic License.txt"); assert FileCopy("res/elementary Icons License.txt", "Essence/Licenses/elementary Icons License.txt"); assert FileCopy("res/Sample Images/Licenses.txt", "Essence/Licenses/Sample Images.txt");