You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			73 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Diff
		
	
		
		
			
		
	
	
			73 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Diff
		
	
| 
											10 years ago
										 | From b53cd994adaff887ec126de259e37d769ad585cb Mon Sep 17 00:00:00 2001 | ||
|  | From: Kenny Root <kroot@google.com> | ||
|  | Date: Fri, 8 Feb 2013 11:22:25 -0800 | ||
|  | Subject: [PATCH] Fix failures when eng_dyn scans multiple directories | ||
|  | 
 | ||
|  | If DIR_ADD is called with multiple directories, and the target file | ||
|  | does not exist in the first directory scanned, the DSO object will still | ||
|  | be considered "loaded" for the next call of DSO_load(...) and cause | ||
|  | subsequent calls to DSO_load(...) fail with the reason code of "already | ||
|  | loaded" even though the load failed. | ||
|  | 
 | ||
|  | Additionally, with multiple directories used in eng_dyn, another problem | ||
|  | manifests because the errors pushed onto the error stack will linger even | ||
|  | if another library is loaded successfully on subsequent calls to | ||
|  | DSO_load(...) in the directory scanning loop. | ||
|  | 
 | ||
|  | Change-Id: I4ddd24f7b39bd88663e1783f30914870a907acfa | ||
|  | ---
 | ||
|  |  crypto/dso/dso_lib.c    | 8 ++++++++ | ||
|  |  crypto/engine/eng_dyn.c | 5 ++++- | ||
|  |  2 files changed, 12 insertions(+), 1 deletion(-) | ||
|  | 
 | ||
|  | diff --git a/crypto/dso/dso_lib.c b/crypto/dso/dso_lib.c
 | ||
|  | index 8a15b79..7801529 100644
 | ||
|  | --- a/crypto/dso/dso_lib.c
 | ||
|  | +++ b/crypto/dso/dso_lib.c
 | ||
|  | @@ -237,11 +237,19 @@ DSO *DSO_load(DSO *dso, const char *filename, DSO_METHOD *meth, int flags)
 | ||
|  |  	if(ret->meth->dso_load == NULL) | ||
|  |  		{ | ||
|  |  		DSOerr(DSO_F_DSO_LOAD,DSO_R_UNSUPPORTED); | ||
|  | +		/* Make sure we unset the filename on failure, because we use
 | ||
|  | +		 * this to determine when the DSO has been loaded above. */
 | ||
|  | +		OPENSSL_free(ret->filename);
 | ||
|  | +		ret->filename = NULL;
 | ||
|  |  		goto err; | ||
|  |  		} | ||
|  |  	if(!ret->meth->dso_load(ret)) | ||
|  |  		{ | ||
|  |  		DSOerr(DSO_F_DSO_LOAD,DSO_R_LOAD_FAILED); | ||
|  | +		/* Make sure we unset the filename on failure, because we use
 | ||
|  | +		 * this to determine when the DSO has been loaded above. */
 | ||
|  | +		OPENSSL_free(ret->filename);
 | ||
|  | +		ret->filename = NULL;
 | ||
|  |  		goto err; | ||
|  |  		} | ||
|  |  	/* Load succeeded */ | ||
|  | diff --git a/crypto/engine/eng_dyn.c b/crypto/engine/eng_dyn.c
 | ||
|  | index 807da7a..8fb8634 100644
 | ||
|  | --- a/crypto/engine/eng_dyn.c
 | ||
|  | +++ b/crypto/engine/eng_dyn.c
 | ||
|  | @@ -408,7 +408,7 @@ static int int_load(dynamic_data_ctx *ctx)
 | ||
|  |  	int num, loop; | ||
|  |  	/* Unless told not to, try a direct load */ | ||
|  |  	if((ctx->dir_load != 2) && (DSO_load(ctx->dynamic_dso, | ||
|  | -				ctx->DYNAMIC_LIBNAME, NULL, 0)) != NULL)
 | ||
|  | +				ctx->DYNAMIC_LIBNAME, NULL, 0) != NULL))
 | ||
|  |  		return 1; | ||
|  |  	/* If we're not allowed to use 'dirs' or we have none, fail */ | ||
|  |  	if(!ctx->dir_load || (num = sk_OPENSSL_STRING_num(ctx->dirs)) < 1) | ||
|  | @@ -423,6 +423,9 @@ static int int_load(dynamic_data_ctx *ctx)
 | ||
|  |  			{ | ||
|  |  			/* Found what we're looking for */ | ||
|  |  			OPENSSL_free(merge); | ||
|  | +			/* Previous failed loop iterations, if any, will have resulted in
 | ||
|  | +			 * errors. Clear them out before returning success. */
 | ||
|  | +			ERR_clear_error();
 | ||
|  |  			return 1; | ||
|  |  			} | ||
|  |  		OPENSSL_free(merge); | ||
|  | -- 
 | ||
|  | 1.7.12.3-x20-1 | ||
|  | 
 |