X-Git-Url: https://git.ucc.asn.au/?a=blobdiff_plain;f=Kernel%2Fmodules.c;h=eec50200594bc266bc070fb16a7576d2c93758fb;hb=3045a1c5f05fa6f6e3cfe73da753b7500e87aff3;hp=b7209fff379a0ce706562dd706dd934d4445d06f;hpb=8fef00cfab98773a519ab909fa9da9dc7af692db;p=tpg%2Facess2.git diff --git a/Kernel/modules.c b/Kernel/modules.c index b7209fff..eec50200 100644 --- a/Kernel/modules.c +++ b/Kernel/modules.c @@ -6,14 +6,16 @@ #include // === PROTOTYPES === + int Modules_LoadBuiltins(); int Module_LoadMem(void *Buffer, Uint Length, char *ArgString); int Module_LoadFile(char *Path, char *ArgString); int Module_int_ResolveDeps(tModule *Info); int Module_IsLoaded(char *Name); // === IMPORTS === +extern void StartupPrint(char *Str); extern tModule gKernelModules[]; -extern void gKernelModulesEnd; +extern void gKernelModulesEnd; // === GLOBALS === int giNumBuiltinModules = 0; @@ -23,18 +25,84 @@ tModule *gLoadedModules = NULL; // === CODE === int Modules_LoadBuiltins() { - int i; + int i, j, k; + int numToInit = 0; + Uint8 *baIsLoaded; + char **deps; + giNumBuiltinModules = (Uint)&gKernelModulesEnd - (Uint)&gKernelModules; giNumBuiltinModules /= sizeof(tModule); + baIsLoaded = calloc( giNumBuiltinModules, sizeof(*baIsLoaded) ); + + // Pass 1 - Are the dependencies compiled in? for( i = 0; i < giNumBuiltinModules; i++ ) { - Log("Initialising %p '%s' v%i.%i...", - &gKernelModules[i], - gKernelModules[i].Name, - gKernelModules[i].Version>>8, gKernelModules[i].Version & 0xFF - ); - gKernelModules[i].Init(NULL); + deps = gKernelModules[i].Dependencies; + if(deps) + { + for( j = 0; deps[j]; j++ ) + { + for( k = 0; k < giNumBuiltinModules; k++ ) { + if(strcmp(deps[j], gKernelModules[k].Name) == 0) + break; + } + if(k == giNumBuiltinModules) { + Warning("Unable to find dependency '%s' for '%s' in kernel", + deps[j], gKernelModules[i].Name); + + baIsLoaded[i] = -1; // Don't Load + break; + } + } + } + numToInit ++; + } + + // Pass 2 - Intialise + while(numToInit) + { + for( i = 0; i < giNumBuiltinModules; i++ ) + { + if( baIsLoaded[i] ) continue; // Ignore already loaded modules + + deps = gKernelModules[i].Dependencies; + + if( deps ) + { + for( j = 0; deps[j]; j++ ) + { + for( k = 0; k < giNumBuiltinModules; k++ ) { + if(strcmp(deps[j], gKernelModules[k].Name) == 0) + break; + } + // `k` is assumed to be less than `giNumBuiltinModules` + + // If a dependency failed, skip and mark as failed + if( baIsLoaded[k] == -1 ) { + baIsLoaded[i] = -1; + numToInit --; + break; + } + // If a dependency is not intialised, skip + if( !baIsLoaded[k] ) break; + } + // Check if we broke out + if( deps[j] ) continue; + } + + // All Dependencies OK? Initialise + StartupPrint(gKernelModules[i].Name); + Log("Initialising %p '%s' v%i.%i...", + &gKernelModules[i], + gKernelModules[i].Name, + gKernelModules[i].Version>>8, gKernelModules[i].Version & 0xFF + ); + gKernelModules[i].Init(NULL); + // Mark as loaded + baIsLoaded[i] = 1; + numToInit --; + } } return 0; @@ -110,6 +178,12 @@ int Module_LoadFile(char *Path, char *ArgString) return 0; } + Log("Initialising %p '%s' v%i.%i...", + info, + info->Name, + info->Version>>8, info->Version & 0xFF + ); + // Call Initialiser //if( info->Init( ArgString ) != 0 ) if( info->Init( NULL ) == 0 )