2008-02-19 11:06:52 -07:00
|
|
|
/* Copyright (c) 2008, Avian Contributors
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2008-10-03 14:15:47 -06:00
|
|
|
#include <stdint.h>
|
2008-07-13 18:14:37 -06:00
|
|
|
#include <dirent.h>
|
2007-07-26 18:06:05 -06:00
|
|
|
|
|
|
|
#include "jni.h"
|
2007-08-27 07:46:17 -06:00
|
|
|
#include "jni-util.h"
|
2007-07-26 18:06:05 -06:00
|
|
|
|
|
|
|
#ifdef WIN32
|
2008-10-03 14:15:47 -06:00
|
|
|
# include <windows.h>
|
2007-07-26 18:06:05 -06:00
|
|
|
# include <io.h>
|
2007-10-24 11:24:19 -06:00
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
# define OPEN _open
|
|
|
|
# define CLOSE _close
|
|
|
|
# define READ _read
|
|
|
|
# define WRITE _write
|
|
|
|
# define STAT _stat
|
|
|
|
# define STRUCT_STAT struct _stat
|
|
|
|
# define MKDIR(path, mode) _mkdir(path)
|
|
|
|
# define CREAT _creat
|
2007-09-11 19:13:05 -06:00
|
|
|
# define UNLINK _unlink
|
2007-07-26 18:06:05 -06:00
|
|
|
# define OPEN_MASK O_BINARY
|
|
|
|
#else
|
|
|
|
# include <unistd.h>
|
2008-10-03 14:15:47 -06:00
|
|
|
# include "sys/mman.h"
|
2007-10-24 11:24:19 -06:00
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
# define OPEN open
|
|
|
|
# define CLOSE close
|
|
|
|
# define READ read
|
|
|
|
# define WRITE write
|
|
|
|
# define STAT stat
|
|
|
|
# define STRUCT_STAT struct stat
|
|
|
|
# define MKDIR mkdir
|
|
|
|
# define CREAT creat
|
2007-09-11 19:13:05 -06:00
|
|
|
# define UNLINK unlink
|
2007-07-26 18:06:05 -06:00
|
|
|
# define OPEN_MASK 0
|
|
|
|
#endif
|
|
|
|
|
2008-10-03 14:15:47 -06:00
|
|
|
inline void* operator new(size_t, void* p) throw() { return p; }
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
exists(const char* path)
|
|
|
|
{
|
|
|
|
STRUCT_STAT s;
|
|
|
|
return STAT(path, &s) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
doOpen(JNIEnv* e, const char* path, int mask)
|
|
|
|
{
|
|
|
|
int fd = OPEN(path, mask | OPEN_MASK, S_IRUSR | S_IWUSR);
|
|
|
|
if (fd == -1) {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2007-07-26 18:06:05 -06:00
|
|
|
}
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
doClose(JNIEnv* e, jint fd)
|
|
|
|
{
|
|
|
|
int r = CLOSE(fd);
|
|
|
|
if (r == -1) {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2007-07-26 18:06:05 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
doRead(JNIEnv* e, jint fd, jbyte* data, jint length)
|
|
|
|
{
|
|
|
|
int r = READ(fd, data, length);
|
|
|
|
if (r > 0) {
|
|
|
|
return r;
|
|
|
|
} else if (r == 0) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2007-07-26 18:06:05 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
doWrite(JNIEnv* e, jint fd, const jbyte* data, jint length)
|
|
|
|
{
|
|
|
|
int r = WRITE(fd, data, length);
|
|
|
|
if (r != length) {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2008-10-03 14:15:47 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
|
|
|
class Mapping {
|
|
|
|
public:
|
|
|
|
Mapping(uint8_t* start, size_t length, HANDLE mapping, HANDLE file):
|
|
|
|
start(start),
|
|
|
|
length(length),
|
|
|
|
mapping(mapping),
|
|
|
|
file(file)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
uint8_t* start;
|
|
|
|
size_t length;
|
|
|
|
HANDLE mapping;
|
|
|
|
HANDLE file;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline Mapping*
|
|
|
|
map(JNIEnv* e, const char* path)
|
|
|
|
{
|
|
|
|
Mapping* result = 0;
|
|
|
|
HANDLE file = CreateFile(path, FILE_READ_DATA, FILE_SHARE_READ, 0,
|
|
|
|
OPEN_EXISTING, 0, 0);
|
|
|
|
if (file != INVALID_HANDLE_VALUE) {
|
|
|
|
unsigned size = GetFileSize(file, 0);
|
|
|
|
if (size != INVALID_FILE_SIZE) {
|
|
|
|
HANDLE mapping = CreateFileMapping(file, 0, PAGE_READONLY, 0, size, 0);
|
|
|
|
if (mapping) {
|
|
|
|
void* data = MapViewOfFile(mapping, FILE_MAP_READ, 0, 0, 0);
|
|
|
|
if (data) {
|
|
|
|
void* p = allocate(e, sizeof(Mapping));
|
|
|
|
if (not e->ExceptionOccurred()) {
|
|
|
|
result = new (p)
|
|
|
|
Mapping(static_cast<uint8_t*>(data), size, file, mapping);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result == 0) {
|
|
|
|
CloseHandle(mapping);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result == 0) {
|
|
|
|
CloseHandle(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (result == 0 and not e->ExceptionOccurred()) {
|
|
|
|
throwNew(e, "java/io/IOException", "%d", GetLastError());
|
|
|
|
}
|
|
|
|
return result;
|
2007-07-26 18:06:05 -06:00
|
|
|
}
|
|
|
|
|
2008-10-03 14:15:47 -06:00
|
|
|
inline void
|
|
|
|
unmap(JNIEnv*, Mapping* mapping)
|
|
|
|
{
|
|
|
|
UnmapViewOfFile(mapping->start);
|
|
|
|
CloseHandle(mapping->mapping);
|
|
|
|
CloseHandle(mapping->file);
|
|
|
|
free(mapping);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // not WIN32
|
|
|
|
|
|
|
|
class Mapping {
|
|
|
|
public:
|
|
|
|
Mapping(uint8_t* start, size_t length):
|
|
|
|
start(start),
|
|
|
|
length(length)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
uint8_t* start;
|
|
|
|
size_t length;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline Mapping*
|
|
|
|
map(JNIEnv* e, const char* path)
|
|
|
|
{
|
|
|
|
Mapping* result = 0;
|
|
|
|
int fd = open(path, O_RDONLY);
|
|
|
|
if (fd != -1) {
|
|
|
|
struct stat s;
|
|
|
|
int r = fstat(fd, &s);
|
|
|
|
if (r != -1) {
|
|
|
|
void* data = mmap(0, s.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
|
|
|
|
if (data) {
|
|
|
|
void* p = allocate(e, sizeof(Mapping));
|
|
|
|
if (not e->ExceptionOccurred()) {
|
|
|
|
result = new (p) Mapping(static_cast<uint8_t*>(data), s.st_size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
if (result == 0 and not e->ExceptionOccurred()) {
|
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
unmap(JNIEnv*, Mapping* mapping)
|
|
|
|
{
|
|
|
|
munmap(mapping->start, mapping->length);
|
|
|
|
free(mapping);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // not WIN32
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
} // namespace
|
|
|
|
|
2007-08-20 18:24:54 -06:00
|
|
|
extern "C" JNIEXPORT jstring JNICALL
|
|
|
|
Java_java_io_File_toCanonicalPath(JNIEnv* /*e*/, jclass, jstring path)
|
|
|
|
{
|
|
|
|
// todo
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
extern "C" JNIEXPORT jstring JNICALL
|
|
|
|
Java_java_io_File_toAbsolutePath(JNIEnv* /*e*/, jclass, jstring path)
|
|
|
|
{
|
|
|
|
// todo
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT jlong JNICALL
|
|
|
|
Java_java_io_File_length(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
STRUCT_STAT s;
|
|
|
|
int r = STAT(chars, &s);
|
|
|
|
if (r == 0) {
|
|
|
|
return s.st_size;
|
|
|
|
}
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_File_mkdir(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
if (not exists(chars)) {
|
|
|
|
int r = ::MKDIR(chars, 0700);
|
|
|
|
if (r != 0) {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2007-07-26 18:06:05 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_File_createNewFile(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
if (not exists(chars)) {
|
|
|
|
int fd = CREAT(chars, 0600);
|
|
|
|
if (fd == -1) {
|
2007-08-14 07:27:10 -06:00
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
2007-07-26 18:06:05 -06:00
|
|
|
} else {
|
|
|
|
doClose(e, fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-07 17:44:43 -06:00
|
|
|
extern "C" JNIEXPORT void JNICALL
|
2007-09-11 19:13:05 -06:00
|
|
|
Java_java_io_File_delete(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
2008-05-07 17:44:43 -06:00
|
|
|
int r = UNLINK(chars);
|
|
|
|
if (r != 0) {
|
|
|
|
throwNew(e, "java/io/IOException", strerror(errno));
|
|
|
|
}
|
2007-09-11 19:13:05 -06:00
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-29 14:57:33 -06:00
|
|
|
extern "C" JNIEXPORT jboolean JNICALL
|
|
|
|
Java_java_io_File_isDirectory(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
STRUCT_STAT s;
|
|
|
|
int r = STAT(chars, &s);
|
|
|
|
bool v = (r == 0 and S_ISDIR(s.st_mode));
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
return v;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
extern "C" JNIEXPORT jboolean JNICALL
|
|
|
|
Java_java_io_File_exists(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
bool v = exists(chars);
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
return v;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-13 18:14:37 -06:00
|
|
|
extern "C" JNIEXPORT jlong JNICALL
|
|
|
|
Java_java_io_File_openDir(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
jlong handle = reinterpret_cast<jlong>(opendir(chars));
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
return handle;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT jstring JNICALL
|
|
|
|
Java_java_io_File_readDir(JNIEnv* e, jclass, jlong handle)
|
|
|
|
{
|
|
|
|
struct dirent * directoryEntry;
|
|
|
|
|
|
|
|
if (handle!=0) {
|
|
|
|
directoryEntry = readdir(reinterpret_cast<DIR*>(handle));
|
|
|
|
if (directoryEntry == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return e->NewStringUTF(directoryEntry->d_name);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_File_closeDir(JNIEnv* , jclass, jlong handle)
|
|
|
|
{
|
|
|
|
if (handle!=0) {
|
|
|
|
closedir(reinterpret_cast<DIR*>(handle));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-26 18:06:05 -06:00
|
|
|
extern "C" JNIEXPORT jint JNICALL
|
|
|
|
Java_java_io_FileInputStream_open(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
int fd = doOpen(e, chars, O_RDONLY);
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
return fd;
|
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT jint JNICALL
|
|
|
|
Java_java_io_FileInputStream_read__I(JNIEnv* e, jclass, jint fd)
|
|
|
|
{
|
|
|
|
jbyte data;
|
|
|
|
int r = doRead(e, fd, &data, 1);
|
|
|
|
if (r <= 0) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
2007-10-04 13:57:39 -06:00
|
|
|
return data & 0xff;
|
2007-07-26 18:06:05 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT jint JNICALL
|
|
|
|
Java_java_io_FileInputStream_read__I_3BII
|
|
|
|
(JNIEnv* e, jclass, jint fd, jbyteArray b, jint offset, jint length)
|
|
|
|
{
|
|
|
|
jbyte* data = static_cast<jbyte*>(malloc(length));
|
|
|
|
if (data == 0) {
|
|
|
|
throwNew(e, "java/lang/OutOfMemoryError", 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int r = doRead(e, fd, data, length);
|
|
|
|
|
|
|
|
e->SetByteArrayRegion(b, offset, length, data);
|
|
|
|
|
|
|
|
free(data);
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_FileInputStream_close(JNIEnv* e, jclass, jint fd)
|
|
|
|
{
|
|
|
|
doClose(e, fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT jint JNICALL
|
|
|
|
Java_java_io_FileOutputStream_open(JNIEnv* e, jclass, jstring path)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
2008-06-26 15:58:14 -06:00
|
|
|
int fd = doOpen(e, chars, O_WRONLY | O_CREAT | O_TRUNC);
|
2007-07-26 18:06:05 -06:00
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
return fd;
|
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_FileOutputStream_write__II(JNIEnv* e, jclass, jint fd, jint c)
|
|
|
|
{
|
|
|
|
jbyte data = c;
|
|
|
|
doWrite(e, fd, &data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_FileOutputStream_write__I_3BII
|
|
|
|
(JNIEnv* e, jclass, jint fd, jbyteArray b, jint offset, jint length)
|
|
|
|
{
|
|
|
|
jbyte* data = static_cast<jbyte*>(malloc(length));
|
|
|
|
if (data == 0) {
|
|
|
|
throwNew(e, "java/lang/OutOfMemoryError", 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
e->GetByteArrayRegion(b, offset, length, data);
|
|
|
|
|
|
|
|
if (not e->ExceptionCheck()) {
|
|
|
|
doWrite(e, fd, data, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_FileOutputStream_close(JNIEnv* e, jclass, jint fd)
|
|
|
|
{
|
|
|
|
doClose(e, fd);
|
|
|
|
}
|
2008-10-03 14:15:47 -06:00
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_RandomAccessFile_open(JNIEnv* e, jclass, jstring path,
|
|
|
|
jlongArray result)
|
|
|
|
{
|
|
|
|
const char* chars = e->GetStringUTFChars(path, 0);
|
|
|
|
if (chars) {
|
|
|
|
Mapping* mapping = map(e, chars);
|
|
|
|
|
|
|
|
jlong peer = reinterpret_cast<jlong>(mapping);
|
|
|
|
e->SetLongArrayRegion(result, 0, 1, &peer);
|
|
|
|
|
2008-10-06 17:30:07 -06:00
|
|
|
jlong length = (mapping ? mapping->length : 0);
|
2008-10-03 14:15:47 -06:00
|
|
|
e->SetLongArrayRegion(result, 1, 1, &length);
|
|
|
|
|
|
|
|
e->ReleaseStringUTFChars(path, chars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_RandomAccessFile_copy(JNIEnv* e, jclass, jlong peer,
|
|
|
|
jlong position, jbyteArray buffer,
|
|
|
|
int offset, int length)
|
|
|
|
{
|
|
|
|
uint8_t* dst = reinterpret_cast<uint8_t*>
|
|
|
|
(e->GetPrimitiveArrayCritical(buffer, 0));
|
|
|
|
|
|
|
|
memcpy(dst + offset,
|
|
|
|
reinterpret_cast<Mapping*>(peer)->start + position,
|
|
|
|
length);
|
|
|
|
|
|
|
|
e->ReleasePrimitiveArrayCritical(buffer, dst, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL
|
|
|
|
Java_java_io_RandomAccessFile_close(JNIEnv* e, jclass, jlong peer)
|
|
|
|
{
|
|
|
|
unmap(e, reinterpret_cast<Mapping*>(peer));
|
|
|
|
}
|