mirror of https://github.com/perkeep/perkeep.git
Merge branch 'master' of ssh://camlistore.org:29418/camlistore
This commit is contained in:
commit
bc291512a5
|
@ -0,0 +1 @@
|
||||||
|
foo.db
|
|
@ -1,4 +1,5 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include "_cgo_export.h"
|
#include "_cgo_export.h"
|
||||||
|
@ -6,49 +7,130 @@
|
||||||
#define SKIP_SQLITE_VERSION
|
#define SKIP_SQLITE_VERSION
|
||||||
#include "sqlite3.h"
|
#include "sqlite3.h"
|
||||||
|
|
||||||
|
static sqlite3_io_methods g_file_methods;
|
||||||
|
|
||||||
typedef struct GoFile GoFile;
|
typedef struct GoFile GoFile;
|
||||||
struct GoFile {
|
struct GoFile {
|
||||||
sqlite3_io_methods const *pMethod; /* Always the first entry */
|
sqlite3_io_methods const *pMethod; /* Always the first entry */
|
||||||
|
int fd;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* File methods */
|
||||||
|
|
||||||
|
static int go_file_close(sqlite3_file* file) {
|
||||||
|
return GoFileClose(((GoFile*) file)->fd) == 0 ? SQLITE_OK : SQLITE_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_file_read(sqlite3_file* file, void* dest, int iAmt, sqlite3_int64 iOfst) {
|
||||||
|
return GoFileRead(((GoFile*) file)->fd, dest, iAmt, iOfst) == 0 ?
|
||||||
|
SQLITE_OK :
|
||||||
|
SQLITE_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_file_write(sqlite3_file* file, const void* src, int iAmt, sqlite3_int64 iOfst) {
|
||||||
|
fprintf(stderr, "write\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* VFS methods */
|
||||||
|
|
||||||
static int go_vfs_open(sqlite3_vfs* vfs,
|
static int go_vfs_open(sqlite3_vfs* vfs,
|
||||||
const char* zName,
|
const char* zName,
|
||||||
sqlite3_file* file,
|
sqlite3_file* file,
|
||||||
int flags,
|
int flags,
|
||||||
int* pOutFlags) {
|
int* pOutFlags) {
|
||||||
struct GoVFSOpen_return r;
|
fprintf(stderr, "go_vfs_open: %s\n", zName);
|
||||||
r = GoVFSOpen((char*) zName, flags);
|
GoFile* go_file = (GoFile*) file;
|
||||||
if (r.r0 == -1) {
|
memset(go_file, 0, sizeof(go_file));
|
||||||
|
|
||||||
|
const int fd = GoVFSOpen((char*) zName, flags);
|
||||||
|
if (fd == -1) {
|
||||||
return SQLITE_ERROR;
|
return SQLITE_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
return SQLITE_ERROR;
|
go_file->pMethod = &g_file_methods;
|
||||||
|
go_file->fd = fd;
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_delete(sqlite3_vfs* vfs, const char* zName, int syncDir) {
|
||||||
|
fprintf(stderr, "delete: %s\n", zName);
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_access(sqlite3_vfs* vfs,
|
||||||
|
const char* zName,
|
||||||
|
int flags,
|
||||||
|
int* pResOut) {
|
||||||
|
fprintf(stderr, "access: %s\n", zName);
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_full_pathname(sqlite3_vfs* vfs,
|
||||||
|
const char* zName,
|
||||||
|
int nOut,
|
||||||
|
char* zOut) {
|
||||||
|
strncpy(zOut, zName, nOut);
|
||||||
|
zOut[nOut - 1] = '\0';
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void* go_vfs_dl_open(sqlite3_vfs* vfs, const char* zFilename) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void go_vfs_dl_error(sqlite3_vfs* vfs, int nByte, char *zErrMsg) {
|
||||||
|
}
|
||||||
|
|
||||||
|
static void* go_vfs_dl_sym(sqlite3_vfs* vfs,
|
||||||
|
void* handle,
|
||||||
|
const char* zSymbol) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void go_vfs_dl_close(sqlite3_vfs* vfs, void* handle) {
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_randomness(sqlite3_vfs* vfs, int nByte, char *zOut) {
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_sleep(sqlite3_vfs* vfs, int microseconds) {
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_current_time(sqlite3_vfs* vfs, double* now) {
|
||||||
|
return SQLITE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int go_vfs_get_last_error(sqlite3_vfs* vfs, int foo, char* bar) {
|
||||||
|
// Unused, per sqlite3's os_unix.c.
|
||||||
|
return SQLITE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int sqlite3_os_init(void) {
|
int sqlite3_os_init(void) {
|
||||||
static sqlite3_vfs vfs;
|
static sqlite3_vfs vfs;
|
||||||
memset(&vfs, 0, sizeof(vfs));
|
memset(&vfs, 0, sizeof(vfs));
|
||||||
vfs.iVersion = 3;
|
vfs.iVersion = 1;
|
||||||
vfs.szOsFile = sizeof(GoFile);
|
vfs.szOsFile = sizeof(GoFile);
|
||||||
vfs.mxPathname = 512;
|
vfs.mxPathname = 512;
|
||||||
vfs.pNext = NULL;
|
vfs.pNext = NULL;
|
||||||
vfs.zName = "go";
|
vfs.zName = "go";
|
||||||
vfs.pAppData = NULL;
|
vfs.pAppData = NULL;
|
||||||
|
/* Version 1 methods */
|
||||||
vfs.xOpen = go_vfs_open;
|
vfs.xOpen = go_vfs_open;
|
||||||
|
vfs.xDelete = go_vfs_delete;
|
||||||
|
vfs.xAccess = go_vfs_access;
|
||||||
|
vfs.xFullPathname = go_vfs_full_pathname;
|
||||||
|
vfs.xDlOpen = go_vfs_dl_open;
|
||||||
|
vfs.xDlError = go_vfs_dl_error;
|
||||||
|
vfs.xDlSym = go_vfs_dl_sym;
|
||||||
|
vfs.xDlClose = go_vfs_dl_close;
|
||||||
|
vfs.xRandomness = go_vfs_randomness;
|
||||||
|
vfs.xSleep = go_vfs_sleep;
|
||||||
|
vfs.xCurrentTime = go_vfs_current_time;
|
||||||
|
vfs.xGetLastError = go_vfs_get_last_error;
|
||||||
#if 0
|
#if 0
|
||||||
int (*xOpen)(sqlite3_vfs*, const char *zName, sqlite3_file*,
|
|
||||||
int flags, int *pOutFlags);
|
|
||||||
int (*xDelete)(sqlite3_vfs*, const char *zName, int syncDir);
|
|
||||||
int (*xAccess)(sqlite3_vfs*, const char *zName, int flags, int *pResOut);
|
|
||||||
int (*xFullPathname)(sqlite3_vfs*, const char *zName, int nOut, char *zOut);
|
|
||||||
void *(*xDlOpen)(sqlite3_vfs*, const char *zFilename);
|
|
||||||
void (*xDlError)(sqlite3_vfs*, int nByte, char *zErrMsg);
|
|
||||||
void (*(*xDlSym)(sqlite3_vfs*,void*, const char *zSymbol))(void);
|
|
||||||
void (*xDlClose)(sqlite3_vfs*, void*);
|
|
||||||
int (*xRandomness)(sqlite3_vfs*, int nByte, char *zOut);
|
|
||||||
int (*xSleep)(sqlite3_vfs*, int microseconds);
|
|
||||||
int (*xCurrentTime)(sqlite3_vfs*, double*);
|
|
||||||
int (*xGetLastError)(sqlite3_vfs*, int, char *);
|
|
||||||
/*
|
/*
|
||||||
** The methods above are in version 1 of the sqlite_vfs object
|
** The methods above are in version 1 of the sqlite_vfs object
|
||||||
** definition. Those that follow are added in version 2 or later
|
** definition. Those that follow are added in version 2 or later
|
||||||
|
@ -69,6 +151,31 @@ int sqlite3_os_init(void) {
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
sqlite3_vfs_register(&vfs, 1);
|
sqlite3_vfs_register(&vfs, 1);
|
||||||
|
|
||||||
|
memset(&g_file_methods, 0, sizeof(g_file_methods));
|
||||||
|
g_file_methods.iVersion = 1;
|
||||||
|
g_file_methods.xClose = go_file_close;
|
||||||
|
g_file_methods.xRead = go_file_read;
|
||||||
|
g_file_methods.xWrite = go_file_write;
|
||||||
|
#if 0
|
||||||
|
int (*xTruncate)(sqlite3_file*, sqlite3_int64 size);
|
||||||
|
int (*xSync)(sqlite3_file*, int flags);
|
||||||
|
int (*xFileSize)(sqlite3_file*, sqlite3_int64 *pSize);
|
||||||
|
int (*xLock)(sqlite3_file*, int);
|
||||||
|
int (*xUnlock)(sqlite3_file*, int);
|
||||||
|
int (*xCheckReservedLock)(sqlite3_file*, int *pResOut);
|
||||||
|
int (*xFileControl)(sqlite3_file*, int op, void *pArg);
|
||||||
|
int (*xSectorSize)(sqlite3_file*);
|
||||||
|
int (*xDeviceCharacteristics)(sqlite3_file*);
|
||||||
|
/* Methods above are valid for version 1 */
|
||||||
|
int (*xShmMap)(sqlite3_file*, int iPg, int pgsz, int, void volatile**);
|
||||||
|
int (*xShmLock)(sqlite3_file*, int offset, int n, int flags);
|
||||||
|
void (*xShmBarrier)(sqlite3_file*);
|
||||||
|
int (*xShmUnmap)(sqlite3_file*, int deleteFlag);
|
||||||
|
/* Methods above are valid for version 2 */
|
||||||
|
/* Additional methods may be added in future releases */
|
||||||
|
#endif
|
||||||
|
|
||||||
return SQLITE_OK;
|
return SQLITE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,108 @@
|
||||||
package sqlite
|
package sqlite
|
||||||
|
|
||||||
|
/*
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#define SKIP_SQLITE_VERSION
|
||||||
|
#include "sqlite3.h"
|
||||||
|
*/
|
||||||
import "C"
|
import "C"
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"sync"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
var file_map_mutex sync.Mutex
|
||||||
|
var file_map = make(map[int]*os.File)
|
||||||
|
|
||||||
|
func GetFile(fd int) (file *os.File) {
|
||||||
|
file_map_mutex.Lock()
|
||||||
|
defer file_map_mutex.Unlock()
|
||||||
|
return file_map[fd]
|
||||||
|
}
|
||||||
|
|
||||||
|
//export GoFileClose
|
||||||
|
// Returns 0 on success and -1 on error.
|
||||||
|
func GoFileClose(fd C.int) (int) {
|
||||||
|
file := GetFile(int(fd))
|
||||||
|
if file.Close() != nil {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
//export GoFileRead
|
||||||
|
// Returns 0 on success and -1 on error.
|
||||||
|
func GoFileRead(fd C.int, dst *C.char, n C.int, offset C.int) (rv int) {
|
||||||
|
println("reading", n, "bytes at offset", offset, "from fd", fd);
|
||||||
|
defer func() {
|
||||||
|
println("read returning", rv);
|
||||||
|
}()
|
||||||
|
|
||||||
|
file := GetFile(int(fd))
|
||||||
|
if file == nil {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := make([]byte, n)
|
||||||
|
curbuf := buf
|
||||||
|
for n > 0 {
|
||||||
|
read, err := file.ReadAt(curbuf, int64(offset))
|
||||||
|
curbuf = curbuf[read:]
|
||||||
|
n -= C.int(read)
|
||||||
|
if err == os.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if n != 0 {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
C.memcpy(unsafe.Pointer(dst), unsafe.Pointer(&buf[0]), C.size_t(len(buf)))
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//export GoVFSOpen
|
//export GoVFSOpen
|
||||||
// fd is -1 on error.
|
// fd is -1 on error.
|
||||||
func GoVFSOpen(filename *C.char, flags C.int) (int, int) {
|
func GoVFSOpen(filename *C.char, flags C.int) (fd int) {
|
||||||
return -1, 0
|
println("opening", C.GoString(filename), "with flags", int(flags))
|
||||||
|
|
||||||
|
goflags := 0
|
||||||
|
if flags & C.SQLITE_OPEN_READONLY != 0 {
|
||||||
|
goflags |= os.O_RDONLY
|
||||||
|
}
|
||||||
|
if flags & C.SQLITE_OPEN_READWRITE != 0 {
|
||||||
|
goflags |= os.O_RDWR
|
||||||
|
}
|
||||||
|
if flags & C.SQLITE_OPEN_CREATE != 0 {
|
||||||
|
goflags |= os.O_RDWR | os.O_CREATE | os.O_TRUNC
|
||||||
|
}
|
||||||
|
if flags & C.SQLITE_OPEN_DELETEONCLOSE != 0 {
|
||||||
|
// TODO: Do something.
|
||||||
|
}
|
||||||
|
if flags & C.SQLITE_OPEN_EXCLUSIVE != 0{
|
||||||
|
goflags |= os.O_EXCL
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.OpenFile(C.GoString(filename), goflags, 0666)
|
||||||
|
defer func() {
|
||||||
|
if err != nil {
|
||||||
|
println("got error:", err.String())
|
||||||
|
}
|
||||||
|
println("returning fd", fd);
|
||||||
|
}()
|
||||||
|
if err != nil {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
file_map_mutex.Lock()
|
||||||
|
defer file_map_mutex.Unlock()
|
||||||
|
file_map[file.Fd()] = file
|
||||||
|
return file.Fd()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue