295 lines
6.8 KiB
C
295 lines
6.8 KiB
C
#include <stddef.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
typedef struct{
|
|
int capacity;
|
|
int arrsize;
|
|
char** value;
|
|
} nb_arr;
|
|
|
|
#define nb_append_da(nb_arr, ...) \
|
|
nb_append_va(nb_arr, \
|
|
((const char*[]){__VA_ARGS__}), \
|
|
(sizeof((const char*[]){__VA_ARGS__})/sizeof(const char*)))
|
|
|
|
typedef struct{
|
|
FILE *filep;
|
|
size_t filesize;
|
|
int chars;
|
|
char *buf;
|
|
} nb_file;
|
|
|
|
void nb_init(nb_arr *newarr, int initial_capacity); // obsolete
|
|
|
|
void nb_append(nb_arr *newarr, char *newval);
|
|
void nb_append_int(nb_arr *newarr, int myint); // will deprecate soon
|
|
void nb_append_float(nb_arr *newarr, float myfloat); // will deprecate soon
|
|
void nb_append_va(nb_arr *newarr, const char *items[], int count);
|
|
|
|
void nb_free(nb_arr *newarr);
|
|
|
|
|
|
char* nb_strdup(const char* s); // make this void that uses realloc later.
|
|
|
|
void nb_print(nb_arr *newarr);
|
|
void nb_print_info(nb_arr *newarr);
|
|
|
|
|
|
void nb_cmd(nb_arr *newarr);
|
|
|
|
// File utils
|
|
void nb_copy_file(char* old_file_name, char* new_file_name);
|
|
char* nb_read_file(char* file_name);
|
|
bool nb_did_file_change(char *filename);
|
|
bool nb_does_file_exist(char *filename);
|
|
|
|
//bool needs_rebuild(); // need to implement rename file first to .old or something like nob does
|
|
|
|
|
|
void nb_rebuild(int argc, char **argv);
|
|
|
|
|
|
|
|
|
|
#ifdef NB_IMPLEMENTATION // make sure to define this before using the header
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
/*
|
|
char* nb_strdup(const char* s) {
|
|
char* d = malloc(strlen(s) + 1);
|
|
if (d) strcpy(d, s);
|
|
return d;
|
|
}
|
|
*/
|
|
|
|
|
|
void nb_init(nb_arr *newarr, int initial_capacity){
|
|
newarr->value = (char**)malloc(sizeof(char*) * initial_capacity);
|
|
newarr->capacity = initial_capacity;
|
|
newarr->arrsize = 0;
|
|
}
|
|
|
|
|
|
// later increase cap by size of new sheiSSe
|
|
void nb_append(nb_arr *newarr, char *newval){
|
|
if (newarr->value == NULL){
|
|
newarr->capacity =16;
|
|
if ((newarr->capacity > 16) | (newarr->arrsize > newarr->capacity)) {
|
|
newarr->capacity *=2;
|
|
}
|
|
newarr->value = (char**)realloc(newarr->value, sizeof(char*) * newarr->capacity);
|
|
}
|
|
newarr->value[newarr->arrsize++] = strdup(newval);
|
|
}
|
|
|
|
void nb_append_int(nb_arr *newarr, int myint){
|
|
char buf[64];
|
|
sprintf(buf, "%d", myint);
|
|
nb_append(newarr, buf);
|
|
}
|
|
|
|
void nb_append_float(nb_arr *newarr, float myfloat){
|
|
char buf[64];
|
|
sprintf(buf, "%f", myfloat);
|
|
nb_append(newarr, buf);
|
|
}
|
|
|
|
void nb_print(nb_arr *newarr){
|
|
for (int i = 0; i < newarr->arrsize; i++){
|
|
printf("%s\n", newarr->value[i]);
|
|
}
|
|
}
|
|
|
|
void nb_print_info(nb_arr *newarr){
|
|
printf("[INFO] ");
|
|
for (int i = 0; i < newarr->arrsize; i++){
|
|
printf("%s", newarr->value[i]);
|
|
printf(" ");
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
void nb_free(nb_arr *newarr){
|
|
if (newarr->value != NULL){
|
|
for (int i=0; i < newarr->arrsize; i++){
|
|
free(newarr->value[i]);
|
|
newarr->value[i] = NULL;
|
|
}
|
|
free(newarr->value);
|
|
newarr->value = NULL;
|
|
}
|
|
newarr -> capacity = 0;
|
|
newarr -> arrsize = 0;
|
|
}
|
|
|
|
|
|
void nb_cmd(nb_arr *newarr) {
|
|
if (newarr->arrsize < 1) {
|
|
printf("USAGE: provide more parameters\n");
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
// Allocate bufferchar
|
|
char* cmd = (char*) malloc(sizeof(char*) * newarr->capacity);
|
|
|
|
cmd[0] = '\0';
|
|
for (int i = 0; i < newarr->arrsize; i++) {
|
|
strcat(cmd, newarr->value[i]);
|
|
if (i < newarr->arrsize - 1) {
|
|
strcat(cmd, " ");
|
|
}
|
|
}
|
|
|
|
printf("[CMD] %s\n", cmd);
|
|
|
|
if (system(cmd) == -1) {
|
|
perror("system");
|
|
}
|
|
|
|
free(cmd);
|
|
for (int i=0; i < newarr->arrsize; ++i){
|
|
nb_free(newarr);
|
|
}
|
|
}
|
|
|
|
|
|
// compile func that requires c_file to run otherwise returns error like <please return usage>
|
|
void nb_com(nb_arr *newarr){
|
|
char* cmd = (char*)malloc(sizeof(char*) *newarr->capacity);
|
|
for (int i=0; i < newarr->arrsize; i++){
|
|
|
|
strcat(cmd, strcat(newarr->value[i]," "));
|
|
}
|
|
system(cmd);
|
|
}
|
|
|
|
|
|
// void append_c_file(FILE *filepointer){
|
|
// filepointer = NULL;
|
|
// }
|
|
|
|
void nb_copy_file(char* old_file_name, char* new_file_name){ // old name shouldnt be nobuild.c. it should be the name of the current file.
|
|
nb_file old_file;
|
|
nb_file new_file;
|
|
|
|
old_file.filep = fopen(old_file_name, "rb");
|
|
fseek(old_file.filep, 0, SEEK_END);
|
|
|
|
old_file.filesize = ftell(old_file.filep);
|
|
old_file.buf = (char*)malloc(old_file.filesize);
|
|
fseek(old_file.filep, 0, SEEK_SET);
|
|
fread(old_file.buf, 1, old_file.filesize, old_file.filep);
|
|
fclose(old_file.filep);
|
|
|
|
new_file.filep = fopen(new_file_name, "wb");
|
|
fwrite(old_file.buf, 1, old_file.filesize, new_file.filep);
|
|
fclose(new_file.filep);
|
|
}
|
|
|
|
bool nb_did_file_change(char *filename){
|
|
struct stat file_old;
|
|
stat(filename, &file_old);
|
|
|
|
struct stat file_new;
|
|
char buf[64];
|
|
sprintf(buf, "%s.old", filename);
|
|
stat(buf, &file_new);
|
|
|
|
return difftime(file_old.st_mtime, file_new.st_mtime) > 0;
|
|
}
|
|
|
|
|
|
bool nb_does_file_exist(char *filename){
|
|
if (access(filename, F_OK) == 0){
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void nb_rebuild(int argc, char **argv){
|
|
char *filename = "builder.c";
|
|
char cloned_file[128];
|
|
sprintf(cloned_file, "%s.old", filename);
|
|
|
|
if (nb_does_file_exist(cloned_file)){
|
|
// printf("%s does exist\n", cloned_file);
|
|
if (nb_did_file_change(filename)){
|
|
printf("[Rebuilding]\n");
|
|
nb_copy_file(filename, cloned_file);
|
|
|
|
nb_arr cmd;
|
|
char fname[128];
|
|
|
|
nb_init(&cmd, sizeof(fname)*2);
|
|
strncpy(fname, filename, sizeof(fname));
|
|
fname[sizeof(fname)-1] = '\0';
|
|
char *dot = strrchr(fname, '.');
|
|
if (dot != NULL) {
|
|
*dot = '\0';
|
|
}
|
|
nb_append(&cmd, "gcc");
|
|
nb_append(&cmd, "-o");
|
|
nb_append(&cmd, fname);
|
|
nb_append(&cmd, filename);
|
|
// nb_print_info(&cmd);
|
|
nb_cmd(&cmd);
|
|
|
|
printf("[INFO] rebuilt %s\n", filename);
|
|
nb_free(&cmd);
|
|
// printf("[INFO] %s", argv)
|
|
|
|
printf("\n");
|
|
|
|
for (int i=0; i<argc; ++i){
|
|
nb_append_da(&cmd, argv[i]);
|
|
}
|
|
nb_cmd(&cmd);
|
|
exit(1);
|
|
|
|
} else {
|
|
// printf("file did not change\n");
|
|
}
|
|
}else{
|
|
// printf("created %s.old\n", filename);
|
|
nb_copy_file(filename, cloned_file);
|
|
}
|
|
}
|
|
|
|
char* nb_read_file(char* file_name){ // old name shouldnt be nobuild.c. it should be the name of the current file. I should think more about adding error handling
|
|
nb_file file;
|
|
|
|
file.filep = fopen(file_name, "rb");
|
|
fseek(file.filep, 0, SEEK_END);
|
|
|
|
file.filesize = ftell(file.filep);
|
|
file.buf = (char*)malloc(file.filesize+1);
|
|
fseek(file.filep, 0, SEEK_SET);
|
|
fread(file.buf, 1, file.filesize, file.filep);
|
|
fclose(file.filep);
|
|
file.buf[file.filesize] = '\0';
|
|
return file.buf;
|
|
}
|
|
|
|
void nb_append_va(nb_arr *newarr, const char *items[], int count) {
|
|
for (int i = 0; i < count; i++) {
|
|
nb_append(newarr, (char*)items[i]);
|
|
}
|
|
}
|
|
|
|
#endif //NB_IMPLEMENTATION
|
|
|