bottleros/Userland/SampleCodeModule/shell/commands/phylo.c

200 lines
4.3 KiB
C

// This is a personal academic project. Dear PVS-Studio, please check it.
// PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
#include "phyloLib.h"
philosopher_t *firstPhil;
sem_t *mutex;
int philoCount = STARTING;
void printState() {
philosopher_t * phil = firstPhil;
do {
if (phil->state == EATING)
putChar('E');
else putChar('.');
phil = phil->right;
} while (phil != firstPhil);
newline();
}
void test(philosopher_t * phil)
{
if (phil->state == HUNGRY && phil->left->state != EATING && phil->right->state != EATING) {
phil->state = EATING;
sys_semPost(phil->sem);
}
}
void addPhilo() {
philoCount++;
philosopher_t * new = sys_malloc(sizeof(philosopher_t));
new->argv = sys_malloc(sizeof(char *) * ARGV_SIZE);
new->buffer = sys_malloc(sizeof(char) * BUFF_SIZE);
new->state = THINKING;
new->sem = sys_semOpen("filosofo", 0);
new->argv[0] = "filosofo";
strcpy(new->buffer, itoa((uint64_t) new, new->buffer, 10));
new->argv[1] = new->buffer;
new->left = firstPhil->left;
new->right = firstPhil;
firstPhil->left->right = new;
firstPhil->left = new;
new->pid = sys_loadProcess(philosopher, 0, 2, new->argv, NULL);
}
void removePhilo() {
philoCount--;
philosopher_t * del = firstPhil->left;
del->left->right = del->right;
del->right->left = del->left;
sys_semClose(del->sem);
sys_free(del->buffer);
sys_free(del->argv);
sys_free(del);
sys_kill(del->pid);
}
void take_fork(philosopher_t * phil) {
sys_semWait(mutex);
phil->state = HUNGRY;
test(phil);
sys_semPost(mutex);
sys_semWait(phil->sem);
}
void put_fork(philosopher_t * phil) {
sys_semWait(mutex);
phil->state = THINKING;
test(phil->left);
test(phil->right);
sys_semPost(mutex);
}
void philosopher(int argc, char ** argv) {
if (argc != 2) {
sys_exit();
}
philosopher_t * i = (philosopher_t *) ((uint64_t) atoi(argv[1], -1));
while (1) {
sys_sleep(1);
take_fork(i);
printState();
sys_sleep(1);
put_fork(i);
printState();
}
sys_exit();
}
void phylo(int argc, char ** argv) {
if (argc == 1)
philoCount = STARTING;
else if (argc == 2) {
philoCount = atoi(argv[1], -1);
if (philoCount < 1 || philoCount > MAX_PHILO_SIZE) {
printStringLen("amount of philosofers must be between 1 and 10\n", 48);
sys_exit();
}
}
else sys_exit();
mutex = sys_semOpen("Mutex", 1);
philosopher_t * left = NULL;
for (int i = 0; i < philoCount; i++) {
philosopher_t * phil = sys_malloc(sizeof(philosopher_t));
phil->argv = sys_malloc(sizeof(char *) * 2);
phil->buffer = sys_malloc(20);
phil->state = THINKING;
phil->sem = sys_semOpen("filosofo", 0);
phil->argv[0] = "filosofo";
strcpy(phil->buffer, itoa((uint64_t) phil, phil->buffer, 10));
phil->argv[1] = phil->buffer;
if (left != NULL) {
phil->left = left;
phil->left->right = phil;
}
if (firstPhil == NULL) {
firstPhil = phil;
}
left = phil;
}
firstPhil->left = left;
firstPhil->left->right = firstPhil;
philosopher_t * phil = firstPhil;
do {
phil->pid = sys_loadProcess(philosopher, 0, 2, phil->argv, NULL);
phil = phil->right;
} while (phil != firstPhil);
char c;
while ((c = getChar()) != 0 && c != -1) {
if (c == 'a') {
addPhilo();
}
else if (c == 'r') {
removePhilo();
}
else if (c == 'q') {
end();
}
// sys_sleep(1);
}
sys_exit();
}
void freePhilo(philosopher_t * phil) {
sys_semClose(phil->sem);
sys_free(phil);
sys_free(phil->buffer);
sys_free(phil->argv);
sys_kill(phil->pid);
}
void end() {
sys_semWait(mutex);
philosopher_t * phil = firstPhil->right;
do {
freePhilo(phil);
phil = phil->right;
} while (phil != firstPhil);
freePhilo(firstPhil);
sys_semPost(mutex);
sys_semClose(mutex);
sys_exit();
}
/*
* Inspirados en "Operating Systems - Design and Implementation" de Tanenbaum
* Sección 2.3, pág. 91
*/