Received: by 2002:a05:6358:9144:b0:117:f937:c515 with SMTP id r4csp4278728rwr; Sun, 23 Apr 2023 02:09:51 -0700 (PDT) X-Google-Smtp-Source: AKy350byuO1JjjRK12upe/5MDcxnK7ZYUBTx/x6etFojxz8Xr6C6PFuOtCBcYVFM57V7AQlZXci0 X-Received: by 2002:a05:6a00:1496:b0:63a:fb57:63c5 with SMTP id v22-20020a056a00149600b0063afb5763c5mr11350340pfu.3.1682240990821; Sun, 23 Apr 2023 02:09:50 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1682240990; cv=none; d=google.com; s=arc-20160816; b=tnSTA4t3QaGjE3rR7oldp93FQDElxvbcpiwGVpolaAXKX2/t19rtzD3Uol0sQRMbZW 8JP01itksAyzSGohiTgrL6sJjTJYTGy/mxRtu0JMBvLAJGtuT4Tlx/NuXtUXFPYVp6Ns 3v4YyGIyFmlxmUMj4UIEkP5URxCHLsk7nuyBScxoRFFGBTVPycviejwAZ9X/hYGiiGAG 33HoyRN6XeZxURfPUGDz2PC0Pr+x4HSlHDK34xGbexQBTv6V9e4vylI21vQgp7U3OMrV cskjT+es0tBxzco3QcQY+bNCgZJiSzA/urWhwqwg9SV/39GZhBFK8S7gHT1UR2Nym1FI WZfQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:mime-version:message-id:date:user-agent :references:in-reply-to:subject:cc:to:from:dkim-signature; bh=j3+0zfn8cNlVMJMbmm5Vq5FMuYDoSo3umxlEbpqHEH4=; b=SMc+8mgZSLgnEADD4esUxyVnCOfrgp1pNVVHcrs0iE3NZ7iqBcJT/6cHgs/G8TivbT X81KOsN2zBK9K5S3eh/NrsbxqJrm6F07gyLzp/yCAnrB9NslndiYNpkhzlDDuvSEaNSQ 2OBgLx/9UNsr4WRBSUYRYFjyk9vO/+BoZZ3DpQHpaqPFnm7T0CK+rKTN4+qQ7d9l9QPN 6Y+rmBrpRHRVmpEEc+GSUQFciV1txmfODWIl+77OFmakZA1CTHqxSZmEnXRmLAZRH3EV 5SGE0n6tdF1FcAt1aLa9twWN2q8FZtp24F2G7NJCGOmfJxpkmIls8dcU7Z5pdkfPxZIj Y0AQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=YxfZlPmJ; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id v10-20020aa799ca000000b0063b8e23415esi8634795pfi.92.2023.04.23.02.09.34; Sun, 23 Apr 2023 02:09:50 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=YxfZlPmJ; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229808AbjDWItL (ORCPT + 99 others); Sun, 23 Apr 2023 04:49:11 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:48468 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229441AbjDWItK (ORCPT ); Sun, 23 Apr 2023 04:49:10 -0400 Received: from mga17.intel.com (mga17.intel.com [192.55.52.151]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 314DB1726; Sun, 23 Apr 2023 01:49:09 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=intel.com; i=@intel.com; q=dns/txt; s=Intel; t=1682239749; x=1713775749; h=from:to:cc:subject:in-reply-to:references:date: message-id:mime-version; bh=SVZBxnuP4vBlqDAzvKI6hIM4GkXNlEyFfSy/CkYXJ7w=; b=YxfZlPmJFokwDKZ3WvY6N266OkxHZ9fJF0/7E20kxDNVKOEjgy+htF7+ UQA0lOVIOkS81WpSpYw++tvfrV3TwKOHETx7aoAGE1aZAE5Um3czKkrfX kplFXiMkEdKh/9AQe2UpGyx94Lj0Yv0Jd4jWxbbjo9MAdupBY3h5SciG1 t9cKY2u/y9ccALPev3tdZbptFOYjHuBx2mNLekAIejXMp5rNLL+XeLvU6 INBrbFeQMVDv3EfwLDCSidCRD0iEUuJBmNklO7uvyPOnhaS3wJ+QybT8c KhEByZ/EeY8n9wnFP4Tm+7e3A0mMCbPQQQCkyVhO18RCK6+JrTSt68gv6 g==; X-IronPort-AV: E=McAfee;i="6600,9927,10688"; a="326585497" X-IronPort-AV: E=Sophos;i="5.99,220,1677571200"; d="scan'208";a="326585497" Received: from orsmga007.jf.intel.com ([10.7.209.58]) by fmsmga107.fm.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 23 Apr 2023 01:49:08 -0700 X-ExtLoop1: 1 X-IronPort-AV: E=McAfee;i="6600,9927,10688"; a="686359883" X-IronPort-AV: E=Sophos;i="5.99,220,1677571200"; d="scan'208";a="686359883" Received: from yhuang6-desk2.sh.intel.com (HELO yhuang6-desk2.ccr.corp.intel.com) ([10.238.208.55]) by orsmga007-auth.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 23 Apr 2023 01:49:02 -0700 From: "Huang, Ying" To: Douglas Anderson Cc: Andrew Morton , Mel Gorman , Vlastimil Babka , Alexander Viro , Christian Brauner , linux-kernel@vger.kernel.org, linux-mm@kvack.org, Yu Zhao , linux-fsdevel@vger.kernel.org, Matthew Wilcox , Bart Van Assche , Ben Segall , Daniel Bristot de Oliveira , Dietmar Eggemann , Ingo Molnar , Jan Kara , Juri Lelli , Mel Gorman , Mikulas Patocka , Peter Zijlstra , "Ritesh Harjani (IBM)" , Steven Rostedt , Valentin Schneider , Vincent Guittot , Will Deacon , Zhang Yi Subject: Re: [PATCH v2 2/4] buffer: Add lock_buffer_timeout() In-Reply-To: <20230421151135.v2.2.Ie146eec4d41480ebeb15f0cfdfb3bc9095e4ebd9@changeid> (Douglas Anderson's message of "Fri, 21 Apr 2023 15:12:46 -0700") References: <20230421221249.1616168-1-dianders@chromium.org> <20230421151135.v2.2.Ie146eec4d41480ebeb15f0cfdfb3bc9095e4ebd9@changeid> User-Agent: Gnus/5.13 (Gnus v5.13) Emacs/27.1 (gnu/linux) Date: Sun, 23 Apr 2023 16:47:58 +0800 Message-ID: <87bkjfkmrl.fsf@yhuang6-desk2.ccr.corp.intel.com> MIME-Version: 1.0 Content-Type: text/plain; charset=ascii X-Spam-Status: No, score=-4.6 required=5.0 tests=BAYES_00,DKIMWL_WL_HIGH, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,RCVD_IN_DNSWL_MED, SPF_HELO_NONE,SPF_NONE,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Douglas Anderson writes: > Add a variant of lock_buffer() that can timeout. This is useful to > avoid unbounded waits for the page lock in kcompactd. > > Signed-off-by: Douglas Anderson > --- > > Changes in v2: > - "Add lock_buffer_timeout()" new for v2. > > fs/buffer.c | 7 +++++++ > include/linux/buffer_head.h | 10 ++++++++++ > include/linux/wait_bit.h | 24 ++++++++++++++++++++++++ > kernel/sched/wait_bit.c | 14 ++++++++++++++ > 4 files changed, 55 insertions(+) > > diff --git a/fs/buffer.c b/fs/buffer.c > index 9e1e2add541e..fcd19c270024 100644 > --- a/fs/buffer.c > +++ b/fs/buffer.c > @@ -71,6 +71,13 @@ void __lock_buffer(struct buffer_head *bh) > } > EXPORT_SYMBOL(__lock_buffer); > > +int __lock_buffer_timeout(struct buffer_head *bh, unsigned long timeout) > +{ > + return wait_on_bit_lock_io_timeout(&bh->b_state, BH_Lock, > + TASK_UNINTERRUPTIBLE, timeout); > +} > +EXPORT_SYMBOL(__lock_buffer_timeout); > + > void unlock_buffer(struct buffer_head *bh) > { > clear_bit_unlock(BH_Lock, &bh->b_state); > diff --git a/include/linux/buffer_head.h b/include/linux/buffer_head.h > index 8f14dca5fed7..2bae464f89d5 100644 > --- a/include/linux/buffer_head.h > +++ b/include/linux/buffer_head.h > @@ -237,6 +237,7 @@ struct buffer_head *alloc_buffer_head(gfp_t gfp_flags); > void free_buffer_head(struct buffer_head * bh); > void unlock_buffer(struct buffer_head *bh); > void __lock_buffer(struct buffer_head *bh); > +int __lock_buffer_timeout(struct buffer_head *bh, unsigned long timeout); > int sync_dirty_buffer(struct buffer_head *bh); > int __sync_dirty_buffer(struct buffer_head *bh, blk_opf_t op_flags); > void write_dirty_buffer(struct buffer_head *bh, blk_opf_t op_flags); > @@ -400,6 +401,15 @@ static inline void lock_buffer(struct buffer_head *bh) > __lock_buffer(bh); > } > > +static inline int lock_buffer_timeout(struct buffer_head *bh, > + unsigned long timeout) > +{ > + might_sleep(); > + if (!trylock_buffer(bh)) > + return __lock_buffer_timeout(bh, timeout); > + return 0; > +} > + Add document about return value of lock_buffer_timeout()? Otherwise looks good to me. Best Regards, Huang, Ying > static inline struct buffer_head *getblk_unmovable(struct block_device *bdev, > sector_t block, > unsigned size) > diff --git a/include/linux/wait_bit.h b/include/linux/wait_bit.h > index 7725b7579b78..33f0f60b1c8c 100644 > --- a/include/linux/wait_bit.h > +++ b/include/linux/wait_bit.h > @@ -30,6 +30,7 @@ void wake_up_bit(void *word, int bit); > int out_of_line_wait_on_bit(void *word, int, wait_bit_action_f *action, unsigned int mode); > int out_of_line_wait_on_bit_timeout(void *word, int, wait_bit_action_f *action, unsigned int mode, unsigned long timeout); > int out_of_line_wait_on_bit_lock(void *word, int, wait_bit_action_f *action, unsigned int mode); > +int out_of_line_wait_on_bit_lock_timeout(void *word, int, wait_bit_action_f *action, unsigned int mode, unsigned long timeout); > struct wait_queue_head *bit_waitqueue(void *word, int bit); > extern void __init wait_bit_init(void); > > @@ -208,6 +209,29 @@ wait_on_bit_lock_io(unsigned long *word, int bit, unsigned mode) > return out_of_line_wait_on_bit_lock(word, bit, bit_wait_io, mode); > } > > +/** > + * wait_on_bit_lock_io_timeout - wait_on_bit_lock_io() with a timeout > + * @word: the word being waited on, a kernel virtual address > + * @bit: the bit of the word being waited on > + * @mode: the task state to sleep in > + * @timeout: the timeout in jiffies; %MAX_SCHEDULE_TIMEOUT means wait forever > + * > + * Returns zero if the bit was (eventually) found to be clear and was > + * set. Returns non-zero if a timeout happened or a signal was delivered to > + * the process and the @mode allows that signal to wake the process. > + */ > +static inline int > +wait_on_bit_lock_io_timeout(unsigned long *word, int bit, unsigned mode, > + unsigned long timeout) > +{ > + might_sleep(); > + if (!test_and_set_bit(bit, word)) > + return 0; > + return out_of_line_wait_on_bit_lock_timeout(word, bit, > + bit_wait_io_timeout, > + mode, timeout); > +} > + > /** > * wait_on_bit_lock_action - wait for a bit to be cleared, when wanting to set it > * @word: the word being waited on, a kernel virtual address > diff --git a/kernel/sched/wait_bit.c b/kernel/sched/wait_bit.c > index 0b1cd985dc27..629acd1c6c79 100644 > --- a/kernel/sched/wait_bit.c > +++ b/kernel/sched/wait_bit.c > @@ -118,6 +118,20 @@ int __sched out_of_line_wait_on_bit_lock(void *word, int bit, > } > EXPORT_SYMBOL(out_of_line_wait_on_bit_lock); > > +int __sched out_of_line_wait_on_bit_lock_timeout(void *word, int bit, > + wait_bit_action_f *action, > + unsigned mode, > + unsigned long timeout) > +{ > + struct wait_queue_head *wq_head = bit_waitqueue(word, bit); > + DEFINE_WAIT_BIT(wq_entry, word, bit); > + > + wq_entry.key.timeout = jiffies + timeout; > + > + return __wait_on_bit_lock(wq_head, &wq_entry, action, mode); > +} > +EXPORT_SYMBOL(out_of_line_wait_on_bit_lock_timeout); > + > void __wake_up_bit(struct wait_queue_head *wq_head, void *word, int bit) > { > struct wait_bit_key key = __WAIT_BIT_KEY_INITIALIZER(word, bit);