mirror of
https://git.proxmox.com/git/libgit2
synced 2025-05-02 16:34:37 +00:00
513 lines
8.8 KiB
C
513 lines
8.8 KiB
C
#define GIT__NO_HIDE_MALLOC
|
|
#include <git2.h>
|
|
#include "common.h"
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <ctype.h>
|
|
|
|
void git_libgit2_version(int *major, int *minor, int *rev)
|
|
{
|
|
*major = LIBGIT2_VER_MAJOR;
|
|
*minor = LIBGIT2_VER_MINOR;
|
|
*rev = LIBGIT2_VER_REVISION;
|
|
}
|
|
|
|
void git_strarray_free(git_strarray *array)
|
|
{
|
|
size_t i;
|
|
for (i = 0; i < array->count; ++i)
|
|
free(array->strings[i]);
|
|
|
|
free(array->strings);
|
|
}
|
|
|
|
int git__strtol32(long *result, const char *nptr, const char **endptr, int base)
|
|
{
|
|
const char *p;
|
|
long n, nn;
|
|
int c, ovfl, v, neg, ndig;
|
|
|
|
p = nptr;
|
|
neg = 0;
|
|
n = 0;
|
|
ndig = 0;
|
|
ovfl = 0;
|
|
|
|
/*
|
|
* White space
|
|
*/
|
|
while (isspace(*p))
|
|
p++;
|
|
|
|
/*
|
|
* Sign
|
|
*/
|
|
if (*p == '-' || *p == '+')
|
|
if (*p++ == '-')
|
|
neg = 1;
|
|
|
|
/*
|
|
* Base
|
|
*/
|
|
if (base == 0) {
|
|
if (*p != '0')
|
|
base = 10;
|
|
else {
|
|
base = 8;
|
|
if (p[1] == 'x' || p[1] == 'X') {
|
|
p += 2;
|
|
base = 16;
|
|
}
|
|
}
|
|
} else if (base == 16 && *p == '0') {
|
|
if (p[1] == 'x' || p[1] == 'X')
|
|
p += 2;
|
|
} else if (base < 0 || 36 < base)
|
|
goto Return;
|
|
|
|
/*
|
|
* Non-empty sequence of digits
|
|
*/
|
|
for (;; p++,ndig++) {
|
|
c = *p;
|
|
v = base;
|
|
if ('0'<=c && c<='9')
|
|
v = c - '0';
|
|
else if ('a'<=c && c<='z')
|
|
v = c - 'a' + 10;
|
|
else if ('A'<=c && c<='Z')
|
|
v = c - 'A' + 10;
|
|
if (v >= base)
|
|
break;
|
|
nn = n*base + v;
|
|
if (nn < n)
|
|
ovfl = 1;
|
|
n = nn;
|
|
}
|
|
|
|
Return:
|
|
if (ndig == 0)
|
|
return git__throw(GIT_ENOTNUM, "Failed to convert string to long. Not a number");
|
|
|
|
if (endptr)
|
|
*endptr = p;
|
|
|
|
if (ovfl)
|
|
return git__throw(GIT_EOVERFLOW, "Failed to convert string to long. Overflow error");
|
|
|
|
*result = neg ? -n : n;
|
|
return GIT_SUCCESS;
|
|
}
|
|
|
|
int git__fmt(char *buf, size_t buf_sz, const char *fmt, ...)
|
|
{
|
|
va_list va;
|
|
int r;
|
|
|
|
va_start(va, fmt);
|
|
r = vsnprintf(buf, buf_sz, fmt, va);
|
|
va_end(va);
|
|
if (r < 0 || ((size_t) r) >= buf_sz)
|
|
return git__throw(GIT_ERROR, "Failed to format string");
|
|
return r;
|
|
}
|
|
|
|
void git__strntolower(char *str, int len)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
str[i] = (char) tolower(str[i]);
|
|
}
|
|
}
|
|
|
|
void git__strtolower(char *str)
|
|
{
|
|
git__strntolower(str, strlen(str));
|
|
}
|
|
|
|
int git__prefixcmp(const char *str, const char *prefix)
|
|
{
|
|
for (;;) {
|
|
char p = *(prefix++), s;
|
|
if (!p)
|
|
return 0;
|
|
if ((s = *(str++)) != p)
|
|
return s - p;
|
|
}
|
|
}
|
|
|
|
int git__suffixcmp(const char *str, const char *suffix)
|
|
{
|
|
size_t a = strlen(str);
|
|
size_t b = strlen(suffix);
|
|
if (a < b)
|
|
return -1;
|
|
return strcmp(str + (a - b), suffix);
|
|
}
|
|
|
|
/*
|
|
* Based on the Android implementation, BSD licensed.
|
|
* Check http://android.git.kernel.org/
|
|
*/
|
|
int git__basename_r(char *buffer, size_t bufflen, const char *path)
|
|
{
|
|
const char *endp, *startp;
|
|
int len, result;
|
|
|
|
/* Empty or NULL string gets treated as "." */
|
|
if (path == NULL || *path == '\0') {
|
|
startp = ".";
|
|
len = 1;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Strip trailing slashes */
|
|
endp = path + strlen(path) - 1;
|
|
while (endp > path && *endp == '/')
|
|
endp--;
|
|
|
|
/* All slashes becomes "/" */
|
|
if (endp == path && *endp == '/') {
|
|
startp = "/";
|
|
len = 1;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Find the start of the base */
|
|
startp = endp;
|
|
while (startp > path && *(startp - 1) != '/')
|
|
startp--;
|
|
|
|
len = endp - startp +1;
|
|
|
|
Exit:
|
|
result = len;
|
|
if (buffer == NULL) {
|
|
return result;
|
|
}
|
|
if (len > (int)bufflen-1) {
|
|
len = (int)bufflen-1;
|
|
result = GIT_ENOMEM;
|
|
}
|
|
|
|
if (len >= 0) {
|
|
memmove(buffer, startp, len);
|
|
buffer[len] = 0;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Based on the Android implementation, BSD licensed.
|
|
* Check http://android.git.kernel.org/
|
|
*/
|
|
int git__dirname_r(char *buffer, size_t bufflen, const char *path)
|
|
{
|
|
const char *endp;
|
|
int result, len;
|
|
|
|
/* Empty or NULL string gets treated as "." */
|
|
if (path == NULL || *path == '\0') {
|
|
path = ".";
|
|
len = 1;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Strip trailing slashes */
|
|
endp = path + strlen(path) - 1;
|
|
while (endp > path && *endp == '/')
|
|
endp--;
|
|
|
|
/* Find the start of the dir */
|
|
while (endp > path && *endp != '/')
|
|
endp--;
|
|
|
|
/* Either the dir is "/" or there are no slashes */
|
|
if (endp == path) {
|
|
path = (*endp == '/') ? "/" : ".";
|
|
len = 1;
|
|
goto Exit;
|
|
}
|
|
|
|
do {
|
|
endp--;
|
|
} while (endp > path && *endp == '/');
|
|
|
|
len = endp - path +1;
|
|
|
|
Exit:
|
|
result = len;
|
|
if (len+1 > GIT_PATH_MAX) {
|
|
return GIT_ENOMEM;
|
|
}
|
|
if (buffer == NULL)
|
|
return result;
|
|
|
|
if (len > (int)bufflen-1) {
|
|
len = (int)bufflen-1;
|
|
result = GIT_ENOMEM;
|
|
}
|
|
|
|
if (len >= 0) {
|
|
memmove(buffer, path, len);
|
|
buffer[len] = 0;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
char *git__dirname(const char *path)
|
|
{
|
|
char *dname = NULL;
|
|
int len;
|
|
|
|
len = (path ? strlen(path) : 0) + 2;
|
|
dname = (char *)git__malloc(len);
|
|
if (dname == NULL)
|
|
return NULL;
|
|
|
|
if (git__dirname_r(dname, len, path) < GIT_SUCCESS) {
|
|
free(dname);
|
|
return NULL;
|
|
}
|
|
|
|
return dname;
|
|
}
|
|
|
|
char *git__basename(const char *path)
|
|
{
|
|
char *bname = NULL;
|
|
int len;
|
|
|
|
len = (path ? strlen(path) : 0) + 2;
|
|
bname = (char *)git__malloc(len);
|
|
if (bname == NULL)
|
|
return NULL;
|
|
|
|
if (git__basename_r(bname, len, path) < GIT_SUCCESS) {
|
|
free(bname);
|
|
return NULL;
|
|
}
|
|
|
|
return bname;
|
|
}
|
|
|
|
|
|
const char *git__topdir(const char *path)
|
|
{
|
|
size_t len;
|
|
int i;
|
|
|
|
assert(path);
|
|
len = strlen(path);
|
|
|
|
if (!len || path[len - 1] != '/')
|
|
return NULL;
|
|
|
|
for (i = len - 2; i >= 0; --i)
|
|
if (path[i] == '/')
|
|
break;
|
|
|
|
return &path[i + 1];
|
|
}
|
|
|
|
void git__joinpath_n(char *buffer_out, int count, ...)
|
|
{
|
|
va_list ap;
|
|
int i;
|
|
char *buffer_start = buffer_out;
|
|
|
|
va_start(ap, count);
|
|
for (i = 0; i < count; ++i) {
|
|
const char *path;
|
|
int len;
|
|
|
|
path = va_arg(ap, const char *);
|
|
|
|
assert((i == 0) || path != buffer_start);
|
|
|
|
if (i > 0 && *path == '/' && buffer_out > buffer_start && buffer_out[-1] == '/')
|
|
path++;
|
|
|
|
if (!*path)
|
|
continue;
|
|
|
|
len = strlen(path);
|
|
memmove(buffer_out, path, len);
|
|
buffer_out = buffer_out + len;
|
|
|
|
if (i < count - 1 && buffer_out[-1] != '/')
|
|
*buffer_out++ = '/';
|
|
}
|
|
va_end(ap);
|
|
|
|
*buffer_out = '\0';
|
|
}
|
|
|
|
char *git__strtok(char **end, const char *sep)
|
|
{
|
|
char *ptr = *end;
|
|
|
|
while (*ptr && strchr(sep, *ptr))
|
|
++ptr;
|
|
|
|
if (*ptr) {
|
|
char *start = ptr;
|
|
*end = start + 1;
|
|
|
|
while (**end && !strchr(sep, **end))
|
|
++*end;
|
|
|
|
if (**end) {
|
|
**end = '\0';
|
|
++*end;
|
|
}
|
|
|
|
return start;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void git__hexdump(const char *buffer, size_t len)
|
|
{
|
|
static const size_t LINE_WIDTH = 16;
|
|
|
|
size_t line_count, last_line, i, j;
|
|
const char *line;
|
|
|
|
line_count = (len / LINE_WIDTH);
|
|
last_line = (len % LINE_WIDTH);
|
|
|
|
for (i = 0; i < line_count; ++i) {
|
|
line = buffer + (i * LINE_WIDTH);
|
|
for (j = 0; j < LINE_WIDTH; ++j, ++line)
|
|
printf("%02X ", (unsigned char)*line & 0xFF);
|
|
|
|
printf("| ");
|
|
|
|
line = buffer + (i * LINE_WIDTH);
|
|
for (j = 0; j < LINE_WIDTH; ++j, ++line)
|
|
printf("%c", (*line >= 32 && *line <= 126) ? *line : '.');
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
if (last_line > 0) {
|
|
|
|
line = buffer + (line_count * LINE_WIDTH);
|
|
for (j = 0; j < last_line; ++j, ++line)
|
|
printf("%02X ", (unsigned char)*line & 0xFF);
|
|
|
|
for (j = 0; j < (LINE_WIDTH - last_line); ++j)
|
|
printf(" ");
|
|
|
|
printf("| ");
|
|
|
|
line = buffer + (line_count * LINE_WIDTH);
|
|
for (j = 0; j < last_line; ++j, ++line)
|
|
printf("%c", (*line >= 32 && *line <= 126) ? *line : '.');
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
#ifdef GIT_LEGACY_HASH
|
|
uint32_t git__hash(const void *key, int len, unsigned int seed)
|
|
{
|
|
const uint32_t m = 0x5bd1e995;
|
|
const int r = 24;
|
|
uint32_t h = seed ^ len;
|
|
|
|
const unsigned char *data = (const unsigned char *)key;
|
|
|
|
while(len >= 4) {
|
|
uint32_t k = *(uint32_t *)data;
|
|
|
|
k *= m;
|
|
k ^= k >> r;
|
|
k *= m;
|
|
|
|
h *= m;
|
|
h ^= k;
|
|
|
|
data += 4;
|
|
len -= 4;
|
|
}
|
|
|
|
switch(len) {
|
|
case 3: h ^= data[2] << 16;
|
|
case 2: h ^= data[1] << 8;
|
|
case 1: h ^= data[0];
|
|
h *= m;
|
|
};
|
|
|
|
h ^= h >> 13;
|
|
h *= m;
|
|
h ^= h >> 15;
|
|
|
|
return h;
|
|
}
|
|
#else
|
|
/*
|
|
Cross-platform version of Murmurhash3
|
|
http://code.google.com/p/smhasher/wiki/MurmurHash3
|
|
by Austin Appleby (aappleby@gmail.com)
|
|
|
|
This code is on the public domain.
|
|
*/
|
|
uint32_t git__hash(const void *key, int len, uint32_t seed)
|
|
{
|
|
|
|
#define MURMUR_BLOCK() {\
|
|
k1 *= c1; \
|
|
k1 = git__rotl(k1,11);\
|
|
k1 *= c2;\
|
|
h1 ^= k1;\
|
|
h1 = h1*3 + 0x52dce729;\
|
|
c1 = c1*5 + 0x7b7d159c;\
|
|
c2 = c2*5 + 0x6bce6396;\
|
|
}
|
|
|
|
const uint8_t *data = (const uint8_t*)key;
|
|
const int nblocks = len / 4;
|
|
|
|
const uint32_t *blocks = (const uint32_t *)(data + nblocks * 4);
|
|
const uint8_t *tail = (const uint8_t *)(data + nblocks * 4);
|
|
|
|
uint32_t h1 = 0x971e137b ^ seed;
|
|
uint32_t k1;
|
|
|
|
uint32_t c1 = 0x95543787;
|
|
uint32_t c2 = 0x2ad7eb25;
|
|
|
|
int i;
|
|
|
|
for (i = -nblocks; i; i++) {
|
|
k1 = blocks[i];
|
|
MURMUR_BLOCK();
|
|
}
|
|
|
|
k1 = 0;
|
|
|
|
switch(len & 3) {
|
|
case 3: k1 ^= tail[2] << 16;
|
|
case 2: k1 ^= tail[1] << 8;
|
|
case 1: k1 ^= tail[0];
|
|
MURMUR_BLOCK();
|
|
}
|
|
|
|
h1 ^= len;
|
|
h1 ^= h1 >> 16;
|
|
h1 *= 0x85ebca6b;
|
|
h1 ^= h1 >> 13;
|
|
h1 *= 0xc2b2ae35;
|
|
h1 ^= h1 >> 16;
|
|
|
|
return h1;
|
|
}
|
|
#endif
|