| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2016-05-18 02:52:22 +08:00
										 |  |  |  * Copyright 2004-2016 The OpenSSL Project Authors. All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-18 02:52:22 +08:00
										 |  |  |  * Licensed under the OpenSSL license (the "License").  You may not use | 
					
						
							|  |  |  |  * this file except in compliance with the License.  You can obtain a copy | 
					
						
							|  |  |  |  * in the file LICENSE in the source distribution or at | 
					
						
							|  |  |  |  * https://www.openssl.org/source/license.html
 | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-05-18 02:52:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-16 00:40:53 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2017-06-20 10:48:25 +08:00
										 |  |  |  * This file is dual-licensed and is also available under the following | 
					
						
							| 
									
										
										
										
											2017-06-16 01:15:26 +08:00
										 |  |  |  * terms: | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-16 00:40:53 +08:00
										 |  |  |  * Copyright (c) 2004, Richard Levitte <richard@levitte.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions | 
					
						
							|  |  |  |  * are met: | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright | 
					
						
							|  |  |  |  *    notice, this list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright | 
					
						
							|  |  |  |  *    notice, this list of conditions and the following disclaimer in the | 
					
						
							|  |  |  |  *    documentation and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
					
						
							|  |  |  |  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
					
						
							|  |  |  |  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
					
						
							|  |  |  |  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
					
						
							|  |  |  |  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
					
						
							|  |  |  |  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
					
						
							|  |  |  |  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
					
						
							|  |  |  |  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
					
						
							|  |  |  |  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
					
						
							|  |  |  |  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | #include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | #include <tchar.h>
 | 
					
						
							| 
									
										
										
										
											2016-07-18 17:29:46 +08:00
										 |  |  | #include "internal/numbers.h"
 | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | #ifndef LPDIR_H
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | # include "LPdir.h"
 | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * We're most likely overcautious here, but let's reserve for broken WinCE | 
					
						
							|  |  |  |  * headers and explicitly opt for UNICODE call. Keep in mind that our WinCE | 
					
						
							|  |  |  |  * builds are compiled with -DUNICODE [as well as -D_UNICODE]. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2004-07-22 05:16:21 +08:00
										 |  |  | #if defined(LP_SYS_WINCE) && !defined(FindFirstFile)
 | 
					
						
							|  |  |  | # define FindFirstFile FindFirstFileW
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-11-29 06:06:20 +08:00
										 |  |  | #if defined(LP_SYS_WINCE) && !defined(FindNextFile)
 | 
					
						
							| 
									
										
										
										
											2004-07-22 21:00:14 +08:00
										 |  |  | # define FindNextFile FindNextFileW
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-07-22 05:16:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-26 00:48:28 +08:00
										 |  |  | #ifndef NAME_MAX
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | # define NAME_MAX 255
 | 
					
						
							| 
									
										
										
										
											2004-07-26 00:48:28 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  | #ifdef CP_UTF8
 | 
					
						
							|  |  |  | # define CP_DEFAULT CP_UTF8
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | # define CP_DEFAULT CP_ACP
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | struct LP_dir_context_st { | 
					
						
							|  |  |  |     WIN32_FIND_DATA ctx; | 
					
						
							|  |  |  |     HANDLE handle; | 
					
						
							|  |  |  |     char entry_name[NAME_MAX + 1]; | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *LP_find_file(LP_DIR_CTX **ctx, const char *directory) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     if (ctx == NULL || directory == NULL) { | 
					
						
							|  |  |  |         errno = EINVAL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     errno = 0; | 
					
						
							|  |  |  |     if (*ctx == NULL) { | 
					
						
							|  |  |  |         size_t dirlen = strlen(directory); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |         if (dirlen == 0 || dirlen > INT_MAX - 3) { | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |             errno = ENOENT; | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-02 11:10:31 +08:00
										 |  |  |         *ctx = malloc(sizeof(**ctx)); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         if (*ctx == NULL) { | 
					
						
							|  |  |  |             errno = ENOMEM; | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-05-05 06:00:15 +08:00
										 |  |  |         memset(*ctx, 0, sizeof(**ctx)); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (sizeof(TCHAR) != sizeof(char)) { | 
					
						
							|  |  |  |             TCHAR *wdir = NULL; | 
					
						
							|  |  |  |             /* len_0 denotes string length *with* trailing 0 */ | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |             size_t index = 0, len_0 = dirlen + 1; | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | #ifdef LP_MULTIBYTE_AVAILABLE
 | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |             int sz = 0; | 
					
						
							|  |  |  |             UINT cp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             do { | 
					
						
							|  |  |  | # ifdef CP_UTF8
 | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |                 if ((sz = MultiByteToWideChar((cp = CP_UTF8), 0, | 
					
						
							|  |  |  |                                               directory, len_0, | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |                                               NULL, 0)) > 0 || | 
					
						
							|  |  |  |                     GetLastError() != ERROR_NO_UNICODE_TRANSLATION) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  | # endif
 | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |                 sz = MultiByteToWideChar((cp = CP_ACP), 0, | 
					
						
							|  |  |  |                                          directory, len_0, | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |                                          NULL, 0); | 
					
						
							|  |  |  |             } while (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (sz > 0) { | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |                 /*
 | 
					
						
							|  |  |  |                  * allocate two additional characters in case we need to | 
					
						
							|  |  |  |                  * concatenate asterisk, |sz| covers trailing '\0'! | 
					
						
							|  |  |  |                  */ | 
					
						
							|  |  |  |                 wdir = _alloca((sz + 2) * sizeof(TCHAR)); | 
					
						
							|  |  |  |                 if (!MultiByteToWideChar(cp, 0, directory, len_0, | 
					
						
							|  |  |  |                                          (WCHAR *)wdir, sz)) { | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |                     free(*ctx); | 
					
						
							|  |  |  |                     *ctx = NULL; | 
					
						
							|  |  |  |                     errno = EINVAL; | 
					
						
							|  |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |                 sz = len_0; | 
					
						
							|  |  |  |                 /*
 | 
					
						
							|  |  |  |                  * allocate two additional characters in case we need to | 
					
						
							|  |  |  |                  * concatenate asterisk, |sz| covers trailing '\0'! | 
					
						
							|  |  |  |                  */ | 
					
						
							|  |  |  |                 wdir = _alloca((sz + 2) * sizeof(TCHAR)); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |                 for (index = 0; index < len_0; index++) | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |                     wdir[index] = (TCHAR)directory[index]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             sz--; /* wdir[sz] is trailing '\0' now */ | 
					
						
							|  |  |  |             if (wdir[sz - 1] != TEXT('*')) { | 
					
						
							|  |  |  |                 if (wdir[sz - 1] != TEXT('/') && wdir[sz - 1] != TEXT('\\')) | 
					
						
							|  |  |  |                     _tcscpy(wdir + sz, TEXT("/*")); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     _tcscpy(wdir + sz, TEXT("*")); | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             (*ctx)->handle = FindFirstFile(wdir, &(*ctx)->ctx); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2016-07-16 19:10:02 +08:00
										 |  |  |             if (directory[dirlen - 1] != '*') { | 
					
						
							|  |  |  |                 char *buf = _alloca(dirlen + 3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 strcpy(buf, directory); | 
					
						
							|  |  |  |                 if (buf[dirlen - 1] != '/' && buf[dirlen - 1] != '\\') | 
					
						
							|  |  |  |                     strcpy(buf + dirlen, "/*"); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     strcpy(buf + dirlen, "*"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 directory = buf; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             (*ctx)->handle = FindFirstFile((TCHAR *)directory, &(*ctx)->ctx); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ((*ctx)->handle == INVALID_HANDLE_VALUE) { | 
					
						
							|  |  |  |             free(*ctx); | 
					
						
							|  |  |  |             *ctx = NULL; | 
					
						
							|  |  |  |             errno = EINVAL; | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (FindNextFile((*ctx)->handle, &(*ctx)->ctx) == FALSE) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     if (sizeof(TCHAR) != sizeof(char)) { | 
					
						
							|  |  |  |         TCHAR *wdir = (*ctx)->ctx.cFileName; | 
					
						
							|  |  |  |         size_t index, len_0 = 0; | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         while (wdir[len_0] && len_0 < (sizeof((*ctx)->entry_name) - 1)) | 
					
						
							|  |  |  |             len_0++; | 
					
						
							|  |  |  |         len_0++; | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef LP_MULTIBYTE_AVAILABLE
 | 
					
						
							| 
									
										
										
										
											2016-07-15 21:21:00 +08:00
										 |  |  |         if (!WideCharToMultiByte(CP_DEFAULT, 0, (WCHAR *)wdir, len_0, | 
					
						
							|  |  |  |                                  (*ctx)->entry_name, | 
					
						
							|  |  |  |                                  sizeof((*ctx)->entry_name), NULL, 0)) | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |             for (index = 0; index < len_0; index++) | 
					
						
							|  |  |  |                 (*ctx)->entry_name[index] = (char)wdir[index]; | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         strncpy((*ctx)->entry_name, (const char *)(*ctx)->ctx.cFileName, | 
					
						
							|  |  |  |                 sizeof((*ctx)->entry_name) - 1); | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     (*ctx)->entry_name[sizeof((*ctx)->entry_name) - 1] = '\0'; | 
					
						
							| 
									
										
										
										
											2004-07-21 05:24:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     return (*ctx)->entry_name; | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int LP_find_file_end(LP_DIR_CTX **ctx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     if (ctx != NULL && *ctx != NULL) { | 
					
						
							|  |  |  |         FindClose((*ctx)->handle); | 
					
						
							|  |  |  |         free(*ctx); | 
					
						
							|  |  |  |         *ctx = NULL; | 
					
						
							|  |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     errno = EINVAL; | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2004-07-10 21:16:02 +08:00
										 |  |  | } |