summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/posix.c42
-rw-r--r--src/posix.h17
-rw-r--r--src/win32/dir.c1
-rw-r--r--src/win32/posix.c177
-rw-r--r--src/win32/posix.h13
-rw-r--r--src/win32/utf8-conv.c68
-rw-r--r--src/win32/utf8-conv.h17
7 files changed, 278 insertions, 57 deletions
diff --git a/src/posix.c b/src/posix.c
index 61c79b071..1b85b053d 100644
--- a/src/posix.c
+++ b/src/posix.c
@@ -10,6 +10,8 @@
#include <stdio.h>
#include <ctype.h>
+#ifndef GIT_WIN32
+
int p_open(const char *path, int flags)
{
return open(path, flags | O_BINARY);
@@ -20,6 +22,25 @@ int p_creat(const char *path, int mode)
return open(path, O_WRONLY | O_CREAT | O_TRUNC | O_BINARY, mode);
}
+int p_getcwd(char *buffer_out, size_t size)
+{
+ char *cwd_buffer;
+
+ assert(buffer_out && size > 0);
+
+ cwd_buffer = getcwd(buffer_out, size);
+
+ if (cwd_buffer == NULL)
+ return git__throw(GIT_EOSERR, "Failed to retrieve current working directory");
+
+ git_path_mkposix(buffer_out);
+
+ git_path_join(buffer_out, buffer_out, ""); //Ensure the path ends with a trailing slash
+ return GIT_SUCCESS;
+}
+
+#endif
+
int p_read(git_file fd, void *buf, size_t cnt)
{
char *b = buf;
@@ -57,24 +78,3 @@ int p_write(git_file fd, const void *buf, size_t cnt)
}
return GIT_SUCCESS;
}
-
-int p_getcwd(char *buffer_out, size_t size)
-{
- char *cwd_buffer;
-
- assert(buffer_out && size > 0);
-
-#ifdef GIT_WIN32
- cwd_buffer = _getcwd(buffer_out, size);
-#else
- cwd_buffer = getcwd(buffer_out, size);
-#endif
-
- if (cwd_buffer == NULL)
- return git__throw(GIT_EOSERR, "Failed to retrieve current working directory");
-
- git_path_mkposix(buffer_out);
-
- git_path_join(buffer_out, buffer_out, ""); //Ensure the path ends with a trailing slash
- return GIT_SUCCESS;
-}
diff --git a/src/posix.h b/src/posix.h
index 7c1ac66c5..48b0255bc 100644
--- a/src/posix.h
+++ b/src/posix.h
@@ -33,19 +33,26 @@ typedef int git_file;
* Use your manpages to check the docs on these.
* Straightforward
*/
-extern int p_open(const char *path, int flags);
-extern int p_creat(const char *path, int mode);
+
extern int p_read(git_file fd, void *buf, size_t cnt);
extern int p_write(git_file fd, const void *buf, size_t cnt);
-extern int p_getcwd(char *buffer_out, size_t size);
+#define p_fstat(f,b) fstat(f, b)
#define p_lseek(f,n,w) lseek(f, n, w)
+#define p_close(fd) close(fd)
+
+extern int p_open(const char *path, int flags);
+extern int p_creat(const char *path, int mode);
+extern int p_getcwd(char *buffer_out, size_t size);
+
+#ifndef GIT_WIN32
+
#define p_stat(p,b) stat(p, b)
-#define p_fstat(f,b) fstat(f, b)
#define p_chdir(p) chdir(p)
#define p_rmdir(p) rmdir(p)
#define p_chmod(p,m) chmod(p, m)
-#define p_close(fd) close(fd)
+
+#endif
/**
* Platform-dependent methods
diff --git a/src/win32/dir.c b/src/win32/dir.c
index 01339577b..380dc0c13 100644
--- a/src/win32/dir.c
+++ b/src/win32/dir.c
@@ -6,6 +6,7 @@
*/
#define GIT__WIN32_NO_WRAP_DIR
#include "dir.h"
+#include "utf8-conv.h"
static int init_filter(char *filter, size_t n, const char *dir)
{
diff --git a/src/win32/posix.c b/src/win32/posix.c
index 8038fdea2..af54e7f82 100644
--- a/src/win32/posix.c
+++ b/src/win32/posix.c
@@ -6,13 +6,23 @@
*/
#include "posix.h"
#include "path.h"
+#include "utf8-conv.h"
#include <errno.h>
#include <io.h>
+#include <fcntl.h>
+
int p_unlink(const char *path)
{
- chmod(path, 0666);
- return unlink(path);
+ int ret = 0;
+ wchar_t* buf;
+
+ buf = conv_utf8_to_utf16(path);
+ _wchmod(buf, 0666);
+ ret = _wunlink(buf);
+ free(buf);
+
+ return ret;
}
int p_fsync(int fd)
@@ -49,8 +59,9 @@ GIT_INLINE(time_t) filetime_to_time_t(const FILETIME *ft)
static int do_lstat(const char *file_name, struct stat *buf)
{
WIN32_FILE_ATTRIBUTE_DATA fdata;
+ wchar_t* fbuf = conv_utf8_to_utf16(file_name);
- if (GetFileAttributesExA(file_name, GetFileExInfoStandard, &fdata)) {
+ if (GetFileAttributesExW(fbuf, GetFileExInfoStandard, &fdata)) {
int fMode = S_IREAD;
if (fdata.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
@@ -74,22 +85,26 @@ static int do_lstat(const char *file_name, struct stat *buf)
buf->st_atime = filetime_to_time_t(&(fdata.ftLastAccessTime));
buf->st_mtime = filetime_to_time_t(&(fdata.ftLastWriteTime));
buf->st_ctime = filetime_to_time_t(&(fdata.ftCreationTime));
+
+ free(fbuf);
return GIT_SUCCESS;
}
- switch (GetLastError()) {
- case ERROR_ACCESS_DENIED:
- case ERROR_SHARING_VIOLATION:
- case ERROR_LOCK_VIOLATION:
- case ERROR_SHARING_BUFFER_EXCEEDED:
- return GIT_EOSERR;
-
- case ERROR_BUFFER_OVERFLOW:
- case ERROR_NOT_ENOUGH_MEMORY:
- return GIT_ENOMEM;
+ free(fbuf);
- default:
- return GIT_EINVALIDPATH;
+ switch (GetLastError()) {
+ case ERROR_ACCESS_DENIED:
+ case ERROR_SHARING_VIOLATION:
+ case ERROR_LOCK_VIOLATION:
+ case ERROR_SHARING_BUFFER_EXCEEDED:
+ return GIT_EOSERR;
+
+ case ERROR_BUFFER_OVERFLOW:
+ case ERROR_NOT_ENOUGH_MEMORY:
+ return GIT_ENOMEM;
+
+ default:
+ return GIT_EINVALIDPATH;
}
}
@@ -124,10 +139,12 @@ int p_lstat(const char *file_name, struct stat *buf)
int p_readlink(const char *link, char *target, size_t target_len)
{
- typedef DWORD (WINAPI *fpath_func)(HANDLE, LPTSTR, DWORD, DWORD);
+ typedef DWORD (WINAPI *fpath_func)(HANDLE, LPWSTR, DWORD, DWORD);
static fpath_func pGetFinalPath = NULL;
HANDLE hFile;
DWORD dwRet;
+ wchar_t* link_w;
+ wchar_t* target_w;
/*
* Try to load the pointer to pGetFinalPath dynamically, because
@@ -137,28 +154,47 @@ int p_readlink(const char *link, char *target, size_t target_len)
HINSTANCE library = LoadLibrary("kernel32");
if (library != NULL)
- pGetFinalPath = (fpath_func)GetProcAddress(library, "GetFinalPathNameByHandleA");
+ pGetFinalPath = (fpath_func)GetProcAddress(library, "GetFinalPathNameByHandleW");
if (pGetFinalPath == NULL)
return git__throw(GIT_EOSERR,
- "'GetFinalPathNameByHandleA' is not available in this platform");
+ "'GetFinalPathNameByHandleW' is not available in this platform");
}
- hFile = CreateFile(link, // file to open
- GENERIC_READ, // open for reading
- FILE_SHARE_READ, // share for reading
- NULL, // default security
- OPEN_EXISTING, // existing file only
- FILE_FLAG_BACKUP_SEMANTICS, // normal file
- NULL); // no attr. template
+ link_w = conv_utf8_to_utf16(link);
+
+ hFile = CreateFileW(link_w, // file to open
+ GENERIC_READ, // open for reading
+ FILE_SHARE_READ, // share for reading
+ NULL, // default security
+ OPEN_EXISTING, // existing file only
+ FILE_FLAG_BACKUP_SEMANTICS, // normal file
+ NULL); // no attr. template
+
+ free(link_w);
if (hFile == INVALID_HANDLE_VALUE)
return GIT_EOSERR;
- dwRet = pGetFinalPath(hFile, target, target_len, 0x0);
- if (dwRet >= target_len)
+ if (target_len <= 0) {
+ return GIT_EINVALIDARGS;
+ }
+
+ target_w = (wchar_t*)git__malloc(target_len * sizeof(wchar_t));
+
+ dwRet = pGetFinalPath(hFile, target_w, target_len, 0x0);
+ if (dwRet >= target_len) {
+ free(target_w);
+ CloseHandle(hFile);
return GIT_ENOMEM;
+ }
+
+ if (!WideCharToMultiByte(CP_UTF8, 0, target_w, -1, target, target_len * sizeof(char), NULL, NULL)) {
+ free(target_w);
+ return GIT_EOSERR;
+ }
+ free(target_w);
CloseHandle(hFile);
if (dwRet > 4) {
@@ -184,13 +220,82 @@ int p_readlink(const char *link, char *target, size_t target_len)
return dwRet;
}
+int p_open(const char *path, int flags)
+{
+ int fd;
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ fd = _wopen(buf, flags | _O_BINARY);
+
+ free(buf);
+ return fd;
+}
+
+int p_creat(const char *path, int mode)
+{
+ int fd;
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ fd = _wopen(buf, _O_WRONLY | _O_CREAT | _O_TRUNC | _O_BINARY, mode);
+
+ free(buf);
+ return fd;
+}
+
+int p_getcwd(char *buffer_out, size_t size)
+{
+ wchar_t* buf = (wchar_t*)git__malloc(sizeof(wchar_t) * (int)size);
+ _wgetcwd(buf, (int)size);
+
+ if (!WideCharToMultiByte(CP_UTF8, 0, buf, -1, buffer_out, size, NULL, NULL)) {
+ free(buf);
+ return GIT_EOSERR;
+ }
+
+ free(buf);
+ return GIT_SUCCESS;
+}
+
+int p_stat(const char* path, struct stat* buf)
+{
+ return do_lstat(path, buf);
+}
+
+int p_chdir(const char* path)
+{
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ int ret = _wchdir(buf);
+
+ free(buf);
+ return ret;
+}
+
+int p_chmod(const char* path, int mode)
+{
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ int ret = _wchmod(buf, mode);
+
+ free(buf);
+ return ret;
+}
+
+int p_rmdir(const char* path)
+{
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ int ret = _wrmdir(buf);
+
+ free(buf);
+ return ret;
+}
+
int p_hide_directory__w32(const char *path)
{
int error;
+ wchar_t* buf = conv_utf8_to_utf16(path);
- error = SetFileAttributes(path, FILE_ATTRIBUTE_HIDDEN) != 0 ?
+ error = SetFileAttributesW(buf, FILE_ATTRIBUTE_HIDDEN) != 0 ?
GIT_SUCCESS : GIT_ERROR; /* MSDN states a "non zero" value indicates a success */
+ free(buf);
+
if (error < GIT_SUCCESS)
error = git__throw(GIT_EOSERR, "Failed to hide directory '%s'", path);
@@ -200,18 +305,30 @@ int p_hide_directory__w32(const char *path)
char *p_realpath(const char *orig_path, char *buffer)
{
int ret, alloc = 0;
-
+ wchar_t* orig_path_w = conv_utf8_to_utf16(orig_path);
+ wchar_t* buffer_w = (wchar_t*)git__malloc(GIT_PATH_MAX * sizeof(wchar_t));
+
if (buffer == NULL) {
buffer = (char *)git__malloc(GIT_PATH_MAX);
alloc = 1;
}
- ret = GetFullPathName(orig_path, GIT_PATH_MAX, buffer, NULL);
+ ret = GetFullPathNameW(orig_path_w, GIT_PATH_MAX, buffer_w, NULL);
+ free(orig_path_w);
+
if (!ret || ret > GIT_PATH_MAX) {
+ free(buffer_w);
if (alloc) free(buffer);
+
return NULL;
}
+ if (!WideCharToMultiByte(CP_UTF8, 0, buffer_w, -1, buffer, GIT_PATH_MAX, NULL, NULL)) {
+ free(buffer_w);
+ if (alloc) free(buffer);
+ }
+
+ free(buffer_w);
git_path_mkposix(buffer);
return buffer;
}
diff --git a/src/win32/posix.h b/src/win32/posix.h
index 58fd05007..4c45fd3e4 100644
--- a/src/win32/posix.h
+++ b/src/win32/posix.h
@@ -9,6 +9,7 @@
#include "common.h"
#include "fnmatch.h"
+#include "utf8-conv.h"
GIT_INLINE(int) p_link(const char *GIT_UNUSED(old), const char *GIT_UNUSED(new))
{
@@ -20,8 +21,13 @@ GIT_INLINE(int) p_link(const char *GIT_UNUSED(old), const char *GIT_UNUSED(new))
GIT_INLINE(int) p_mkdir(const char *path, int GIT_UNUSED(mode))
{
+ wchar_t* buf = conv_utf8_to_utf16(path);
+ int ret = _wmkdir(buf);
+
GIT_UNUSED_ARG(mode)
- return mkdir(path);
+
+ free(buf);
+ return ret;
}
extern int p_unlink(const char *path);
@@ -33,5 +39,10 @@ extern int p_vsnprintf(char *buffer, size_t count, const char *format, va_list a
extern int p_snprintf(char *buffer, size_t count, const char *format, ...) GIT_FORMAT_PRINTF(3, 4);
extern int p_mkstemp(char *tmp_path);
extern int p_setenv(const char* name, const char* value, int overwrite);
+extern int p_stat(const char* path, struct stat* buf);
+extern int p_chdir(const char* path);
+extern int p_chmod(const char* path, int mode);
+extern int p_rmdir(const char* path);
+
#endif
diff --git a/src/win32/utf8-conv.c b/src/win32/utf8-conv.c
new file mode 100644
index 000000000..dec6f8e79
--- /dev/null
+++ b/src/win32/utf8-conv.c
@@ -0,0 +1,68 @@
+/*
+ * Copyright (C) 2009-2011 the libgit2 contributors
+ *
+ * This file is part of libgit2, distributed under the GNU GPL v2 with
+ * a Linking Exception. For full terms see the included COPYING file.
+ */
+
+#include "common.h"
+#include "utf8-conv.h"
+
+wchar_t* conv_utf8_to_utf16(const char* str)
+{
+ wchar_t* ret;
+ int cb;
+
+ if (!str) {
+ return NULL;
+ }
+
+ cb = strlen(str) * sizeof(wchar_t);
+ if (cb == 0) {
+ ret = (wchar_t*)git__malloc(sizeof(wchar_t));
+ ret[0] = 0;
+ return ret;
+ }
+
+ /* Add space for null terminator */
+ cb += sizeof(wchar_t);
+
+ ret = (wchar_t*)git__malloc(cb);
+
+ if (MultiByteToWideChar(CP_UTF8, 0, str, -1, ret, cb) == 0) {
+ free(ret);
+ ret = NULL;
+ }
+
+ return ret;
+}
+
+char* conv_utf16_to_utf8(const wchar_t* str)
+{
+ char* ret;
+ int cb;
+
+ if (!str) {
+ return NULL;
+ }
+
+ cb = wcslen(str) * sizeof(char);
+ if (cb == 0) {
+ ret = (char*)git__malloc(sizeof(char));
+ ret[0] = 0;
+ return ret;
+ }
+
+ /* Add space for null terminator */
+ cb += sizeof(char);
+
+ ret = (char*)git__malloc(cb);
+
+ if (WideCharToMultiByte(CP_UTF8, 0, str, -1, ret, cb, NULL, NULL) == 0) {
+ free(ret);
+ ret = NULL;
+ }
+
+ return ret;
+
+}
diff --git a/src/win32/utf8-conv.h b/src/win32/utf8-conv.h
new file mode 100644
index 000000000..1967ac3a1
--- /dev/null
+++ b/src/win32/utf8-conv.h
@@ -0,0 +1,17 @@
+/*
+ * Copyright (C) 2009-2011 the libgit2 contributors
+ *
+ * This file is part of libgit2, distributed under the GNU GPL v2 with
+ * a Linking Exception. For full terms see the included COPYING file.
+ */
+
+#include <wchar.h>
+
+#ifndef INCLUDE_git_utf8conv_h__
+#define INCLUDE_git_utf8conv_h__
+
+wchar_t* conv_utf8_to_utf16(const char* str);
+char* conv_utf16_to_utf8(const wchar_t* str);
+
+#endif
+