Use strbuf for reading configuration files

Use struct strbuf from Git instead of fixed-size buffers to remove the
limit on the length of configuration file lines and refactor
read_config_line() to improve readability.

Note that this also fixes a buffer overflow that existed with the
original fixed-size buffer implementation.

Signed-off-by: Lukas Fleischer <cgit@cryptocrack.de>
This commit is contained in:
Lukas Fleischer 2013-06-04 14:47:53 +00:00 committed by Jason A. Donenfeld
parent 820df9c660
commit 50e70d32f0
2 changed files with 35 additions and 31 deletions

View file

@ -31,45 +31,45 @@ static void skip_line(FILE *f)
;
}
static int read_config_line(FILE *f, char *line, const char **value, int bufsize)
static int read_config_line(FILE *f, struct strbuf *name, struct strbuf *value)
{
int i = 0, isname = 0;
int c = next_char(f);
*value = NULL;
while (i < bufsize - 1) {
int c = next_char(f);
if (!isname && (c == '#' || c == ';')) {
strbuf_reset(name);
strbuf_reset(value);
/* Skip comments and preceding spaces. */
for(;;) {
if (c == '#' || c == ';')
skip_line(f);
continue;
}
if (!isname && isspace(c))
continue;
if (c == '=' && !*value) {
line[i] = 0;
*value = &line[i + 1];
} else if (c == '\n' && !isname) {
i = 0;
continue;
} else if (c == '\n' || c == EOF) {
line[i] = 0;
else if (!isspace(c))
break;
} else {
line[i] = c;
}
isname = 1;
i++;
c = next_char(f);
}
line[i + 1] = 0;
return i;
/* Read variable name. */
while (c != '=') {
if (c == '\n' || c == EOF)
return 0;
strbuf_addch(name, c);
c = next_char(f);
}
/* Read variable value. */
c = next_char(f);
while (c != '\n' && c != EOF) {
strbuf_addch(value, c);
c = next_char(f);
}
return 1;
}
int parse_configfile(const char *filename, configfile_value_fn fn)
{
static int nesting;
int len;
char line[256];
const char *value;
struct strbuf name = STRBUF_INIT;
struct strbuf value = STRBUF_INIT;
FILE *f;
/* cancel deeply nested include-commands */
@ -78,10 +78,12 @@ int parse_configfile(const char *filename, configfile_value_fn fn)
if (!(f = fopen(filename, "r")))
return -1;
nesting++;
while ((len = read_config_line(f, line, &value, sizeof(line))) > 0)
fn(line, value);
while (read_config_line(f, &name, &value))
fn(name.buf, value.buf);
nesting--;
fclose(f);
strbuf_release(&name);
strbuf_release(&value);
return 0;
}

View file

@ -1,6 +1,8 @@
#ifndef CONFIGFILE_H
#define CONFIGFILE_H
#include "cgit.h"
typedef void (*configfile_value_fn)(const char *name, const char *value);
extern int parse_configfile(const char *filename, configfile_value_fn fn);