| 1 |
#include <stdio.h> |
| 2 |
#include <stdlib.h> |
| 3 |
#include <stdbool.h> |
| 4 |
#include <string.h> |
| 5 |
#include <ctype.h> |
| 6 |
|
| 7 |
#include "inifile.h" |
| 8 |
|
| 9 |
char* inifile_cleanstr(char* str) |
| 10 |
{ |
| 11 |
int i; |
| 12 |
for (i = strlen(str) - 1; i >= 0; i --) |
| 13 |
if (isspace(str[i])) |
| 14 |
str[i] = '\0'; |
| 15 |
else |
| 16 |
break; |
| 17 |
|
| 18 |
while (isspace(*str)) |
| 19 |
str++; |
| 20 |
|
| 21 |
return str; |
| 22 |
} |
| 23 |
|
| 24 |
int64_t inifile_parseint(const char* str, bool issigned) |
| 25 |
{ |
| 26 |
int64_t ret = 0; |
| 27 |
bool neg = false; |
| 28 |
if (str[0] == '0' && str[1] == 'x') |
| 29 |
{ |
| 30 |
int i; |
| 31 |
if (str[2] == '\0') |
| 32 |
return 0x100000000LL; |
| 33 |
|
| 34 |
for (i = 0, str += 2; *str; i++, str++) |
| 35 |
{ |
| 36 |
if (i == 8) |
| 37 |
return 0x100000000LL; |
| 38 |
|
| 39 |
ret <<= 4; |
| 40 |
if (*str >= '0' && *str <= '9') |
| 41 |
ret |= *str - '0'; |
| 42 |
else if (*str >= 'a' && *str <= 'f') |
| 43 |
ret |= *str - 'a' + 10; |
| 44 |
else if (*str >= 'A' && *str <= 'F') |
| 45 |
ret |= *str - 'A' + 10; |
| 46 |
else |
| 47 |
return 0x100000000LL; |
| 48 |
} |
| 49 |
return ret; |
| 50 |
} |
| 51 |
else if ((*str >= '0' && *str <= '9') || (neg = (*str == '-'))) |
| 52 |
{ |
| 53 |
int i; |
| 54 |
if (neg) |
| 55 |
str++; |
| 56 |
for (i = 0; *str; i++, str++) |
| 57 |
{ |
| 58 |
if (i == 10) |
| 59 |
return 0x100000000LL; |
| 60 |
else if (i == 9 && !issigned && (ret > 429496729LL || (ret == 429496729LL && *str > '5'))) |
| 61 |
return 0x100000000LL; |
| 62 |
else if (i == 9 && issigned && (ret > 214748364LL || (ret == 214748364LL && *str > (neg ? '8' : '7')))) |
| 63 |
return 0x100000000LL; |
| 64 |
|
| 65 |
ret *= 10; |
| 66 |
if (*str >= '0' && *str <= '9') |
| 67 |
ret += *str - '0'; |
| 68 |
else |
| 69 |
return 0x100000000LL; |
| 70 |
} |
| 71 |
if (neg) |
| 72 |
ret *= -1; |
| 73 |
return ret; |
| 74 |
} |
| 75 |
else |
| 76 |
return 0x100000000LL; |
| 77 |
} |
| 78 |
|
| 79 |
bool inifile_read(const char* filename, inifile_callback callback) |
| 80 |
{ |
| 81 |
FILE* fp = fopen(filename, "r"); |
| 82 |
char* inisection = ""; |
| 83 |
char readbuf[4096] = ""; |
| 84 |
char* readptr; |
| 85 |
bool success = true; |
| 86 |
bool newsection = false; |
| 87 |
|
| 88 |
if (!fp) |
| 89 |
return inifile_cantread; |
| 90 |
|
| 91 |
while ((readptr = fgets(readbuf, sizeof(readbuf), fp))) // Loop through each line. |
| 92 |
{ |
| 93 |
while (isspace(readptr[0])) // Skip whitespace. |
| 94 |
readptr++; |
| 95 |
|
| 96 |
if (readptr[0] == '\0' || readptr[0] == '#' || readptr[0] == '!') // Skip empty lines and comments. |
| 97 |
continue; |
| 98 |
else if (readptr[0] == '[' && readptr[1] != ']') // It's a section header. |
| 99 |
{ |
| 100 |
int i; |
| 101 |
for (i = 2; readptr[i]; i ++) // Look for the ] |
| 102 |
if (readptr[i] == ']') |
| 103 |
break; |
| 104 |
|
| 105 |
if (readptr[i]) // Replace with a null or crash with error. |
| 106 |
readptr[i] = '\0'; |
| 107 |
else |
| 108 |
{ |
| 109 |
success = false; |
| 110 |
break; |
| 111 |
} |
| 112 |
|
| 113 |
if (inisection[0]) |
| 114 |
free(inisection); |
| 115 |
inisection = strdup(readptr + 1); // Skip the first [ |
| 116 |
newsection = true; |
| 117 |
} |
| 118 |
else // It's a value. |
| 119 |
{ |
| 120 |
int i; |
| 121 |
int equals = 0; |
| 122 |
for (i = 0; readptr[i]; i ++) // Find the = |
| 123 |
if (readptr[i] == '=') |
| 124 |
equals = i; |
| 125 |
|
| 126 |
if (readptr[i - 1] == '\n') |
| 127 |
readptr[i - 1] = '\0'; // Remove the trailing newline. |
| 128 |
|
| 129 |
if (equals) |
| 130 |
{ |
| 131 |
readptr[equals] = '\0'; |
| 132 |
if (!callback(inisection, newsection, readptr, readptr + equals + 1)) // If the callback is false, exit. |
| 133 |
break; |
| 134 |
newsection = false; |
| 135 |
} |
| 136 |
else // If there's no equals, crash with error. |
| 137 |
{ |
| 138 |
success = false; |
| 139 |
break; |
| 140 |
} |
| 141 |
} |
| 142 |
} |
| 143 |
|
| 144 |
if (inisection[0]) |
| 145 |
free(inisection); |
| 146 |
|
| 147 |
fclose(fp); |
| 148 |
return success; |
| 149 |
} |