Commit a5ac0663 authored by Yoni Fogel's avatar Yoni Fogel

End of day, complete with comments

git-svn-id: file:///svn/tokudb@395 c7de825b-a66e-492c-adef-691d508d4ae1
parent 50bd83f2
...@@ -16,7 +16,7 @@ UTILS= \ ...@@ -16,7 +16,7 @@ UTILS= \
all: $(UTILS) all: $(UTILS)
test: test_gen test: test_gen test_load
test_gen: test_gen_hex test_gen: test_gen_hex
...@@ -41,7 +41,9 @@ test_gen_hex: ...@@ -41,7 +41,9 @@ test_gen_hex:
./tokudb_gen -Fh >> $@.gen_sorted.temp ./tokudb_gen -Fh >> $@.gen_sorted.temp
if ! diff -q $@.load_dump.temp $@.gen_sorted.temp; then echo "Test Failed!"; exit 1; fi if ! diff -q $@.load_dump.temp $@.gen_sorted.temp; then echo "Test Failed!"; exit 1; fi
rm $@.*.temp rm $@.*.temp
test_load: test_load_hex
#Generating
#if diff -q <(echo "foo") <(echo "foo") > /dev/null; then echo yes; else echo no; fi #if diff -q <(echo "foo") <(echo "foo") > /dev/null; then echo yes; else echo no; fi
clean: clean:
......
...@@ -15,6 +15,7 @@ typedef uint8_t bool; ...@@ -15,6 +15,7 @@ typedef uint8_t bool;
#define REMOVE_BITS(bitvector, bits) ((bitvector) &= ~(bits)) #define REMOVE_BITS(bitvector, bits) ((bitvector) &= ~(bits))
#define IS_SET_ANY(bitvector, bits) ((bitvector) & (bits)) #define IS_SET_ANY(bitvector, bits) ((bitvector) & (bits))
#define IS_SET_ALL(bitvector, bits) (((bitvector) & (bits)) == (bits)) #define IS_SET_ALL(bitvector, bits) (((bitvector) & (bits)) == (bits))
#define IS_POWER_OF_2(num) ((num) > 0 && ((num) & ((num) - 1))) #define IS_POWER_OF_2(num) ((num) > 0 && ((num) & ((num) - 1)))
int strtoint32 (DB* db, char* progname, char* str, int32_t* num, int32_t min, int32_t max, int base); int strtoint32 (DB* db, char* progname, char* str, int32_t* num, int32_t min, int32_t max, int base);
...@@ -23,15 +24,16 @@ int strtoint64 (DB* db, char* progname, char* str, int64_t* num, int64_t mi ...@@ -23,15 +24,16 @@ int strtoint64 (DB* db, char* progname, char* str, int64_t* num, int64_t mi
int strtouint64 (DB* db, char* progname, char* str, uint64_t* num, uint64_t min, uint64_t max, int base); int strtouint64 (DB* db, char* progname, char* str, uint64_t* num, uint64_t min, uint64_t max, int base);
/* /*
* Convert a string to an "type". Uses base 10. * Convert a string to an integer of type "type".
* Allow range of [min, max].
* *
* *
* Sets errno and returns: * Sets errno and returns:
* EINVAL: str == NULL, num == NULL, or string not of the form [ ]+[+-]?[0-9]+ * EINVAL: str == NULL, num == NULL, or string not of the form [ \t]*[+-]?[0-9]+
* ERANGE: value out of range specified. * ERANGE: value out of range specified. (Range of [min, max])
* *
* *num is unchanged on error. * *num is unchanged on error.
* Returns:
*
*/ */
#define DEF_STR_TO(name, type, bigtype, strtofunc, frmt) \ #define DEF_STR_TO(name, type, bigtype, strtofunc, frmt) \
int name(DB* db, char* progname, char* str, type* num, type min, type max, int base) \ int name(DB* db, char* progname, char* str, type* num, type min, type max, int base) \
...@@ -46,6 +48,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b ...@@ -46,6 +48,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b
assert(base == 0 || (base >= 2 && base <= 36)); \ assert(base == 0 || (base >= 2 && base <= 36)); \
\ \
errno = 0; \ errno = 0; \
while (isspace(*str)) str++; \
value = strtofunc(str, &test, base); \ value = strtofunc(str, &test, base); \
if ((*test != '\0' && *test != '\n') || test == str) { \ if ((*test != '\0' && *test != '\n') || test == str) { \
if (db == NULL) fprintf(stderr, "%s: %s: Invalid numeric argument\n", progname, str); \ if (db == NULL) fprintf(stderr, "%s: %s: Invalid numeric argument\n", progname, str); \
...@@ -53,7 +56,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b ...@@ -53,7 +56,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b
errno = EINVAL; \ errno = EINVAL; \
goto error; \ goto error; \
} \ } \
if (errno == ERANGE) { \ if (errno != 0) { \
if (db == NULL) fprintf(stderr, "%s: %s: %s\n", progname, str, strerror(errno)); \ if (db == NULL) fprintf(stderr, "%s: %s: %s\n", progname, str, strerror(errno)); \
else db->err(db, errno, "%s", str); \ else db->err(db, errno, "%s", str); \
goto error; \ goto error; \
...@@ -69,7 +72,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b ...@@ -69,7 +72,7 @@ int name(DB* db, char* progname, char* str, type* num, type min, type max, int b
goto error; \ goto error; \
} \ } \
*num = value; \ *num = value; \
return 0; \ return EXIT_SUCCESS; \
error: \ error: \
return errno; \ return errno; \
} }
......
...@@ -43,7 +43,6 @@ load_globals g; ...@@ -43,7 +43,6 @@ load_globals g;
int usage (); int usage ();
int longusage (); int longusage ();
int load_database (); int load_database ();
int create_init_env(); int create_init_env();
int read_header (); int read_header ();
int open_database (); int open_database ();
...@@ -62,14 +61,13 @@ int main(int argc, char *argv[]) { ...@@ -62,14 +61,13 @@ int main(int argc, char *argv[]) {
{ "plain_text", no_argument, NULL, 'T' }, { "plain_text", no_argument, NULL, 'T' },
{ "Version", no_argument, NULL, 'V' }, { "Version", no_argument, NULL, 'V' },
{ "config", required_argument, NULL, 'c' }, { "config", required_argument, NULL, 'c' },
{ "file", required_argument, NULL, 'f' }, { "input_file", required_argument, NULL, 'f' },
{ "home", required_argument, NULL, 'h' }, { "home", required_argument, NULL, 'h' },
{ "password", required_argument, NULL, 'p' }, { "password", required_argument, NULL, 'p' },
{ "type", required_argument, NULL, 't' }, { "type", required_argument, NULL, 't' },
{ NULL, 0, NULL, 0 } { NULL, 0, NULL, 0 }
}; };
char** next_config_option; char** next_config_option;
DB_ENV* dbenv;
/* Set up the globals. */ /* Set up the globals. */
memset(&g, 0, sizeof(g)); memset(&g, 0, sizeof(g));
...@@ -153,18 +151,22 @@ int main(int argc, char *argv[]) { ...@@ -153,18 +151,22 @@ int main(int argc, char *argv[]) {
//TODO: /* Handle possible interruptions/signals. */ //TODO: /* Handle possible interruptions/signals. */
g.database = argv[0]; g.database = argv[0];
//TODO: flockfile(stdin);
if (!create_init_env()) { if (create_init_env() != 0) goto error;
while (!g.eof) { while (!g.eof) {
if (!load_database()) goto cleanup; //BOOKMARK
} if (!load_database()) goto errorcleanup;
}
if (false) {
errorcleanup:
g.exitcode = EXIT_FAILURE;
} }
cleanup: cleanup:
if ((retval = dbenv->close(dbenv, 0)) != 0) { if ((retval = g.dbenv->close(g.dbenv, 0)) != 0) {
g.exitcode = EXIT_FAILURE; g.exitcode = EXIT_FAILURE;
fprintf(stderr, "%s: dbenv->close: %s\n", g.progname, db_strerror(retval)); fprintf(stderr, "%s: dbenv->close: %s\n", g.progname, db_strerror(retval));
} }
//TODO: funlockfile(stdin);
//TODO: /* Resend any caught signal. */ //TODO: /* Resend any caught signal. */
free(g.config_options); free(g.config_options);
...@@ -173,6 +175,7 @@ int main(int argc, char *argv[]) { ...@@ -173,6 +175,7 @@ int main(int argc, char *argv[]) {
error: error:
fprintf(stderr, "%s: Quitting out due to errors.\n", g.progname); fprintf(stderr, "%s: Quitting out due to errors.\n", g.progname);
return EXIT_FAILURE; return EXIT_FAILURE;
//TODO:free all malloced memory (may require static stuff turning global)
} }
int load_database() int load_database()
...@@ -185,15 +188,16 @@ int load_database() ...@@ -185,15 +188,16 @@ int load_database()
retval = db_create(&g.db, g.dbenv, 0); retval = db_create(&g.db, g.dbenv, 0);
if (retval != 0) { if (retval != 0) {
dbenv->err(dbenv, retval, "db_create"); dbenv->err(dbenv, retval, "db_create");
goto cleanup; return EXIT_FAILURE;
} }
db = g.db; db = g.db;
if (g.header && read_header()) goto error; if (g.header && read_header() != 0) goto error;
if (g.eof) goto cleanup; if (g.eof) goto cleanup;
if (apply_commandline_options()) goto error; if (apply_commandline_options() != 0) goto error;
if (g.eof) goto cleanup; if (g.eof) goto cleanup;
//TODO: Only quit out if DB does NOT EXIST.
if (g.dbtype == DB_UNKNOWN) { if (g.dbtype == DB_UNKNOWN) {
dbenv->errx(dbenv, "no database type specified"); dbenv->errx(dbenv, "no database type specified");
goto error; goto error;
...@@ -205,9 +209,9 @@ int load_database() ...@@ -205,9 +209,9 @@ int load_database()
goto error; goto error;
} }
*/ */
if (open_database()) goto error; if (open_database() != 0) goto error;
if (g.eof) goto cleanup; if (g.eof) goto cleanup;
if (read_keys()) goto error; if (read_keys() != 0) goto error;
if (g.eof) goto cleanup; if (g.eof) goto cleanup;
if (false) { if (false) {
...@@ -215,7 +219,7 @@ int load_database() ...@@ -215,7 +219,7 @@ int load_database()
g.exitcode = EXIT_FAILURE; g.exitcode = EXIT_FAILURE;
} }
cleanup: cleanup:
if (close_database()) g.exitcode = EXIT_FAILURE; if (close_database() != 0) g.exitcode = EXIT_FAILURE;
return g.exitcode; return g.exitcode;
} }
...@@ -244,6 +248,7 @@ int create_init_env() ...@@ -244,6 +248,7 @@ int create_init_env()
int retval; int retval;
DB_ENV* dbenv; DB_ENV* dbenv;
int flags; int flags;
//TODO: Experiments to determine right cache size for tokudb, or maybe command line argument.
int cache = 1 << 20; /* 1 megabyte */ int cache = 1 << 20; /* 1 megabyte */
retval = db_env_create(&dbenv, 0); retval = db_env_create(&dbenv, 0);
...@@ -263,21 +268,25 @@ int create_init_env() ...@@ -263,21 +268,25 @@ int create_init_env()
/* Open the dbenvironment. */ /* Open the dbenvironment. */
g.is_private = false; g.is_private = false;
flags = DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_MPOOL | DB_INIT_TXN | DB_USE_ENVIRON; flags = DB_JOINENV | DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_MPOOL | DB_USE_ENVIRON;
//TODO: Transactions.. SET_BITS(flags, DB_INIT_TXN);
if (!dbenv->open(dbenv, g.homedir, flags, 0)) goto success; if (!dbenv->open(dbenv, g.homedir, flags, 0)) goto success;
retval = dbenv->set_cachesize(dbenv, 0, cache, 1); retval = dbenv->set_cachesize(dbenv, 0, cache, 1);
if (retval) { if (retval) {
dbenv->err(dbenv, retval, "set_cachesize"); dbenv->err(dbenv, retval, "DB_ENV->set_cachesize");
goto error; goto error;
} }
g.is_private = true; g.is_private = true;
//TODO: Do we want to support transactions/logging even in single-process mode?
//Maybe if the db already exists.
//If db does not exist.. makes sense not to log or have transactions
REMOVE_BITS(flags, DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_TXN); REMOVE_BITS(flags, DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_TXN);
SET_BITS(flags, DB_CREATE | DB_PRIVATE); SET_BITS(flags, DB_CREATE | DB_PRIVATE);
retval = dbenv->open(dbenv, g.homedir, flags, 0); retval = dbenv->open(dbenv, g.homedir, flags, 0);
if (retval) { if (retval) {
dbenv->err(dbenv, retval, "DB_dbenv->open"); dbenv->err(dbenv, retval, "DB_ENV->open");
goto error; goto error;
} }
success: success:
...@@ -425,6 +434,7 @@ int read_header() ...@@ -425,6 +434,7 @@ int read_header()
while (true) { while (true) {
if ((ch = getchar()) == EOF) { if ((ch = getchar()) == EOF) {
g.eof = true; g.eof = true;
//TODO: Check ferror for every EOF test.
if (ferror(stdin)) goto formaterror; if (ferror(stdin)) goto formaterror;
break; break;
} }
...@@ -471,7 +481,7 @@ int read_header() ...@@ -471,7 +481,7 @@ int read_header()
continue; continue;
} }
if (!strcmp(value, "hash") || strcmp(value, "recno") || strcmp(value, "queue")) { if (!strcmp(value, "hash") || strcmp(value, "recno") || strcmp(value, "queue")) {
fprintf(stderr, "%s: db type %s not supported.\n", g.progname, value); db->errx(db, "db type %s not supported.\n", value);
goto error; goto error;
} }
db->errx(db, "line %lu: unknown type %s", g.linenumber, value); db->errx(db, "line %lu: unknown type %s", g.linenumber, value);
...@@ -479,6 +489,7 @@ int read_header() ...@@ -479,6 +489,7 @@ int read_header()
} }
if (!strcmp(field, "database") || !strcmp(field, "subdatabase")) { if (!strcmp(field, "database") || !strcmp(field, "subdatabase")) {
if (g.subdatabase != NULL) { if (g.subdatabase != NULL) {
//TODO: Free before quitting main., clear at start too.. could be a new db without a name?
free(g.subdatabase); free(g.subdatabase);
g.subdatabase = NULL; g.subdatabase = NULL;
} }
...@@ -636,7 +647,7 @@ int open_database() ...@@ -636,7 +647,7 @@ int open_database()
int open_flags = DB_CREATE; int open_flags = DB_CREATE;
//TODO: Transaction auto commit stuff //TODO: Transaction auto commit stuff
//if (TXN_ON(dbenv)) SET_BITS(open_flags, DB_AUTO_COMMIT); //if (TXN_ON(dbenv)) SET_BITS(open_flags, DB_AUTO_COMMIT);
//TODO: First see if it exists.. THEN create it?
retval = db->open(db, NULL, g.database, g.subdatabase, g.dbtype, open_flags, 0666); retval = db->open(db, NULL, g.database, g.subdatabase, g.dbtype, open_flags, 0666);
if (retval != 0) { if (retval != 0) {
db->err(db, retval, "DB->open: %s", g.database); db->err(db, retval, "DB->open: %s", g.database);
...@@ -715,6 +726,7 @@ int get_dbt(DBT* pdbt) ...@@ -715,6 +726,7 @@ int get_dbt(DBT* pdbt)
/* *pdbt should have been memset to 0 before being called. */ /* *pdbt should have been memset to 0 before being called. */
which = 1 - which; which = 1 - which;
if (data[which] == NULL) data[which] = (char*)malloc(datasize[which] * sizeof(char)); if (data[which] == NULL) data[which] = (char*)malloc(datasize[which] * sizeof(char));
//TODO: Test for ENOMEM/error here.
datum = data[which]; datum = data[which];
if (g.plaintext) { if (g.plaintext) {
...@@ -942,9 +954,3 @@ int close_database() ...@@ -942,9 +954,3 @@ int close_database()
error: error:
return EXIT_FAILURE; return EXIT_FAILURE;
} }
void read_footer(bool plaintext)
{
printf("TODO: Implement %s:%s:%d\n", __FILE__, __FUNCTION__, __LINE__);
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment