⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 rcutorture.c

📁 Kernel code of linux kernel
💻 C
📖 第 1 页 / 共 3 页
字号:
		       atomic_read(&n_rcu_torture_free),		       atomic_read(&n_rcu_torture_mberror),		       n_rcu_torture_timers);	if (atomic_read(&n_rcu_torture_mberror) != 0)		cnt += sprintf(&page[cnt], " !!!");	cnt += sprintf(&page[cnt], "\n%s%s ", torture_type, TORTURE_FLAG);	if (i > 1) {		cnt += sprintf(&page[cnt], "!!! ");		atomic_inc(&n_rcu_torture_error);		WARN_ON_ONCE(1);	}	cnt += sprintf(&page[cnt], "Reader Pipe: ");	for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++)		cnt += sprintf(&page[cnt], " %ld", pipesummary[i]);	cnt += sprintf(&page[cnt], "\n%s%s ", torture_type, TORTURE_FLAG);	cnt += sprintf(&page[cnt], "Reader Batch: ");	for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++)		cnt += sprintf(&page[cnt], " %ld", batchsummary[i]);	cnt += sprintf(&page[cnt], "\n%s%s ", torture_type, TORTURE_FLAG);	cnt += sprintf(&page[cnt], "Free-Block Circulation: ");	for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++) {		cnt += sprintf(&page[cnt], " %d",			       atomic_read(&rcu_torture_wcount[i]));	}	cnt += sprintf(&page[cnt], "\n");	if (cur_ops->stats)		cnt += cur_ops->stats(&page[cnt]);	return cnt;}/* * Print torture statistics.  Caller must ensure that there is only * one call to this function at a given time!!!  This is normally * accomplished by relying on the module system to only have one copy * of the module loaded, and then by giving the rcu_torture_stats * kthread full control (or the init/cleanup functions when rcu_torture_stats * thread is not running). */static voidrcu_torture_stats_print(void){	int cnt;	cnt = rcu_torture_printk(printk_buf);	printk(KERN_ALERT "%s", printk_buf);}/* * Periodically prints torture statistics, if periodic statistics printing * was specified via the stat_interval module parameter. * * No need to worry about fullstop here, since this one doesn't reference * volatile state or register callbacks. */static intrcu_torture_stats(void *arg){	VERBOSE_PRINTK_STRING("rcu_torture_stats task started");	do {		schedule_timeout_interruptible(stat_interval * HZ);		rcu_torture_stats_print();	} while (!kthread_should_stop());	VERBOSE_PRINTK_STRING("rcu_torture_stats task stopping");	return 0;}static int rcu_idle_cpu;	/* Force all torture tasks off this CPU *//* Shuffle tasks such that we allow @rcu_idle_cpu to become idle. A special case * is when @rcu_idle_cpu = -1, when we allow the tasks to run on all CPUs. */static void rcu_torture_shuffle_tasks(void){	cpumask_t tmp_mask;	int i;	cpus_setall(tmp_mask);	get_online_cpus();	/* No point in shuffling if there is only one online CPU (ex: UP) */	if (num_online_cpus() == 1) {		put_online_cpus();		return;	}	if (rcu_idle_cpu != -1)		cpu_clear(rcu_idle_cpu, tmp_mask);	set_cpus_allowed_ptr(current, &tmp_mask);	if (reader_tasks) {		for (i = 0; i < nrealreaders; i++)			if (reader_tasks[i])				set_cpus_allowed_ptr(reader_tasks[i],						     &tmp_mask);	}	if (fakewriter_tasks) {		for (i = 0; i < nfakewriters; i++)			if (fakewriter_tasks[i])				set_cpus_allowed_ptr(fakewriter_tasks[i],						     &tmp_mask);	}	if (writer_task)		set_cpus_allowed_ptr(writer_task, &tmp_mask);	if (stats_task)		set_cpus_allowed_ptr(stats_task, &tmp_mask);	if (rcu_idle_cpu == -1)		rcu_idle_cpu = num_online_cpus() - 1;	else		rcu_idle_cpu--;	put_online_cpus();}/* Shuffle tasks across CPUs, with the intent of allowing each CPU in the * system to become idle at a time and cut off its timer ticks. This is meant * to test the support for such tickless idle CPU in RCU. */static intrcu_torture_shuffle(void *arg){	VERBOSE_PRINTK_STRING("rcu_torture_shuffle task started");	do {		schedule_timeout_interruptible(shuffle_interval * HZ);		rcu_torture_shuffle_tasks();	} while (!kthread_should_stop());	VERBOSE_PRINTK_STRING("rcu_torture_shuffle task stopping");	return 0;}/* Cause the rcutorture test to "stutter", starting and stopping all * threads periodically. */static intrcu_torture_stutter(void *arg){	VERBOSE_PRINTK_STRING("rcu_torture_stutter task started");	do {		schedule_timeout_interruptible(stutter * HZ);		stutter_pause_test = 1;		if (!kthread_should_stop())			schedule_timeout_interruptible(stutter * HZ);		stutter_pause_test = 0;	} while (!kthread_should_stop());	VERBOSE_PRINTK_STRING("rcu_torture_stutter task stopping");	return 0;}static inline voidrcu_torture_print_module_parms(char *tag){	printk(KERN_ALERT "%s" TORTURE_FLAG		"--- %s: nreaders=%d nfakewriters=%d "		"stat_interval=%d verbose=%d test_no_idle_hz=%d "		"shuffle_interval=%d stutter=%d irqreader=%d\n",		torture_type, tag, nrealreaders, nfakewriters,		stat_interval, verbose, test_no_idle_hz, shuffle_interval,		stutter, irqreader);}static voidrcu_torture_cleanup(void){	int i;	fullstop = 1;	if (stutter_task) {		VERBOSE_PRINTK_STRING("Stopping rcu_torture_stutter task");		kthread_stop(stutter_task);	}	stutter_task = NULL;	if (shuffler_task) {		VERBOSE_PRINTK_STRING("Stopping rcu_torture_shuffle task");		kthread_stop(shuffler_task);	}	shuffler_task = NULL;	if (writer_task) {		VERBOSE_PRINTK_STRING("Stopping rcu_torture_writer task");		kthread_stop(writer_task);	}	writer_task = NULL;	if (reader_tasks) {		for (i = 0; i < nrealreaders; i++) {			if (reader_tasks[i]) {				VERBOSE_PRINTK_STRING(					"Stopping rcu_torture_reader task");				kthread_stop(reader_tasks[i]);			}			reader_tasks[i] = NULL;		}		kfree(reader_tasks);		reader_tasks = NULL;	}	rcu_torture_current = NULL;	if (fakewriter_tasks) {		for (i = 0; i < nfakewriters; i++) {			if (fakewriter_tasks[i]) {				VERBOSE_PRINTK_STRING(					"Stopping rcu_torture_fakewriter task");				kthread_stop(fakewriter_tasks[i]);			}			fakewriter_tasks[i] = NULL;		}		kfree(fakewriter_tasks);		fakewriter_tasks = NULL;	}	if (stats_task) {		VERBOSE_PRINTK_STRING("Stopping rcu_torture_stats task");		kthread_stop(stats_task);	}	stats_task = NULL;	/* Wait for all RCU callbacks to fire.  */	if (cur_ops->cb_barrier != NULL)		cur_ops->cb_barrier();	rcu_torture_stats_print();  /* -After- the stats thread is stopped! */	if (cur_ops->cleanup)		cur_ops->cleanup();	if (atomic_read(&n_rcu_torture_error))		rcu_torture_print_module_parms("End of test: FAILURE");	else		rcu_torture_print_module_parms("End of test: SUCCESS");}static int __initrcu_torture_init(void){	int i;	int cpu;	int firsterr = 0;	static struct rcu_torture_ops *torture_ops[] =		{ &rcu_ops, &rcu_sync_ops, &rcu_bh_ops, &rcu_bh_sync_ops,		  &srcu_ops, &sched_ops, &sched_ops_sync, };	/* Process args and tell the world that the torturer is on the job. */	for (i = 0; i < ARRAY_SIZE(torture_ops); i++) {		cur_ops = torture_ops[i];		if (strcmp(torture_type, cur_ops->name) == 0)			break;	}	if (i == ARRAY_SIZE(torture_ops)) {		printk(KERN_ALERT "rcutorture: invalid torture type: \"%s\"\n",		       torture_type);		return (-EINVAL);	}	if (cur_ops->init)		cur_ops->init(); /* no "goto unwind" prior to this point!!! */	if (nreaders >= 0)		nrealreaders = nreaders;	else		nrealreaders = 2 * num_online_cpus();	rcu_torture_print_module_parms("Start of test");	fullstop = 0;	/* Set up the freelist. */	INIT_LIST_HEAD(&rcu_torture_freelist);	for (i = 0; i < ARRAY_SIZE(rcu_tortures); i++) {		rcu_tortures[i].rtort_mbtest = 0;		list_add_tail(&rcu_tortures[i].rtort_free,			      &rcu_torture_freelist);	}	/* Initialize the statistics so that each run gets its own numbers. */	rcu_torture_current = NULL;	rcu_torture_current_version = 0;	atomic_set(&n_rcu_torture_alloc, 0);	atomic_set(&n_rcu_torture_alloc_fail, 0);	atomic_set(&n_rcu_torture_free, 0);	atomic_set(&n_rcu_torture_mberror, 0);	atomic_set(&n_rcu_torture_error, 0);	for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++)		atomic_set(&rcu_torture_wcount[i], 0);	for_each_possible_cpu(cpu) {		for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++) {			per_cpu(rcu_torture_count, cpu)[i] = 0;			per_cpu(rcu_torture_batch, cpu)[i] = 0;		}	}	/* Start up the kthreads. */	VERBOSE_PRINTK_STRING("Creating rcu_torture_writer task");	writer_task = kthread_run(rcu_torture_writer, NULL,				  "rcu_torture_writer");	if (IS_ERR(writer_task)) {		firsterr = PTR_ERR(writer_task);		VERBOSE_PRINTK_ERRSTRING("Failed to create writer");		writer_task = NULL;		goto unwind;	}	fakewriter_tasks = kzalloc(nfakewriters * sizeof(fakewriter_tasks[0]),	                           GFP_KERNEL);	if (fakewriter_tasks == NULL) {		VERBOSE_PRINTK_ERRSTRING("out of memory");		firsterr = -ENOMEM;		goto unwind;	}	for (i = 0; i < nfakewriters; i++) {		VERBOSE_PRINTK_STRING("Creating rcu_torture_fakewriter task");		fakewriter_tasks[i] = kthread_run(rcu_torture_fakewriter, NULL,		                                  "rcu_torture_fakewriter");		if (IS_ERR(fakewriter_tasks[i])) {			firsterr = PTR_ERR(fakewriter_tasks[i]);			VERBOSE_PRINTK_ERRSTRING("Failed to create fakewriter");			fakewriter_tasks[i] = NULL;			goto unwind;		}	}	reader_tasks = kzalloc(nrealreaders * sizeof(reader_tasks[0]),			       GFP_KERNEL);	if (reader_tasks == NULL) {		VERBOSE_PRINTK_ERRSTRING("out of memory");		firsterr = -ENOMEM;		goto unwind;	}	for (i = 0; i < nrealreaders; i++) {		VERBOSE_PRINTK_STRING("Creating rcu_torture_reader task");		reader_tasks[i] = kthread_run(rcu_torture_reader, NULL,					      "rcu_torture_reader");		if (IS_ERR(reader_tasks[i])) {			firsterr = PTR_ERR(reader_tasks[i]);			VERBOSE_PRINTK_ERRSTRING("Failed to create reader");			reader_tasks[i] = NULL;			goto unwind;		}	}	if (stat_interval > 0) {		VERBOSE_PRINTK_STRING("Creating rcu_torture_stats task");		stats_task = kthread_run(rcu_torture_stats, NULL,					"rcu_torture_stats");		if (IS_ERR(stats_task)) {			firsterr = PTR_ERR(stats_task);			VERBOSE_PRINTK_ERRSTRING("Failed to create stats");			stats_task = NULL;			goto unwind;		}	}	if (test_no_idle_hz) {		rcu_idle_cpu = num_online_cpus() - 1;		/* Create the shuffler thread */		shuffler_task = kthread_run(rcu_torture_shuffle, NULL,					  "rcu_torture_shuffle");		if (IS_ERR(shuffler_task)) {			firsterr = PTR_ERR(shuffler_task);			VERBOSE_PRINTK_ERRSTRING("Failed to create shuffler");			shuffler_task = NULL;			goto unwind;		}	}	if (stutter < 0)		stutter = 0;	if (stutter) {		/* Create the stutter thread */		stutter_task = kthread_run(rcu_torture_stutter, NULL,					  "rcu_torture_stutter");		if (IS_ERR(stutter_task)) {			firsterr = PTR_ERR(stutter_task);			VERBOSE_PRINTK_ERRSTRING("Failed to create stutter");			stutter_task = NULL;			goto unwind;		}	}	return 0;unwind:	rcu_torture_cleanup();	return firsterr;}module_init(rcu_torture_init);module_exit(rcu_torture_cleanup);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -