Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp1595039ybi; Wed, 17 Jul 2019 18:23:08 -0700 (PDT) X-Google-Smtp-Source: APXvYqyvBqnmjin4colXJW+CJx6a4gVsTy9X9BkF4AsfQ5k+FHXqf07B3ERtjyZEF2+BbKbW7ufM X-Received: by 2002:a17:902:744c:: with SMTP id e12mr46690410plt.287.1563412988390; Wed, 17 Jul 2019 18:23:08 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1563412988; cv=none; d=google.com; s=arc-20160816; b=NISvSLbn5w2uuZc6/x99F6CkYUcRqGRz6m/w+iZdOEfXneZigKE6fnwdbkD2cpRTa4 nSWJM+WHcfiGTZGSwrFATGJuLcFlkYZoGUOQ+sYNicmlnyKCTDjYzoqNpS+QMMGGuH33 sT+h3H3ylITAD3rP80+/lJD+nvHQZVqnp1EsVo6k7FLubU9PzD1ageGAEtfjt4v2Fv30 cRB9n/5VZy69Uh1lIm63U1nNS975fLfqCsO/V6PpTmzEMesGWZQ2NGjRZ9grVaKo4g6j sthV5wdY4oaziCNnCBIpCe2fqWh/HPSy/nLmPxNEireXyUhWyOUuGPtEfEJ9EQb1gN39 RLCQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:cc:to:from :subject; bh=JmNJWK3amMQOLdJR/UK4VCiLf5BlQpwJcfVRQmI9YOc=; b=tGvnnb7+bGFcdlvLSBKhoZXeS6+qioptGduyxeS2g6ywoq94bNjL6+RCKyB2VETkNb y6a4WcWqO6td976uCGHvTMHvZ1KORMxB+ev4ec31S504EXrxOXoUxVWfF4ganxEfGfGr SGNB5bdahLzSms7lEJjw4cF9jEnG9rPR9R+AhnV6CtZLhgqYrCX/2RDfXmWC84hpsqOR RVHLfaP9JeZD0T/NcKwty3+Bf0Ow4XPKafAcpVS8igG7VT1IvrUpnKZlyg184uQbsp1a wsBUSgjFHb3QZzchz6VI9rE5d+/mjYRH0BapalRjCJdvXIedImuGVfKQ4KZYbf7Uskjs J4Nw== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id p125si17101110pfp.35.2019.07.17.18.22.52; Wed, 17 Jul 2019 18:23:08 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1732440AbfGRBWR (ORCPT + 99 others); Wed, 17 Jul 2019 21:22:17 -0400 Received: from mga11.intel.com ([192.55.52.93]:43704 "EHLO mga11.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727658AbfGRBWQ (ORCPT ); Wed, 17 Jul 2019 21:22:16 -0400 X-Amp-Result: SKIPPED(no attachment in message) X-Amp-File-Uploaded: False Received: from orsmga001.jf.intel.com ([10.7.209.18]) by fmsmga102.fm.intel.com with ESMTP/TLS/DHE-RSA-AES256-GCM-SHA384; 17 Jul 2019 18:22:15 -0700 X-IronPort-AV: E=Sophos;i="5.64,276,1559545200"; d="scan'208";a="251660620" Received: from dwillia2-desk3.jf.intel.com (HELO dwillia2-desk3.amr.corp.intel.com) ([10.54.39.16]) by orsmga001-auth.jf.intel.com with ESMTP/TLS/DHE-RSA-AES256-GCM-SHA384; 17 Jul 2019 18:22:15 -0700 Subject: [PATCH v2 2/7] libnvdimm/bus: Prevent duplicate device_unregister() calls From: Dan Williams To: linux-nvdimm@lists.01.org Cc: Jane Chu , Erwin Tsaur , stable@vger.kernel.org, Jane Chu , peterz@infradead.org, vishal.l.verma@intel.com, linux-kernel@vger.kernel.org Date: Wed, 17 Jul 2019 18:07:58 -0700 Message-ID: <156341207846.292348.10435719262819764054.stgit@dwillia2-desk3.amr.corp.intel.com> In-Reply-To: <156341206785.292348.1660822720191643298.stgit@dwillia2-desk3.amr.corp.intel.com> References: <156341206785.292348.1660822720191643298.stgit@dwillia2-desk3.amr.corp.intel.com> User-Agent: StGit/0.18-2-gc94f MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org A multithreaded namespace creation/destruction stress test currently fails with signatures like the following: sysfs group 'power' not found for kobject 'dax1.1' RIP: 0010:sysfs_remove_group+0x76/0x80 Call Trace: device_del+0x73/0x370 device_unregister+0x16/0x50 nd_async_device_unregister+0x1e/0x30 [libnvdimm] async_run_entry_fn+0x39/0x160 process_one_work+0x23c/0x5e0 worker_thread+0x3c/0x390 BUG: kernel NULL pointer dereference, address: 0000000000000020 RIP: 0010:klist_put+0x1b/0x6c Call Trace: klist_del+0xe/0x10 device_del+0x8a/0x2c9 ? __switch_to_asm+0x34/0x70 ? __switch_to_asm+0x40/0x70 device_unregister+0x44/0x4f nd_async_device_unregister+0x22/0x2d [libnvdimm] async_run_entry_fn+0x47/0x15a process_one_work+0x1a2/0x2eb worker_thread+0x1b8/0x26e Use the kill_device() helper to atomically resolve the race of multiple threads issuing kill, device_unregister(), requests. Reported-by: Jane Chu Reported-by: Erwin Tsaur Fixes: 4d88a97aa9e8 ("libnvdimm, nvdimm: dimm driver and base libnvdimm device-driver...") Cc: Link: https://github.com/pmem/ndctl/issues/96 Tested-by: Tested-by: Jane Chu Signed-off-by: Dan Williams --- drivers/nvdimm/bus.c | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/drivers/nvdimm/bus.c b/drivers/nvdimm/bus.c index 2dca3034fee0..42713b210f51 100644 --- a/drivers/nvdimm/bus.c +++ b/drivers/nvdimm/bus.c @@ -547,13 +547,38 @@ EXPORT_SYMBOL(nd_device_register); void nd_device_unregister(struct device *dev, enum nd_async_mode mode) { + bool killed; + switch (mode) { case ND_ASYNC: + /* + * In the async case this is being triggered with the + * device lock held and the unregistration work needs to + * be moved out of line iff this is thread has won the + * race to schedule the deletion. + */ + if (!kill_device(dev)) + return; + get_device(dev); async_schedule_domain(nd_async_device_unregister, dev, &nd_async_domain); break; case ND_SYNC: + /* + * In the sync case the device is being unregistered due + * to a state change of the parent. Claim the kill state + * to synchronize against other unregistration requests, + * or otherwise let the async path handle it if the + * unregistration was already queued. + */ + device_lock(dev); + killed = kill_device(dev); + device_unlock(dev); + + if (!killed) + return; + nd_synchronize(); device_unregister(dev); break;