diff --git a/Kernel/asm/interrupts.asm b/Kernel/asm/interrupts.asm index 7d7104b..2c7c0ad 100755 --- a/Kernel/asm/interrupts.asm +++ b/Kernel/asm/interrupts.asm @@ -19,8 +19,6 @@ GLOBAL _exception6Handler EXTERN irqDispatcher EXTERN exceptionDispatcher EXTERN systemCallsDispatcher -; EXTERN preserveStack -; EXTERN newStack EXTERN changeWindow EXTERN updateRSP EXTERN nextProcess @@ -28,7 +26,6 @@ EXTERN nextProcess GLOBAL switchContext GLOBAL loadProcess GLOBAL _initialize_stack_frame -;GLOBAL _switchContext, forceTimerAux EXTERN getFPUaddress, getSSEaddress EXTERN checkSleeping @@ -76,10 +73,9 @@ SECTION .text fsave [bytesForFPU] fxsave [bytesForSSEAligned] - mov rdi, %1 ; pasaje de parametro + mov rdi, %1 call irqDispatcher - ; signal pic EOI (End of Interrupt) mov al, 20h out 20h, al @@ -99,7 +95,7 @@ SECTION .text pushState - mov rdi, %1 ; pasaje de parametro + mov rdi, %1 mov rsi, [insPointer] mov rdx, [rspPointer] mov rcx, rsp @@ -168,7 +164,7 @@ picMasterMask: picSlaveMask: push rbp mov rbp, rsp - mov ax, di ; ax = mascara de 16 bits + mov ax, di out 0A1h,al pop rbp retn diff --git a/Kernel/asm/libasm.asm b/Kernel/asm/libasm.asm index 5539fea..613bf13 100644 --- a/Kernel/asm/libasm.asm +++ b/Kernel/asm/libasm.asm @@ -94,4 +94,4 @@ getRegs: ret section .bss - regs resb 120 ; 8 bytes * 16 regs \ No newline at end of file + regs resb 120 \ No newline at end of file diff --git a/Kernel/include/lib.h b/Kernel/include/lib.h index db6c7e9..ebf2e91 100644 --- a/Kernel/include/lib.h +++ b/Kernel/include/lib.h @@ -13,5 +13,6 @@ char * itoa(int value, char * buffer, int base, int length); void strlen(const char *str, int *len); int strcpy(char * strDest, const char * strSrc); char addSpaces(char * str, char qty); +int strcmp(const char *s1, const char *s2); #endif \ No newline at end of file diff --git a/Kernel/interruptions/include/irqDispatcher.h b/Kernel/interruptions/include/irqDispatcher.h new file mode 100644 index 0000000..676eee3 --- /dev/null +++ b/Kernel/interruptions/include/irqDispatcher.h @@ -0,0 +1,12 @@ +#ifndef IRQDISP_H +#define IRQDISP_H + +#include + +void keyboard_handler(); +void timer_handler(); + +static void int_20(); +static void int_21(); + +#endif \ No newline at end of file diff --git a/Kernel/interruptions/include/systemCallsDispatcher.h b/Kernel/interruptions/include/systemCallsDispatcher.h new file mode 100644 index 0000000..a9e8110 --- /dev/null +++ b/Kernel/interruptions/include/systemCallsDispatcher.h @@ -0,0 +1,11 @@ +#ifndef SYSCALLSDISP_H +#define SYSCALLSDISP_H + +#include +#include "systemCalls.h" +#include "memManager.h" +#include "pipeLib.h" +#include "semLib.h" +#include "schedulerLib.h" + +#endif \ No newline at end of file diff --git a/Kernel/interruptions/irqDispatcher.c b/Kernel/interruptions/irqDispatcher.c index 2a7c886..925a929 100755 --- a/Kernel/interruptions/irqDispatcher.c +++ b/Kernel/interruptions/irqDispatcher.c @@ -1,9 +1,4 @@ -#include - -static void int_20(); -static void int_21(); -void keyboard_handler(); -void timer_handler(); +#include "irqDispatcher.h" void irqDispatcher(uint64_t irq) { switch (irq) { diff --git a/Kernel/interruptions/systemCallsDispatcher.c b/Kernel/interruptions/systemCallsDispatcher.c index ceae7f0..4f1fd13 100644 --- a/Kernel/interruptions/systemCallsDispatcher.c +++ b/Kernel/interruptions/systemCallsDispatcher.c @@ -1,9 +1,4 @@ -#include -#include "systemCalls.h" -#include "memManager.h" -#include "pipeLib.h" -#include "semLib.h" -#include "schedulerLib.h" +#include "systemCallsDispatcher.h" uint64_t systemCallsDispatcher(uint64_t rdi, uint64_t rsi, uint64_t rdx, uint64_t rcx, uint64_t r8, uint64_t r9) { switch (rdi) { @@ -59,6 +54,8 @@ uint64_t systemCallsDispatcher(uint64_t rdi, uint64_t rsi, uint64_t rdx, uint64_ return wait(); case 22: return pipes(); + case 23: + return quitCPU(); default: return -1; } diff --git a/Kernel/kernel.c b/Kernel/kernel.c index 3de6b0f..3882a50 100644 --- a/Kernel/kernel.c +++ b/Kernel/kernel.c @@ -1,5 +1,4 @@ #include -// #include #include #include #include @@ -29,11 +28,7 @@ void clearBSS(void *bssAddress, uint64_t bssSize) { } void *getStackBase() { - return (void *) ( - (uint64_t) & endOfKernel - + PageSize * 8 //The size of the stack itself, 32KiB - - sizeof(uint64_t) //Begin at the top of the stack - ); + return (void *) ((uint64_t) &endOfKernel + PageSize * 8 - sizeof(uint64_t)); } void *initializeKernelBinary() { @@ -49,140 +44,24 @@ void *initializeKernelBinary() { } void load_idt(); - uint64_t getRSP(); - -void printBottlerAndWait(); - -#include "test_util.h" - -#ifdef BUDDY -#define MAX_BLOCKS 128 -#define MAX_MEMORY 800 -#else -#define MAX_BLOCKS 42//1024/24 //128 -#define MAX_MEMORY 352//1024-42x16 //Should be around 80% of memory managed by the MM -#endif - -typedef struct MM_rq { - void *address; - uint32_t size; -} mm_rq; - -void test_mm() { - mm_rq mm_rqs[MAX_BLOCKS]; - uint8_t rq; - uint32_t total; - - int i = 1000; - while (i-- != 0) { -// while (1){ - rq = 0; - total = 0; - - // Request as many blocks as we can - while (rq < MAX_BLOCKS && total < MAX_MEMORY) { - mm_rqs[rq].size = GetUniform(MAX_MEMORY - total - 1) + 1; - mm_rqs[rq].address = pvPortMalloc(mm_rqs[rq].size); - // mm_rqs[rq].address = memMalloc(mm_rqs[rq].size); - if (mm_rqs[rq].address == NULL) { - for (int rqAux = 0; rqAux < rq; rqAux++) { - ncPrintDec(mm_rqs[rqAux].size); - ncPrint(" - "); - } - // printStringLen(13, "malloc() -- ERROR", 20); - // new_line(); - ncPrint("Malloc -- null"); - ncNewline(); - } - total += mm_rqs[rq].size; - rq++; - } - - // ncPrint("libre dps de malloc: "); - // ncPrintDec(xPortGetFreeHeapSize()); - // ncNewline(); - // ncPrint("cumpli maximo"); - // ncNewline(); - - // Set - uint32_t i; - for (i = 0; i < rq; i++) - if (mm_rqs[i].address != NULL) - memset(mm_rqs[i].address, i, mm_rqs[i].size); - - // Check - for (i = 0; i < rq; i++) - if (mm_rqs[i].address != NULL) - if (!memcheck(mm_rqs[i].address, i, mm_rqs[i].size)) { - // printStringLen(13, "memCheck() -- ERROR", 20); - // new_line(); - ncPrint("memCheck -- null"); - ncNewline(); - } - - // Free - for (i = 0; i < rq; i++) { - if (mm_rqs[i].address != NULL) - vPortFree(mm_rqs[i].address); - } - // memFree(mm_rqs[i].address); - - // ncPrint("libre dps de free: "); - // ncPrintDec(xPortGetFreeHeapSize()); - // ncNewline(); - ncPrint("termine un loop regio"); - ncNewline(); - - // if (i == 5) - // wait(3); - // ncClear(); - - // printStringLen(13, "ALL GOOD", 9); - // new_line(); - } -} - -// void enqueueProcess(void (*fn) (int, char **), char foreground, int argc, char *argv[], int * fd); void initScheduler(); - void _cli(); - void _sti(); - void haltcpu(); - void forceTimer(); int main() { - // _cli(); - load_idt(); - // if (initMemoryManager(memoryModuleAddress, memoryModuleAddress + sizeof(void *)) == -1) { - // printStringLen(13, "createMemoryManager() -- ERROR", 31); - // new_line(); - // return EXIT_FAILURE; - // } - initMemoryManager(memoryModuleAddress); initScheduler(); -#ifndef BUDDY - // SACAR DESPUÉS! ES SOLO PARA TESTEO... CORRER DE A UNO! - // if (testOne() == EXIT_FAILURE) - // return EXIT_FAILURE; - // if (testTwo() == EXIT_FAILURE) - // return EXIT_FAILURE; -#endif - // test_mm(); - saveSampleRSP(getRSP()); char *argv[] = {"SampleCode"}; enqueueProcess(sampleCodeModuleAddress, 1, 1, argv, NULL); clear(); - // haltcpu(); _sti(); forceTimer(); diff --git a/Kernel/lib.c b/Kernel/lib.c index a936417..8f4cda0 100644 --- a/Kernel/lib.c +++ b/Kernel/lib.c @@ -95,4 +95,12 @@ int strcpy(char *strDest, const char *strSrc) { i++; } return i; +} + +int strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *s1 - *s2; } \ No newline at end of file diff --git a/Kernel/tests/test_mm.c b/Kernel/tests/test_mm.c deleted file mode 100644 index 5696d4b..0000000 --- a/Kernel/tests/test_mm.c +++ /dev/null @@ -1,54 +0,0 @@ -// #include -// #include -// #include -// #include "test_util.h" - -// #define MAX_BLOCKS 128 -// #define MAX_MEMORY 1024 //Should be around 80% of memory managed by the MM - -// typedef struct MM_rq{ -// void *address; -// uint32_t size; -// }mm_rq; - -// void test_mm(){ -// mm_rq mm_rqs[MAX_BLOCKS]; -// uint8_t rq; -// uint32_t total; - -// while (1){ -// rq = 0; -// total = 0; - -// // Request as many blocks as we can -// while(rq < MAX_BLOCKS && total < MAX_MEMORY){ -// mm_rqs[rq].size = GetUniform(MAX_MEMORY - total - 1) + 1; -// mm_rqs[rq].address = malloc(mm_rqs[rq].size); // TODO: Port this call as required -// //TODO: check if NULL -// total += mm_rqs[rq].size; -// rq++; -// } - -// // Set -// uint32_t i; -// for (i = 0; i < rq; i++) -// if (mm_rqs[i].address != NULL) -// memset(mm_rqs[i].address, i, mm_rqs[i].size); // TODO: Port this call as required - -// // Check -// for (i = 0; i < rq; i++) -// if (mm_rqs[i].address != NULL) -// if(!memcheck(mm_rqs[i].address, i, mm_rqs[i].size)) -// printf("ERROR!\n"); // TODO: Port this call as required - -// // Free -// for (i = 0; i < rq; i++) -// if (mm_rqs[i].address != NULL) -// free(mm_rqs[i].address); // TODO: Port this call as required -// } -// } - -// int main(){ -// test_mm(); -// return 0; -// } diff --git a/Kernel/tests/test_prio.c b/Kernel/tests/test_prio.c deleted file mode 100644 index e660008..0000000 --- a/Kernel/tests/test_prio.c +++ /dev/null @@ -1,107 +0,0 @@ -// #include -// #include - -// #define MINOR_WAIT 1000000 // TODO: To prevent a process from flooding the screen -// #define WAIT 10000000 // TODO: Long enough to see theese processes beeing run at least twice - -// uint64_t my_getpid(){ -// return 0; -// } - -// uint64_t my_create_process(char * name){ -// return 0; -// } - -// uint64_t my_nice(uint64_t pid, uint64_t newPrio){ -// return 0; -// } - -// uint64_t my_kill(uint64_t pid){ -// return 0; -// } - -// uint64_t my_block(uint64_t pid){ -// return 0; -// } - -// uint64_t my_unblock(uint64_t pid){ -// return 0; -// } - -// void bussy_wait(uint64_t n){ -// uint64_t i; -// for (i = 0; i < n; i++); -// } - -// void endless_loop(){ -// uint64_t pid = my_getpid(); - -// while(1){ -// printf("%d ",pid); -// bussy_wait(MINOR_WAIT); -// } -// } - -// #define TOTAL_PROCESSES 3 - -// void test_prio(){ -// uint64_t pids[TOTAL_PROCESSES]; -// uint64_t i; - -// for(i = 0; i < TOTAL_PROCESSES; i++) -// pids[i] = my_create_process("endless_loop"); - -// bussy_wait(WAIT); -// printf("\nCHANGING PRIORITIES...\n"); - -// for(i = 0; i < TOTAL_PROCESSES; i++){ -// switch (i % 3){ -// case 0: -// my_nice(pids[i], 0); //lowest priority -// break; -// case 1: -// my_nice(pids[i], 1); //medium priority -// break; -// case 2: -// my_nice(pids[i], 2); //highest priority -// break; -// } -// } - -// bussy_wait(WAIT); -// printf("\nBLOCKING...\n"); - -// for(i = 0; i < TOTAL_PROCESSES; i++) -// my_block(pids[i]); - -// printf("CHANGING PRIORITIES WHILE BLOCKED...\n"); -// for(i = 0; i < TOTAL_PROCESSES; i++){ -// switch (i % 3){ -// case 0: -// my_nice(pids[i], 1); //medium priority -// break; -// case 1: -// my_nice(pids[i], 1); //medium priority -// break; -// case 2: -// my_nice(pids[i], 1); //medium priority -// break; -// } -// } - -// printf("UNBLOCKING...\n"); - -// for(i = 0; i < TOTAL_PROCESSES; i++) -// my_unblock(pids[i]); - -// bussy_wait(WAIT); -// printf("\nKILLING...\n"); - -// for(i = 0; i < TOTAL_PROCESSES; i++) -// my_kill(pids[i]); -// } - -// int main(){ -// test_prio(); -// return 0; -// } diff --git a/Kernel/tests/test_processes.c b/Kernel/tests/test_processes.c deleted file mode 100644 index e89b577..0000000 --- a/Kernel/tests/test_processes.c +++ /dev/null @@ -1,101 +0,0 @@ -// #include -// #include "test_util.h" - -// //TO BE INCLUDED -// void endless_loop(){ -// while(1); -// } - -// uint32_t my_create_process(char * name){ -// return 0; -// } - -// uint32_t my_kill(uint32_t pid){ -// return 0; -// } - -// uint32_t my_block(uint32_t pid){ -// return 0; -// } - -// uint32_t my_unblock(uint32_t pid){ -// return 0; -// } - -// #define MAX_PROCESSES 10 //Should be around 80% of the the processes handled by the kernel - -// enum State {ERROR, RUNNING, BLOCKED, KILLED}; - -// typedef struct P_rq{ -// uint32_t pid; -// enum State state; -// }p_rq; - -// void test_processes(){ -// p_rq p_rqs[MAX_PROCESSES]; -// uint8_t rq; -// uint8_t alive = 0; -// uint8_t action; - -// while (1){ - -// // Create MAX_PROCESSES processes -// for(rq = 0; rq < MAX_PROCESSES; rq++){ -// p_rqs[rq].pid = my_create_process("endless_loop"); // TODO: Port this call as required - -// if (p_rqs[rq].pid == -1){ // TODO: Port this as required -// printf("Error creating process\n"); // TODO: Port this as required -// return; -// }else{ -// p_rqs[rq].state = RUNNING; -// alive++; -// } -// } - -// // Randomly kills, blocks or unblocks processes until every one has been killed -// while (alive > 0){ - -// for(rq = 0; rq < MAX_PROCESSES; rq++){ -// action = GetUniform(2) % 2; - -// switch(action){ -// case 0: -// if (p_rqs[rq].state == RUNNING || p_rqs[rq].state == BLOCKED){ -// if (my_kill(p_rqs[rq].pid) == -1){ // TODO: Port this as required -// printf("Error killing process\n"); // TODO: Port this as required -// return; -// } -// p_rqs[rq].state = KILLED; -// alive--; -// } -// break; - -// case 1: -// if (p_rqs[rq].state == RUNNING){ -// if(my_block(p_rqs[rq].pid) == -1){ // TODO: Port this as required -// printf("Error blocking process\n"); // TODO: Port this as required -// return; -// } -// p_rqs[rq].state = BLOCKED; -// } -// break; -// } -// } - -// // Randomly unblocks processes -// for(rq = 0; rq < MAX_PROCESSES; rq++) -// if (p_rqs[rq].state == BLOCKED && GetUniform(2) % 2){ -// if(my_unblock(p_rqs[rq].pid) == -1){ // TODO: Port this as required -// printf("Error unblocking process\n"); // TODO: Port this as required -// return; -// } -// p_rqs[rq].state = RUNNING; -// } -// } -// } -// } - -// int main(){ -// test_processes(); -// return 0; -// } diff --git a/Kernel/tests/test_sync.c b/Kernel/tests/test_sync.c deleted file mode 100644 index d174e44..0000000 --- a/Kernel/tests/test_sync.c +++ /dev/null @@ -1,84 +0,0 @@ -// #include -// #include - -// uint64_t my_create_process(char * name){ -// return 0; -// } - -// uint64_t my_sem_open(char *sem_id, uint64_t initialValue){ -// return 0; -// } - -// uint64_t my_sem_wait(char *sem_id){ -// return 0; -// } - -// uint64_t my_sem_post(char *sem_id){ -// return 0; -// } - -// uint64_t my_sem_close(char *sem_id){ -// return 0; -// } - -// #define TOTAL_PAIR_PROCESSES 2 -// #define SEM_ID "sem" - -// int64_t global; //shared memory - -// void slowInc(int64_t *p, int64_t inc){ -// int64_t aux = *p; -// aux += inc; -// yield(); -// *p = aux; -// } - -// void inc(uint64_t sem, int64_t value, uint64_t N){ -// uint64_t i; - -// if (sem && !my_sem_open(SEM_ID, 1)){ -// printf("ERROR OPENING SEM\n"); -// return; -// } - -// for (i = 0; i < N; i++){ -// if (sem) my_sem_wait(SEM_ID); -// slowInc(&global, value); -// if (sem) my_sem_post(SEM_ID); -// } - -// if (sem) my_sem_close(SEM_ID); - -// printf("Final value: %d\n", global); -// } - -// void test_sync(){ -// uint64_t i; - -// global = 0; - -// printf("CREATING PROCESSES...(WITH SEM)\n"); - -// for(i = 0; i < TOTAL_PAIR_PROCESSES; i++){ -// my_create_process("inc", 1, 1, 1000000); -// my_create_process("inc", 1, -1, 1000000); -// } -// } - -// void test_no_sync(){ -// uint64_t i; - -// global = 0; - -// printf("CREATING PROCESSES...(WITHOUT SEM)\n"); - -// for(i = 0; i < TOTAL_PAIR_PROCESSES; i++){ -// my_create_process("inc", 0, 1, 1000000); -// my_create_process("inc", 0, -1, 1000000); -// } -// } - -// int main(){ -// test_sync(); -// return 0; -// } diff --git a/Kernel/tests/test_util.c b/Kernel/tests/test_util.c deleted file mode 100644 index 7d2eb88..0000000 --- a/Kernel/tests/test_util.c +++ /dev/null @@ -1,27 +0,0 @@ -#include -#include "test_util.h" - -static uint32_t m_z = 362436069; -static uint32_t m_w = 521288629; - -uint32_t GetUint(){ - m_z = 36969 * (m_z & 65535) + (m_z >> 16); - m_w = 18000 * (m_w & 65535) + (m_w >> 16); - return (m_z << 16) + m_w; -} - -uint32_t GetUniform(uint32_t max){ - uint32_t u = GetUint(); - return (u + 1.0) * 2.328306435454494e-10 * max; -} - -uint8_t memcheck(void *start, uint8_t value, uint32_t size){ - uint8_t *p = (uint8_t *) start; - uint32_t i; - - for (i = 0; i < size; i++, p++) - if (*p != value) - return 0; - - return 1; -} diff --git a/Kernel/utils/include/schedulerLib.h b/Kernel/utils/include/schedulerLib.h index 1f3e9e5..ec061e6 100644 --- a/Kernel/utils/include/schedulerLib.h +++ b/Kernel/utils/include/schedulerLib.h @@ -20,5 +20,6 @@ char unblockFirst(int pid); void sleep(int secs); char blockIO(); void unblockIO(); +char getState(int pid); #endif \ No newline at end of file diff --git a/Kernel/utils/memManagerBuddy.c b/Kernel/utils/memManagerBuddy.c index 89bc563..5ccf2fc 100644 --- a/Kernel/utils/memManagerBuddy.c +++ b/Kernel/utils/memManagerBuddy.c @@ -1,268 +1,231 @@ #ifdef BUDDY -// #include + +// Basado en: https://github.com/cloudwu/buddy/blob/master/buddy.c + #include -// #include -// #include -#include -#define MANAGED_MEMORY_SIZE 1024 +#include +#include -// https://github.com/sdpetrides/BuddyAllocator +#define NODE_UNUSED 0 +#define NODE_USED 1 +#define NODE_SPLIT 2 +#define NODE_FULL 3 -// char initMemoryManager(void *const restrict memoryForMemoryManager, void *const restrict managedMemory) { -// return 1; -// } +#define LEVEL 20 -// void * memMalloc(const size_t memoryToAllocate) { -// return NULL; -// } +#define SIZE (1< 1) ? 1 + log2(n / 2) : 0; +int is_pow_of_2(uint32_t x) { + return !(x & (x-1)); } -// /* Returns log base 2 of a double d */ -// double log2(double d) { -// return log(d) / log(2); -// } - -/* Returns the level a reqSize will fit in */ -int size_to_n(size_t reqSize) { - reqSize+=1; - double d = log2((double)reqSize); - return (int)ceil(d); +uint32_t next_pow_of_2(uint32_t x) { + if ( is_pow_of_2(x) ) + return x; + x |= x>>1; + x |= x>>2; + x |= x>>4; + x |= x>>8; + x |= x>>16; + return x+1; } -/* Returns the position of the next block of the correct size */ -int jump_next(int n, int pos) { - int bits = pos>>(n); - bits+=1; - int ret = bits<<(n); - - if (ret == MANAGED_MEMORY_SIZE) { - return ret; - } else { - return ret; - } +#define PAGE_SIZE 0x1000 +int _index_offset(int index, int level, int max_level) { + return (int*)((((index + 1) - (1 << level)) << (max_level - level)) * PAGE_SIZE + heapAddress); } -/* Returns the position of the left half of a pair */ -int jump_back(int n, int pos) { - int bits = pos>>(n); - bits-=1; - return bits<<(n); +void _mark_parent(int index) { + + while(1) { + int buddy = index - 1 + (index & 1) * 2; + if (buddy > 0 && (self.tree[buddy] == NODE_USED || self.tree[buddy] == NODE_FULL)) { + index = (index + 1) / 2 - 1; + self.tree[index] = NODE_FULL; + } else { + return; + } + } } -/* Fills a Meta struct with metadata at pos */ -void unpack(Meta * m, int pos) { - memset(m, myblock[pos], 1); +void * buddy_alloc(int s) { + + if(s%PAGE_SIZE != 0){ + s=(s/PAGE_SIZE)+1; + }else{ + s/=PAGE_SIZE; + } + int size; + if (s==0) { + size = 1; + } else { + size = (int)next_pow_of_2(s); + } + int length = 1 << self.level; + + if (size > length) + return NULL; + + int index = 0; + int level = 0; + int nextStep = 1; + + while (index >= 0) { + nextStep = 1; + if (size == length) { + if (self.tree[index] == NODE_UNUSED) { + self.tree[index] = NODE_USED; + _mark_parent(index); + return _index_offset(index, level, self.level); + } + } else { + switch (self.tree[index]) { + case NODE_USED: break; + case NODE_FULL: break; + case NODE_UNUSED: + self.tree[index] = NODE_SPLIT; + self.tree[index*2+1] = NODE_UNUSED; + self.tree[index*2+2] = NODE_UNUSED; + default: + index = index * 2 + 1; + length /= 2; + level++; + nextStep = 0; + } + } + if( nextStep ){ + if (index & 1) { + ++index; + } else { + int cont = 1; + while(cont) { + level--; + length *= 2; + index = (index+1)/2 -1; + if (index < 0) + return NULL; + if (index & 1) { + ++index; + cont = 0; + } + } + } + + } + } + + return NULL; } -/* Returns whether position at level n is left or right partner */ -int is_left(int n, int pos) { - - // Manipulate bits to set nth bit on - int k = 1; - k<<=(n); - - // Manipulate bits to zero bits above n - unsigned int p = (unsigned int)pos; - p<<=(31-n); - p>>=(31-n); - - if (k == p) { - return 0; // Right - } else { - return 1; // Left - } +void _combine(int index) { + while(1) { + int buddy = index - 1 + (index & 1) * 2; + if (buddy < 0 || self.tree[buddy] != NODE_UNUSED) { + self.tree[index] = NODE_UNUSED; + while (((index = (index + 1) / 2 - 1) >= 0) && self.tree[index] == NODE_FULL){ + self.tree[index] = NODE_SPLIT; + } + return; + } + index = (index + 1) / 2 - 1; + } } -/* Mergee two unallocated blocks with same size */ -void merge(int pos, int pos2, int n) { +int buddy_free(void * pointer) { - // Create new meta and set size field - char newMeta = (n+1)<<2; + int offset=(int)pointer; - // Set left field - if (is_left(n+1, pos)) { - newMeta+=2; - } + offset=(offset- heapAddress)/PAGE_SIZE; - // Add new meta - myblock[pos] = newMeta; - // Delete meta on right partner - myblock[pos2] = 0; + if (offset >= (1<< self.level)){ + return -1; + } + int left = 0; + int length = 1 << self.level; + int index = 0; + + while(1) { + switch (self.tree[index]) { + case NODE_USED: + if (offset != left){ + return -1; + } + _combine(index); + return 0; + case NODE_UNUSED: + return -1; + default: + length /= 2; + if (offset < left + length) { + index = index * 2 + 1; + } else { + left += length; + index = index * 2 + 2; + } + break; + } + } } -/* MYmymalloc */ -void * pvPortMalloc(size_t reqSize) { - - // Check if too big - if (reqSize > MANAGED_MEMORY_SIZE - 1) { - // fprintf(stderr, "Error: Requested size too large\n"); - return NULL; - } - - // Traverse heap to find block of correct size - algo(n) - int n = size_to_n(reqSize); - int pos = 0; - unsigned char c = 0; - Meta * m = memset(&c, 0, 1); - - while (pos < MANAGED_MEMORY_SIZE) { - // Read metadata - unpack(m, pos); - - // // Debugging - // if (m->size == 0) { - // exit(0); - // } - - if (n <= m->size) { - if (m->allo == 1) { - // Jump - pos = jump_next(n, pos); - continue; - } else if (m->size == n) { - // Allocate - myblock[pos]+=1; - pos+=1; - return (void*)((long int)&myblock+pos); - } else { - // Partition - - // Get partner position - int partner = jump_next((m->size)-1, pos); - - // Set Left - char meta_1 = 2; - char meta_2 = 0; - - // Set Size - char s = ((m->size)-1)<<2; - meta_1 = (meta_1 | s); - meta_2 = (meta_2 | s); - - // Fill in metadata - myblock[pos] = meta_1; - myblock[partner] = meta_2; - - // Continue on same position with new size of block - continue; - } - } else { - // Jump - pos = jump_next(n, pos); - continue; - } - } - - //error - - return 0; +static void dumpMM(struct buddy * self, int index , int level) { + char buffer[100]; + switch (self->tree[index]) { + case NODE_UNUSED: + printStringLen("(%d:%d)", _index_offset(index, level, self->level) , 1 << (self->level - level)); + break; + case NODE_USED: + printStringLen("(", 1); + printStringLen(itoa(_index_offset(index, level, self->level), buffer, 10)); + printStringLen(":", 1); + printStringLen(itoa(1 << (self->level - level), buffer, 10)); + printStringLen(")", 1); + break; + case NODE_FULL: + printStringLen("{", 1); + _dump(self, index * 2 + 1, level+1); + _dump(self, index * 2 + 2, level+1); + printStringLen("}", 1); + break; + default: + printStringLen("(", 1); + _dump(self, index * 2 + 1, level+1); + _dump(self, index * 2 + 2, level+1); + printStringLen(")", 1); + break; + } } -/* MYmyfree */ -void vPortFree(void * ptr) { - - // Error Checking - if (ptr <= (void *)&myblock || ptr > (void *)(&myblock + MANAGED_MEMORY_SIZE)) { - //error - return; - } - - // Get position - // int pos = (int)(ptr-(void *)&myblock-1); - int pos = (int)((char *)ptr-(char *)&myblock-1); - - // Check if valid metadata location - if (pos%2 == 1 || myblock[pos] == 0) { - //error - return; - } - - - // Initialize variables for merge - unsigned char c1 = 0; - unsigned char c2 = 0; - Meta * m1 = memset(&c1, 0, 1); - Meta * m2 = memset(&c2, 0, 1); - unpack(m1,pos); - - // Change allocated field - myblock[pos] = myblock[pos] - 1; - - while (pos >= 0 && pos <= 8196){ - // Read metadata - unpack(m1,pos); - - if (m1->left) { // Left Partner - - // Get position of other partner and read metadata - int pos2 = jump_next(m1->size, pos); - - if (pos2 >= 0 && pos2 <= MANAGED_MEMORY_SIZE - 2) { - unpack(m2,pos2); - } else { - break; - } - - // Merge or break - if (m2->allo || m2->size != m1->size) { - break; - } else { - merge(pos, pos2, m1->size); - } - - } else { // Right Partner - - // Get position of other partner and read metadata - int pos2 = jump_back(m2->size,pos); - - if (pos2 >= 0 && pos2 <= MANAGED_MEMORY_SIZE -2) { - unpack(m2,pos2); - } else { - break; - } - - // Merge or break - if (m2->allo || m2->size != m1->size) { - break; - } else { - merge(pos2, pos, m1->size); - } - } - } +void buddy_dumpMM(struct buddy * self) { + dumpMM(self, 0 , 0); + printStringLen("\n", 1); } #endif \ No newline at end of file diff --git a/Kernel/utils/memManagerFRT4.c b/Kernel/utils/memManagerFRT4.c index be02304..1d6b38b 100644 --- a/Kernel/utils/memManagerFRT4.c +++ b/Kernel/utils/memManagerFRT4.c @@ -188,14 +188,6 @@ void *pvPortMalloc(size_t xWantedSize) { } } else { - // ncNewline(); - // ncPrint("MALLOC: "); - // ncPrintDec(xFreeBytesRemaining); - // ncPrint(" "); - // ncPrintDec(xWantedSize); - // ncPrint(" "); - // ncPrintDec(configADJUSTED_HEAP_SIZE); - // ncNewline(); } } @@ -227,12 +219,12 @@ void vPortFree(void *pv) { allocated. */ pxLink->xBlockSize &= ~xBlockAllocatedBit; - { + // { /* Add this block to the list of free blocks. */ xFreeBytesRemaining += pxLink->xBlockSize; prvInsertBlockIntoFreeList(((BlockLink_t *) pxLink)); xNumberOfSuccessfulFrees++; - } + // } } } } diff --git a/Kernel/utils/scheduler.c b/Kernel/utils/scheduler.c index ca62fab..de04c33 100644 --- a/Kernel/utils/scheduler.c +++ b/Kernel/utils/scheduler.c @@ -58,7 +58,7 @@ uint64_t nextProcess(uint64_t currentRSP) { currentProcess->rsp = currentRSP; processCDT *prev = currentProcess; if (currentProcess != NULL && currentProcess->state == READY && - currentProcess->executions == MAX_PRIORITY - currentProcess->priority + 1) { + currentProcess->executions == MAX_PRIORITY - currentProcess->priority) { currentProcess->executions = 0; currentProcess = currentProcess->next; } @@ -115,8 +115,7 @@ int enqueueProcess(void (*fn)(int, char **), char foreground, int argc, char *ar block(IDLE_PID); if (currentProcess != NULL) { - if (foreground) - currentProcess->children++; + currentProcess->children++; if (currentProcess->foreground && foreground) { currentProcess->foreground = 0; currentProcess->backWait = 1; @@ -230,18 +229,9 @@ void unblockIO() { char blockIO() { if (currentProcess == NULL || currentProcess->state == DEAD) return EXIT_FAILURE; - else { - currentProcess->state = BLOCKEDIO; - } - - // processCDT *prev = NULL; - // processADT parent = searchProcess(&prev, currentProcess->ppid, firstProcess); - // if (currentProcess->foreground) { - // if (parent->backWait) { - // parent->backWait = 0; - // parent->foreground = 1; - // } - // } + + currentProcess->state = BLOCKEDIO; + currentProcess->executions = 0; forceTimer(); return EXIT_SUCCESS; @@ -264,6 +254,7 @@ char block(int pid) { } } if (pid == currentProcess->pid) { + currentProcess->executions = 0; forceTimer(); } @@ -275,9 +266,9 @@ char unblock(int pid) { processADT del = searchProcess(&prev, pid, firstProcess); if (del == NULL || del->state == DEAD) { return EXIT_FAILURE; - } else { - del->state = READY; } + + del->state = READY; processADT parent = searchProcess(&prev, del->ppid, firstProcess); if (del->foreground) { @@ -289,6 +280,46 @@ char unblock(int pid) { return EXIT_SUCCESS; } +char unblockFirst(int pid) { + processADT prev = NULL; + processADT del = searchProcess(&prev, pid, firstProcess); + if (del == NULL || del->state == DEAD) { + return EXIT_FAILURE; + } + + removeProcess(del, &firstProcess, &lastProcess); + if (currentProcess != NULL) { + currentProcess->executions = MAX_PRIORITY - currentProcess->priority; + del->next = currentProcess->next; + currentProcess->next = del; + if (currentProcess == lastProcess) + lastProcess = del; + } + else { + if (firstProcess == NULL) { + del->next = NULL; + firstProcess = del; + lastProcess = del; + } + else { + del->next = firstProcess->next; + firstProcess = del; + } + } + + del->state = READY; + + processADT parent = searchProcess(&prev, del->ppid, firstProcess); + if (del->foreground) { + if (parent->foreground) { + parent->backWait = 1; + parent->foreground = 0; + } + } + forceTimer(); + return EXIT_SUCCESS; +} + char kill(int pid) { processADT prev = NULL; processADT del = searchProcess(&prev, pid, firstProcess); @@ -297,19 +328,21 @@ char kill(int pid) { } processADT parent = searchProcess(&prev, del->ppid, firstProcess); - if (parent != NULL && del->foreground) { + if (parent != NULL) { parent->children--; if (!parent->children && parent->state == WAITING) { parent->state = READY; } - if (parent->backWait) { + if (parent->backWait && del->foreground) { parent->backWait = 0; parent->foreground = 1; } } vPortFree(del->fd); - vPortFree((void *) del->rbp - STACK_SIZE); + vPortFree(del->name); + void *auxi = del->rbp - STACK_SIZE; + vPortFree((void *) auxi); del->state = DEAD; if (pid == currentProcess->pid) { @@ -367,15 +400,27 @@ char updateRSP(uint64_t newRsp) { int getPid() { if (currentProcess == NULL) - return EXIT_FAILURE; + return -1; return currentProcess->pid; } +char getState(int pid){ + processCDT * aux = firstProcess; + while (aux != NULL) { + if (aux->pid == pid){ + break; + } + aux = aux -> next; + } + return aux == NULL ? -1 : aux->state; +} + char quitCPU() { int pid = getPid(); - if (pid == EXIT_FAILURE) + if (pid == -1) return EXIT_FAILURE; // return block(pid); + currentProcess->executions = MAX_PRIORITY - currentProcess->priority; forceTimer(); return EXIT_SUCCESS; } diff --git a/Kernel/utils/sem.c b/Kernel/utils/sem.c index a70bc6e..9bbda2e 100644 --- a/Kernel/utils/sem.c +++ b/Kernel/utils/sem.c @@ -10,9 +10,27 @@ typedef struct node_t { node_t *firstSem = NULL; static char counter = 0; +node_t * searchSem(char * name, node_t ** prev) { + *prev = NULL; + node_t * aux = firstSem; + while (aux != NULL) { + if (!strcmp(name, aux->sem->name)) + break; + } + return aux; +} + sem_t *semOpen(char *name, unsigned int value) { enter_region(&semLock); + node_t * prev = NULL; + node_t * aux = searchSem(name, &prev); + + if (aux != NULL) { + leave_region(&semLock); + return aux->sem; + } + sem_t *sem = pvPortMalloc(sizeof(sem_t)); node_t *node = pvPortMalloc(sizeof(node_t)); node->sem = sem; @@ -74,6 +92,8 @@ char semClose(sem_t *sem) { return EXIT_SUCCESS; } +#include "video.h" + void semWait(sem_t *sem) { enter_region(&semLock); @@ -109,8 +129,10 @@ void semPost(sem_t *sem) { sem->entering = sem->entering->next; if (sem->entering == NULL) sem->last = NULL; - unblock(aux->pid); + leave_region(&semLock); + unblockFirst(aux->pid); vPortFree(aux); + enter_region(&semLock); } leave_region(&semLock); diff --git a/Userland/SampleCodeModule/asm/libasm.asm b/Userland/SampleCodeModule/asm/libasm.asm index 5826f43..f68ed6c 100644 --- a/Userland/SampleCodeModule/asm/libasm.asm +++ b/Userland/SampleCodeModule/asm/libasm.asm @@ -5,7 +5,7 @@ GLOBAL _getRegs, sys_switchContext GLOBAL cpu_id, cpu_id_support GLOBAL sys_exit, sys_ps, sys_free, sys_malloc, sys_sem, sys_openPipe, sys_semClose GLOBAL sys_nice, sys_semWait, sys_semPost, sys_semOpen, sys_sleep, sys_kill, sys_getPid, -GLOBAL sys_block, sys_unblock, sys_wait, sys_pipes +GLOBAL sys_block, sys_unblock, sys_wait, sys_pipes, sys_quitCPU section .text @@ -227,6 +227,21 @@ sys_getPid: pop rbp ret +sys_quitCPU: + push rbp + mov rbp, rsp + + push rdi + + mov rdi, 23 + int 80h + + pop rdi + + mov rsp, rbp + pop rbp + ret + sys_wait: push rbp mov rbp, rsp diff --git a/Userland/SampleCodeModule/bottler/bottler.c b/Userland/SampleCodeModule/bottler/bottler.c index fa99336..9039716 100644 --- a/Userland/SampleCodeModule/bottler/bottler.c +++ b/Userland/SampleCodeModule/bottler/bottler.c @@ -2,63 +2,63 @@ void bottler(int argc, char **argv) { printStringLen(" ", 80); - new_line(); + newline(); printStringLen(" (%( ", 80); - new_line(); + newline(); printStringLen(" Welcome to", 17); printStringLen(" %%%%% ", 80); - new_line(); + newline(); printStringLen(" BottlerOS", 18); printStringLen(" %%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%% %%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%% %%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%% %%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%% ", 27); printStringLen(" %%%% %%%% ", 22); printStringLen(" %%%%% ", 30); - new_line(); + newline(); printStringLen(" %%%%% ", 28); printStringLen(" (% %( ", 21); printStringLen(" %%%%% ", 30); - new_line(); + newline(); printStringLen(" %%%%%%% %%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%%%%%%%%%%%% ", 41); printStringLen(" % %* ", 8); printStringLen(" %%%%%%%%% ", 30); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%**%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/%% ", 80); - new_line(); + newline(); printStringLen(" %%* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% /%% ", 80); - new_line(); + newline(); printStringLen(" %%* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% /%% ", 80); - new_line(); + newline(); printStringLen(" %%% %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% %%% ", 80); - new_line(); + newline(); printStringLen(" ,%%%%% %%%%%%%%%%%%%%%%%%%%%%%%%%% %%%%%. ", 80); - new_line(); + newline(); printStringLen(" %%. %%%%%%%%%%%%%%%%% .%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%%%%%%%% ", 80); - new_line(); + newline(); printStringLen(" %%%%%%% ", 80); - new_line(); + newline(); printStringLen(" ", 80); - new_line(); + newline(); sys_sleep(3); winClear(); diff --git a/Userland/SampleCodeModule/include/system.h b/Userland/SampleCodeModule/include/system.h index 0eb036c..2c58746 100644 --- a/Userland/SampleCodeModule/include/system.h +++ b/Userland/SampleCodeModule/include/system.h @@ -30,8 +30,10 @@ void sys_semPost(sem_t *); sem_t * sys_semOpen(char *, unsigned int); int sys_getPid(); char sys_semClose(void *); -void sys_kill(int pid); -void sys_block(int pid); -void sys_unblock(int pid); +char sys_kill(int pid); +char sys_block(int pid); +char sys_unblock(int pid); +void sys_quitCPU(); +void sys_wait(); #endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/libc.c b/Userland/SampleCodeModule/libc.c index 10ea8bd..0bb6faa 100644 --- a/Userland/SampleCodeModule/libc.c +++ b/Userland/SampleCodeModule/libc.c @@ -42,7 +42,7 @@ void putChar(char c) { sys_write(1, &buffer, 0); } -void new_line() { +void newline() { putChar('\n'); } @@ -50,6 +50,10 @@ void backspace() { putChar('\b'); } +void addEOF() { + putChar(-1); +} + void winClear() { sys_write(1, "\e\f", 2); } diff --git a/Userland/SampleCodeModule/sampleCodeModule.c b/Userland/SampleCodeModule/sampleCodeModule.c index b65c822..28843ff 100644 --- a/Userland/SampleCodeModule/sampleCodeModule.c +++ b/Userland/SampleCodeModule/sampleCodeModule.c @@ -8,7 +8,7 @@ int main(int argc, char *argv[]) { winClear(); char *argv1[] = {"bottler"}; - sys_loadProcess(bottler, 1, 1, argv1, NULL); + sys_loadProcess(bottler, 0, 1, argv1, NULL); sys_wait(); char *argv2[] = {"shell"}; diff --git a/Userland/SampleCodeModule/shell/commands/phylo.c b/Userland/SampleCodeModule/shell/commands/phylo.c index 5dd16f0..aada59a 100644 --- a/Userland/SampleCodeModule/shell/commands/phylo.c +++ b/Userland/SampleCodeModule/shell/commands/phylo.c @@ -1,4 +1,4 @@ -#include "phylo.h" +#include "phyloLib.h" philosopher_t *firstPhil; sem_t *mutex; @@ -154,7 +154,6 @@ void phylo(int argc, char ** argv) { } while (phil != firstPhil); char c; - // while (1) { while ((c = getChar()) != 0 && c != -1) { if (c == 'a') { addPhilo(); @@ -165,8 +164,8 @@ void phylo(int argc, char ** argv) { else if (c == 'q') { end(); } + // sys_sleep(1); } - // } sys_exit(); } @@ -195,6 +194,6 @@ void end() { } /* - * Taken from "Operating Systems - Design and Implementation" by Tanenbaum - * Section 2.3, page 91, Figure 2-10 -*/ + * Inspirados en "Operating Systems - Design and Implementation" de Tanenbaum + * Sección 2.3, pág. 91 +*/ \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/commands/pipes.c b/Userland/SampleCodeModule/shell/commands/pipe.c similarity index 62% rename from Userland/SampleCodeModule/shell/commands/pipes.c rename to Userland/SampleCodeModule/shell/commands/pipe.c index 118dece..640fb6a 100644 --- a/Userland/SampleCodeModule/shell/commands/pipes.c +++ b/Userland/SampleCodeModule/shell/commands/pipe.c @@ -1,7 +1,6 @@ -#include "libc.h" -#include "shell.h" +#include "pipes.h" -void pipes(int argc, char *argv[]) { +void pipe(int argc, char *argv[]) { char * output = sys_pipes(); printString(output); newline(); diff --git a/Userland/SampleCodeModule/shell/commands/semCom.c b/Userland/SampleCodeModule/shell/commands/sems.c similarity index 100% rename from Userland/SampleCodeModule/shell/commands/semCom.c rename to Userland/SampleCodeModule/shell/commands/sems.c diff --git a/Userland/SampleCodeModule/shell/commands/test_mm.c b/Userland/SampleCodeModule/shell/commands/test_mm.c new file mode 100644 index 0000000..0f25d89 --- /dev/null +++ b/Userland/SampleCodeModule/shell/commands/test_mm.c @@ -0,0 +1,69 @@ +#include "test_mm.h" + +typedef struct MM_rq{ + void *address; + uint32_t size; +} mm_rq; + +static uint32_t m_z = 362436069; +static uint32_t m_w = 521288629; + +uint32_t GetUint(){ + m_z = 36969 * (m_z & 65535) + (m_z >> 16); + m_w = 18000 * (m_w & 65535) + (m_w >> 16); + return (m_z << 16) + m_w; +} + +uint32_t GetUniform(uint32_t max){ + uint32_t u = GetUint(); + return (u + 1.0) * 2.328306435454494e-10 * max; +} + +uint8_t memcheck(void *start, uint8_t value, uint32_t size){ + uint8_t *p = (uint8_t *) start; + uint32_t i; + + for (i = 0; i < size; i++, p++) + if (*p != value) + return 0; + + return 1; +} + +void test_mm(int argc, char ** argv){ + mm_rq mm_rqs[MAX_BLOCKS]; + uint8_t rq; + uint32_t total; + + while (1){ + rq = 0; + total = 0; + + while(rq < MAX_BLOCKS && total < MAX_MEMORY){ + mm_rqs[rq].size = GetUniform(MAX_MEMORY - total - 1) + 1; + mm_rqs[rq].address = sys_malloc(mm_rqs[rq].size); +//TODO: check if NULL + total += mm_rqs[rq].size; + rq++; + } + + // Set + uint32_t i; + for (i = 0; i < rq; i++) + if (mm_rqs[i].address != NULL) + memset(mm_rqs[i].address, i, mm_rqs[i].size); + + // Check + for (i = 0; i < rq; i++) + if (mm_rqs[i].address != NULL) + if(!memcheck(mm_rqs[i].address, i, mm_rqs[i].size)) + printStringLen("ERROR!\n", 8); + + // Free + for (i = 0; i < rq; i++) + if (mm_rqs[i].address != NULL) + sys_free(mm_rqs[i].address); + } + + sys_exit(); +} diff --git a/Userland/SampleCodeModule/shell/commands/test_prio.c b/Userland/SampleCodeModule/shell/commands/test_prio.c new file mode 100644 index 0000000..ee05ae3 --- /dev/null +++ b/Userland/SampleCodeModule/shell/commands/test_prio.c @@ -0,0 +1,88 @@ +#include "test_prio.h" + +#define MINOR_WAIT 10000000 // TODO: To prevent a process from flooding the screen +#define WAIT 1000000000 // TODO: Long enough to see theese processes beeing run at least twice + +void bussy_wait(uint64_t n){ + uint64_t i; + for (i = 0; i < n; i++); +} + +void endless_loop2(){ + uint64_t pid = sys_getPid(); + + while(1){ + char buffer[10]; + printStringLen(itoa(pid, buffer, 10), 10); + // sys_sleep(1); + bussy_wait(MINOR_WAIT); + } + + sys_exit(); +} + +#define TOTAL_PROCESSES 3 + +void test_prio(int argc, char ** argv){ + uint64_t pids[TOTAL_PROCESSES]; + uint64_t i; + + for(i = 0; i < TOTAL_PROCESSES; i++) { + char *argv[] = {"test_prio"}; + pids[i] = sys_loadProcess(endless_loop2, 0, 1, argv, NULL); + } + + bussy_wait(WAIT); + // sys_sleep(5); + printStringLen("\nCHANGING PRIORITIES...\n", 25); + + for(i = 0; i < TOTAL_PROCESSES; i++){ + switch (i % 3){ + case 0: + sys_nice(pids[i], 19); //lowest priority + break; + case 1: + sys_nice(pids[i], 0); //medium priority + break; + case 2: + sys_nice(pids[i], -20); //highest priority + break; + } + } + + bussy_wait(WAIT); +// sys_sleep(5); + printStringLen("\nBLOCKING...\n", 14); + + for(i = 0; i < TOTAL_PROCESSES; i++) + sys_block(pids[i]); + + printStringLen("CHANGING PRIORITIES WHILE BLOCKED...\n", 38); + for(i = 0; i < TOTAL_PROCESSES; i++){ + switch (i % 3){ + case 0: + sys_nice(pids[i], 0); //medium priority + break; + case 1: + sys_nice(pids[i], 0); //medium priority + break; + case 2: + sys_nice(pids[i], 0); //medium priority + break; + } + } + + printStringLen("UNBLOCKING...\n", 15); + + for(i = 0; i < TOTAL_PROCESSES; i++) + sys_unblock(pids[i]); + + bussy_wait(WAIT); +// sys_sleep(5); + printStringLen("\nKILLING...\n", 13); + + for(i = 0; i < TOTAL_PROCESSES; i++) + sys_kill(pids[i]); + + sys_exit(); +} \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/commands/test_processes.c b/Userland/SampleCodeModule/shell/commands/test_processes.c new file mode 100644 index 0000000..f8f7fad --- /dev/null +++ b/Userland/SampleCodeModule/shell/commands/test_processes.c @@ -0,0 +1,103 @@ +#include "test_processes.h" +#include "ps.h" + +#define MAX_PROCESSES 10 //Should be around 80% of the the processes handled by the kernel + +enum State {ERROR, RUNNING, BLOCKED, KILLED}; + +typedef struct P_rq{ + uint32_t pid; + enum State state; +}p_rq; + + +void endless_loop(){ + while(1); + sys_exit(); +} + +void debugPSA() { + return; +} + +void test_processes(int argc, char ** argv){ + p_rq p_rqs[MAX_PROCESSES]; + uint8_t rq; + uint8_t alive = 0; + uint8_t action; + + int i = 0; + while (1) { + + debugPSA(); + // Create MAX_PROCESSES processes + for(rq = 0; rq < MAX_PROCESSES; rq++){ + char *argv[] = {"endless"}; + p_rqs[rq].pid = sys_loadProcess(endless_loop, 0, 1, argv, NULL); // TODO: Port this call as required + sys_nice(p_rqs[rq].pid, 19); + i++; + + if (p_rqs[rq].pid == -1){ // TODO: Port this as required + printStringLen("Error creating process\n", 24); // TODO: Port this as required + return; + }else{ + p_rqs[rq].state = RUNNING; + alive++; + } + } + + // Randomly kills, blocks or unblocks processes until every one has been killed + while (alive > 0){ + + for(rq = 0; rq < MAX_PROCESSES; rq++){ + action = GetUniform(2) % 2; + + switch(action){ + case 0: + if (p_rqs[rq].state == RUNNING || p_rqs[rq].state == BLOCKED){ + sys_kill(p_rqs[rq].pid); + p_rqs[rq].state = KILLED; + alive--; + } + break; + + case 1: + if (p_rqs[rq].state == RUNNING){ + if(sys_block(p_rqs[rq].pid) == -1){ // TODO: Port this as required + printStringLen("Error blocking process\n", 24); // TODO: Port this as required + return; + } + p_rqs[rq].state = BLOCKED; + } + break; + } + } + + // Randomly unblocks processes + for(rq = 0; rq < MAX_PROCESSES; rq++) + if (p_rqs[rq].state == BLOCKED && GetUniform(2) % 2){ + if(sys_unblock(p_rqs[rq].pid) == -1){ // TODO: Port this as required + printStringLen("Error unblocking process\n", 26); // TODO: Port this as required + return; + } + p_rqs[rq].state = RUNNING; + } + } + if (i % 10000 == 0) { + printStringLen("Procesos: ", 11); + char buffer[4]; + printString(itoa(i, buffer, 10)); + newline(); + } + // char buffer[6] = {0}; + // winClear(); + // char buffer[4]; + // printString(itoa(i, buffer, 10)); + // printStringLen(" procesos creados\n", 19); + + // char * output = sys_ps(); + // printString(output); + // newline(); + } + sys_exit(); +} \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/commands/test_sync.c b/Userland/SampleCodeModule/shell/commands/test_sync.c new file mode 100644 index 0000000..167215c --- /dev/null +++ b/Userland/SampleCodeModule/shell/commands/test_sync.c @@ -0,0 +1,81 @@ +#include "test_sync.h" + +#define TOTAL_PAIR_PROCESSES 2 +#define SEM_ID "sem" +#define ITERS "2000" + +static int global; + +void slowInc(int *p, int inc){ + int aux = *p; + aux += inc; + sys_quitCPU(); + *p = aux; +} + +void inc(int argc, char ** argv) { + if (argc != 4) { + printStringLen("inc recibe 4 argumentos\n", 25); + sys_exit(); + return; + } + + uint64_t semDir = (uint64_t) atoi(argv[1], 14); + int64_t value = atoi(argv[2], 10); + uint64_t N = atoi(argv[3], 10); + uint64_t i; + + char buff[10] = {0}; + for (i = 0; i < N; i++){ + if (semDir) sys_semWait(semDir); + slowInc(&global, value); + if (semDir) sys_semPost(semDir); + } + + char buffer[10] = {0}; + printStringLen("Final value: ", 14); + printStringLen(itoa(global, buffer, 10), 10); + newline(); + sys_exit(); +} + +void test_sync(int argc, char ** argv){ + uint64_t i; + + global = 0; + + printStringLen("CREATING PROCESSES...(WITH SEM)\n", 33); + + sem_t * sem = sys_semOpen(SEM_ID, 1); + char semDir[10] = {0}; + itoa((int) sem, semDir, 10); + + for(i = 0; i < TOTAL_PAIR_PROCESSES; i++){ + char * argv1[] = {"inc", semDir, "1", ITERS}; + char * argv2[] = {"inc", semDir, "-1", ITERS}; + sys_loadProcess(inc, 0, 4, argv1, NULL); + sys_loadProcess(inc, 0, 4, argv2, NULL); + } + + sys_wait(); + sys_semClose(sem); + + sys_exit(); +} + +void test_no_sync(int argc, char ** argv){ + uint64_t i; + + global = 0; + + printStringLen("CREATING PROCESSES...(WITHOUT SEM)\n", 36); + + for(i = 0; i < TOTAL_PAIR_PROCESSES; i++){ + char * argv1[] = {"inc", "0", "1", ITERS}; + char * argv2[] = {"inc", "0", "-1", ITERS}; + sys_loadProcess(inc, 0, 4, argv1, NULL); + sys_loadProcess(inc, 0, 4, argv2, NULL); + } + + sys_exit(); +} diff --git a/Userland/SampleCodeModule/shell/commands/test_util.c b/Userland/SampleCodeModule/shell/commands/test_util.c new file mode 100644 index 0000000..495f254 --- /dev/null +++ b/Userland/SampleCodeModule/shell/commands/test_util.c @@ -0,0 +1,27 @@ +#include +#include "test_util.h" + +// static uint32_t m_z = 362436069; +// static uint32_t m_w = 521288629; + +// uint32_t GetUint(){ +// m_z = 36969 * (m_z & 65535) + (m_z >> 16); +// m_w = 18000 * (m_w & 65535) + (m_w >> 16); +// return (m_z << 16) + m_w; +// } + +// uint32_t GetUniform(uint32_t max){ +// uint32_t u = GetUint(); +// return (u + 1.0) * 2.328306435454494e-10 * max; +// } + +// uint8_t memcheck(void *start, uint8_t value, uint32_t size){ +// uint8_t *p = (uint8_t *) start; +// uint32_t i; + +// for (i = 0; i < size; i++, p++) +// if (*p != value) +// return 0; + +// return 1; +// } diff --git a/Userland/SampleCodeModule/shell/include/phylo.h b/Userland/SampleCodeModule/shell/include/phylo.h index b867818..bbec5e4 100644 --- a/Userland/SampleCodeModule/shell/include/phylo.h +++ b/Userland/SampleCodeModule/shell/include/phylo.h @@ -1,33 +1,6 @@ #ifndef PHYLO_H #define PHYLO_H -#include -#include -#include "libc.h" -#include "system.h" -#include "phyloLib.h" - -#define ARGV_SIZE 2 -#define BUFF_SIZE 20 -#define MAX_PHILO_SIZE 3 -#define MAX_NAME_SIZE 10 -#define STARTING 5 - -int * state; -typedef enum states {EATING = 0, HUNGRY, THINKING} states; - -typedef struct philosopher_t { - int debug; - char ** argv; - char * buffer; - sem_t * sem; - int pid; - states state; - - struct philosopher_t * left; - struct philosopher_t * right; -} philosopher_t; - -void philosopher(int argc, char ** argv); +void phylo(int argc, char *argv[]); #endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/phyloLib.h b/Userland/SampleCodeModule/shell/include/phyloLib.h index 74d3b5b..84f2db3 100644 --- a/Userland/SampleCodeModule/shell/include/phyloLib.h +++ b/Userland/SampleCodeModule/shell/include/phyloLib.h @@ -1,6 +1,33 @@ #ifndef PHYLOLIB_H #define PHYLOLIB_H -void phylo(int argc, char *argv[]); +#include +#include +#include "libc.h" +#include "system.h" +#include "phylo.h" + +#define ARGV_SIZE 2 +#define BUFF_SIZE 20 +#define MAX_PHILO_SIZE 3 +#define MAX_NAME_SIZE 10 +#define STARTING 5 + +int * state; +typedef enum states {EATING = 0, HUNGRY, THINKING} states; + +typedef struct philosopher_t { + int debug; + char ** argv; + char * buffer; + sem_t * sem; + int pid; + states state; + + struct philosopher_t * left; + struct philosopher_t * right; +} philosopher_t; + +void philosopher(int argc, char ** argv); #endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/pipes.h b/Userland/SampleCodeModule/shell/include/pipes.h index 1dc2723..c703cc1 100644 --- a/Userland/SampleCodeModule/shell/include/pipes.h +++ b/Userland/SampleCodeModule/shell/include/pipes.h @@ -1,6 +1,9 @@ #ifndef PIPES_LIB #define PIPES_LIB -void pipes(int argc, char *argv[]); +#include "libc.h" +#include "shell.h" + +void pipe(int argc, char *argv[]); #endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/semCom.h b/Userland/SampleCodeModule/shell/include/sems.h similarity index 100% rename from Userland/SampleCodeModule/shell/include/semCom.h rename to Userland/SampleCodeModule/shell/include/sems.h diff --git a/Userland/SampleCodeModule/shell/include/shell.h b/Userland/SampleCodeModule/shell/include/shell.h index a05f30c..20166ef 100644 --- a/Userland/SampleCodeModule/shell/include/shell.h +++ b/Userland/SampleCodeModule/shell/include/shell.h @@ -1,5 +1,5 @@ -#ifndef SHELL -#define SHELL +#ifndef SHELL_H +#define SHELL_H #include "system.h" #include "libc.h" @@ -18,14 +18,18 @@ #include "wc.h" #include "filter.h" #include "cat.h" -#include "semCom.h" +#include "sems.h" #include "stddef.h" #include "nice.h" -#include "phyloLib.h" +#include "phylo.h" #include "kill.h" #include "block.h" #include "unblock.h" #include "loop.h" +#include "test_prio.h" +#include "test_sync.h" +#include "test_processes.h" +#include "test_mm.h" #define EXIT_FAILURE 1 #define EXIT_SUCCESS 0 diff --git a/Userland/SampleCodeModule/shell/include/test_mm.h b/Userland/SampleCodeModule/shell/include/test_mm.h new file mode 100644 index 0000000..f5f83d3 --- /dev/null +++ b/Userland/SampleCodeModule/shell/include/test_mm.h @@ -0,0 +1,14 @@ +#ifndef TESTMM_H +#define TESTMM_H + +#include "test_util.h" +#include "system.h" +#include +#include "libc.h" + +void test_mm(int argc, char *argv[]); + +#define MAX_BLOCKS 128 +#define MAX_MEMORY 0.8 * 1024 * 1024 * 512 //Should be around 80% of memory managed by the MM + +#endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/test_prio.h b/Userland/SampleCodeModule/shell/include/test_prio.h new file mode 100644 index 0000000..69476e0 --- /dev/null +++ b/Userland/SampleCodeModule/shell/include/test_prio.h @@ -0,0 +1,12 @@ +#ifndef TESTPRIO_H +#define TESTPRIO_H + +#include +#include +#include "libc.h" +#include "system.h" +#include "libc.h" + +void test_prio(int argc, char *argv[]); + +#endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/test_processes.h b/Userland/SampleCodeModule/shell/include/test_processes.h new file mode 100644 index 0000000..84e8f12 --- /dev/null +++ b/Userland/SampleCodeModule/shell/include/test_processes.h @@ -0,0 +1,13 @@ +#ifndef TESTPROC_H +#define TESTPROC_H + +#include +#include +#include "libc.h" +#include "system.h" +#include "test_util.h" +#include "libc.h" + +void test_processes(int argc, char *argv[]); + +#endif \ No newline at end of file diff --git a/Userland/SampleCodeModule/shell/include/test_sync.h b/Userland/SampleCodeModule/shell/include/test_sync.h new file mode 100644 index 0000000..872485b --- /dev/null +++ b/Userland/SampleCodeModule/shell/include/test_sync.h @@ -0,0 +1,12 @@ +#ifndef TESTSYNC_H +#define TESTSYNC_H + +#include +#include +#include "system.h" +#include "libc.h" + +void test_sync(int argc, char *argv[]); +void test_no_sync(int argc, char ** argv); + +#endif \ No newline at end of file diff --git a/Kernel/tests/include/test_util.h b/Userland/SampleCodeModule/shell/include/test_util.h similarity index 100% rename from Kernel/tests/include/test_util.h rename to Userland/SampleCodeModule/shell/include/test_util.h diff --git a/Userland/SampleCodeModule/shell/shell.c b/Userland/SampleCodeModule/shell/shell.c index 663c6e0..c513dfc 100644 --- a/Userland/SampleCodeModule/shell/shell.c +++ b/Userland/SampleCodeModule/shell/shell.c @@ -14,15 +14,19 @@ cmd_t commands[] = { {"cpufeatures", cpufeatures, 0, 1}, {"nice", nice, 0, 1}, {"ps", ps, 0, 1}, - {"pipes", pipes, 0, 1}, + {"pipe", pipe, 0, 1}, {"kill", kill, 1, 1}, {"sem", sem, 0, 1}, {"quadratic", quadratic, 0, 1}, {"printmem", printmem, 0, 1}, {"phylo", phylo, 0, 1}, {"wc", wc, 0, 1}, - {"loop", loop, 0, 0}, - {"loopcaca", loop, 0, 1}, + {"loop", loop, 0, 1}, + {"test_mm", test_mm, 0, 1}, + {"test_prio", test_prio, 0, 1}, + {"test_processes", test_processes, 0, 1}, + {"test_sync", test_sync, 0, 1}, + {"test_no_sync", test_no_sync, 0, 1}, {NULL, NULL, 0, 0} }; @@ -112,6 +116,10 @@ void processInput(char *input) { if (comm_flag0 && pipe != -1) { for (int i = 0; commands[i].name != NULL; i++) { if (!strcmp(tokens[pipe + 1], commands[i].name)) { + if (commands->isBuiltIn && ampersand) { + printStringLen("built-in commands are not able to run in background\n", 53); + return; + } comm_flag1 = 1; comm1 = i; break; @@ -151,7 +159,7 @@ void shell(int argc, char *argv[]) { char buffer[SIZE] = {0}; while (scanfNoPrint(buffer) != 0) { - new_line(); + newline(); processInput(buffer); printStringLen("$> ", 3); } @@ -165,5 +173,5 @@ void incorrect_comm(char *buffer) { void incorrect_arg(char *command) { printStringLen("Incorrect arguments for command ", 33); printString(command); - new_line(); + newline(); } \ No newline at end of file diff --git a/flags.txt b/flags.txt index 4e9b0ab..898febc 100644 --- a/flags.txt +++ b/flags.txt @@ -1 +1 @@ -ke +be