예제 #1
0
파일: path.c 프로젝트: Mailaender/bam
/* is it absolute and normalized? */
int path_isnice(const char *path)
{
	/* check for initial "../../" */
	while(path[0] == '.')
	{
		if(path[1] == '.')
		{
			if(path_is_separator(path[2]))
			{
				/* found "../" case */
				path += 3;
			}
			else
				return 0;
		}
		else if(path_is_separator(path[1]))
			return 0;
		else
			break;
	}
	
	while(path[0])
	{
		if(path_is_separator(path[0]))
		{
			/* check for // */
			if(path_is_separator(path[1]))
				return 0;
			
			if(path[1] == '.')
			{
				/* check for /.. */
				if(path[2] == '.')
					return 0;
				
				/* check for /./ */
				if(path_is_separator(path[2]))
					return 0;
			}
			
			/* check so that the path doesn't end on / */
			if(path[1] == 0)
				return 0;
		}
		path++;
	}
	
	return 1;
}
예제 #2
0
파일: path.c 프로젝트: Mailaender/bam
int path_directory(const char *path, char *directory, int size)
{
	char *dest = directory;
	char *dest_end = directory+size-1;
	const char *read = path;
	const char *cur;

	for(cur = path; *cur; cur++)
	{
		if(path_is_separator(*cur))
		{
			/* ok, copy the directory */
			for(; read != cur; read++, dest++)
			{
				if(dest == dest_end)
				{
					*dest =  0;
					return 1;
				}
				*dest = *read;
			}
		}
	}
	
	*dest = 0;
	if(0)
		printf("path_directory:\n\tinput:'%s'\n\toutput:'%s'\n", path, directory);
	return 0;
}
예제 #3
0
파일: path.c 프로젝트: EEmmanuel7/bam
int lf_path_base(lua_State *L)
{
	int n = lua_gettop(L);
	size_t org_len;
	size_t new_len;
	size_t count = 0;
	const char *cur = 0;
	const char *path = 0;
	if(n < 1)
		luaL_error(L, "path_base: incorrect number of arguments");
	
	path = lua_tolstring(L, 1, &org_len);
	if(!path)
		luaL_error(L, "path_base: argument is not a string");

	/* cut off the ext */
	new_len = org_len;
	for(cur = path; *cur; cur++, count++)
	{
		if(*cur == '.')
			new_len = count;
		else if(path_is_separator(*cur))
			new_len = org_len;
	}
		
	lua_pushlstring(L, path, new_len);
	return 1;
}
예제 #4
0
파일: path.c 프로젝트: Mailaender/bam
/* returns true if a path is absolute */
int path_isabs(const char *path)
{
#if defined(BAM_FAMILY_WINDOWS)
	if(strlen(path) > 2 && isalpha(path[0]) && path[1] == ':' && path_is_separator(path[2]))
		return 1;
#elif defined(BAM_FAMILY_UNIX)
	if(path_is_separator(path[0]))
		return 1;
#elif defined(BAM_FAMILY_BEOS)
	if(path_is_separator(path[0]))
		return 1;
#else
	#error path_isabs(const char *path) not implemented on this platform
#endif
	
	return 0;
}
예제 #5
0
파일: path.c 프로젝트: Mailaender/bam
/* return zero on success */
int path_join(const char *base, int base_len, const char *extend, int extend_len, char *output, int size)
{
	int i;
	if(extend_len < 0)
		extend_len = strlen(extend);
	
	if(path_isabs(extend))
	{
		/* just copy the extend path */
		if(extend_len+1 > size)
		{
			fprintf(stderr, "'%s' + '%s' results in a too long path\n", base, extend);
			return __LINE__;
		}
		
		memcpy(output, extend, extend_len+1);
		path_normalize(output);
		return 0;
	}
	
	if(base_len < 0)
		base_len = strlen(base);
	
	/* +2 for separator and null terminator */
	if(base_len+extend_len+2 > size)
	{
		fprintf(stderr, "'%s' + '%s' results in a too long path\n", base, extend);
		return __LINE__;
	}

	/* no base path, just use extend path then */
	if(base_len == 0)
	{
		memcpy(output, extend, extend_len+1);
		path_normalize(output);
		return 0;
	}
	
	/* copy base path */
	memcpy(output, base, base_len);
	
	/* append path separator if needed */
	if(!path_is_separator(base[base_len-1]))
	{
		output[base_len] = PATH_SEPARATOR;
		base_len++;
	}
	
	/* append the extra path, and null-terminator*/
	for(i = 0; i < extend_len+1; i++)
		output[base_len+i] = extend[i];
	
	/* normalize path and return success */
	path_normalize(output);
	return 0;
}
예제 #6
0
파일: path.c 프로젝트: Mailaender/bam
const char *path_filename(const char *path)
{
	const char *ret = path;
	const char *cur;
	for(cur = path; *cur; cur++)
	{
		if(path_is_separator(*cur))
			ret = cur+1;
	}
	return ret;
}
예제 #7
0
파일: path.c 프로젝트: EEmmanuel7/bam
static int path_dir_length(const char *path)
{
	const char *cur = path;
	int total = 0;
	int len = -1;
	for(; *cur; cur++, total++)
	{
		if(path_is_separator(*cur))
			len = (int)(cur-path);
	}

	if(len == -1)
		return 0;
	return len;
}
예제 #8
0
파일: path.c 프로젝트: Mailaender/bam
const char *path_ext(const char *filename)
{
	const char *cur = filename;
	const char *ext = 0;
	
	for(; *cur; cur++)
	{
		if(*cur == '.')
			ext = cur;
		if(path_is_separator(*cur))
			ext = (const char *)0x0;
	}
	if(!ext)
		return "";
	return ext+1;
}
예제 #9
0
파일: path.c 프로젝트: Mailaender/bam
/* normalizes the path, it rewrites the path */
int path_normalize(char *path)
{
	char *dirs[128];
	int depth = 0;
	char *dstptr = path;
	char *srcptr = path;
	
	/* add the start */
	dirs[0] = path;
	depth++;
	
	while(1)
	{
		if(srcptr[0] == '.' && path_is_separator(srcptr[1]))
		{
			/* "./" case, just skip the data */
			srcptr += 2;
		}
		else if(srcptr[0] == '.'  && srcptr[1] == '.' && path_is_separator(srcptr[2]))
		{
			/* "../" case */
			if(depth == 1)
			{
				/* case where we are at the start so append ../ to the start of the string */
				dstptr[0] = '.';
				dstptr[1] = '.';
				dstptr[2] = PATH_SEPARATOR;
				dstptr += 3;
				srcptr += 3;
				
				dirs[0] = dstptr;
			}
			else
			{
				/* normal case where we are in the middle like "a/b/../c" */
				depth--;
				dstptr = dirs[depth-1];
				srcptr += 3;
			}
		}
		else if(srcptr[0] == '.'  && srcptr[1] == '.' && srcptr[2] == 0)
		{
			/* ".." case, .. at end of string */
			if(depth == 1)
			{
				dstptr[0] = '.';
				dstptr[1] = '.';
				dstptr += 2;
				srcptr += 2;
				
				dirs[0] = dstptr;
			}
			else
			{
				depth--;
				dstptr = dirs[depth-1];
				srcptr += 2;
			}
		}
		else
		{
			/* search for separator */
			while(!path_is_separator(srcptr[0]) && srcptr[0])
				*dstptr++ = *srcptr++;
			
			if(srcptr[0] == 0)
			{
				/* end of string, zero terminate and return, strip ending '/' if it exists */
				if(dstptr != path && path_is_separator(dstptr[-1]))
					dstptr[-1] = 0;
				dstptr[0] = 0;
				return 0;
			}
			else if(path_is_separator(srcptr[0]))
			{
				/* store the point of this directory */
				*dstptr++ = *srcptr++;
				dirs[depth] = dstptr;
				depth++;
				
				/* condense multiple // */
				while(path_is_separator(srcptr[0]) && srcptr[0])
					srcptr++;
			}
			else
			{
				/* non reachable case */
				return -1;
			}
		}
	}
	
	return 0;
}