blob: b853d1d1ea81e0b6d80dec12b0b4d7d18519dc13 [file] [log] [blame]
/**
* This file has no copyright assigned and is placed in the Public Domain.
* This file is part of the mingw-w64 runtime package.
* No warranty is given; refer to the file DISCLAIMER.PD within this package.
*/
#ifndef WIN32_LEAN_AND_MEAN
#define WIN32_LEAN_AND_MEAN
#endif
#include <sys/stat.h>
#include <stdlib.h>
#include <locale.h>
#include <windows.h>
#include "__mingw_fix_stat.h"
static const char* next_char (unsigned int cp, const char* p)
{
/* If it is a lead byte, skip the next byte except if it is \0.
* If it is \0, it's not a valid DBCS string. */
return (IsDBCSLeadByteEx (cp, *p) && p[1] != '\0') ? p + 2 : p + 1;
}
/**
* Returns _path without trailing slash if any
*
* - if _path has no trailing slash, the function returns it
* - if _path has a trailing slash, but is of the form C:/, then it returns it
* - otherwise, the function creates a new string, which is a copy of _path
* without the trailing slash. It is then the responsibility of the caller
* to free it.
*/
char* __mingw_fix_stat_path (const char* _path)
{
const unsigned int cp = __mingw_filename_cp ();
size_t len;
char *p;
p = (char*)_path;
if (_path && *_path) {
len = strlen (_path);
/* Ignore X:\
* No ANSI or OEM code page uses ':' as a trail byte. (The code page 1361
* cannot be used as ANSI or OEM code page.) */
if (len <= 1 || ((len == 2 || len == 3) && _path[1] == ':'))
return p;
const char *r = _path;
/* Check UNC \\abc\<name>\ */
if ((_path[0] == '\\' || _path[0] == '/')
&& (_path[1] == '\\' || _path[1] == '/'))
{
r = &_path[2];
while (*r != 0 && *r != '\\' && *r != '/')
r = next_char (cp, r);
if (*r != 0)
++r;
if (*r == 0)
return p;
while (*r != 0 && *r != '\\' && *r != '/')
r = next_char (cp, r);
if (*r != 0)
++r;
if (*r == 0)
return p;
}
if (_path[len - 1] == '/' || _path[len - 1] == '\\')
{
/* Return if the last character is a double-byte character.
* Its trail byte could be a '\' which must not be interpret
* as a directory separator. */
while (r[1] != '\0')
{
r = next_char (cp, r);
if (*r == '\0')
return p;
}
p = (char*)malloc (len);
if (p == NULL)
return NULL; /* malloc has set errno. */
memcpy (p, _path, len - 1);
p[len - 1] = '\0';
}
}
return p;
}