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

📄 rcutorture.c

📁 linux 2.6.19 kernel source code before patching
💻 C
📖 第 1 页 / 共 2 页
字号:
	.name = "sched"};/* * RCU torture writer kthread.  Repeatedly substitutes a new structure * for that pointed to by rcu_torture_current, freeing the old structure * after a series of grace periods (the "pipeline"). */static intrcu_torture_writer(void *arg){	int i;	long oldbatch = rcu_batches_completed();	struct rcu_torture *rp;	struct rcu_torture *old_rp;	static DEFINE_RCU_RANDOM(rand);	VERBOSE_PRINTK_STRING("rcu_torture_writer task started");	set_user_nice(current, 19);	current->flags |= PF_NOFREEZE;	do {		schedule_timeout_uninterruptible(1);		if ((rp = rcu_torture_alloc()) == NULL)			continue;		rp->rtort_pipe_count = 0;		udelay(rcu_random(&rand) & 0x3ff);		old_rp = rcu_torture_current;		rp->rtort_mbtest = 1;		rcu_assign_pointer(rcu_torture_current, rp);		smp_wmb();		if (old_rp) {			i = old_rp->rtort_pipe_count;			if (i > RCU_TORTURE_PIPE_LEN)				i = RCU_TORTURE_PIPE_LEN;			atomic_inc(&rcu_torture_wcount[i]);			old_rp->rtort_pipe_count++;			cur_ops->deferredfree(old_rp);		}		rcu_torture_current_version++;		oldbatch = cur_ops->completed();	} while (!kthread_should_stop() && !fullstop);	VERBOSE_PRINTK_STRING("rcu_torture_writer task stopping");	while (!kthread_should_stop())		schedule_timeout_uninterruptible(1);	return 0;}/* * RCU torture fake writer kthread.  Repeatedly calls sync, with a random * delay between calls. */static intrcu_torture_fakewriter(void *arg){	DEFINE_RCU_RANDOM(rand);	VERBOSE_PRINTK_STRING("rcu_torture_fakewriter task started");	set_user_nice(current, 19);	current->flags |= PF_NOFREEZE;	do {		schedule_timeout_uninterruptible(1 + rcu_random(&rand)%10);		udelay(rcu_random(&rand) & 0x3ff);		cur_ops->sync();	} while (!kthread_should_stop() && !fullstop);	VERBOSE_PRINTK_STRING("rcu_torture_fakewriter task stopping");	while (!kthread_should_stop())		schedule_timeout_uninterruptible(1);	return 0;}/* * RCU torture reader kthread.  Repeatedly dereferences rcu_torture_current, * incrementing the corresponding element of the pipeline array.  The * counter in the element should never be greater than 1, otherwise, the * RCU implementation is broken. */static intrcu_torture_reader(void *arg){	int completed;	int idx;	DEFINE_RCU_RANDOM(rand);	struct rcu_torture *p;	int pipe_count;	VERBOSE_PRINTK_STRING("rcu_torture_reader task started");	set_user_nice(current, 19);	current->flags |= PF_NOFREEZE;	do {		idx = cur_ops->readlock();		completed = cur_ops->completed();		p = rcu_dereference(rcu_torture_current);		if (p == NULL) {			/* Wait for rcu_torture_writer to get underway */			cur_ops->readunlock(idx);			schedule_timeout_interruptible(HZ);			continue;		}		if (p->rtort_mbtest == 0)			atomic_inc(&n_rcu_torture_mberror);		cur_ops->readdelay(&rand);		preempt_disable();		pipe_count = p->rtort_pipe_count;		if (pipe_count > RCU_TORTURE_PIPE_LEN) {			/* Should not happen, but... */			pipe_count = RCU_TORTURE_PIPE_LEN;		}		++__get_cpu_var(rcu_torture_count)[pipe_count];		completed = cur_ops->completed() - completed;		if (completed > RCU_TORTURE_PIPE_LEN) {			/* Should not happen, but... */			completed = RCU_TORTURE_PIPE_LEN;		}		++__get_cpu_var(rcu_torture_batch)[completed];		preempt_enable();		cur_ops->readunlock(idx);		schedule();	} while (!kthread_should_stop() && !fullstop);	VERBOSE_PRINTK_STRING("rcu_torture_reader task stopping");	while (!kthread_should_stop())		schedule_timeout_uninterruptible(1);	return 0;}/* * Create an RCU-torture statistics message in the specified buffer. */static intrcu_torture_printk(char *page){	int cnt = 0;	int cpu;	int i;	long pipesummary[RCU_TORTURE_PIPE_LEN + 1] = { 0 };	long batchsummary[RCU_TORTURE_PIPE_LEN + 1] = { 0 };	for_each_possible_cpu(cpu) {		for (i = 0; i < RCU_TORTURE_PIPE_LEN + 1; i++) {			pipesummary[i] += per_cpu(rcu_torture_count, cpu)[i];			batchsummary[i] += per_cpu(rcu_torture_batch, cpu)[i];		}	}	for (i = RCU_TORTURE_PIPE_LEN - 1; i >= 0; i--) {		if (pipesummary[i] != 0)			break;	}	cnt += sprintf(&page[cnt], "%s%s ", torture_type, TORTURE_FLAG);	cnt += sprintf(&page[cnt],		       "rtc: %p ver: %ld tfle: %d rta: %d rtaf: %d rtf: %d "		       "rtmbe: %d",		       rcu_torture_current,		       rcu_torture_current_version,		       list_empty(&rcu_torture_freelist),		       atomic_read(&n_rcu_torture_alloc),		       atomic_read(&n_rcu_torture_alloc_fail),		       atomic_read(&n_rcu_torture_free),		       atomic_read(&n_rcu_torture_mberror));	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);	}	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 = CPU_MASK_ALL;	int i;	lock_cpu_hotplug();	/* No point in shuffling if there is only one online CPU (ex: UP) */	if (num_online_cpus() == 1) {		unlock_cpu_hotplug();		return;	}	if (rcu_idle_cpu != -1)		cpu_clear(rcu_idle_cpu, tmp_mask);	set_cpus_allowed(current, tmp_mask);	if (reader_tasks) {		for (i = 0; i < nrealreaders; i++)			if (reader_tasks[i])				set_cpus_allowed(reader_tasks[i], tmp_mask);	}	if (fakewriter_tasks) {		for (i = 0; i < nfakewriters; i++)			if (fakewriter_tasks[i])				set_cpus_allowed(fakewriter_tasks[i], tmp_mask);	}	if (writer_task)		set_cpus_allowed(writer_task, tmp_mask);	if (stats_task)		set_cpus_allowed(stats_task, tmp_mask);	if (rcu_idle_cpu == -1)		rcu_idle_cpu = num_online_cpus() - 1;	else		rcu_idle_cpu--;	unlock_cpu_hotplug();}/* 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;}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\n",		torture_type, tag, nrealreaders, nfakewriters,		stat_interval, verbose, test_no_idle_hz, shuffle_interval);}static voidrcu_torture_cleanup(void){	int i;	fullstop = 1;	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.  */	rcu_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, };	/* 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;		}	}	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 + -