mirror of https://github.com/openssl/openssl.git
				
				
				
			
		
			
	
	
		
			68 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
		
		
			
		
	
	
			68 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
|  | /*
 | ||
|  |  * Copyright 2015-2019 The OpenSSL Project Authors. All Rights Reserved. | ||
|  |  * | ||
|  |  * Licensed under the Apache License 2.0 (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
 | ||
|  |  */ | ||
|  | 
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <openssl/crypto.h>
 | ||
|  | #include "platform.h"            /* for copy_argv() */
 | ||
|  | #include "apps.h"                /* for app_malloc() */
 | ||
|  | 
 | ||
|  | char **newargv = NULL; | ||
|  | 
 | ||
|  | static void cleanup_argv(void) | ||
|  | { | ||
|  |     OPENSSL_free(newargv); | ||
|  |     newargv = NULL; | ||
|  | } | ||
|  | 
 | ||
|  | char **copy_argv(int *argc, char *argv[]) | ||
|  | { | ||
|  |     /*-
 | ||
|  |      * The note below is for historical purpose.  On VMS now we always | ||
|  |      * copy argv "safely." | ||
|  |      * | ||
|  |      * 2011-03-22 SMS. | ||
|  |      * If we have 32-bit pointers everywhere, then we're safe, and | ||
|  |      * we bypass this mess, as on non-VMS systems. | ||
|  |      * Problem 1: Compaq/HP C before V7.3 always used 32-bit | ||
|  |      * pointers for argv[]. | ||
|  |      * Fix 1: For a 32-bit argv[], when we're using 64-bit pointers | ||
|  |      * everywhere else, we always allocate and use a 64-bit | ||
|  |      * duplicate of argv[]. | ||
|  |      * Problem 2: Compaq/HP C V7.3 (Alpha, IA64) before ECO1 failed | ||
|  |      * to NULL-terminate a 64-bit argv[].  (As this was written, the | ||
|  |      * compiler ECO was available only on IA64.) | ||
|  |      * Fix 2: Unless advised not to (VMS_TRUST_ARGV), we test a | ||
|  |      * 64-bit argv[argc] for NULL, and, if necessary, use a | ||
|  |      * (properly) NULL-terminated (64-bit) duplicate of argv[]. | ||
|  |      * The same code is used in either case to duplicate argv[]. | ||
|  |      * Some of these decisions could be handled in preprocessing, | ||
|  |      * but the code tends to get even uglier, and the penalty for | ||
|  |      * deciding at compile- or run-time is tiny. | ||
|  |      */ | ||
|  | 
 | ||
|  |     int i, count = *argc; | ||
|  |     char **p = newargv; | ||
|  | 
 | ||
|  |     cleanup_argv(); | ||
|  | 
 | ||
|  |     newargv = app_malloc(sizeof(*newargv) * (count + 1), "argv copy"); | ||
|  |     if (newargv == NULL) | ||
|  |         return NULL; | ||
|  | 
 | ||
|  |     /* Register automatic cleanup on first use */ | ||
|  |     if (p == NULL) | ||
|  |         OPENSSL_atexit(cleanup_argv); | ||
|  | 
 | ||
|  |     for (i = 0; i < count; i++) | ||
|  |         newargv[i] = argv[i]; | ||
|  |     newargv[i] = NULL; | ||
|  |     *argc = i; | ||
|  |     return newargv; | ||
|  | } |