innodb_thread_concurrency算法介绍:

1、server层到innodb层读写数据是一条一条记录进行的,每次读写都会进/出一次InnoDB层(row_search_for_mysql),进入InnoDB层的时候会检查当前并发线程数目,当超过innodb_thread_concurrency时,线程将尝试spin和sleep并再次检查,如果并发数还是超过innodb_thread_concurrency,线程将进入到一个FIFO中等待被唤醒,读写记录结束后退出InnoDB层时会将当前并发线程数减1,并检查其是否低于innodb_thread_concurrency,如果是的话,从FIFO中唤醒一个等待的线程,保证并发线程不会超过innodb_thread_concurrency参数。

2、当线程进入InnoDB层后,但在获取数据时由于锁请求无法得到满足而需要挂起时,线程将强制退出InnoDB层,当锁请求满足后,线程继续运行并强制进入到InnoDB层,这会导致实际并发线程数不是严格控制在innodb_thread_concurrency之内

代码调用逻辑

主要函数:

innodb_srv_conc_enter_innodb

innodb_srv_conc_exit_innodb

srv_conc_force_enter_innodb

srv_conc_force_exit_innodb

调用逻辑

进入/退出InnoDB层

ha_innobase::index_read

ha_innobase::general_fetch

row_check_index_for_mysql

...

innodb_srv_conc_enter_innodb(prebuilt->trx);

ret = row_search_for_mysql((byte*) buf, mode,

prebuilt,  match_mode, 0);

innodb_srv_conc_exit_innodb(prebuilt->trx);

...

锁等待逻辑:

srv_suspend_mysql_thread

...

if (trx->declared_to_be_inside_innodb)

{

was_declared_inside_innodb =

TRUE;

/* We must declare this OS

thread to exit InnoDB, since a

possible other thread holding

a lock which this thread waits

for must be allowed to enter,

sooner or later */

srv_conc_force_exit_innodb(trx);

}

/* Suspend this thread and wait for the

event. */

thd_wait_begin(trx->mysql_thd,

THD_WAIT_ROW_LOCK);

os_event_wait(event);

thd_wait_end(trx->mysql_thd);

if (was_declared_inside_innodb) {

/* Return back inside InnoDB

*/

srv_conc_force_enter_innodb(trx);

}

5.6的改进

从5.6开始,默认编译会使用GCC atomic,可避免热点锁srv_conc_mutex的频繁加锁/释放。

Pre-5.5如果超过并发限制时,就给其线程分配一个slot,让其进入信号量等待,原子操作则无需使用信号量。

function srv_conc_enter_innodb:

ifdef HAVE_ATOMIC_BUILTINS

srv_conc_enter_innodb_with_atomics(trx);

#else

srv_conc_enter_innodb_without_atomics(trx);

#endif /* HAVE_ATOMIC_BUILTINS */

5.6引入adaptive sleep,允许innodb根据系统负载进行自适应调整,当innodb_adaptive_max_sleep_delay>0,innodb_thread_sleep_delay则会动态调整(以前者为上限)。

ada还ptive 的算法是:

如果sleep了当前值以后还是不能进入,就把sleep时间+1;

如果还有线程在sleep时,已经有了空闲线程,就把当前值的sleep 时间减半。

innodb_thread_sleep_delay降低比增加的更快,这样在并发线程数很高时,当限制并发数早就达到,其他线程的每次sleep时间会缓慢拉长。而当Innodb层很空闲时,sleep时间又会快速降到非常低

调整sleep到一个优化值的目的是,过小的sleep值可能会产生太多的线程切换,但过长的sleep时间,在并发比较空闲的时候又会影响性能。新的并发控制策略有利于随着负载的变化而做自适应调整。

Innodb_thread_concurrency不足

1线程因为锁等待而退出innodb层,当获取锁时可以直接重入innodb(跳过此参数检查),因此系统并发执行的线程数可能大于此参数值。

2代码路径靠后,此时线程已经开始执行命令,进入到ha_innobase,应该在mysql_start_query前限制即thread_running。

为此好多技术高手开发了相应patch,从server层控制并发执行的线程数,下文将做描述。