541 lines
11 KiB
Text
541 lines
11 KiB
Text
|
/*
|
||
|
* lar - Lua Archive Library
|
||
|
*
|
||
|
* 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 "lua.h"
|
||
|
#include "lualib.h"
|
||
|
#include "lauxlib.h"
|
||
|
#include "lar.h"
|
||
|
|
||
|
typedef struct {
|
||
|
int fd;
|
||
|
char *data;
|
||
|
size_t length;
|
||
|
} mmap_handle;
|
||
|
|
||
|
static int larlib_perror( lua_State *L, const char *message )
|
||
|
{
|
||
|
lua_pushnil(L);
|
||
|
lua_pushstring(L, message);
|
||
|
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
int larlib_open( lua_State *L )
|
||
|
{
|
||
|
lar_archive *ar, **udata;
|
||
|
const char *filename = luaL_checkstring( L, 1 );
|
||
|
|
||
|
if( filename != NULL && (ar = lar_open(filename)) != NULL )
|
||
|
{
|
||
|
if( (udata = lua_newuserdata(L, sizeof(lar_archive *))) != NULL )
|
||
|
{
|
||
|
*udata = ar;
|
||
|
luaL_getmetatable(L, "lar.archive");
|
||
|
lua_setmetatable(L, -2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Archive not found");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_find( lua_State *L )
|
||
|
{
|
||
|
const char *filename = luaL_checkstring( L, 1 );
|
||
|
const char *basepath = luaL_optstring( L, 2, "./" );
|
||
|
int is_pkg = strstr(filename, "/") ? 0 : 1;
|
||
|
lar_archive *ar, **udata;
|
||
|
|
||
|
if( ((ar = lar_find_archive(filename, basepath, is_pkg)) != NULL) ||
|
||
|
((ar = lar_find_archive(filename, LUA_LDIR, is_pkg)) != NULL) ||
|
||
|
((ar = lar_find_archive(filename, LUA_CDIR, is_pkg)) != NULL) )
|
||
|
{
|
||
|
if( (udata = lua_newuserdata(L, sizeof(lar_archive *))) != NULL )
|
||
|
{
|
||
|
*udata = ar;
|
||
|
luaL_getmetatable(L, "lar.archive");
|
||
|
lua_setmetatable(L, -2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Archive not found");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_md5( lua_State *L )
|
||
|
{
|
||
|
int i;
|
||
|
char md5[16], md5_hex[33];
|
||
|
const char *data = luaL_checkstring( L, 1 );
|
||
|
md5_state_t state;
|
||
|
|
||
|
md5_init(&state);
|
||
|
md5_append(&state, (const md5_byte_t *)data, strlen(data));
|
||
|
md5_finish(&state, (md5_byte_t *)md5);
|
||
|
|
||
|
for( i = 0; i < 16; i++ )
|
||
|
sprintf(&md5_hex[i*2], "%02x", (unsigned char)md5[i]);
|
||
|
|
||
|
lua_pushstring(L, md5_hex);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_md5_file( lua_State *L )
|
||
|
{
|
||
|
int i, fd, len;
|
||
|
char md5[16], md5_hex[33], buffer[1024];
|
||
|
const char *filename = luaL_checkstring( L, 1 );
|
||
|
md5_state_t state;
|
||
|
|
||
|
if( (fd = open(filename, O_RDONLY)) != -1 )
|
||
|
{
|
||
|
md5_init(&state);
|
||
|
|
||
|
while( (len = read(fd, buffer, 1024)) > 0 )
|
||
|
md5_append(&state, (const md5_byte_t *)buffer, len);
|
||
|
|
||
|
md5_finish(&state, (md5_byte_t *)md5);
|
||
|
|
||
|
for( i = 0; i < 16; i++ )
|
||
|
sprintf(&md5_hex[i*2], "%02x", (unsigned char)md5[i]);
|
||
|
|
||
|
close(fd);
|
||
|
lua_pushstring(L, md5_hex);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, strerror(errno));
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int larlib_mkpath( const char *name, const char *path, char *buffer )
|
||
|
{
|
||
|
int nlen = strlen(name);
|
||
|
int plen = strlen(path);
|
||
|
|
||
|
if( (nlen + plen + 1) <= LAR_FNAME_BUFFER )
|
||
|
{
|
||
|
strcpy(buffer, path);
|
||
|
|
||
|
if( buffer[plen-1] != '/' )
|
||
|
buffer[plen++] = '/';
|
||
|
|
||
|
strcpy(&buffer[plen], name);
|
||
|
buffer[plen + nlen] = '\0';
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int larlib__gc( lua_State *L )
|
||
|
{
|
||
|
lar_archive **archive = luaL_checkudata( L, 1, "lar.archive" );
|
||
|
|
||
|
if( *archive )
|
||
|
lar_close(*archive);
|
||
|
|
||
|
*archive = NULL;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int larlib_member__open( lua_State *L, lar_member *mb )
|
||
|
{
|
||
|
lar_archive **archive = NULL;
|
||
|
const char *filename = NULL;
|
||
|
lar_member **udata;
|
||
|
|
||
|
if( mb == NULL )
|
||
|
{
|
||
|
*archive = luaL_checkudata( L, 1, "lar.archive" );
|
||
|
filename = luaL_checkstring( L, 2 );
|
||
|
}
|
||
|
|
||
|
if( mb != NULL || (mb = lar_open_member(*archive, filename)) != NULL )
|
||
|
{
|
||
|
if( (udata = lua_newuserdata(L, sizeof(lar_member *))) != NULL )
|
||
|
{
|
||
|
*udata = mb;
|
||
|
luaL_getmetatable(L, "lar.member");
|
||
|
lua_setmetatable(L, -2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Member not found in archive");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_open( lua_State *L )
|
||
|
{
|
||
|
return larlib_member__open( L, NULL );
|
||
|
}
|
||
|
|
||
|
int larlib_member_find( lua_State *L )
|
||
|
{
|
||
|
lar_archive **archive = luaL_checkudata( L, 1, "lar.archive" );
|
||
|
const char *package = luaL_checkstring( L, 2 );
|
||
|
lar_member *mb, **udata;
|
||
|
|
||
|
if( (mb = lar_find_member(*archive, package)) != NULL )
|
||
|
{
|
||
|
if( (udata = lua_newuserdata(L, sizeof(lar_member *))) != NULL )
|
||
|
{
|
||
|
*udata = mb;
|
||
|
luaL_getmetatable(L, "lar.member");
|
||
|
lua_setmetatable(L, -2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Member not found in archive");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_size( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
lua_pushnumber(L, (*member)->length);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_type( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
lua_pushnumber(L, (*member)->type);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_flags( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
lua_pushnumber(L, (*member)->flags);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_read( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
int start = luaL_checknumber( L, 2 );
|
||
|
int length = luaL_optnumber( L, 3, (*member)->length );
|
||
|
char *stringcopy;
|
||
|
|
||
|
if( (start >= 0) && (start < (*member)->length) && (length > 0) )
|
||
|
{
|
||
|
if( (start + length) >= (*member)->length )
|
||
|
length = (*member)->length - start;
|
||
|
|
||
|
if( (stringcopy = (char *)malloc(length + 1)) != NULL )
|
||
|
{
|
||
|
memcpy(stringcopy, &(*member)->data[start], length);
|
||
|
stringcopy[length] = '\0';
|
||
|
lua_pushstring(L, stringcopy);
|
||
|
free(stringcopy);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Invalid argument");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_data( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
lua_pushstring(L, (*member)->data);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_member_load( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
int status = luaL_loadbuffer( L, (*member)->data, (*member)->length,
|
||
|
"=(lar member)" );
|
||
|
|
||
|
if( status )
|
||
|
{
|
||
|
lua_pushnil(L);
|
||
|
lua_insert(L, -2);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int larlib_member__gc( lua_State *L )
|
||
|
{
|
||
|
lar_member **member = luaL_checkudata( L, 1, "lar.member" );
|
||
|
|
||
|
if( *member )
|
||
|
lar_close_member(*member);
|
||
|
|
||
|
*member = NULL;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int larlib_mmfile__open( lua_State *L, const char *filename )
|
||
|
{
|
||
|
struct stat s;
|
||
|
mmap_handle *fh, **udata;
|
||
|
|
||
|
if( filename == NULL )
|
||
|
filename = (const char *)luaL_checkstring( L, 1 );
|
||
|
|
||
|
if( (fh = (mmap_handle *)malloc(sizeof(mmap_handle))) == NULL )
|
||
|
return larlib_perror(L, "Out of memory");
|
||
|
|
||
|
if( stat(filename, &s) > -1 && (fh->fd = open(filename, O_RDONLY)) > -1 )
|
||
|
{
|
||
|
fh->length = s.st_size;
|
||
|
fh->data = mmap( 0, s.st_size, PROT_READ, MAP_PRIVATE, fh->fd, 0 );
|
||
|
|
||
|
if( fh->data == MAP_FAILED )
|
||
|
return larlib_perror(L, "Failed to mmap() file");
|
||
|
|
||
|
if( (udata = lua_newuserdata(L, sizeof(char *))) != NULL )
|
||
|
{
|
||
|
*udata = fh;
|
||
|
luaL_getmetatable(L, "lar.mmfile");
|
||
|
lua_setmetatable(L, -2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, strerror(errno));
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_mmfile_open( lua_State *L )
|
||
|
{
|
||
|
return larlib_mmfile__open(L, NULL);
|
||
|
}
|
||
|
|
||
|
int larlib_mmfile_size( lua_State *L )
|
||
|
{
|
||
|
mmap_handle **fh = luaL_checkudata( L, 1, "lar.mmfile" );
|
||
|
lua_pushnumber(L, (*fh)->length);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_mmfile_read( lua_State *L )
|
||
|
{
|
||
|
mmap_handle **fh = luaL_checkudata( L, 1, "lar.mmfile" );
|
||
|
int start = luaL_checknumber( L, 2 );
|
||
|
int length = luaL_optnumber( L, 3, (*fh)->length );
|
||
|
char *stringcopy;
|
||
|
|
||
|
if( (start >= 0) && (start < (*fh)->length) && (length > 0) )
|
||
|
{
|
||
|
if( (start + length) >= (*fh)->length )
|
||
|
length = (*fh)->length - start;
|
||
|
|
||
|
if( (stringcopy = (char *)malloc(length + 1)) != NULL )
|
||
|
{
|
||
|
memcpy(stringcopy, &(*fh)->data[start], length);
|
||
|
stringcopy[length] = '\0';
|
||
|
lua_pushstring(L, stringcopy);
|
||
|
free(stringcopy);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return luaL_error(L, "Out of memory");
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return larlib_perror(L, "Invalid argument");
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_mmfile_data( lua_State *L )
|
||
|
{
|
||
|
mmap_handle **fh = luaL_checkudata( L, 1, "lar.mmfile" );
|
||
|
lua_pushstring(L, (*fh)->data);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int larlib_mmfile_load( lua_State *L )
|
||
|
{
|
||
|
mmap_handle **fh = luaL_checkudata( L, 1, "lar.mmfile" );
|
||
|
int status = luaL_loadbuffer(L, (*fh)->data, (*fh)->length, "=(mmap file)");
|
||
|
|
||
|
if( status )
|
||
|
{
|
||
|
lua_pushnil(L);
|
||
|
lua_insert(L, -2);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int larlib_mmfile__gc( lua_State *L )
|
||
|
{
|
||
|
mmap_handle **fh = luaL_checkudata( L, 1, "lar.mmfile" );
|
||
|
|
||
|
if( *fh )
|
||
|
{
|
||
|
close((*fh)->fd);
|
||
|
munmap((*fh)->data, (*fh)->length);
|
||
|
free(*fh);
|
||
|
*fh = NULL;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
int larlib_findfile( lua_State *L )
|
||
|
{
|
||
|
int i;
|
||
|
const char *filename = luaL_checkstring( L, 1 );
|
||
|
const char *basepath = luaL_optstring( L, 2, "./" );
|
||
|
struct stat s;
|
||
|
lar_archive *ar;
|
||
|
lar_member *mb;
|
||
|
LAR_FNAME(filepath);
|
||
|
|
||
|
const char *searchpath[3] = { basepath, LUA_LDIR, LUA_CDIR };
|
||
|
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
if( !larlib_mkpath(filename, searchpath[i], filepath) )
|
||
|
if( stat(filepath, &s) > -1 && (s.st_mode & S_IFREG) )
|
||
|
return larlib_mmfile__open( L, filepath );
|
||
|
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
if( (ar = lar_find_archive(filename, searchpath[i], 0)) != NULL )
|
||
|
if( (mb = lar_open_member(ar, filename)) != NULL )
|
||
|
return larlib_member__open( L, mb );
|
||
|
|
||
|
return larlib_perror(L, "File not found");
|
||
|
}
|
||
|
|
||
|
|
||
|
static const luaL_reg LAR_REG[] = {
|
||
|
{ "open", larlib_open },
|
||
|
{ "find", larlib_find },
|
||
|
{ "md5", larlib_md5 },
|
||
|
{ "md5_file", larlib_md5_file },
|
||
|
{ "mmap", larlib_mmfile_open },
|
||
|
{ "findfile", larlib_findfile },
|
||
|
{ NULL, NULL }
|
||
|
};
|
||
|
|
||
|
static const luaL_reg LAR_ARCHIVE_REG[] = {
|
||
|
{ "member", larlib_member_open },
|
||
|
{ "find", larlib_member_find },
|
||
|
{ "__gc", larlib__gc },
|
||
|
{ NULL, NULL }
|
||
|
};
|
||
|
|
||
|
static const luaL_reg LAR_MEMBER_REG[] = {
|
||
|
{ "size", larlib_member_size },
|
||
|
{ "type", larlib_member_type },
|
||
|
{ "flags", larlib_member_flags },
|
||
|
{ "read", larlib_member_read },
|
||
|
{ "data", larlib_member_data },
|
||
|
{ "load", larlib_member_load },
|
||
|
{ "__gc", larlib_member__gc },
|
||
|
{ NULL, NULL }
|
||
|
};
|
||
|
|
||
|
static const luaL_reg LAR_MMFILE_REG[] = {
|
||
|
{ "size", larlib_mmfile_size },
|
||
|
{ "read", larlib_mmfile_read },
|
||
|
{ "data", larlib_mmfile_data },
|
||
|
{ "load", larlib_mmfile_load },
|
||
|
{ "__gc", larlib_mmfile__gc },
|
||
|
{ NULL, NULL }
|
||
|
};
|
||
|
|
||
|
|
||
|
LUALIB_API int luaopen_larlib( lua_State *L )
|
||
|
{
|
||
|
luaL_newmetatable(L, "lar");
|
||
|
luaL_register(L, NULL, LAR_REG);
|
||
|
lua_pushvalue(L, -1);
|
||
|
lua_setfield(L, -2, "__index");
|
||
|
lua_setglobal(L, "lar");
|
||
|
|
||
|
luaL_newmetatable(L, "lar.archive");
|
||
|
luaL_register(L, NULL, LAR_ARCHIVE_REG);
|
||
|
lua_pushvalue(L, -1);
|
||
|
lua_setfield(L, -2, "__index");
|
||
|
lua_setglobal(L, "lar.archive");
|
||
|
|
||
|
luaL_newmetatable(L, "lar.member");
|
||
|
luaL_register(L, NULL, LAR_MEMBER_REG);
|
||
|
lua_pushvalue(L, -1);
|
||
|
lua_setfield(L, -2, "__index");
|
||
|
lua_setglobal(L, "lar.member");
|
||
|
|
||
|
luaL_newmetatable(L, "lar.mmfile");
|
||
|
luaL_register(L, NULL, LAR_MMFILE_REG);
|
||
|
lua_pushvalue(L, -1);
|
||
|
lua_setfield(L, -2, "__index");
|
||
|
lua_setglobal(L, "lar.mmfile");
|
||
|
|
||
|
return 1;
|
||
|
}
|