diff --git a/apps/app_voicemail.c b/apps/app_voicemail.c
index a2f6c36b84c5923793a7da7988b88509ea419bce..6a29a20c81c1c83eb6e07fcfc2ee510dd11afa79 100644
--- a/apps/app_voicemail.c
+++ b/apps/app_voicemail.c
@@ -3838,7 +3838,7 @@ static SQLHSTMT generic_prepare(struct odbc_obj *obj, void *data)
 		ast_log(AST_LOG_WARNING, "SQL Alloc Handle failed!\n");
 		return NULL;
 	}
-	res = SQLPrepare(stmt, (unsigned char *) gps->sql, SQL_NTS);
+	res = ast_odbc_prepare(obj, stmt, gps->sql);
 	if (!SQL_SUCCEEDED(res)) {
 		ast_log(AST_LOG_WARNING, "SQL Prepare failed![%s]\n", gps->sql);
 		SQLFreeHandle(SQL_HANDLE_STMT, stmt);
@@ -4361,7 +4361,7 @@ static SQLHSTMT insert_data_cb(struct odbc_obj *obj, void *vdata)
 	if (!ast_strlen_zero(data->category)) {
 		SQLBindParameter(stmt, 13, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, strlen(data->category), 0, (void *) data->category, 0, NULL);
 	}
-	res = SQLExecDirect(stmt, (unsigned char *) data->sql, SQL_NTS);
+	res = ast_odbc_execute_sql(obj, stmt, data->sql);
 	if (!SQL_SUCCEEDED(res)) {
 		ast_log(AST_LOG_WARNING, "SQL Direct Execute failed!\n");
 		SQLFreeHandle(SQL_HANDLE_STMT, stmt);
diff --git a/cdr/cdr_adaptive_odbc.c b/cdr/cdr_adaptive_odbc.c
index 321a646314d40892e059e574f7214b86d58f107f..d810f9c3fd22e6ae3fba2c49e0475a3c09f819cf 100644
--- a/cdr/cdr_adaptive_odbc.c
+++ b/cdr/cdr_adaptive_odbc.c
@@ -327,7 +327,7 @@ static SQLHSTMT generic_prepare(struct odbc_obj *obj, void *data)
 		return NULL;
 	}
 
-	res = SQLPrepare(stmt, (unsigned char *) data, SQL_NTS);
+	res = ast_odbc_prepare(obj, stmt, data);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		ast_log(LOG_WARNING, "SQL Prepare failed![%s]\n", (char *) data);
 		SQLGetDiagField(SQL_HANDLE_STMT, stmt, 1, SQL_DIAG_NUMBER, &numfields, SQL_IS_INTEGER, &diagbytes);
diff --git a/cdr/cdr_odbc.c b/cdr/cdr_odbc.c
index aed7499501275dae3bc289ce9cec83c2515a9910..455217afa9aef3b8cc622d3d9223b0514f42469d 100644
--- a/cdr/cdr_odbc.c
+++ b/cdr/cdr_odbc.c
@@ -155,7 +155,7 @@ static SQLHSTMT execute_cb(struct odbc_obj *obj, void *data)
 		SQLBindParameter(stmt, i + 2, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER, 0, 0, &cdr->sequence, 0, NULL);
 	}
 
-	ODBC_res = SQLExecDirect(stmt, (unsigned char *)sqlcmd, SQL_NTS);
+	ODBC_res = ast_odbc_execute_sql(obj, stmt, sqlcmd);
 
 	if ((ODBC_res != SQL_SUCCESS) && (ODBC_res != SQL_SUCCESS_WITH_INFO)) {
 		ast_log(LOG_WARNING, "cdr_odbc: Error in ExecDirect: %d, query is: %s\n", ODBC_res, sqlcmd);
diff --git a/cel/cel_odbc.c b/cel/cel_odbc.c
index ec2feaba950792d3f6c52dc1e194971b78805cbf..f6b6321bad0356ae33433e7922f7b1b627e83a2e 100644
--- a/cel/cel_odbc.c
+++ b/cel/cel_odbc.c
@@ -325,7 +325,7 @@ static SQLHSTMT generic_prepare(struct odbc_obj *obj, void *data)
 		return NULL;
 	}
 
