/**
* @file db_manage.c
* Management of a DB file
*
* @remark Copyright 2002 OProfile authors
* @remark Read the file COPYING
*
* @author Philippe Elie
*/
#define _GNU_SOURCE
#include <stdlib.h>
#ifndef ANDROID
#include <sys/fcntl.h>
#else
#include <fcntl.h>
#endif
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <stdio.h>
#include "odb.h"
#include "op_string.h"
#include "op_libiberty.h"
static __inline odb_descr_t * odb_to_descr(odb_data_t * data)
{
return (odb_descr_t *)(((char*)data->base_memory) + data->sizeof_header);
}
static __inline odb_node_t * odb_to_node_base(odb_data_t * data)
{
return (odb_node_t *)(((char *)data->base_memory) + data->offset_node);
}
static __inline odb_index_t * odb_to_hash_base(odb_data_t * data)
{
return (odb_index_t *)(((char *)data->base_memory) +
data->offset_node +
(data->descr->size * sizeof(odb_node_t)));
}
/**
* return the number of bytes used by hash table, node table and header.
*/
static unsigned int tables_size(odb_data_t const * data, odb_node_nr_t node_nr)
{
size_t size;
size = node_nr * (sizeof(odb_index_t) * BUCKET_FACTOR);
size += node_nr * sizeof(odb_node_t);
size += data->offset_node;
return size;
}
int odb_grow_hashtable(odb_data_t * data)
{
unsigned int old_file_size;
unsigned int new_file_size;
unsigned int pos;
void * new_map;
old_file_size = tables_size(data, data->descr->size);
new_file_size = tables_size(data, data->descr->size * 2);
if (ftruncate(data->fd, new_file_size))
return 1;
new_map = mremap(data->base_memory,
old_file_size, new_file_size, MREMAP_MAYMOVE);
if (new_map == MAP_FAILED)
return 1;
data->base_memory = new_map;
data->descr = odb_to_descr(data);
data->descr->size *= 2;
data->node_base = odb_to_node_base(data);
data->hash_base = odb_to_hash_base(data);
data->hash_mask = (data->descr->size * BUCKET_FACTOR) - 1;
/* rebuild the hash table, node zero is never used. This works
* because layout of file is node table then hash table,
* sizeof(node) > sizeof(bucket) and when we grow table we
* double size ==> old hash table and new hash table can't
* overlap so on the new hash table is entirely in the new
* memory area (the grown part) and we know the new hash
* hash table is zeroed. That's why we don't need to zero init
* the new table */
/* OK: the above is not exact
* if BUCKET_FACTOR < sizeof(bd_node_t) / sizeof(bd_node_nr_t)
* all things are fine and we don't need to init the hash
* table because in this case the new hash table is completely
* inside the new growed part. Avoiding to touch this memory is
* useful.
*/
#if 0
for (pos = 0 ; pos < data->descr->size*BUCKET_FACTOR ; ++pos)
data->hash_base[pos] = 0;
#endif
for (pos = 1; pos < data->descr->current_size; ++pos) {
odb_node_t * node = &data->node_base[pos];
size_t index = odb_do_hash(data, node->key);
node->next = data->hash_base[index];
data->hash_base[index] = pos;
}
return 0;
}
void odb_init(odb_t * odb)
{
odb->data = NULL;
}
/* the default number of page, calculated to fit in 4096 bytes */
#define DEFAULT_NODE_NR(offset_node) 128
#define FILES_HASH_SIZE 512
static struct list_head files_hash[FILES_HASH_SIZE];
static void init_hash()
{
size_t i;
for (i = 0; i < FILES_HASH_SIZE; ++i)
list_init(&files_hash[i]);
}
static odb_data_t *
find_samples_data(size_t hash, char const * filename)
{
struct list_head * pos;
/* FIXME: maybe an initial init routine ? */
if (files_hash[0].next == NULL) {
init_hash();
return NULL;
}
list_for_each(pos, &files_hash[hash]) {
odb_data_t * entry = list_entry(pos, odb_data_t, list);
if (strcmp(entry->filename, filename) == 0)
return entry;
}
return NULL;
}
int odb_open(odb_t * odb, char const * filename, enum odb_rw rw,
size_t sizeof_header)
{
struct stat stat_buf;
odb_node_nr_t nr_node;
odb_data_t * data;
size_t hash;
int err = 0;
int flags = (rw == ODB_RDWR) ? (O_CREAT | O_RDWR) : O_RDONLY;
int mmflags = (rw == ODB_RDWR) ? (PROT_READ | PROT_WRITE) : PROT_READ;
hash = op_hash_string(filename) % FILES_HASH_SIZE;
data = find_samples_data(hash, filename);
if (data) {
odb->data = data;
data->ref_count++;
return 0;
}
data = xmalloc(sizeof(odb_data_t));
memset(data, '\0', sizeof(odb_data_t));
list_init(&data->list);
data->offset_node = sizeof_header + sizeof(odb_descr_t);
data->sizeof_header = sizeof_header;
data->ref_count = 1;
data->filename = xstrdup(filename);
data->fd = open(filename, flags, 0644);
if (data->fd < 0) {
err = errno;
goto out;
}
if (fstat(data->fd, &stat_buf)) {
err = errno;
goto fail;
}
if (stat_buf.st_size == 0) {
size_t file_size;
if (rw == ODB_RDONLY) {
err = EIO;
goto fail;
}
nr_node = DEFAULT_NODE_NR(data->offset_node);
file_size = tables_size(data, nr_node);
if (ftruncate(data->fd, file_size)) {
err = errno;
goto fail;
}
} else {
/* Calculate nr node allowing a sanity check later */
nr_node = (stat_buf.st_size - data->offset_node) /
((sizeof(odb_index_t) * BUCKET_FACTOR) + sizeof(odb_node_t));
}
data->base_memory = mmap(0, tables_size(data, nr_node), mmflags,
MAP_SHARED, data->fd, 0);
if (data->base_memory == MAP_FAILED) {
err = errno;
goto fail;
}
data->descr = odb_to_descr(data);
if (stat_buf.st_size == 0) {
data->descr->size = nr_node;
/* page zero is not used */
data->descr->current_size = 1;
} else {
/* file already exist, sanity check nr node */
if (nr_node != data->descr->size) {
err = EINVAL;
goto fail_unmap;
}
}
data->hash_base = odb_to_hash_base(data);
data->node_base = odb_to_node_base(data);
data->hash_mask = (data->descr->size * BUCKET_FACTOR) - 1;
list_add(&data->list, &files_hash[hash]);
odb->data = data;
out:
return err;
fail_unmap:
munmap(data->base_memory, tables_size(data, nr_node));
fail:
close(data->fd);
free(data->filename);
free(data);
odb->data = NULL;
goto out;
}
void odb_close(odb_t * odb)
{
odb_data_t * data = odb->data;
if (data) {
data->ref_count--;
if (data->ref_count == 0) {
size_t size = tables_size(data, data->descr->size);
list_del(&data->list);
munmap(data->base_memory, size);
if (data->fd >= 0)
close(data->fd);
free(data->filename);
free(data);
odb->data = NULL;
}
}
}
int odb_open_count(odb_t const * odb)
{
if (!odb->data)
return 0;
return odb->data->ref_count;
}
void * odb_get_data(odb_t * odb)
{
return odb->data->base_memory;
}
void odb_sync(odb_t const * odb)
{
odb_data_t * data = odb->data;
size_t size;
if (!data)
return;
size = tables_size(data, data->descr->size);
msync(data->base_memory, size, MS_ASYNC);
}