libs: introduce lmo - Lua Machine Objects, an implementation of binary hash tables
This commit is contained in:
parent
fb64c14609
commit
d9d3c71435
9 changed files with 872 additions and 0 deletions
46
libs/lmo/Makefile
Normal file
46
libs/lmo/Makefile
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
ifneq (,$(wildcard ../../build/config.mk))
|
||||||
|
include ../../build/config.mk
|
||||||
|
include ../../build/module.mk
|
||||||
|
include ../../build/gccconfig.mk
|
||||||
|
else
|
||||||
|
include standalone.mk
|
||||||
|
endif
|
||||||
|
|
||||||
|
LMO_LDFLAGS =
|
||||||
|
LMO_CFLAGS =
|
||||||
|
LMO_SO = lmo.so
|
||||||
|
LMO_PO2LMO = po2lmo
|
||||||
|
LMO_LOOKUP = lookup
|
||||||
|
LMO_COMMON_OBJ = src/lmo_core.o src/lmo_hash.o
|
||||||
|
LMO_PO2LMO_OBJ = src/lmo_po2lmo.o
|
||||||
|
LMO_LOOKUP_OBJ = src/lmo_lookup.o
|
||||||
|
LMO_LUALIB_OBJ = src/lmo_lualib.o
|
||||||
|
|
||||||
|
%.o: %.c
|
||||||
|
$(COMPILE) $(LMO_CFLAGS) $(LUA_CFLAGS) $(FPIC) -c -o $@ $<
|
||||||
|
|
||||||
|
compile: build-clean $(LMO_COMMON_OBJ) $(LMO_PO2LMO_OBJ) $(LMO_LOOKUP_OBJ) $(LMO_LUALIB_OBJ)
|
||||||
|
$(LINK) $(SHLIB_FLAGS) $(LMO_LDFLAGS) -o src/$(LMO_SO) \
|
||||||
|
$(LMO_COMMON_OBJ) $(LMO_LUALIB_OBJ)
|
||||||
|
$(LINK) $(LMO_LDFLAGS) -o src/$(LMO_PO2LMO) $(LMO_COMMON_OBJ) $(LMO_PO2LMO_OBJ)
|
||||||
|
$(LINK) $(LMO_LDFLAGS) -o src/$(LMO_LOOKUP) $(LMO_COMMON_OBJ) $(LMO_LOOKUP_OBJ)
|
||||||
|
mkdir -p dist$(LUA_LIBRARYDIR)
|
||||||
|
cp src/$(LMO_SO) dist$(LUA_LIBRARYDIR)/$(LMO_SO)
|
||||||
|
|
||||||
|
install: build
|
||||||
|
cp -pR dist$(LUA_LIBRARYDIR)/* $(LUA_LIBRARYDIR)
|
||||||
|
|
||||||
|
clean: build-clean
|
||||||
|
|
||||||
|
build-clean:
|
||||||
|
rm -f src/*.o src/lookup src/po2lmo src/lmo.so
|
||||||
|
|
||||||
|
host-compile: build-clean host-clean $(LMO_COMMON_OBJ) $(LMO_PO2LMO_OBJ)
|
||||||
|
$(LINK) $(LMO_LDFLAGS) -o src/$(LMO_PO2LMO) $(LMO_COMMON_OBJ) $(LMO_PO2LMO_OBJ)
|
||||||
|
|
||||||
|
host-install: host-compile
|
||||||
|
cp src/$(LMO_PO2LMO) ../../build/$(LMO_PO2LMO)
|
||||||
|
|
||||||
|
host-clean:
|
||||||
|
rm -f ../../build/$(LMO_PO2LMO)
|
||||||
|
|
72
libs/lmo/src/lmo.h
Normal file
72
libs/lmo/src/lmo.h
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - General header
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _LMO_H_
|
||||||
|
#define _LMO_H_
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
|
||||||
|
#if (defined(__GNUC__) && defined(__i386__))
|
||||||
|
#define sfh_get16(d) (*((const uint16_t *) (d)))
|
||||||
|
#else
|
||||||
|
#define sfh_get16(d) ((((uint32_t)(((const uint8_t *)(d))[1])) << 8)\
|
||||||
|
+(uint32_t)(((const uint8_t *)(d))[0]) )
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
struct lmo_entry {
|
||||||
|
uint32_t key_id;
|
||||||
|
uint32_t val_id;
|
||||||
|
uint32_t offset;
|
||||||
|
uint32_t length;
|
||||||
|
struct lmo_entry *next;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
typedef struct lmo_entry lmo_entry_t;
|
||||||
|
|
||||||
|
|
||||||
|
struct lmo_archive {
|
||||||
|
int fd;
|
||||||
|
uint32_t length;
|
||||||
|
lmo_entry_t *index;
|
||||||
|
char *mmap;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct lmo_archive lmo_archive_t;
|
||||||
|
|
||||||
|
|
||||||
|
uint32_t sfh_hash(const char * data, int len);
|
||||||
|
|
||||||
|
char _lmo_error[1024];
|
||||||
|
const char * lmo_error(void);
|
||||||
|
|
||||||
|
lmo_archive_t * lmo_open(const char *file);
|
||||||
|
int lmo_lookup(lmo_archive_t *ar, const char *key, char *dest, int len);
|
||||||
|
void lmo_close(lmo_archive_t *ar);
|
||||||
|
|
||||||
|
#endif
|
231
libs/lmo/src/lmo_core.c
Normal file
231
libs/lmo/src/lmo_core.c
Normal file
|
@ -0,0 +1,231 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - Base functions
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lmo.h"
|
||||||
|
|
||||||
|
extern char _lmo_error[1024];
|
||||||
|
|
||||||
|
static int lmo_read32( int fd, uint32_t *val )
|
||||||
|
{
|
||||||
|
uint8_t buffer[5];
|
||||||
|
|
||||||
|
if( read(fd, buffer, 4) < 4 )
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
buffer[4] = 0;
|
||||||
|
*val = ntohl(*((uint32_t *) buffer));
|
||||||
|
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char * error(const char *message, int add_errno)
|
||||||
|
{
|
||||||
|
memset(_lmo_error, 0, sizeof(_lmo_error));
|
||||||
|
|
||||||
|
if( add_errno )
|
||||||
|
snprintf(_lmo_error, sizeof(_lmo_error),
|
||||||
|
"%s: %s", message, strerror(errno));
|
||||||
|
else
|
||||||
|
snprintf(_lmo_error, sizeof(_lmo_error), "%s", message);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char * lmo_error(void)
|
||||||
|
{
|
||||||
|
return _lmo_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
lmo_archive_t * lmo_open(const char *file)
|
||||||
|
{
|
||||||
|
int in = -1;
|
||||||
|
uint32_t idx_offset = 0;
|
||||||
|
uint32_t i;
|
||||||
|
struct stat s;
|
||||||
|
|
||||||
|
lmo_archive_t *ar = NULL;
|
||||||
|
lmo_entry_t *head = NULL;
|
||||||
|
lmo_entry_t *entry = NULL;
|
||||||
|
|
||||||
|
if( stat(file, &s) == -1 )
|
||||||
|
{
|
||||||
|
error("Can not stat file", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( (in = open(file, O_RDONLY)) == -1 )
|
||||||
|
{
|
||||||
|
error("Can not open file", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( lseek(in, -sizeof(uint32_t), SEEK_END) == -1 )
|
||||||
|
{
|
||||||
|
error("Can not seek to eof", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( lmo_read32(in, &idx_offset) != 4 )
|
||||||
|
{
|
||||||
|
error("Unexpected EOF while reading index offset", 0);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( lseek(in, idx_offset, SEEK_SET) == -1 )
|
||||||
|
{
|
||||||
|
error("Can not seek to index offset", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( (ar = (lmo_archive_t *) malloc(sizeof(lmo_archive_t))) != NULL )
|
||||||
|
{
|
||||||
|
ar->fd = in;
|
||||||
|
ar->length = idx_offset;
|
||||||
|
|
||||||
|
for( i = idx_offset;
|
||||||
|
i < (s.st_size - sizeof(uint32_t));
|
||||||
|
i += (4 * sizeof(uint32_t))
|
||||||
|
) {
|
||||||
|
if( (entry = (lmo_entry_t *) malloc(sizeof(lmo_entry_t))) != NULL )
|
||||||
|
{
|
||||||
|
if( (lmo_read32(ar->fd, &entry->key_id) == 4) &&
|
||||||
|
(lmo_read32(ar->fd, &entry->val_id) == 4) &&
|
||||||
|
(lmo_read32(ar->fd, &entry->offset) == 4) &&
|
||||||
|
(lmo_read32(ar->fd, &entry->length) == 4)
|
||||||
|
) {
|
||||||
|
entry->next = head;
|
||||||
|
head = entry;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error("Unexpected EOF while reading index entry", 0);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error("Out of memory", 0);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ar->index = head;
|
||||||
|
|
||||||
|
if( lseek(ar->fd, 0, SEEK_SET) == -1 )
|
||||||
|
{
|
||||||
|
error("Can not seek to start", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( (ar->mmap = mmap(NULL, ar->length, PROT_READ, MAP_SHARED, ar->fd, 0)) == MAP_FAILED )
|
||||||
|
{
|
||||||
|
error("Failed to memory map archive contents", 1);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ar;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error("Out of memory", 0);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
|
||||||
|
if( in > -1 )
|
||||||
|
close(in);
|
||||||
|
|
||||||
|
if( head != NULL )
|
||||||
|
{
|
||||||
|
entry = head;
|
||||||
|
|
||||||
|
while( entry != NULL )
|
||||||
|
{
|
||||||
|
head = entry->next;
|
||||||
|
free(entry);
|
||||||
|
entry = head;
|
||||||
|
}
|
||||||
|
|
||||||
|
head = entry = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( ar != NULL )
|
||||||
|
{
|
||||||
|
if( (ar->mmap != NULL) && (ar->mmap != MAP_FAILED) )
|
||||||
|
munmap(ar->mmap, ar->length);
|
||||||
|
|
||||||
|
free(ar);
|
||||||
|
ar = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void lmo_close(lmo_archive_t *ar)
|
||||||
|
{
|
||||||
|
lmo_entry_t *head = NULL;
|
||||||
|
lmo_entry_t *entry = NULL;
|
||||||
|
|
||||||
|
if( ar != NULL )
|
||||||
|
{
|
||||||
|
entry = ar->index;
|
||||||
|
|
||||||
|
while( entry != NULL )
|
||||||
|
{
|
||||||
|
head = entry->next;
|
||||||
|
free(entry);
|
||||||
|
entry = head;
|
||||||
|
}
|
||||||
|
|
||||||
|
head = entry = NULL;
|
||||||
|
|
||||||
|
if( (ar->mmap != NULL) && (ar->mmap != MAP_FAILED) )
|
||||||
|
munmap(ar->mmap, ar->length);
|
||||||
|
|
||||||
|
close(ar->fd);
|
||||||
|
free(ar);
|
||||||
|
|
||||||
|
ar = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int lmo_lookup(lmo_archive_t *ar, const char *key, char *dest, int len)
|
||||||
|
{
|
||||||
|
uint32_t look_key = sfh_hash(key, strlen(key));
|
||||||
|
int copy_len = -1;
|
||||||
|
|
||||||
|
lmo_entry_t *entry = ar->index;
|
||||||
|
|
||||||
|
while( entry != NULL )
|
||||||
|
{
|
||||||
|
if( entry->key_id == look_key )
|
||||||
|
{
|
||||||
|
copy_len = (len > entry->length) ? entry->length : len;
|
||||||
|
memcpy(dest, &ar->mmap[entry->offset], copy_len);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
entry = entry->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return copy_len;
|
||||||
|
}
|
||||||
|
|
53
libs/lmo/src/lmo_hash.c
Normal file
53
libs/lmo/src/lmo_hash.c
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
/*
|
||||||
|
* Hash function from http://www.azillionmonkeys.com/qed/hash.html
|
||||||
|
* Copyright (C) 2004-2008 by Paul Hsieh
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lmo.h"
|
||||||
|
|
||||||
|
uint32_t sfh_hash(const char * data, int len)
|
||||||
|
{
|
||||||
|
uint32_t hash = len, tmp;
|
||||||
|
int rem;
|
||||||
|
|
||||||
|
if (len <= 0 || data == NULL) return 0;
|
||||||
|
|
||||||
|
rem = len & 3;
|
||||||
|
len >>= 2;
|
||||||
|
|
||||||
|
/* Main loop */
|
||||||
|
for (;len > 0; len--) {
|
||||||
|
hash += sfh_get16(data);
|
||||||
|
tmp = (sfh_get16(data+2) << 11) ^ hash;
|
||||||
|
hash = (hash << 16) ^ tmp;
|
||||||
|
data += 2*sizeof(uint16_t);
|
||||||
|
hash += hash >> 11;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Handle end cases */
|
||||||
|
switch (rem) {
|
||||||
|
case 3: hash += sfh_get16(data);
|
||||||
|
hash ^= hash << 16;
|
||||||
|
hash ^= data[sizeof(uint16_t)] << 18;
|
||||||
|
hash += hash >> 11;
|
||||||
|
break;
|
||||||
|
case 2: hash += sfh_get16(data);
|
||||||
|
hash ^= hash << 11;
|
||||||
|
hash += hash >> 17;
|
||||||
|
break;
|
||||||
|
case 1: hash += *data;
|
||||||
|
hash ^= hash << 10;
|
||||||
|
hash += hash >> 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Force "avalanching" of final 127 bits */
|
||||||
|
hash ^= hash << 3;
|
||||||
|
hash += hash >> 5;
|
||||||
|
hash ^= hash << 4;
|
||||||
|
hash += hash >> 17;
|
||||||
|
hash ^= hash << 25;
|
||||||
|
hash += hash >> 6;
|
||||||
|
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
58
libs/lmo/src/lmo_lookup.c
Normal file
58
libs/lmo/src/lmo_lookup.c
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - Lookup utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lmo.h"
|
||||||
|
|
||||||
|
extern char _lmo_error[1024];
|
||||||
|
|
||||||
|
static void die(const char *msg)
|
||||||
|
{
|
||||||
|
printf("Error: %s\n", msg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usage(const char *name)
|
||||||
|
{
|
||||||
|
printf("Usage: %s input.lmo key\n", name);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
char val[4096];
|
||||||
|
lmo_archive_t *ar = NULL;
|
||||||
|
|
||||||
|
if( argc != 3 )
|
||||||
|
usage(argv[0]);
|
||||||
|
|
||||||
|
if( (ar = (lmo_archive_t *) lmo_open(argv[1])) != NULL )
|
||||||
|
{
|
||||||
|
if( lmo_lookup(ar, argv[2], val, sizeof(val)) > -1 )
|
||||||
|
{
|
||||||
|
printf("%s\n", val);
|
||||||
|
}
|
||||||
|
|
||||||
|
lmo_close(ar);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
die(lmo_error());
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
124
libs/lmo/src/lmo_lualib.c
Normal file
124
libs/lmo/src/lmo_lualib.c
Normal file
|
@ -0,0 +1,124 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - Lookup utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lmo_lualib.h"
|
||||||
|
|
||||||
|
extern char _lmo_error[1024];
|
||||||
|
|
||||||
|
|
||||||
|
static int lmo_L_open(lua_State *L) {
|
||||||
|
const char *filename = luaL_checklstring(L, 1, NULL);
|
||||||
|
lmo_archive_t *ar, **udata;
|
||||||
|
|
||||||
|
if( (ar = lmo_open(filename)) != NULL )
|
||||||
|
{
|
||||||
|
if( (udata = lua_newuserdata(L, sizeof(lmo_archive_t *))) != NULL )
|
||||||
|
{
|
||||||
|
*udata = ar;
|
||||||
|
luaL_getmetatable(L, LMO_ARCHIVE_META);
|
||||||
|
lua_setmetatable(L, -2);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pushnil(L);
|
||||||
|
lua_pushstring(L, "out of memory");
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pushnil(L);
|
||||||
|
lua_pushstring(L, lmo_error());
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int lmo_L_hash(lua_State *L) {
|
||||||
|
const char *data = luaL_checkstring(L, 1);
|
||||||
|
uint32_t hash = sfh_hash(data, strlen(data));
|
||||||
|
lua_pushnumber(L, hash);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int lmo_L_lookup(lua_State *L) {
|
||||||
|
lmo_archive_t **ar = luaL_checkudata(L, 1, LMO_ARCHIVE_META);
|
||||||
|
lmo_entry_t *e = (*ar)->index;
|
||||||
|
const char *key = luaL_checkstring(L, 2);
|
||||||
|
uint32_t hash = sfh_hash(key, strlen(key));
|
||||||
|
|
||||||
|
while( e != NULL )
|
||||||
|
{
|
||||||
|
if( e->key_id == hash )
|
||||||
|
{
|
||||||
|
lua_pushlstring(L, &(*ar)->mmap[e->offset], e->length);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
e = e->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pushnil(L);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int lmo_L__gc(lua_State *L) {
|
||||||
|
lmo_archive_t **ar = luaL_checkudata(L, 1, LMO_ARCHIVE_META);
|
||||||
|
|
||||||
|
if( (*ar) != NULL )
|
||||||
|
lmo_close(*ar);
|
||||||
|
|
||||||
|
*ar = NULL;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int lmo_L__tostring(lua_State *L) {
|
||||||
|
lmo_archive_t **ar = luaL_checkudata(L, 1, LMO_ARCHIVE_META);
|
||||||
|
lua_pushfstring(L, "LMO Archive (%d bytes)", (*ar)->length);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* method table */
|
||||||
|
static const luaL_reg M[] = {
|
||||||
|
{"close", lmo_L__gc},
|
||||||
|
{"lookup", lmo_L_lookup},
|
||||||
|
{"__tostring", lmo_L__tostring},
|
||||||
|
{"__gc", lmo_L__gc},
|
||||||
|
{NULL, NULL}
|
||||||
|
};
|
||||||
|
|
||||||
|
/* module table */
|
||||||
|
static const luaL_reg R[] = {
|
||||||
|
{"open", lmo_L_open},
|
||||||
|
{"hash", lmo_L_hash},
|
||||||
|
{NULL, NULL}
|
||||||
|
};
|
||||||
|
|
||||||
|
LUALIB_API int luaopen_lmo(lua_State *L) {
|
||||||
|
luaL_newmetatable(L, LMO_LUALIB_META);
|
||||||
|
luaL_register(L, NULL, R);
|
||||||
|
lua_pushvalue(L, -1);
|
||||||
|
lua_setfield(L, -2, "__index");
|
||||||
|
lua_setglobal(L, LMO_LUALIB_META);
|
||||||
|
|
||||||
|
luaL_newmetatable(L, LMO_ARCHIVE_META);
|
||||||
|
luaL_register(L, NULL, M);
|
||||||
|
lua_pushvalue(L, -1);
|
||||||
|
lua_setfield(L, -2, "__index");
|
||||||
|
lua_setglobal(L, LMO_ARCHIVE_META);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
33
libs/lmo/src/lmo_lualib.h
Normal file
33
libs/lmo/src/lmo_lualib.h
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - Lua library header
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _LMO_LUALIB_H_
|
||||||
|
#define _LMO_LUALIB_H_
|
||||||
|
|
||||||
|
#include <lua.h>
|
||||||
|
#include <lualib.h>
|
||||||
|
#include <lauxlib.h>
|
||||||
|
|
||||||
|
#include "lmo.h"
|
||||||
|
|
||||||
|
#define LMO_LUALIB_META "lmo"
|
||||||
|
#define LMO_ARCHIVE_META "lmo.archive"
|
||||||
|
|
||||||
|
LUALIB_API int luaopen_lmo(lua_State *L);
|
||||||
|
|
||||||
|
#endif
|
199
libs/lmo/src/lmo_po2lmo.c
Normal file
199
libs/lmo/src/lmo_po2lmo.c
Normal file
|
@ -0,0 +1,199 @@
|
||||||
|
/*
|
||||||
|
* lmo - Lua Machine Objects - PO to LMO conversion tool
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Jo-Philipp Wich <xm@subsignal.org>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lmo.h"
|
||||||
|
|
||||||
|
static void die(const char *msg)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Error: %s\n", msg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usage(const char *name)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Usage: %s input.po output.lmo\n", name);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void print(const void *ptr, size_t size, size_t nmemb, FILE *stream)
|
||||||
|
{
|
||||||
|
if( fwrite(ptr, size, nmemb, stream) == 0 )
|
||||||
|
die("Failed to write stdout");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int extract_string(const char *src, char *dest, int len)
|
||||||
|
{
|
||||||
|
int pos = 0;
|
||||||
|
int esc = 0;
|
||||||
|
int off = -1;
|
||||||
|
|
||||||
|
for( pos = 0; (pos < strlen(src)) && (pos < len); pos++ )
|
||||||
|
{
|
||||||
|
if( (off == -1) && (src[pos] == '"') )
|
||||||
|
{
|
||||||
|
off = pos + 1;
|
||||||
|
}
|
||||||
|
else if( off >= 0 )
|
||||||
|
{
|
||||||
|
if( esc == 1 )
|
||||||
|
{
|
||||||
|
dest[pos-off] = src[pos];
|
||||||
|
esc = 0;
|
||||||
|
}
|
||||||
|
else if( src[pos] == '\\' )
|
||||||
|
{
|
||||||
|
off++;
|
||||||
|
esc = 1;
|
||||||
|
|
||||||
|
}
|
||||||
|
else if( src[pos] != '"' )
|
||||||
|
{
|
||||||
|
dest[pos-off] = src[pos];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dest[pos-off] = '\0';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (off > -1) ? strlen(dest) : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
char line[4096];
|
||||||
|
char key[4096];
|
||||||
|
char val[4096];
|
||||||
|
char tmp[4096];
|
||||||
|
int state = 0;
|
||||||
|
int offset = 0;
|
||||||
|
int length = 0;
|
||||||
|
|
||||||
|
FILE *in;
|
||||||
|
FILE *out;
|
||||||
|
|
||||||
|
lmo_entry_t *head = NULL;
|
||||||
|
lmo_entry_t *entry = NULL;
|
||||||
|
|
||||||
|
if( (argc != 3) || ((in = fopen(argv[1], "r")) == NULL) || ((out = fopen(argv[2], "w")) == NULL) )
|
||||||
|
usage(argv[0]);
|
||||||
|
|
||||||
|
memset(line, 0, sizeof(key));
|
||||||
|
memset(key, 0, sizeof(val));
|
||||||
|
memset(val, 0, sizeof(val));
|
||||||
|
|
||||||
|
while( (NULL != fgets(line, sizeof(line), in)) || (state >= 2 && feof(in)) )
|
||||||
|
{
|
||||||
|
if( state == 0 && strstr(line, "msgid \"") == line )
|
||||||
|
{
|
||||||
|
switch(extract_string(line, key, sizeof(key)))
|
||||||
|
{
|
||||||
|
case -1:
|
||||||
|
die("Syntax error in msgid");
|
||||||
|
case 0:
|
||||||
|
continue;
|
||||||
|
default:
|
||||||
|
state = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if( state == 1 && strstr(line, "msgstr \"") == line )
|
||||||
|
{
|
||||||
|
switch(extract_string(line, val, sizeof(val)))
|
||||||
|
{
|
||||||
|
case -1:
|
||||||
|
die("Syntax error in msgstr");
|
||||||
|
case 0:
|
||||||
|
state = 2;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
state = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if( state == 2 )
|
||||||
|
{
|
||||||
|
switch(extract_string(line, tmp, sizeof(tmp)))
|
||||||
|
{
|
||||||
|
case -1:
|
||||||
|
state = 3;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
strcat(val, tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if( state == 3 )
|
||||||
|
{
|
||||||
|
if( strlen(key) > 0 && strlen(val) > 0 )
|
||||||
|
{
|
||||||
|
if( (entry = (lmo_entry_t *) malloc(sizeof(lmo_entry_t))) != NULL )
|
||||||
|
{
|
||||||
|
memset(entry, 0, sizeof(entry));
|
||||||
|
length = strlen(val) + ((4 - (strlen(val) % 4)) % 4);
|
||||||
|
|
||||||
|
entry->key_id = htonl(sfh_hash(key, strlen(key)));
|
||||||
|
entry->val_id = htonl(sfh_hash(val, strlen(val)));
|
||||||
|
entry->offset = htonl(offset);
|
||||||
|
entry->length = htonl(strlen(val));
|
||||||
|
|
||||||
|
print(val, length, 1, out);
|
||||||
|
offset += length;
|
||||||
|
|
||||||
|
entry->next = head;
|
||||||
|
head = entry;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
die("Out of memory");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
state = 0;
|
||||||
|
memset(key, 0, sizeof(key));
|
||||||
|
memset(val, 0, sizeof(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(line, 0, sizeof(line));
|
||||||
|
}
|
||||||
|
|
||||||
|
entry = head;
|
||||||
|
while( entry != NULL )
|
||||||
|
{
|
||||||
|
print(&entry->key_id, sizeof(uint32_t), 1, out);
|
||||||
|
print(&entry->val_id, sizeof(uint32_t), 1, out);
|
||||||
|
print(&entry->offset, sizeof(uint32_t), 1, out);
|
||||||
|
print(&entry->length, sizeof(uint32_t), 1, out);
|
||||||
|
entry = entry->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( offset > 0 )
|
||||||
|
{
|
||||||
|
offset = htonl(offset);
|
||||||
|
print(&offset, sizeof(uint32_t), 1, out);
|
||||||
|
fsync(fileno(out));
|
||||||
|
fclose(out);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
fclose(out);
|
||||||
|
unlink(argv[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(in);
|
||||||
|
return(0);
|
||||||
|
}
|
56
libs/lmo/standalone.mk
Normal file
56
libs/lmo/standalone.mk
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
LUAC = luac
|
||||||
|
LUAC_OPTIONS = -s
|
||||||
|
LUA_TARGET ?= source
|
||||||
|
|
||||||
|
LUA_MODULEDIR = /usr/local/share/lua/5.1
|
||||||
|
LUA_LIBRARYDIR = /usr/local/lib/lua/5.1
|
||||||
|
|
||||||
|
OS ?= $(shell uname)
|
||||||
|
|
||||||
|
LUA_SHLIBS = $(shell pkg-config --silence-errors --libs lua5.1 || pkg-config --silence-errors --libs lua-5.1 || pkg-config --silence-errors --libs lua)
|
||||||
|
LUA_LIBS = $(if $(LUA_SHLIBS),$(LUA_SHLIBS),$(firstword $(wildcard /usr/lib/liblua.a /usr/local/lib/liblua.a /opt/local/lib/liblua.a)))
|
||||||
|
LUA_CFLAGS = $(shell pkg-config --silence-errors --cflags lua5.1 || pkg-config --silence-errors --cflags lua-5.1 || pkg-config --silence-errors --cflags lua)
|
||||||
|
|
||||||
|
CC = gcc
|
||||||
|
AR = ar
|
||||||
|
RANLIB = ranlib
|
||||||
|
CFLAGS = -O2
|
||||||
|
FPIC = -fPIC
|
||||||
|
EXTRA_CFLAGS = --std=gnu99
|
||||||
|
WFLAGS = -Wall -Werror -pedantic
|
||||||
|
CPPFLAGS =
|
||||||
|
COMPILE = $(CC) $(CPPFLAGS) $(CFLAGS) $(EXTRA_CFLAGS) $(WFLAGS)
|
||||||
|
ifeq ($(OS),Darwin)
|
||||||
|
SHLIB_FLAGS = -bundle -undefined dynamic_lookup
|
||||||
|
else
|
||||||
|
SHLIB_FLAGS = -shared
|
||||||
|
endif
|
||||||
|
LINK = $(CC) $(LDFLAGS)
|
||||||
|
|
||||||
|
.PHONY: all build compile luacompile luasource clean luaclean
|
||||||
|
|
||||||
|
all: build
|
||||||
|
|
||||||
|
build: luabuild gccbuild
|
||||||
|
|
||||||
|
luabuild: lua$(LUA_TARGET)
|
||||||
|
|
||||||
|
gccbuild: compile
|
||||||
|
compile:
|
||||||
|
|
||||||
|
clean: luaclean
|
||||||
|
|
||||||
|
luasource:
|
||||||
|
mkdir -p dist$(LUA_MODULEDIR)
|
||||||
|
cp -pR root/* dist 2>/dev/null || true
|
||||||
|
cp -pR lua/* dist$(LUA_MODULEDIR) 2>/dev/null || true
|
||||||
|
for i in $$(find dist -name .svn); do rm -rf $$i || true; done
|
||||||
|
|
||||||
|
luastrip: luasource
|
||||||
|
for i in $$(find dist -type f -name '*.lua'); do perl -e 'undef $$/; open( F, "< $$ARGV[0]" ) || die $$!; $$src = <F>; close F; $$src =~ s/--\[\[.*?\]\](--)?//gs; $$src =~ s/^\s*--.*?\n//gm; open( F, "> $$ARGV[0]" ) || die $$!; print F $$src; close F' $$i; done
|
||||||
|
|
||||||
|
luacompile: luasource
|
||||||
|
for i in $$(find dist -name *.lua -not -name debug.lua); do $(LUAC) $(LUAC_OPTIONS) -o $$i $$i; done
|
||||||
|
|
||||||
|
luaclean:
|
||||||
|
rm -rf dist
|
Loading…
Reference in a new issue