mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-20 19:53:10 +00:00 
			
		
		
		
	res_config_odbc: Fix old and new ast_string_field memory leaks.
The ODBC realtime driver uses ^NN parameter encoding to cope with the special meaning of the semi-colon. A semi-colon in a field is interpreted as if the key was supplied twice, something which isn't otherwise possible with fixed database columns. E.g. allow=alaw;ulaw is parsed as allow=alaw and allow=ulaw. A literal semi-colon is rewritten to ^3B when stored in the database. The module uses a stringfield to efficiently store the encoded parameters. However, this stringfield wasn't always freed in some off-nominal cases. Commit r413241 fixed initialization so the encoding for INSERT and DELETE queries wouldn't crash. (Only SELECTs and UPDATEs worked apparently.) But that commit forgot the frees. This change cleans that up. Review: https://reviewboard.asterisk.org/r/3555/ ........ Merged revisions 414564 from http://svn.asterisk.org/svn/asterisk/branches/1.8 ........ Merged revisions 414565 from http://svn.asterisk.org/svn/asterisk/branches/11 ........ Merged revisions 414566 from http://svn.asterisk.org/svn/asterisk/branches/12 git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@414567 65c4cc65-6c06-0410-ace0-fbb531ad65f3
This commit is contained in:
		| @@ -184,16 +184,7 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 	struct custom_prepare_struct cps = { .sql = sql, .fields = fields, }; | 	struct custom_prepare_struct cps = { .sql = sql, .fields = fields, }; | ||||||
