Blame tests/clar/fs.h

Packit Service 20376f
#ifdef _WIN32
Packit Service 20376f
Packit Service 20376f
#define RM_RETRY_COUNT	5
Packit Service 20376f
#define RM_RETRY_DELAY	10
Packit Service 20376f
Packit Service 20376f
#ifdef __MINGW32__
Packit Service 20376f
Packit Service 20376f
/* These security-enhanced functions are not available
Packit Service 20376f
 * in MinGW, so just use the vanilla ones */
Packit Service 20376f
#define wcscpy_s(a, b, c) wcscpy((a), (c))
Packit Service 20376f
#define wcscat_s(a, b, c) wcscat((a), (c))
Packit Service 20376f
Packit Service 20376f
#endif /* __MINGW32__ */
Packit Service 20376f
Packit Service 20376f
static int
Packit Service 20376f
fs__dotordotdot(WCHAR *_tocheck)
Packit Service 20376f
{
Packit Service 20376f
	return _tocheck[0] == '.' &&
Packit Service 20376f
		(_tocheck[1] == '\0' ||
Packit Service 20376f
		 (_tocheck[1] == '.' && _tocheck[2] == '\0'));
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int
Packit Service 20376f
fs_rmdir_rmdir(WCHAR *_wpath)
Packit Service 20376f
{
Packit Service 20376f
	unsigned retries = 1;
Packit Service 20376f
Packit Service 20376f
	while (!RemoveDirectoryW(_wpath)) {
Packit Service 20376f
		/* Only retry when we have retries remaining, and the
Packit Service 20376f
		 * error was ERROR_DIR_NOT_EMPTY. */
Packit Service 20376f
		if (retries++ > RM_RETRY_COUNT ||
Packit Service 20376f
			ERROR_DIR_NOT_EMPTY != GetLastError())
Packit Service 20376f
			return -1;
Packit Service 20376f
Packit Service 20376f
		/* Give whatever has a handle to a child item some time
Packit Service 20376f
		 * to release it before trying again */
Packit Service 20376f
		Sleep(RM_RETRY_DELAY * retries * retries);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_rmdir_helper(WCHAR *_wsource)
Packit Service 20376f
{
Packit Service 20376f
	WCHAR buffer[MAX_PATH];
Packit Service 20376f
	HANDLE find_handle;
Packit Service 20376f
	WIN32_FIND_DATAW find_data;
Packit Service 20376f
	size_t buffer_prefix_len;
Packit Service 20376f
Packit Service 20376f
	/* Set up the buffer and capture the length */
Packit Service 20376f
	wcscpy_s(buffer, MAX_PATH, _wsource);
Packit Service 20376f
	wcscat_s(buffer, MAX_PATH, L"\\");
Packit Service 20376f
	buffer_prefix_len = wcslen(buffer);
Packit Service 20376f
Packit Service 20376f
	/* FindFirstFile needs a wildcard to match multiple items */
Packit Service 20376f
	wcscat_s(buffer, MAX_PATH, L"*");
Packit Service 20376f
	find_handle = FindFirstFileW(buffer, &find_data);
Packit Service 20376f
	cl_assert(INVALID_HANDLE_VALUE != find_handle);
Packit Service 20376f
Packit Service 20376f
	do {
Packit Service 20376f
		/* FindFirstFile/FindNextFile gives back . and ..
Packit Service 20376f
		 * entries at the beginning */
Packit Service 20376f
		if (fs__dotordotdot(find_data.cFileName))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		wcscpy_s(buffer + buffer_prefix_len, MAX_PATH - buffer_prefix_len, find_data.cFileName);
Packit Service 20376f
Packit Service 20376f
		if (FILE_ATTRIBUTE_DIRECTORY & find_data.dwFileAttributes)
Packit Service 20376f
			fs_rmdir_helper(buffer);
Packit Service 20376f
		else {
Packit Service 20376f
			/* If set, the +R bit must be cleared before deleting */
Packit Service 20376f
			if (FILE_ATTRIBUTE_READONLY & find_data.dwFileAttributes)
Packit Service 20376f
				cl_assert(SetFileAttributesW(buffer, find_data.dwFileAttributes & ~FILE_ATTRIBUTE_READONLY));
Packit Service 20376f
Packit Service 20376f
			cl_assert(DeleteFileW(buffer));
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	while (FindNextFileW(find_handle, &find_data));
Packit Service 20376f
Packit Service 20376f
	/* Ensure that we successfully completed the enumeration */
Packit Service 20376f
	cl_assert(ERROR_NO_MORE_FILES == GetLastError());
Packit Service 20376f
Packit Service 20376f
	/* Close the find handle */
Packit Service 20376f
	FindClose(find_handle);
Packit Service 20376f
Packit Service 20376f
	/* Now that the directory is empty, remove it */
Packit Service 20376f
	cl_assert(0 == fs_rmdir_rmdir(_wsource));
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int
Packit Service 20376f
fs_rm_wait(WCHAR *_wpath)
Packit Service 20376f
{
Packit Service 20376f
	unsigned retries = 1;
Packit Service 20376f
	DWORD last_error;
Packit Service 20376f
Packit Service 20376f
	do {
Packit Service 20376f
		if (INVALID_FILE_ATTRIBUTES == GetFileAttributesW(_wpath))
Packit Service 20376f
			last_error = GetLastError();
Packit Service 20376f
		else
Packit Service 20376f
			last_error = ERROR_SUCCESS;
Packit Service 20376f
Packit Service 20376f
		/* Is the item gone? */
Packit Service 20376f
		if (ERROR_FILE_NOT_FOUND == last_error ||
Packit Service 20376f
			ERROR_PATH_NOT_FOUND == last_error)
Packit Service 20376f
			return 0;
Packit Service 20376f
Packit Service 20376f
		Sleep(RM_RETRY_DELAY * retries * retries);	
Packit Service 20376f
	}
Packit Service 20376f
	while (retries++ <= RM_RETRY_COUNT);
Packit Service 20376f
Packit Service 20376f
	return -1;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_rm(const char *_source)
Packit Service 20376f
{
Packit Service 20376f
	WCHAR wsource[MAX_PATH];
Packit Service 20376f
	DWORD attrs;
Packit Service 20376f
Packit Service 20376f
	/* The input path is UTF-8. Convert it to wide characters
Packit Service 20376f
	 * for use with the Windows API */
Packit Service 20376f
	cl_assert(MultiByteToWideChar(CP_UTF8,
Packit Service 20376f
				MB_ERR_INVALID_CHARS,
Packit Service 20376f
				_source,
Packit Service 20376f
				-1, /* Indicates NULL termination */
Packit Service 20376f
				wsource,
Packit Service 20376f
				MAX_PATH));
Packit Service 20376f
Packit Service 20376f
	/* Does the item exist? If not, we have no work to do */
Packit Service 20376f
	attrs = GetFileAttributesW(wsource);
Packit Service 20376f
Packit Service 20376f
	if (INVALID_FILE_ATTRIBUTES == attrs)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	if (FILE_ATTRIBUTE_DIRECTORY & attrs)
Packit Service 20376f
		fs_rmdir_helper(wsource);
Packit Service 20376f
	else {
Packit Service 20376f
		/* The item is a file. Strip the +R bit */
Packit Service 20376f
		if (FILE_ATTRIBUTE_READONLY & attrs)
Packit Service 20376f
			cl_assert(SetFileAttributesW(wsource, attrs & ~FILE_ATTRIBUTE_READONLY));
Packit Service 20376f
Packit Service 20376f
		cl_assert(DeleteFileW(wsource));
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* Wait for the DeleteFile or RemoveDirectory call to complete */
Packit Service 20376f
	cl_assert(0 == fs_rm_wait(wsource));
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_copydir_helper(WCHAR *_wsource, WCHAR *_wdest)
Packit Service 20376f
{
Packit Service 20376f
	WCHAR buf_source[MAX_PATH], buf_dest[MAX_PATH];
Packit Service 20376f
	HANDLE find_handle;
Packit Service 20376f
	WIN32_FIND_DATAW find_data;
Packit Service 20376f
	size_t buf_source_prefix_len, buf_dest_prefix_len;
Packit Service 20376f
Packit Service 20376f
	wcscpy_s(buf_source, MAX_PATH, _wsource);
Packit Service 20376f
	wcscat_s(buf_source, MAX_PATH, L"\\");
Packit Service 20376f
	buf_source_prefix_len = wcslen(buf_source);
Packit Service 20376f
Packit Service 20376f
	wcscpy_s(buf_dest, MAX_PATH, _wdest);
Packit Service 20376f
	wcscat_s(buf_dest, MAX_PATH, L"\\");
Packit Service 20376f
	buf_dest_prefix_len = wcslen(buf_dest);
Packit Service 20376f
Packit Service 20376f
	/* Get an enumerator for the items in the source. */
Packit Service 20376f
	wcscat_s(buf_source, MAX_PATH, L"*");
Packit Service 20376f
	find_handle = FindFirstFileW(buf_source, &find_data);
Packit Service 20376f
	cl_assert(INVALID_HANDLE_VALUE != find_handle);
Packit Service 20376f
Packit Service 20376f
	/* Create the target directory. */
Packit Service 20376f
	cl_assert(CreateDirectoryW(_wdest, NULL));
Packit Service 20376f
Packit Service 20376f
	do {
Packit Service 20376f
		/* FindFirstFile/FindNextFile gives back . and ..
Packit Service 20376f
		 * entries at the beginning */
Packit Service 20376f
		if (fs__dotordotdot(find_data.cFileName))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		wcscpy_s(buf_source + buf_source_prefix_len, MAX_PATH - buf_source_prefix_len, find_data.cFileName);
Packit Service 20376f
		wcscpy_s(buf_dest + buf_dest_prefix_len, MAX_PATH - buf_dest_prefix_len, find_data.cFileName);
Packit Service 20376f
Packit Service 20376f
		if (FILE_ATTRIBUTE_DIRECTORY & find_data.dwFileAttributes)
Packit Service 20376f
			fs_copydir_helper(buf_source, buf_dest);
Packit Service 20376f
		else
Packit Service 20376f
			cl_assert(CopyFileW(buf_source, buf_dest, TRUE));
Packit Service 20376f
	}
Packit Service 20376f
	while (FindNextFileW(find_handle, &find_data));
Packit Service 20376f
Packit Service 20376f
	/* Ensure that we successfully completed the enumeration */
Packit Service 20376f
	cl_assert(ERROR_NO_MORE_FILES == GetLastError());
Packit Service 20376f
Packit Service 20376f
	/* Close the find handle */
Packit Service 20376f
	FindClose(find_handle);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_copy(const char *_source, const char *_dest)
Packit Service 20376f
{
Packit Service 20376f
	WCHAR wsource[MAX_PATH], wdest[MAX_PATH];
Packit Service 20376f
	DWORD source_attrs, dest_attrs;
Packit Service 20376f
	HANDLE find_handle;
Packit Service 20376f
	WIN32_FIND_DATAW find_data;
Packit Service 20376f
Packit Service 20376f
	/* The input paths are UTF-8. Convert them to wide characters
Packit Service 20376f
	 * for use with the Windows API. */
Packit Service 20376f
	cl_assert(MultiByteToWideChar(CP_UTF8,
Packit Service 20376f
				MB_ERR_INVALID_CHARS,
Packit Service 20376f
				_source,
Packit Service 20376f
				-1,
Packit Service 20376f
				wsource,
Packit Service 20376f
				MAX_PATH));
Packit Service 20376f
Packit Service 20376f
	cl_assert(MultiByteToWideChar(CP_UTF8,
Packit Service 20376f
				MB_ERR_INVALID_CHARS,
Packit Service 20376f
				_dest,
Packit Service 20376f
				-1,
Packit Service 20376f
				wdest,
Packit Service 20376f
				MAX_PATH));
Packit Service 20376f
Packit Service 20376f
	/* Check the source for existence */
Packit Service 20376f
	source_attrs = GetFileAttributesW(wsource);
Packit Service 20376f
	cl_assert(INVALID_FILE_ATTRIBUTES != source_attrs);
Packit Service 20376f
Packit Service 20376f
	/* Check the target for existence */
Packit Service 20376f
	dest_attrs = GetFileAttributesW(wdest);
Packit Service 20376f
Packit Service 20376f
	if (INVALID_FILE_ATTRIBUTES != dest_attrs) {
Packit Service 20376f
		/* Target exists; append last path part of source to target.
Packit Service 20376f
		 * Use FindFirstFile to parse the path */
Packit Service 20376f
		find_handle = FindFirstFileW(wsource, &find_data);
Packit Service 20376f
		cl_assert(INVALID_HANDLE_VALUE != find_handle);
Packit Service 20376f
		wcscat_s(wdest, MAX_PATH, L"\\");
Packit Service 20376f
		wcscat_s(wdest, MAX_PATH, find_data.cFileName);
Packit Service 20376f
		FindClose(find_handle);
Packit Service 20376f
Packit Service 20376f
		/* Check the new target for existence */
Packit Service 20376f
		cl_assert(INVALID_FILE_ATTRIBUTES == GetFileAttributesW(wdest));
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (FILE_ATTRIBUTE_DIRECTORY & source_attrs)
Packit Service 20376f
		fs_copydir_helper(wsource, wdest);
Packit Service 20376f
	else
Packit Service 20376f
		cl_assert(CopyFileW(wsource, wdest, TRUE));
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void
Packit Service 20376f
cl_fs_cleanup(void)
Packit Service 20376f
{
Packit Service 20376f
	fs_rm(fixture_path(_clar_path, "*"));
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
#else
Packit Service 20376f
Packit Service 20376f
#include <errno.h>
Packit Service 20376f
#include <string.h>
Packit Service 20376f
Packit Service 20376f
static int
Packit Service 20376f
shell_out(char * const argv[])
Packit Service 20376f
{
Packit Service 20376f
	int status, piderr;
Packit Service 20376f
	pid_t pid;
Packit Service 20376f
Packit Service 20376f
	pid = fork();
Packit Service 20376f
Packit Service 20376f
	if (pid < 0) {
Packit Service 20376f
		fprintf(stderr,
Packit Service 20376f
			"System error: `fork()` call failed (%d) - %s\n",
Packit Service 20376f
			errno, strerror(errno));
Packit Service 20376f
		exit(-1);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (pid == 0) {
Packit Service 20376f
		execv(argv[0], argv);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	do {
Packit Service 20376f
		piderr = waitpid(pid, &status, WUNTRACED);
Packit Service 20376f
	} while (piderr < 0 && (errno == EAGAIN || errno == EINTR));
Packit Service 20376f
Packit Service 20376f
	return WEXITSTATUS(status);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_copy(const char *_source, const char *dest)
Packit Service 20376f
{
Packit Service 20376f
	char *argv[5];
Packit Service 20376f
	char *source;
Packit Service 20376f
	size_t source_len;
Packit Service 20376f
Packit Service 20376f
	source = strdup(_source);
Packit Service 20376f
	source_len = strlen(source);
Packit Service 20376f
Packit Service 20376f
	if (source[source_len - 1] == '/')
Packit Service 20376f
		source[source_len - 1] = 0;
Packit Service 20376f
Packit Service 20376f
	argv[0] = "/bin/cp";
Packit Service 20376f
	argv[1] = "-R";
Packit Service 20376f
	argv[2] = source;
Packit Service 20376f
	argv[3] = (char *)dest;
Packit Service 20376f
	argv[4] = NULL;
Packit Service 20376f
Packit Service 20376f
	cl_must_pass_(
Packit Service 20376f
		shell_out(argv),
Packit Service 20376f
		"Failed to copy test fixtures to sandbox"
Packit Service 20376f
	);
Packit Service 20376f
Packit Service 20376f
	free(source);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
fs_rm(const char *source)
Packit Service 20376f
{
Packit Service 20376f
	char *argv[4];
Packit Service 20376f
Packit Service 20376f
	argv[0] = "/bin/rm";
Packit Service 20376f
	argv[1] = "-Rf";
Packit Service 20376f
	argv[2] = (char *)source;
Packit Service 20376f
	argv[3] = NULL;
Packit Service 20376f
Packit Service 20376f
	cl_must_pass_(
Packit Service 20376f
		shell_out(argv),
Packit Service 20376f
		"Failed to cleanup the sandbox"
Packit Service 20376f
	);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void
Packit Service 20376f
cl_fs_cleanup(void)
Packit Service 20376f
{
Packit Service 20376f
	clar_unsandbox();
Packit Service 20376f
	clar_sandbox();
Packit Service 20376f
}
Packit Service 20376f
#endif