2 * Translate between Windows and Unix paths formats
4 * Copyright 2002 Mike Wetherell
5 * Copyright 2005 Dmitry Timoshkov
6 * Copyright 2005 Francois Gouget
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23 #define WIN32_LEAN_AND_MEAN
31 #include "wine/debug.h"
41 static const char progname[] = "winepath";
46 static int option(int shortopt, const WCHAR *longopt)
48 static const char helpmsg[] =
49 "Convert PATH(s) to Unix or Windows long or short paths.\n"
51 " -u, --unix converts a Windows path to a Unix path\n"
52 " -w, --windows converts a Unix path to a long Windows path\n"
53 " -l, --long converts the short Windows path of an existing file or\n"
54 " directory to the long format\n"
55 " -s, --short converts the long Windows path of an existing file or\n"
56 " directory to the short format\n"
57 " -0 separate output with \\0 character, instead of a newline\n"
58 " -h, --help output this help message and exit\n"
59 " -v, --version output version information and exit\n"
61 "If more than one option is given then the input paths are output in\n"
62 "all formats specified, in the order long, short, Unix, Windows.\n"
63 "If no option is given the default is Unix format.\n";
67 printf("Usage: %s [OPTION] [PATH]...\n", progname);
71 printf("%s version " PACKAGE_VERSION "\n", progname);
85 fprintf(stderr, "%s: invalid option ", progname);
87 fprintf(stderr, "%s\n", wine_dbgstr_w(longopt));
89 fprintf(stderr, "'-%c'\n", shortopt);
90 fprintf(stderr, "Try '%s --help' for help\n", progname);
95 * Parse command line options
97 static int parse_options(WCHAR *argv[])
99 static const WCHAR longW[] = { 'l','o','n','g',0 };
100 static const WCHAR shortW[] = { 's','h','o','r','t',0 };
101 static const WCHAR unixW[] = { 'u','n','i','x',0 };
102 static const WCHAR windowsW[] = { 'w','i','n','d','o','w','s',0 };
103 static const WCHAR helpW[] = { 'h','e','l','p',0 };
104 static const WCHAR versionW[] = { 'v','e','r','s','i','o','n',0 };
105 static const WCHAR nullW[] = { 0 };
106 static const WCHAR *longopts[] = { longW, shortW, unixW, windowsW, helpW, versionW, nullW };
107 int outputformats = 0;
111 for (i = 1; argv[i] && !done; )
113 if (argv[i][0] != '-') {
119 if (argv[i][1] == '-') {
120 if (argv[i][2] == 0) {
121 /* '--' end of options */
125 for (j = 0; longopts[j][0]; j++)
126 if (!lstrcmpiW(argv[i]+2, longopts[j]))
128 outputformats |= option(longopts[j][0], argv[i]);
132 for (j = 1; argv[i][j]; j++)
133 outputformats |= option(argv[i][j], NULL);
137 for (j = i + 1; argv[j - 1]; j++)
138 argv[j - 1] = argv[j];
141 return outputformats;
147 int wmain(int argc, WCHAR *argv[])
149 LPSTR (*CDECL wine_get_unix_file_name_ptr)(LPCWSTR) = NULL;
150 LPWSTR (*CDECL wine_get_dos_file_name_ptr)(LPCSTR) = NULL;
151 WCHAR dos_pathW[MAX_PATH];
157 outputformats = parse_options(argv);
159 if (outputformats & PRINT0)
162 outputformats ^= PRINT0;
167 if (outputformats == 0)
168 outputformats = UNIXFORMAT;
170 if (outputformats & UNIXFORMAT) {
171 wine_get_unix_file_name_ptr = (void*)
172 GetProcAddress(GetModuleHandleA("KERNEL32"),
173 "wine_get_unix_file_name");
174 if (wine_get_unix_file_name_ptr == NULL) {
175 fprintf(stderr, "%s: cannot get the address of "
176 "'wine_get_unix_file_name'\n", progname);
181 if (outputformats & WINDOWSFORMAT) {
182 wine_get_dos_file_name_ptr = (void*)
183 GetProcAddress(GetModuleHandleA("KERNEL32"),
184 "wine_get_dos_file_name");
185 if (wine_get_dos_file_name_ptr == NULL) {
186 fprintf(stderr, "%s: cannot get the address of "
187 "'wine_get_dos_file_name'\n", progname);
192 for (i = 1; argv[i]; i++)
195 if (outputformats & LONGFORMAT) {
196 if (GetLongPathNameW(argv[i], dos_pathW, MAX_PATH))
197 WideCharToMultiByte(CP_UNIXCP, 0, dos_pathW, -1, path, MAX_PATH, NULL, NULL);
198 printf("%s%c", path, separator);
200 if (outputformats & SHORTFORMAT) {
201 if (GetShortPathNameW(argv[i], dos_pathW, MAX_PATH))
202 WideCharToMultiByte(CP_UNIXCP, 0, dos_pathW, -1, path, MAX_PATH, NULL, NULL);
203 printf("%s%c", path, separator);
205 if (outputformats & UNIXFORMAT) {
206 WCHAR *ntpath, *tail;
207 int ntpathlen=lstrlenW(argv[i]);
208 ntpath=HeapAlloc(GetProcessHeap(), 0, sizeof(*ntpath)*(ntpathlen+1));
209 lstrcpyW(ntpath, argv[i]);
216 unix_name = wine_get_unix_file_name_ptr(ntpath);
221 WideCharToMultiByte(CP_UNIXCP, 0, tail+1, -1, path, MAX_PATH, NULL, NULL);
222 printf("%s/%s%c", unix_name, path, separator);
226 printf("%s%c", unix_name, separator);
228 HeapFree( GetProcessHeap(), 0, unix_name );
232 slash=(tail ? tail : ntpath+ntpathlen);
233 while (slash != ntpath && *slash != '/' && *slash != '\\')
237 /* This is a complete path conversion failure.
238 * It would typically happen if ntpath == "".
240 printf("%c", separator);
244 while (*c != '\0' && *c != '*' && *c != '?' &&
245 *c != '<' && *c != '>' && *c != '|' && *c != '"')
249 /* If this is not a valid NT path to start with,
250 * then obviously we cannot convert it.
252 printf("%c", separator);
260 HeapFree(GetProcessHeap(), 0, ntpath);
262 if (outputformats & WINDOWSFORMAT) {
267 size=WideCharToMultiByte(CP_UNIXCP, 0, argv[i], -1, NULL, 0, NULL, NULL);
268 unix_name=HeapAlloc(GetProcessHeap(), 0, size);
269 WideCharToMultiByte(CP_UNIXCP, 0, argv[i], -1, unix_name, size, NULL, NULL);
271 if ((windows_name = wine_get_dos_file_name_ptr(unix_name)))
273 WideCharToMultiByte(CP_UNIXCP, 0, windows_name, -1, path, MAX_PATH, NULL, NULL);
274 printf("%s%c", path, separator);
275 HeapFree( GetProcessHeap(), 0, windows_name );
277 else printf("%c", separator);
278 HeapFree( GetProcessHeap(), 0, unix_name );