| 	struct ast_flags connected_flag = { RES_ODBC_CONNECTED }; | 	struct ast_flags connected_flag = { RES_ODBC_CONNECTED }; | ||||||
|  |  | ||||||
| 	if (!fields) { | 	if (!table || !field) { | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if (ast_string_field_init(&cps, 256)) { |  | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if (!table) { |  | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -201,7 +192,6 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
|  |  | ||||||
| 	if (!obj) { | 	if (!obj) { | ||||||
| 		ast_log(LOG_ERROR, "No database handle available with the name of '%s' (check res_odbc.conf)\n", database); | 		ast_log(LOG_ERROR, "No database handle available with the name of '%s' (check res_odbc.conf)\n", database); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -214,11 +204,15 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 			strcasestr(field->name, "LIKE") && !ast_odbc_backslash_is_escape(obj) ? " ESCAPE '\\'" : ""); | 			strcasestr(field->name, "LIKE") && !ast_odbc_backslash_is_escape(obj) ? " ESCAPE '\\'" : ""); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if (ast_string_field_init(&cps, 256)) { | ||||||
|  | 		ast_odbc_release_obj(obj); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
| 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | ||||||
|  | 	ast_string_field_free_memory(&cps); | ||||||
|  |  | ||||||
| 	if (!stmt) { | 	if (!stmt) { | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -227,7 +221,6 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 		ast_log(LOG_WARNING, "SQL Column Count error!\n[%s]\n\n", sql); | 		ast_log(LOG_WARNING, "SQL Column Count error!\n[%s]\n\n", sql); | ||||||
| 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -235,14 +228,12 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 	if (res == SQL_NO_DATA) { | 	if (res == SQL_NO_DATA) { | ||||||
| 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
| 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) { | 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) { | ||||||
| 		ast_log(LOG_WARNING, "SQL Fetch error!\n[%s]\n\n", sql); | 		ast_log(LOG_WARNING, "SQL Fetch error!\n[%s]\n\n", sql); | ||||||
| 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | 		SQLFreeHandle (SQL_HANDLE_STMT, stmt); | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
| 	for (x = 0; x < colcount; x++) { | 	for (x = 0; x < colcount; x++) { | ||||||
| @@ -256,7 +247,6 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 			if (var) | 			if (var) | ||||||
| 				ast_variables_destroy(var); | 				ast_variables_destroy(var); | ||||||
| 			ast_odbc_release_obj(obj); | 			ast_odbc_release_obj(obj); | ||||||
| 			ast_string_field_free_memory(&cps); |  | ||||||
| 			return NULL; | 			return NULL; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @@ -296,10 +286,8 @@ static struct ast_variable *realtime_odbc(const char *database, const char *tabl | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
| 	SQLFreeHandle(SQL_HANDLE_STMT, stmt); | 	SQLFreeHandle(SQL_HANDLE_STMT, stmt); | ||||||
| 	ast_odbc_release_obj(obj); | 	ast_odbc_release_obj(obj); | ||||||
| 	ast_string_field_free_memory(&cps); |  | ||||||
| 	return var; | 	return var; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -344,13 +332,12 @@ static struct ast_config *realtime_multi_odbc(const char *database, const char * | |||||||
| 	SQLLEN indicator; | 	SQLLEN indicator; | ||||||
| 	struct custom_prepare_struct cps = { .sql = sql, .fields = fields, }; | 	struct custom_prepare_struct cps = { .sql = sql, .fields = fields, }; | ||||||
|  |  | ||||||
| 	if (!table || !field || ast_string_field_init(&cps, 256)) { | 	if (!table || !field) { | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	obj = ast_odbc_request_obj2(database, connected_flag); | 	obj = ast_odbc_request_obj2(database, connected_flag); | ||||||
| 	if (!obj) { | 	if (!obj) { | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -370,11 +357,15 @@ static struct ast_config *realtime_multi_odbc(const char *database, const char * | |||||||
|  |  | ||||||
| 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), " ORDER BY %s", initfield); | 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), " ORDER BY %s", initfield); | ||||||
|  |  | ||||||
|  | 	if (ast_string_field_init(&cps, 256)) { | ||||||
|  | 		ast_odbc_release_obj(obj); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
| 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | ||||||
|  | 	ast_string_field_free_memory(&cps); | ||||||
|  |  | ||||||
| 	if (!stmt) { | 	if (!stmt) { | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -383,7 +374,6 @@ static struct ast_config *realtime_multi_odbc(const char *database, const char * | |||||||
| 		ast_log(LOG_WARNING, "SQL Column Count error!\n[%s]\n\n", sql); | 		ast_log(LOG_WARNING, "SQL Column Count error!\n[%s]\n\n", sql); | ||||||
| 		SQLFreeHandle(SQL_HANDLE_STMT, stmt); | 		SQLFreeHandle(SQL_HANDLE_STMT, stmt); | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -392,7 +382,6 @@ static struct ast_config *realtime_multi_odbc(const char *database, const char * | |||||||
| 		ast_log(LOG_WARNING, "Out of memory!\n"); | 		ast_log(LOG_WARNING, "Out of memory!\n"); | ||||||
| 		SQLFreeHandle(SQL_HANDLE_STMT, stmt); | 		SQLFreeHandle(SQL_HANDLE_STMT, stmt); | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -450,7 +439,6 @@ next_sql_fetch:; | |||||||
|  |  | ||||||
| 	SQLFreeHandle(SQL_HANDLE_STMT, stmt); | 	SQLFreeHandle(SQL_HANDLE_STMT, stmt); | ||||||
| 	ast_odbc_release_obj(obj); | 	ast_odbc_release_obj(obj); | ||||||
| 	ast_string_field_free_memory(&cps); |  | ||||||
| 	return cfg; | 	return cfg; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -486,14 +474,9 @@ static int update_odbc(const char *database, const char *table, const char *keyf | |||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (ast_string_field_init(&cps, 256)) { |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	tableptr = ast_odbc_find_table(database, table); | 	tableptr = ast_odbc_find_table(database, table); | ||||||
| 	if (!(obj = ast_odbc_request_obj2(database, connected_flag))) { | 	if (!(obj = ast_odbc_request_obj2(database, connected_flag))) { | ||||||
| 		ast_odbc_release_table(tableptr); | 		ast_odbc_release_table(tableptr); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -524,18 +507,21 @@ static int update_odbc(const char *database, const char *table, const char *keyf | |||||||
| 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), " WHERE %s=?", keyfield); | 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), " WHERE %s=?", keyfield); | ||||||
| 	ast_odbc_release_table(tableptr); | 	ast_odbc_release_table(tableptr); | ||||||
|  |  | ||||||
|  | 	if (ast_string_field_init(&cps, 256)) { | ||||||
|  | 		ast_odbc_release_obj(obj); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
| 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | ||||||
|  | 	ast_string_field_free_memory(&cps); | ||||||
|  |  | ||||||
| 	if (!stmt) { | 	if (!stmt) { | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| 		ast_string_field_free_memory(&cps); |  | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	res = SQLRowCount(stmt, &rowcount); | 	res = SQLRowCount(stmt, &rowcount); | ||||||
| 	SQLFreeHandle (SQL_HANDLE_STMT, stmt); | 	SQLFreeHandle (SQL_HANDLE_STMT, stmt); | ||||||
| 	ast_odbc_release_obj(obj); | 	ast_odbc_release_obj(obj); | ||||||
| 	ast_string_field_free_memory(&cps); |  | ||||||
|  |  | ||||||
| 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) { | 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) { | ||||||
| 		ast_log(LOG_WARNING, "SQL Row Count error!\n[%s]\n\n", sql); | 		ast_log(LOG_WARNING, "SQL Row Count error!\n[%s]\n\n", sql); | ||||||
| @@ -704,10 +690,6 @@ static int store_odbc(const char *database, const char *table, const struct ast_ | |||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (ast_string_field_init(&cps, 256)) { |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	obj = ast_odbc_request_obj2(database, connected_flag); | 	obj = ast_odbc_request_obj2(database, connected_flag); | ||||||
| 	if (!obj) { | 	if (!obj) { | ||||||
| 		return -1; | 		return -1; | ||||||
| @@ -721,7 +703,12 @@ static int store_odbc(const char *database, const char *table, const struct ast_ | |||||||
| 	} | 	} | ||||||
| 	snprintf(sql, sizeof(sql), "INSERT INTO %s (%s) VALUES (%s)", table, keys, vals); | 	snprintf(sql, sizeof(sql), "INSERT INTO %s (%s) VALUES (%s)", table, keys, vals); | ||||||
|  |  | ||||||
|  | 	if (ast_string_field_init(&cps, 256)) { | ||||||
|  | 		ast_odbc_release_obj(obj); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
| 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | ||||||
|  | 	ast_string_field_free_memory(&cps); | ||||||
|  |  | ||||||
| 	if (!stmt) { | 	if (!stmt) { | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
| @@ -773,10 +760,6 @@ static int destroy_odbc(const char *database, const char *table, const char *key | |||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (ast_string_field_init(&cps, 256)) { |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	obj = ast_odbc_request_obj2(database, connected_flag); | 	obj = ast_odbc_request_obj2(database, connected_flag); | ||||||
| 	if (!obj) { | 	if (!obj) { | ||||||
| 		return -1; | 		return -1; | ||||||
| @@ -789,7 +772,12 @@ static int destroy_odbc(const char *database, const char *table, const char *key | |||||||
| 	} | 	} | ||||||
| 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), "%s=?", keyfield); | 	snprintf(sql + strlen(sql), sizeof(sql) - strlen(sql), "%s=?", keyfield); | ||||||
|  |  | ||||||
|  | 	if (ast_string_field_init(&cps, 256)) { | ||||||
|  | 		ast_odbc_release_obj(obj); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
| 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | 	stmt = ast_odbc_prepare_and_execute(obj, custom_prepare, &cps); | ||||||
|  | 	ast_string_field_free_memory(&cps); | ||||||
|  |  | ||||||
| 	if (!stmt) { | 	if (!stmt) { | ||||||
| 		ast_odbc_release_obj(obj); | 		ast_odbc_release_obj(obj); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user