@@ -930,7 +930,7 @@ nothrow:
930
930
}
931
931
932
932
/* *****************************************
933
- * Return canonical version of name in a malloc'd buffer .
933
+ * Return canonical version of name.
934
934
* This code is high risk.
935
935
*/
936
936
extern (C++ ) static const (char )* canonicalName(const (char )* name)
@@ -975,14 +975,16 @@ nothrow:
975
975
char [PATH_MAX ] buf = void ;
976
976
auto path = name.toCStringThen! ((n) => realpath(n.ptr, buf.ptr));
977
977
if (path ! is null )
978
- return mem.xstrdup (path) .toDString;
978
+ return xarraydup (path.toDString) ;
979
979
}
980
980
else static if (__traits(compiles, canonicalize_file_name))
981
981
{
982
982
// Have canonicalize_file_name, which malloc's memory.
983
+ // We need a dmd.root.rmem allocation though.
983
984
auto path = name.toCStringThen! ((n) => canonicalize_file_name(n.ptr));
985
+ scope (exit) .free(path.ptr);
984
986
if (path ! is null )
985
- return path.toDString;
987
+ return xarraydup ( path.toDString) ;
986
988
}
987
989
else static if (__traits(compiles, _PC_PATH_MAX))
988
990
{
@@ -994,14 +996,14 @@ nothrow:
994
996
scope (exit) mem.xfree(buf);
995
997
auto path = name.toCStringThen! ((n) => realpath(n.ptr, buf));
996
998
if (path ! is null )
997
- return mem.xstrdup (path) .toDString;
999
+ return xarraydup (path.toDString) ;
998
1000
}
999
1001
}
1000
1002
// Give up trying to support this platform, just duplicate the filename
1001
1003
// unless there is nothing to copy from.
1002
1004
if (! name.length)
1003
1005
return null ;
1004
- return mem.xstrdup (name.ptr)[ 0 .. name.length] ;
1006
+ return xarraydup (name) ;
1005
1007
}
1006
1008
else version (Windows )
1007
1009
{
@@ -1011,18 +1013,18 @@ nothrow:
1011
1013
/* Apparently, there is no good way to do this on Windows.
1012
1014
* GetFullPathName isn't it, but use it anyway.
1013
1015
*/
1014
- // First find out how long the buffer has to be.
1015
- const fullPathLength = GetFullPathNameW(&wname[0 ], 0 , null , null );
1016
- if (! fullPathLength ) return null ;
1017
- auto fullPath = ( cast (wchar * ) mem.xmalloc_noscan((fullPathLength + 1 ) * wchar .sizeof))[ 0 .. fullPathLength + 1 ] ;
1018
- scope (exit) mem.xfree(fullPath.ptr );
1019
-
1020
- // Actually get the full path name
1021
- const length = GetFullPathNameW(
1022
- &wname[0 ], cast ( DWORD ) fullPath.length, &fullPath[ 0 ] , null /* filePart*/ );
1023
- assert (length == fullPathLength );
1024
-
1025
- return toNarrowStringz (fullPath [0 .. length]);
1016
+ // First find out how long the buffer has to be, incl. terminating null .
1017
+ const capacity = GetFullPathNameW(&wname[0 ], 0 , null , null );
1018
+ if (! capacity ) return null ;
1019
+ auto buffer = cast (wchar * ) mem.xmalloc_noscan(capacity * wchar .sizeof);
1020
+ scope (exit) mem.xfree(buffer );
1021
+
1022
+ // Actually get the full path name. If the buffer is large enough,
1023
+ // the returned length does NOT include the terminating null...
1024
+ const length = GetFullPathNameW( &wname[0 ], capacity , buffer , null /* filePart*/ );
1025
+ assert (length == capacity - 1 );
1026
+
1027
+ return toNarrowStringz (buffer [0 .. length]);
1026
1028
});
1027
1029
}
1028
1030
else
@@ -1031,6 +1033,15 @@ nothrow:
1031
1033
}
1032
1034
}
1033
1035
1036
+ unittest
1037
+ {
1038
+ string filename = " foo.bar" ;
1039
+ const path = canonicalName(filename);
1040
+ scope (exit) free (path.ptr);
1041
+ assert (path.length >= filename.length);
1042
+ assert (path[$ - filename.length .. $] == filename);
1043
+ }
1044
+
1034
1045
/* *******************************
1035
1046
* Free memory allocated by FileName routines
1036
1047
*/
0 commit comments