-	res = SQLPrepare(stmt, (unsigned char *)sql, SQL_NTS);
+	res = ast_odbc_prepare(obj, stmt, sql);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		ast_log(LOG_WARNING, "SQL Prepare failed![%s]\n", sql);
 		SQLGetDiagField(SQL_HANDLE_STMT, stmt, 1, SQL_DIAG_NUMBER, &numfields, SQL_IS_INTEGER, &diagbytes);
diff --git a/configs/samples/res_odbc.conf.sample b/configs/samples/res_odbc.conf.sample
index c69f55637555ccc428e49969ce355ed079e69acc..58671fc4af235b78a583aa2594f3d2e75bdd5826 100644
--- a/configs/samples/res_odbc.conf.sample
+++ b/configs/samples/res_odbc.conf.sample
@@ -65,6 +65,20 @@ pre-connect => yes
 ; information before we attempt another connection?  This increases
 ; responsiveness, when a database resource is not working.
 ;negative_connection_cache => 300
+;
+; Enable query logging. This keeps track of the number of prepared queries
+; and executed queries as well as the query that has taken the longest to
+; execute. This can be useful for determining the latency with a database.
+; The data can be viewed using the "odbc show" CLI command.
+; Note that only successful queries are logged currently.
+;logging => yes
+;
+; Slow query limit. If a query exceeds the given amount of time (in milliseconds)
+; when executing then a WARNING message will be output to indicate that there
+; may be a problem. Note that logging must be set to "yes" for this to occur. By
+; default this is set to 5000 milliseconds (or 5 seconds). If you would like to
+; disable the WARNING message it can be set to "0".
+;slow_query_limit => 5000
 
 [mysql2]
 enabled => no
diff --git a/funcs/func_odbc.c b/funcs/func_odbc.c
index 70f02b8570bc0175ec2d06d1c2a655d422d6d0ff..fc5fe2bd0c6fd899eb02d19fb3f93661b872bed3 100644
--- a/funcs/func_odbc.c
+++ b/funcs/func_odbc.c
@@ -464,7 +464,7 @@ static SQLHSTMT execute(struct odbc_obj *obj, void *data, int silent)
 		return NULL;
 	}
 
-	res = SQLExecDirect(stmt, (unsigned char *)sql, SQL_NTS);
+	res = ast_odbc_execute_sql(obj, stmt, sql);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO) && (res != SQL_NO_DATA)) {
 		if (res == SQL_ERROR && !silent) {
 			int i;
diff --git a/include/asterisk/res_odbc.h b/include/asterisk/res_odbc.h
index 4286272fcce0e37a3be02db479fd164deed6c574..9a38915bc1b61d75f6e7362d85ed5dd8678e31f8 100644
--- a/include/asterisk/res_odbc.h
+++ b/include/asterisk/res_odbc.h
@@ -51,6 +51,7 @@ struct odbc_obj {
 	char function[80];
 	int lineno;
 #endif
+	char *sql_text;					/*!< The SQL text currently executing */
 	AST_LIST_ENTRY(odbc_obj) list;
 };
 
@@ -160,6 +161,22 @@ SQLHSTMT ast_odbc_direct_execute(struct odbc_obj *obj, SQLHSTMT (*exec_cb)(struc
  */
 SQLHSTMT ast_odbc_prepare_and_execute(struct odbc_obj *obj, SQLHSTMT (*prepare_cb)(struct odbc_obj *obj, void *data), void *data);
 
+/*!
+ * \brief Prepares a SQL query on a statement.
+ * \param obj The ODBC object
+ * \param stmt The statement
+ * \parma sql The SQL query
+ * \note This should be used in place of SQLPrepare
+ */
+int ast_odbc_prepare(struct odbc_obj *obj, SQLHSTMT *stmt, const char *sql);
+
+/*! \brief Execute a nonprepared SQL query.
+ * \param obj The ODBC object
+ * \param sql The SQL query
+ * \note This should be used in place of SQLExecDirect
+ */
+SQLRETURN ast_odbc_execute_sql(struct odbc_obj *obj, SQLHSTMT *stmt, const char *sql);
+
 /*!
  * \brief Find or create an entry describing the table specified.
  * \param database Name of an ODBC class on which to query the table
diff --git a/res/res_config_odbc.c b/res/res_config_odbc.c
index 8e54b47882337bb9993ef0f69528520607fe3580..a9c349566d0e9429c3d1e4de53b343f64d0f9358 100644
--- a/res/res_config_odbc.c
+++ b/res/res_config_odbc.c
@@ -116,7 +116,7 @@ static SQLHSTMT custom_prepare(struct odbc_obj *obj, void *data)
 
 	ast_debug(1, "Skip: %llu; SQL: %s\n", cps->skip, cps->sql);
 
-	res = SQLPrepare(stmt, (unsigned char *)cps->sql, SQL_NTS);
+	res = ast_odbc_prepare(obj, stmt, cps->sql);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		if (res == SQL_ERROR) {
 			ast_odbc_print_errors(SQL_HANDLE_STMT, stmt, "SQL Prepare");
@@ -634,7 +634,7 @@ static SQLHSTMT update2_prepare(struct odbc_obj *obj, void *data)
 	/* Done with the table metadata */
 	ast_odbc_release_table(tableptr);
 
-	res = SQLPrepare(stmt, (unsigned char *)ast_str_buffer(sql), SQL_NTS);
+	res = ast_odbc_prepare(obj, stmt, ast_str_buffer(sql));
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		if (res == SQL_ERROR) {
 			ast_odbc_print_errors(SQL_HANDLE_STMT, stmt, "SQL Prepare");
@@ -872,7 +872,7 @@ static SQLHSTMT length_determination_odbc_prepare(struct odbc_obj *obj, void *da
 		return NULL;
 	}
 
-	res = SQLPrepare(sth, (unsigned char *)q->sql, SQL_NTS);
+	res = ast_odbc_prepare(obj, sth, q->sql);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		ast_verb(4, "Error in PREPARE %d\n", res);
 		SQLFreeHandle(SQL_HANDLE_STMT, sth);
@@ -896,7 +896,7 @@ static SQLHSTMT config_odbc_prepare(struct odbc_obj *obj, void *data)
 		return NULL;
 	}
 
-	res = SQLPrepare(sth, (unsigned char *)q->sql, SQL_NTS);
+	res = ast_odbc_prepare(obj, sth, q->sql);
 	if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
 		ast_verb(4, "Error in PREPARE %d\n", res);
 		SQLFreeHandle(SQL_HANDLE_STMT, sth);
diff --git a/res/res_odbc.c b/res/res_odbc.c
index cd588ee69afe7275b38c6396f37e0b2d6df68da3..5b79bd433f607fdd9626f6d44487581904a46c0c 100644
--- a/res/res_odbc.c
+++ b/res/res_odbc.c
@@ -91,6 +91,18 @@ struct odbc_class
 	ast_cond_t cond;
 	/*! The total number of current connections */
 	size_t connection_cnt;
+	/*! Whether logging is enabled on this class or not */
+	unsigned int logging;
+	/*! The number of prepares executed on this class (total from all connections */
+	int prepares_executed;
+	/*! The number of queries executed on this class (total from all connections) */
+	int queries_executed;
+	/*! The longest execution time for a query executed on this class */
+	long longest_query_execution_time;
+	/*! The SQL query that took the longest to execute */
+	char *sql_text;
+	/*! Slow query limit (in milliseconds) */
+	unsigned int slowquerylimit;
 };
 
 static struct ao2_container *class_container;
@@ -188,6 +200,7 @@ static void odbc_class_destructor(void *data)
 	SQLFreeHandle(SQL_HANDLE_ENV, class->env);
 	ast_mutex_destroy(&class->lock);
 	ast_cond_destroy(&class->cond);
+	ast_free(class->sql_text);
 }
 
 static void odbc_obj_destructor(void *data)
@@ -367,18 +380,52 @@ int ast_odbc_clear_cache(const char *database, const char *tablename)
 
 SQLHSTMT ast_odbc_direct_execute(struct odbc_obj *obj, SQLHSTMT (*exec_cb)(struct odbc_obj *obj, void *data), void *data)
 {
+	struct timeval start;
 	SQLHSTMT stmt;
 
+	if (obj->parent->logging) {
+		start = ast_tvnow();
+	}
+
 	stmt = exec_cb(obj, data);
 
+	if (obj->parent->logging) {
+		long execution_time = ast_tvdiff_ms(ast_tvnow(), start);
+
+		if (obj->parent->slowquerylimit && execution_time > obj->parent->slowquerylimit) {
+			ast_log(LOG_WARNING, "SQL query '%s' took %ld milliseconds to execute on class '%s', this may indicate a database problem\n",
+				obj->sql_text, execution_time, obj->parent->name);
+		}
+
+		ast_mutex_lock(&obj->parent->lock);
+		if (execution_time > obj->parent->longest_query_execution_time || !obj->parent->sql_text) {
+			obj->parent->longest_query_execution_time = execution_time;
+			/* Due to the callback nature of the res_odbc API it's not possible to ensure that
+			 * the SQL text is removed from the connection in all cases, so only if it becomes the
+			 * new longest executing query do we steal the SQL text. In other cases what will happen
+			 * is that the SQL text will be freed if the connection is released back to the class or
+			 * if a new query is done on the connection.
+			 */
+			ast_free(obj->parent->sql_text);
+			obj->parent->sql_text = obj->sql_text;
+			obj->sql_text = NULL;
+		}
+		ast_mutex_unlock(&obj->parent->lock);
+	}
+
 	return stmt;
 }
 
 SQLHSTMT ast_odbc_prepare_and_execute(struct odbc_obj *obj, SQLHSTMT (*prepare_cb)(struct odbc_obj *obj, void *data), void *data)
 {
+	struct timeval start;
 	int res = 0;
 	SQLHSTMT stmt;
 
+	if (obj->parent->logging) {
+		start = ast_tvnow();
+	}
+
 	/* This prepare callback may do more than just prepare -- it may also
 	 * bind parameters, bind results, etc.  The real key, here, is that
 	 * when we disconnect, all handles become invalid for most databases.
@@ -398,11 +445,59 @@ SQLHSTMT ast_odbc_prepare_and_execute(struct odbc_obj *obj, SQLHSTMT (*prepare_c
 		ast_log(LOG_WARNING, "SQL Execute error %d!\n", res);
 		SQLFreeHandle(SQL_HANDLE_STMT, stmt);
 		stmt = NULL;
+	} else if (obj->parent->logging) {
+		long execution_time = ast_tvdiff_ms(ast_tvnow(), start);
+
+		if (obj->parent->slowquerylimit && execution_time > obj->parent->slowquerylimit) {
+			ast_log(LOG_WARNING, "SQL query '%s' took %ld milliseconds to execute on class '%s', this may indicate a database problem\n",
+				obj->sql_text, execution_time, obj->parent->name);
+		}
+
+		ast_mutex_lock(&obj->parent->lock);
+
+		/* If this takes the record on longest query execution time, update the parent class
+		 * with the information.
+		 */
+		if (execution_time > obj->parent->longest_query_execution_time || !obj->parent->sql_text) {
+			obj->parent->longest_query_execution_time = execution_time;
+			ast_free(obj->parent->sql_text);
+			obj->parent->sql_text = obj->sql_text;
+			obj->sql_text = NULL;
+		}
+		ast_mutex_unlock(&obj->parent->lock);
+
+		ast_atomic_fetchadd_int(&obj->parent->queries_executed, +1);
 	}
 
 	return stmt;
 }
 
+int ast_odbc_prepare(struct odbc_obj *obj, SQLHSTMT *stmt, const char *sql)
+{
+	if (obj->parent->logging) {
+		/* It is possible for this connection to be reused without being
+		 * released back to the class, so we free what may already exist
+		 * and place the new SQL in.
+		 */
+		ast_free(obj->sql_text);
+		obj->sql_text = ast_strdup(sql);
+		ast_atomic_fetchadd_int(&obj->parent->prepares_executed, +1);
+	}
+
+	return SQLPrepare(stmt, (unsigned char *)sql, SQL_NTS);
+}
+
+SQLRETURN ast_odbc_execute_sql(struct odbc_obj *obj, SQLHSTMT *stmt, const char *sql)
+{
+	if (obj->parent->logging) {
+		ast_free(obj->sql_text);
+		obj->sql_text = ast_strdup(sql);
+		ast_atomic_fetchadd_int(&obj->parent->queries_executed, +1);
+	}
+
+	return SQLExecDirect(stmt, (unsigned char *)sql, SQL_NTS);
+}
+
 int ast_odbc_smart_execute(struct odbc_obj *obj, SQLHSTMT stmt)
 {
 	int res = 0;
@@ -414,6 +509,10 @@ int ast_odbc_smart_execute(struct odbc_obj *obj, SQLHSTMT stmt)
 		}
 	}
 
+	if (obj->parent->logging) {
+		ast_atomic_fetchadd_int(&obj->parent->queries_executed, +1);
+	}
+
 	return res;
 }
 
@@ -481,7 +580,7 @@ static int load_odbc_config(void)
 	struct ast_variable *v;
 	char *cat;
 	const char *dsn, *username, *password, *sanitysql;
-	int enabled, bse, conntimeout, forcecommit, isolation, maxconnections;
+	int enabled, bse, conntimeout, forcecommit, isolation, maxconnections, logging, slowquerylimit;
 	struct timeval ncache = { 0, 0 };
 	int preconnect = 0, res = 0;
 	struct ast_flags config_flags = { 0 };
@@ -509,6 +608,8 @@ static int load_odbc_config(void)
 			forcecommit = 0;
 			isolation = SQL_TXN_READ_COMMITTED;
 			maxconnections = 1;
+			logging = 0;
+			slowquerylimit = 5000;
 			for (v = ast_variable_browse(config, cat); v; v = v->next) {
 				if (!strcasecmp(v->name, "pooling") ||
 						!strncasecmp(v->name, "share", 5) ||
@@ -557,6 +658,13 @@ static int load_odbc_config(void)
 						ast_log(LOG_WARNING, "max_connections must be a positive integer\n");
 						maxconnections = 1;
                                         }
+				} else if (!strcasecmp(v->name, "logging")) {
+					logging = ast_true(v->value);
+				} else if (!strcasecmp(v->name, "slow_query_limit")) {
+					if (sscanf(v->value, "%30d", &slowquerylimit) != 1) {
+						ast_log(LOG_WARNING, "slow_query_limit must be a positive integer\n");
+						slowquerylimit = 5000;
+					}
 				}
 			}
 
@@ -583,6 +691,8 @@ static int load_odbc_config(void)
 				new->conntimeout = conntimeout;
 				new->negative_connection_cache = ncache;
 				new->maxconnections = maxconnections;
+				new->logging = logging;
+				new->slowquerylimit = slowquerylimit;
 
 				if (cat)
 					ast_copy_string(new->name, cat, sizeof(new->name));
@@ -665,6 +775,16 @@ static char *handle_cli_odbc_show(struct ast_cli_entry *e, int cmd, struct ast_c
 			ast_cli(a->fd, "  Name:   %s\n  DSN:    %s\n", class->name, class->dsn);
 			ast_cli(a->fd, "    Last connection attempt: %s\n", timestr);
 			ast_cli(a->fd, "    Number of active connections: %zd (out of %d)\n", class->connection_cnt, class->maxconnections);
+			ast_cli(a->fd, "    Logging: %s\n", class->logging ? "Enabled" : "Disabled");
+			if (class->logging) {
+				ast_cli(a->fd, "    Number of prepares executed: %d\n", class->prepares_executed);
+				ast_cli(a->fd, "    Number of queries executed: %d\n", class->queries_executed);
+				ast_mutex_lock(&class->lock);
+				if (class->sql_text) {
+					ast_cli(a->fd, "    Longest running SQL query: %s (%ld milliseconds)\n", class->sql_text, class->longest_query_execution_time);
+				}
+				ast_mutex_unlock(&class->lock);
+			}
 			ast_cli(a->fd, "\n");
 		}
 		ao2_ref(class, -1);
@@ -713,6 +833,12 @@ void ast_odbc_release_obj(struct odbc_obj *obj)
 	 */
 	obj->parent = NULL;
 
+	/* Free the SQL text so that the next user of this connection has
+	 * a fresh start.
+	 */
+	ast_free(obj->sql_text);
+	obj->sql_text = NULL;
+
 	ast_mutex_lock(&class->lock);
 	AST_LIST_INSERT_HEAD(&class->connections, obj, list);
 	ast_cond_signal(&class->cond);