2022-03-27 14:41:24 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
2022-04-01 15:41:54 +02:00
|
|
|
#include "struct/neuron.h"
|
2022-03-27 14:41:24 +02:00
|
|
|
#define MAGIC_NUMBER 2023
|
|
|
|
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-04-01 15:41:54 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
Neuron* read_neuron(uint32_t nb_weights, FILE *ptr) {
|
|
|
|
Neuron* neuron = malloc(sizeof(Neuron));
|
2022-03-28 16:52:17 +02:00
|
|
|
float activation;
|
2022-04-25 14:39:45 +02:00
|
|
|
float bias;
|
2022-03-28 18:03:10 +02:00
|
|
|
float tmp;
|
2022-03-28 16:52:17 +02:00
|
|
|
|
|
|
|
fread(&activation, sizeof(float), 1, ptr);
|
2022-04-25 14:39:45 +02:00
|
|
|
fread(&bias, sizeof(float), 1, ptr);
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
neuron->bias = bias;
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
neuron->z = 0.0;
|
|
|
|
neuron->last_back_bias = 0.0;
|
|
|
|
neuron->back_bias = 0.0;
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
float* weights = malloc(sizeof(float)*nb_weights);
|
2022-03-28 18:03:10 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
neuron->last_back_weights = malloc(sizeof(float)*nb_weights);
|
|
|
|
neuron->back_weights = malloc(sizeof(float)*nb_weights);
|
|
|
|
neuron->weights = weights;
|
2022-03-28 18:03:10 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
for (int i=0; i < nb_weights; i++) {
|
2022-03-28 18:03:10 +02:00
|
|
|
fread(&tmp, sizeof(float), 1, ptr);
|
2022-04-25 14:39:45 +02:00
|
|
|
neuron->weights[i] = tmp;
|
|
|
|
neuron->back_weights[i] = 0.0;
|
|
|
|
neuron->last_back_weights[i] = 0.0;
|
2022-03-28 16:52:17 +02:00
|
|
|
}
|
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
return neuron;
|
2022-03-28 16:52:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Lit une couche de neurones
|
2022-04-25 14:39:45 +02:00
|
|
|
Neuron** read_neurons(uint32_t nb_neurons, uint32_t nb_weights, FILE *ptr) {
|
|
|
|
Neuron** neurons = malloc(sizeof(Neuron*)*nb_neurons);
|
|
|
|
for (int i=0; i < nb_neurons; i++) {
|
|
|
|
neurons[i] = read_neuron(nb_weights, ptr);
|
2022-03-28 16:52:17 +02:00
|
|
|
}
|
2022-04-25 14:39:45 +02:00
|
|
|
return neurons;
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
|
|
|
|
2022-03-28 18:03:10 +02:00
|
|
|
|
2022-03-27 14:41:24 +02:00
|
|
|
// Charge l'entièreté du réseau neuronal depuis un fichier binaire
|
2022-04-25 14:39:45 +02:00
|
|
|
Network* read_network(char* filename) {
|
2022-03-27 14:41:24 +02:00
|
|
|
FILE *ptr;
|
2022-04-25 14:39:45 +02:00
|
|
|
Network* network = malloc(sizeof(Network));
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
ptr = fopen(filename, "rb");
|
|
|
|
|
|
|
|
uint32_t magic_number;
|
2022-04-25 14:39:45 +02:00
|
|
|
uint32_t nb_layers;
|
2022-03-28 18:03:10 +02:00
|
|
|
uint32_t tmp;
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
fread(&magic_number, sizeof(uint32_t), 1, ptr);
|
|
|
|
if (magic_number != MAGIC_NUMBER) {
|
|
|
|
printf("Incorrect magic number !\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
fread(&nb_layers, sizeof(uint32_t), 1, ptr);
|
|
|
|
network->nb_layers = nb_layers;
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
Layer** layers = malloc(sizeof(Layer*)*nb_layers);
|
|
|
|
uint32_t nb_neurons_layer[nb_layers+1];
|
2022-03-27 14:41:24 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
network->layers = layers;
|
2022-03-27 14:41:24 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
for (int i=0; i < nb_layers; i++) {
|
|
|
|
layers[i] = malloc(sizeof(Layer));
|
2022-03-28 18:03:10 +02:00
|
|
|
fread(&tmp, sizeof(tmp), 1, ptr);
|
2022-04-25 14:39:45 +02:00
|
|
|
layers[i]->nb_neurons = tmp;
|
|
|
|
nb_neurons_layer[i] = tmp;
|
2022-03-28 16:52:17 +02:00
|
|
|
}
|
2022-04-25 14:39:45 +02:00
|
|
|
nb_neurons_layer[nb_layers] = 0;
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
for (int i=0; i < nb_layers; i++) {
|
|
|
|
layers[i]->neurons = read_neurons(layers[i]->nb_neurons, nb_neurons_layer[i+1], ptr);
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fclose(ptr);
|
2022-04-25 14:39:45 +02:00
|
|
|
return network;
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
2022-03-28 16:52:17 +02:00
|
|
|
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
|
2022-03-28 18:03:10 +02:00
|
|
|
|
2022-03-27 14:41:24 +02:00
|
|
|
// Écrit un neurone dans le fichier pointé par *ptr
|
2022-04-25 14:39:45 +02:00
|
|
|
void ecrire_neuron(Neuron* neuron, int weights, FILE *ptr) {
|
|
|
|
float buffer[weights+2];
|
2022-03-27 14:41:24 +02:00
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
buffer[1] = neuron->bias;
|
|
|
|
for (int i=0; i < weights; i++) {
|
|
|
|
buffer[i+2] = neuron->weights[i];
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fwrite(buffer, sizeof(buffer), 1, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Stocke l'entièreté du réseau neuronal dans un fichier binaire
|
2022-04-25 14:39:45 +02:00
|
|
|
int write_network(char* filename, Network* network) {
|
2022-03-27 14:41:24 +02:00
|
|
|
FILE *ptr;
|
2022-04-25 14:39:45 +02:00
|
|
|
int nb_layers = network->nb_layers;
|
|
|
|
int nb_neurons[nb_layers+1];
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
ptr = fopen(filename, "wb");
|
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
uint32_t buffer[nb_layers+2];
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
buffer[0] = MAGIC_NUMBER;
|
2022-04-25 14:39:45 +02:00
|
|
|
buffer[1] = nb_layers;
|
|
|
|
for (int i=0; i < nb_layers; i++) {
|
|
|
|
buffer[i+2] = network->layers[i]->nb_neurons;
|
|
|
|
nb_neurons[i] = network->layers[i]->nb_neurons;
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
2022-04-25 14:39:45 +02:00
|
|
|
nb_neurons[nb_layers] = 0;
|
2022-03-27 14:41:24 +02:00
|
|
|
|
|
|
|
fwrite(buffer, sizeof(buffer), 1, ptr);
|
|
|
|
|
2022-04-25 14:39:45 +02:00
|
|
|
for (int i=0; i < nb_layers; i++) {
|
|
|
|
for (int j=0; j < nb_neurons[i]; j++) {
|
|
|
|
ecrire_neuron(network->layers[i]->neurons[j], nb_neurons[i+1], ptr);
|
2022-03-27 14:41:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(ptr);
|
|
|
|
return 1;
|
|
|
|
}
|