Received: by 2002:a05:6a10:f3d0:0:0:0:0 with SMTP id a16csp3686797pxv; Tue, 13 Jul 2021 01:05:46 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyhE13itmzI6fxsaWkR7x9lkOEyvr0djgc/+dw84uYGBdcKGjs+Rbp5s/8lIJA4FLddwXZD X-Received: by 2002:a17:907:1de7:: with SMTP id og39mr4142450ejc.221.1626163545849; Tue, 13 Jul 2021 01:05:45 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1626163545; cv=none; d=google.com; s=arc-20160816; b=mTlqmwOljpWR8O8zL4T1vl8tFp6zyCklom2sXwkI4pENhApDVqBzISu5XyDzyhRXMg Xt8aRp7K+JutxPus+XNl4nUWXsVZ3V2Cd1+YkvZq/MW+nBYctal43ZbHBOk69GQPYCSz Fhb8QepyUtFEX9uvH2rt7hr6/sdgktba/wzMBhJszmYs675wduoxLqoyJcaLaARig1ZE CgMR5kaxZEXoFH60HbZ/KqFgKNmDHw7bEuaXnbsRFURwp1Z8SD/rhLWZbhpIGA19KwAV CQmLx8SxqHSQOBwbtW29ZuNof69WX4jzxrneVVJB9Y0gQXdzQn7DwokDCiuyvCo9Z+bQ rsAw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-language:content-transfer-encoding :in-reply-to:mime-version:user-agent:date:message-id:from:references :cc:to:subject; bh=jKEJCTfrOTdLmjd7D4GzcjxYHA902ijC9d3ziwXBqZQ=; b=bnz2SIEXbFz/PKvpCb94SEIJhS6Q1pmK6QVcQ2EcEvk+/Axktei7lH32Rw/PwIt6uo peVHYg7OXj5tUDFANEubLKxtFI8FP+ULtzgVawz5HSz7tCv8paDsHS4BNwkzgbqsxeYa JeyvPXyfxfwLnItIk63/EECUKOIPNDXbT2vKrIh96+ae+Myw7YBJVmMDjO3LH1LdaLHp 8RI4m4CnuPCVT2j4QZqYLf0dAc5hj57Nn/MltIZFB5Adube1mKn4ULjCin4J5NGLuIkZ mEJOTKhaPKCtWHLSkvIbo55y3H/rqNv0aFNHoL9flhZY6qkJSywgvoys0NKcTotvQt2z URmA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=huawei.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id j26si7486511edt.98.2021.07.13.01.05.22; Tue, 13 Jul 2021 01:05:45 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=huawei.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S234396AbhGMIFP (ORCPT + 99 others); Tue, 13 Jul 2021 04:05:15 -0400 Received: from szxga01-in.huawei.com ([45.249.212.187]:15007 "EHLO szxga01-in.huawei.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S234157AbhGMIFO (ORCPT ); Tue, 13 Jul 2021 04:05:14 -0400 Received: from dggemv703-chm.china.huawei.com (unknown [172.30.72.54]) by szxga01-in.huawei.com (SkyGuard) with ESMTP id 4GPChB21bYzbc4Z; Tue, 13 Jul 2021 15:59:06 +0800 (CST) Received: from dggpeml500017.china.huawei.com (7.185.36.243) by dggemv703-chm.china.huawei.com (10.3.19.46) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.2176.2; Tue, 13 Jul 2021 16:02:21 +0800 Received: from [10.174.178.174] (10.174.178.174) by dggpeml500017.china.huawei.com (7.185.36.243) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.2176.2; Tue, 13 Jul 2021 16:02:20 +0800 Subject: Re: [PATCH v2] workqueue: fix UAF in pwq_unbound_release_workfn() To: Lai Jiangshan , Tejun Heo CC: LKML , Xu Qiang , Pavel Skripkin References: <20210709071100.4057639-1-yangyingliang@huawei.com> From: Yang Yingliang Message-ID: Date: Tue, 13 Jul 2021 16:02:20 +0800 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Thunderbird/68.7.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset="utf-8"; format=flowed Content-Transfer-Encoding: 7bit Content-Language: en-US X-Originating-IP: [10.174.178.174] X-ClientProxiedBy: dggems701-chm.china.huawei.com (10.3.19.178) To dggpeml500017.china.huawei.com (7.185.36.243) X-CFilter-Loop: Reflected Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi, On 2021/7/13 13:56, Lai Jiangshan wrote: > On Tue, Jul 13, 2021 at 1:12 AM Tejun Heo wrote: >> Hello, Yang. >> >>> +static void free_pwq(struct pool_workqueue *pwq) >>> +{ >>> + if (!pwq || --pwq->refcnt) >>> + return; >>> + >>> + put_unbound_pool(pwq->pool); >>> + kmem_cache_free(pwq_cache, pwq); >>> +} >>> + >>> +static void free_wqattrs_ctx(struct apply_wqattrs_ctx *ctx) >>> +{ >>> + int node; >>> + >>> + if (!ctx) >>> + return; >>> + >>> + for_each_node(node) >>> + free_pwq(ctx->pwq_tbl[node]); >>> + free_pwq(ctx->dfl_pwq); >>> + >>> + free_workqueue_attrs(ctx->attrs); >>> + >>> + kfree(ctx); >>> +} >> It bothers me that we're partially replicating the free path including pwq >> refcnting. > The replicating code can be reduced by merging > apply_wqattrs_cleanup() into apply_wqattrs_commit(). > >> Does something like the following work? > It works since it has a flush_scheduled_work() in > alloc_and_link_pwqs(). But I don't think it works for > workqueue_apply_unbound_cpumask() when apply_wqattrs_commit() > is not called. > > If we want to reuse the current apply_wqattrs_cleanup(), I would prefer > something like this: (untested) > > @@ -3680,15 +3676,21 @@ static void pwq_unbound_release_workfn(struct > work_struct *work) > unbound_release_work); > struct workqueue_struct *wq = pwq->wq; > struct worker_pool *pool = pwq->pool; > - bool is_last; > + bool is_last = false; > > - if (WARN_ON_ONCE(!(wq->flags & WQ_UNBOUND))) > - return; > + /* > + * when @pwq is not linked, it doesn't hold any reference to the > + * @wq, and @wq is invalid to access. > + */ > + if (!list_empty(&pwq->pwqs_node)) { > + if (WARN_ON_ONCE(!(wq->flags & WQ_UNBOUND))) > + return; > > - mutex_lock(&wq->mutex); > - list_del_rcu(&pwq->pwqs_node); > - is_last = list_empty(&wq->pwqs); > - mutex_unlock(&wq->mutex); > + mutex_lock(&wq->mutex); > + list_del_rcu(&pwq->pwqs_node); > + is_last = list_empty(&wq->pwqs); > + mutex_unlock(&wq->mutex); > + } > > mutex_lock(&wq_pool_mutex); > put_unbound_pool(pool); I test the code with my testcase, it works. I can send a v3 about this. Thanks, Yang > >> diff --git a/kernel/workqueue.c b/kernel/workqueue.c >> index 104e3ef04e33..0c0ab363edeb 100644 >> --- a/kernel/workqueue.c >> +++ b/kernel/workqueue.c >> @@ -3693,7 +3693,7 @@ static void pwq_unbound_release_workfn(struct work_struct *work) >> * If we're the last pwq going away, @wq is already dead and no one >> * is gonna access it anymore. Schedule RCU free. >> */ >> - if (is_last) { >> + if (is_last && !list_empty(&wq->list)) { >> wq_unregister_lockdep(wq); >> call_rcu(&wq->rcu, rcu_free_wq); >> } >> @@ -4199,6 +4199,10 @@ static int alloc_and_link_pwqs(struct workqueue_struct *wq) >> } >> put_online_cpus(); >> >> + if (ret) { >> + flush_scheduled_work(); >> + } >> + >> return ret; >> } >> >> -- >> tejun > .