cpuset: fix regression when failed to generate sched domains

Impact: properly rebuild sched-domains on kmalloc() failure

When cpuset failed to generate sched domains due to kmalloc()
failure, the scheduler should fallback to the single partition
'fallback_doms' and rebuild sched domains, but now it only
destroys but not rebuilds sched domains.

The regression was introduced by:

| commit dfb512ec4834116124da61d6c1ee10fd0aa32bd6
| Author: Max Krasnyansky <maxk@qualcomm.com>
| Date: Fri Aug 29 13:11:41 2008 -0700
|
| sched: arch_reinit_sched_domains() must destroy domains to force rebuild

After the above commit, partition_sched_domains(0, NULL, NULL) will
only destroy sched domains and partition_sched_domains(1, NULL, NULL)
will create the default sched domain.

Signed-off-by: Li Zefan <lizf@cn.fujitsu.com>
Cc: Max Krasnyansky <maxk@qualcomm.com>
Cc: <stable@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>

authored by Li Zefan and committed by Ingo Molnar 700018e0 ad133ba3

+15 -10
+8 -4
kernel/cpuset.c
··· 587 587 int ndoms; /* number of sched domains in result */ 588 588 int nslot; /* next empty doms[] cpumask_t slot */ 589 589 590 - ndoms = 0; 591 590 doms = NULL; 592 591 dattr = NULL; 593 592 csa = NULL; ··· 673 674 * Convert <csn, csa> to <ndoms, doms> and populate cpu masks. 674 675 */ 675 676 doms = kmalloc(ndoms * sizeof(cpumask_t), GFP_KERNEL); 676 - if (!doms) { 677 - ndoms = 0; 677 + if (!doms) 678 678 goto done; 679 - } 680 679 681 680 /* 682 681 * The rest of the code, including the scheduler, can deal with ··· 728 731 729 732 done: 730 733 kfree(csa); 734 + 735 + /* 736 + * Fallback to the default domain if kmalloc() failed. 737 + * See comments in partition_sched_domains(). 738 + */ 739 + if (doms == NULL) 740 + ndoms = 1; 731 741 732 742 *domains = doms; 733 743 *attributes = dattr;
+7 -6
kernel/sched.c
··· 7789 7789 * 7790 7790 * The passed in 'doms_new' should be kmalloc'd. This routine takes 7791 7791 * ownership of it and will kfree it when done with it. If the caller 7792 - * failed the kmalloc call, then it can pass in doms_new == NULL, 7793 - * and partition_sched_domains() will fallback to the single partition 7794 - * 'fallback_doms', it also forces the domains to be rebuilt. 7792 + * failed the kmalloc call, then it can pass in doms_new == NULL && 7793 + * ndoms_new == 1, and partition_sched_domains() will fallback to 7794 + * the single partition 'fallback_doms', it also forces the domains 7795 + * to be rebuilt. 7795 7796 * 7796 - * If doms_new==NULL it will be replaced with cpu_online_map. 7797 - * ndoms_new==0 is a special case for destroying existing domains. 7798 - * It will not create the default domain. 7797 + * If doms_new == NULL it will be replaced with cpu_online_map. 7798 + * ndoms_new == 0 is a special case for destroying existing domains, 7799 + * and it will not create the default domain. 7799 7800 * 7800 7801 * Call with hotplug lock held 7801 7802 */