Received: by 2002:ab2:620c:0:b0:1ef:ffd0:ce49 with SMTP id o12csp1269341lqt; Tue, 19 Mar 2024 19:55:25 -0700 (PDT) X-Forwarded-Encrypted: i=3; AJvYcCWpdzPjL1tj7fQNdfg5Adwjgwh8/PiJi74f4s/ymGXrFa8qUWbNjJQjrqCjPzv69HLQI124OGO44GEg8rqg4xMy7Qj0+ewL9sPD2CXzpA== X-Google-Smtp-Source: AGHT+IGTzEzfg5TonK/6rb4Y6PiT7vnJ4iIkJQFQL3h54IDCNZmeIJc+iQu6W3pj2CxT4mqPMhjl X-Received: by 2002:a17:907:86aa:b0:a46:ea32:9a19 with SMTP id qa42-20020a17090786aa00b00a46ea329a19mr2379185ejc.23.1710903325491; Tue, 19 Mar 2024 19:55:25 -0700 (PDT) ARC-Seal: i=2; a=rsa-sha256; t=1710903325; cv=pass; d=google.com; s=arc-20160816; b=OZcBZCNLuiqJdX3USueIIRTRfRFhyFzo2v20HOo7rFs3KRllW2u3cMO5Q7ENVw3ddM gAPZKOkFpS3rbE0OJbxxBB7jR8ODKPwklbg08uWFLg1xOnsdlVXJaBAvmltNLwELLSMT hSTnb4t1SB3d2Ng74ZI0mQkmwjKqYMWpD3ErqjygoUebA5GJQAif4xyDrZkGp45IFOlu McCbgxRfY4N2wyGU3LUbOon68fUoe7lQclRbssZF1lUSnBAtPFOHps55Ia1aaEcOZwaf dB/uiGGdNTs5V/sOJLHPBw/6AhHwIBvgrVX+sf2vvxG0Gcs5Kv7FBFC90KtWAk/qWkhS mUhg== ARC-Message-Signature: i=2; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=in-reply-to:content-transfer-encoding:content-disposition :mime-version:list-unsubscribe:list-subscribe:list-id:precedence :references:message-id:subject:cc:to:from:date:dkim-signature; bh=CgE4Mpzc7USCNSk5n/jU7+IJGA1QF8JmfVfBPDD7vCQ=; fh=ToQl55n7GjOFeoWUasSxMY5pwddNywN3E4FU9017R7E=; b=nbsFKV0/7eMIk4eSLcXbFi2CCL4hkuZTACFkpdLdo4I7Sq7vSz7BuEKRqGW54ReG3l tdvZ0op+Vk7mRCisIkKVxl+vCsSzw+gHf1XJQ3q6syflLUBHNaHOo4DfCbo+AfNdpVQl ywDBUSglG/9jK6R2LLHzNI3HW43jxKQy/+thrdz5eH1LDhMj949Y/PuZ8VXnZ2v/iavR mmtVWmtnwpNKhIkFnSdazjNbxD53xIMKe71HWmqJ2N2ArpANjzzyTRcKwyqz+GTsPMq1 cqCbD9rIUOHJbBqbbFO5uyDeTSf8hCkdp/fbFGZ7M5DaKA2egV/Ooa8/Z8VBoebstd4/ C6Qg==; dara=google.com ARC-Authentication-Results: i=2; mx.google.com; dkim=pass header.i=@gmail.com header.s=20230601 header.b=XUzM6E3t; arc=pass (i=1 spf=pass spfdomain=gmail.com dkim=pass dkdomain=gmail.com dmarc=pass fromdomain=gmail.com); spf=pass (google.com: domain of linux-kernel+bounces-108399-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.80.249 as permitted sender) smtp.mailfrom="linux-kernel+bounces-108399-linux.lists.archive=gmail.com@vger.kernel.org"; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from am.mirrors.kernel.org (am.mirrors.kernel.org. [147.75.80.249]) by mx.google.com with ESMTPS id m26-20020a1709066d1a00b00a46eac821bbsi638423ejr.453.2024.03.19.19.55.25 for (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 19 Mar 2024 19:55:25 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel+bounces-108399-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.80.249 as permitted sender) client-ip=147.75.80.249; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20230601 header.b=XUzM6E3t; arc=pass (i=1 spf=pass spfdomain=gmail.com dkim=pass dkdomain=gmail.com dmarc=pass fromdomain=gmail.com); spf=pass (google.com: domain of linux-kernel+bounces-108399-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.80.249 as permitted sender) smtp.mailfrom="linux-kernel+bounces-108399-linux.lists.archive=gmail.com@vger.kernel.org"; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: from smtp.subspace.kernel.org (wormhole.subspace.kernel.org [52.25.139.140]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by am.mirrors.kernel.org (Postfix) with ESMTPS id 11AA71F22A2F for ; Wed, 20 Mar 2024 02:55:25 +0000 (UTC) Received: from localhost.localdomain (localhost.localdomain [127.0.0.1]) by smtp.subspace.kernel.org (Postfix) with ESMTP id E010811181; Wed, 20 Mar 2024 02:55:14 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=gmail.com header.i=@gmail.com header.b="XUzM6E3t" Received: from mail-pf1-f180.google.com (mail-pf1-f180.google.com [209.85.210.180]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 2532212B7D; Wed, 20 Mar 2024 02:55:10 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.210.180 ARC-Seal:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1710903313; cv=none; b=P+WKDD1IyG8KKdhLDqBh5PpMBI95ZLwM/a48e5K8MOqACoDE7RRIpaGWMfYp0g8pE3wgFuaEjEmVGzGG5FggNv0LIZmxrU2Qk2xeKLIhM7s1Nspvx6b9MDIehc7KyhMmvAz6D/J0SCZ/UxBEpGzCGAisJDimr4mys/or8+Eo+js= ARC-Message-Signature:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1710903313; c=relaxed/simple; bh=RzDIHyaQvzbDH+uf+7O1VtdnOx6OPQ/jy6nxAxaRtZY=; h=Date:From:To:Cc:Subject:Message-ID:References:MIME-Version: Content-Type:Content-Disposition:In-Reply-To; b=eXwcmSO5Zy59XDjuNKrZqZ7kPKoVINDyzfWMRGUKk+FZ8fWhUPAFvxscFv/0lsZBJusD+bDmSR7YCTasVDjtvvRMr0q4Br1+mSuwNFpZ+EJj18vVWRRS0+RmLHxTYh5mI+jrV9v9tz3AM63nb+6UARGs7c3tkg/g0aTNiDBKw8g= ARC-Authentication-Results:i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gmail.com; spf=pass smtp.mailfrom=gmail.com; dkim=pass (2048-bit key) header.d=gmail.com header.i=@gmail.com header.b=XUzM6E3t; arc=none smtp.client-ip=209.85.210.180 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=gmail.com Received: by mail-pf1-f180.google.com with SMTP id d2e1a72fcca58-6e73f67adcdso534159b3a.0; Tue, 19 Mar 2024 19:55:10 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20230601; t=1710903310; x=1711508110; darn=vger.kernel.org; h=in-reply-to:content-transfer-encoding:content-disposition :mime-version:references:message-id:subject:cc:to:from:date:from:to :cc:subject:date:message-id:reply-to; bh=CgE4Mpzc7USCNSk5n/jU7+IJGA1QF8JmfVfBPDD7vCQ=; b=XUzM6E3ttyuiMj6ZphibopU3i6hJPUBd/ymb3E9tDM7vCz1qcRgsKlzkjlLKz7+f91 NcF6gQOzk3r4/vrLWDU4911wd3Q0arMPiGBM5L2cPhgJr0uBp8+na+fhEdxifs0B5JdL Kg44y8xVszTbjC0K9ati7kM+Q1CF2eFSmBOlrd/L5dXSzQsTlpy7V/7seaq8DtSKzwAw gImOYucU5Zqz9z7NjNbDxEixOeJqMCmh+lahUkbFLXwKFCRV4utwyb61o8GeC2gqRSm5 fIVfGfNRtG4dngk+E511gNn4Cm9xTArTj5GhBQzdHpIB8AUBhr0xtKxMNLLWxIQs0+Yl vxjg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1710903310; x=1711508110; h=in-reply-to:content-transfer-encoding:content-disposition :mime-version:references:message-id:subject:cc:to:from:date :x-gm-message-state:from:to:cc:subject:date:message-id:reply-to; bh=CgE4Mpzc7USCNSk5n/jU7+IJGA1QF8JmfVfBPDD7vCQ=; b=XRE/ztnCQemc55gs1DWxg9CxtX8NE5gcZ2n1KzSE488H73DLGx97UvdHr9eiNk1kfI eC3Ab90ntzja1Y1zIs4ES0kUBA8FWkXAdHugnGCK90epOsqoInNmAG/WPWeuJrDMF4Tn 88ut7rQO2r9ZYcggf1r222BMRfcBczHEfqELnOCGu7zCcc4iZvMq4b3fqgIway1Kk+aX 1IBZv5rtA3wns4iohY6ifc0ej0hMgwebK8WLBgUBubnwIoH4OCnhl9z6V1xr3eqiGG8x qkfuG4nD4s65kfM5K1qXu5EEtYVwrDKgwfSp2k3Nh+GWd/13vdG4CtrS93O0Bo4FXv3i Z5Dg== X-Forwarded-Encrypted: i=1; AJvYcCVpiuD7mau4m7ADqoiMoBVBbvYinJT0S+3zV3m1jwidu6ReCtYb6ZDArWC8C1gEhIWJ0CPc90u7fwBCAX/c8BRI1xXo0fgt+nTIKnQ6buPbfr+vrf97tm+cwlZNceBCAmmYrsQXR8gGHDkprsisWUUTOxF+9r+3tME5TlbdRJlwpryiaAiklVWEPIowIyn/PBZYSl7epMHGSc9vy86JeKm3U44NfJcp8QE7I8CE X-Gm-Message-State: AOJu0Yz6jqL3+frCstZMUYWC2hZ9LboQ03IzIzm8m30raBXong3lneaw cYoiAzmhIQfstyo8APGCfT8hfVSwgdGA83xc+/yH22h15LQm2YKJ X-Received: by 2002:a17:902:ce91:b0:1dc:df03:ad86 with SMTP id f17-20020a170902ce9100b001dcdf03ad86mr18844326plg.2.1710903310160; Tue, 19 Mar 2024 19:55:10 -0700 (PDT) Received: from visitorckw-System-Product-Name ([140.113.216.168]) by smtp.gmail.com with ESMTPSA id q14-20020a17090311ce00b001dc71ead7e5sm12230682plh.165.2024.03.19.19.55.06 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 19 Mar 2024 19:55:09 -0700 (PDT) Date: Wed, 20 Mar 2024 10:55:05 +0800 From: Kuan-Wei Chiu To: Ian Rogers Cc: colyli@suse.de, kent.overstreet@linux.dev, msakai@redhat.com, peterz@infradead.org, mingo@redhat.com, acme@kernel.org, namhyung@kernel.org, akpm@linux-foundation.org, bfoster@redhat.com, mark.rutland@arm.com, alexander.shishkin@linux.intel.com, jolsa@kernel.org, adrian.hunter@intel.com, jserv@ccns.ncku.edu.tw, linux-bcache@vger.kernel.org, dm-devel@lists.linux.dev, linux-bcachefs@vger.kernel.org, linux-perf-users@vger.kernel.org, linux-kernel@vger.kernel.org Subject: Re: [PATCH 13/13] bcachefs: Remove heap-related macros and switch to generic min_heap Message-ID: References: <20240319180005.246930-1-visitorckw@gmail.com> <20240319180005.246930-14-visitorckw@gmail.com> Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Type: text/plain; charset=utf-8 Content-Disposition: inline Content-Transfer-Encoding: 8bit In-Reply-To: On Tue, Mar 19, 2024 at 01:03:17PM -0700, Ian Rogers wrote: > On Tue, Mar 19, 2024 at 11:01 AM Kuan-Wei Chiu wrote: > > > > Drop the heap-related macros from bcachefs and replaces them with the > > generic min_heap implementation from include/linux. This change > > improves code readability by using functions instead of macros. > > > > Link: https://lkml.kernel.org/ioyfizrzq7w7mjrqcadtzsfgpuntowtjdw5pgn4qhvsdp4mqqg@nrlek5vmisbu > > Signed-off-by: Kuan-Wei Chiu > > --- > > fs/bcachefs/clock.c | 53 +++++++++++----- > > fs/bcachefs/clock_types.h | 2 +- > > fs/bcachefs/ec.c | 99 ++++++++++++++++++----------- > > fs/bcachefs/ec_types.h | 2 +- > > fs/bcachefs/util.h | 127 +++----------------------------------- > > 5 files changed, 109 insertions(+), 174 deletions(-) > > > > diff --git a/fs/bcachefs/clock.c b/fs/bcachefs/clock.c > > index 363644451106..7a7d13f7a629 100644 > > --- a/fs/bcachefs/clock.c > > +++ b/fs/bcachefs/clock.c > > @@ -6,16 +6,29 @@ > > #include > > #include > > > > -static inline long io_timer_cmp(io_timer_heap *h, > > - struct io_timer *l, > > - struct io_timer *r) > > +static inline bool io_timer_cmp(const void *l, const void *r, void *args) > > Should args here and below be marked with the attribute __always_unused? > Thank you for bringing this to my attention. Yes, I will add the __always_unused to the args in v2. Regards, Kuan-Wei > > > { > > - return l->expire - r->expire; > > + struct io_timer *_l = (struct io_timer *)l; > > + struct io_timer *_r = (struct io_timer *)r; > > + > > + return _l->expire >= _r->expire; > > +} > > + > > +static inline void io_timer_swp(void *l, void *r, void *args) > > +{ > > + struct io_timer *_l = (struct io_timer *)l; > > + struct io_timer *_r = (struct io_timer *)r; > > + > > + swap(*_l, *_r); > > } > > > > void bch2_io_timer_add(struct io_clock *clock, struct io_timer *timer) > > { > > size_t i; > > + const struct min_heap_callbacks callbacks = { > > + .less = io_timer_cmp, > > + .swp = io_timer_swp, > > + }; > > > > spin_lock(&clock->timer_lock); > > > > @@ -26,11 +39,11 @@ void bch2_io_timer_add(struct io_clock *clock, struct io_timer *timer) > > return; > > } > > > > - for (i = 0; i < clock->timers.used; i++) > > - if (clock->timers.data[i] == timer) > > + for (i = 0; i < clock->timers.heap.nr; i++) > > + if (min_heap_peek(&clock->timers)[i] == timer) > > goto out; > > > > - BUG_ON(!heap_add(&clock->timers, timer, io_timer_cmp, NULL)); > > + BUG_ON(!min_heap_push(&clock->timers, &timer, &callbacks, NULL)); > > out: > > spin_unlock(&clock->timer_lock); > > } > > @@ -38,12 +51,16 @@ void bch2_io_timer_add(struct io_clock *clock, struct io_timer *timer) > > void bch2_io_timer_del(struct io_clock *clock, struct io_timer *timer) > > { > > size_t i; > > + const struct min_heap_callbacks callbacks = { > > + .less = io_timer_cmp, > > + .swp = io_timer_swp, > > + }; > > > > spin_lock(&clock->timer_lock); > > > > - for (i = 0; i < clock->timers.used; i++) > > - if (clock->timers.data[i] == timer) { > > - heap_del(&clock->timers, i, io_timer_cmp, NULL); > > + for (i = 0; i < clock->timers.heap.nr; i++) > > + if (min_heap_peek(&clock->timers)[i] == timer) { > > + min_heap_pop(&clock->timers, &callbacks, NULL); > > break; > > } > > > > @@ -131,12 +148,16 @@ static struct io_timer *get_expired_timer(struct io_clock *clock, > > unsigned long now) > > { > > struct io_timer *ret = NULL; > > + const struct min_heap_callbacks callbacks = { > > + .less = io_timer_cmp, > > + .swp = io_timer_swp, > > + }; > > > > spin_lock(&clock->timer_lock); > > > > - if (clock->timers.used && > > - time_after_eq(now, clock->timers.data[0]->expire)) > > - heap_pop(&clock->timers, ret, io_timer_cmp, NULL); > > + if (clock->timers.heap.nr && > > + time_after_eq(now, min_heap_peek(&clock->timers)[0]->expire)) > > + min_heap_pop(&clock->timers, &callbacks, NULL); > > > > spin_unlock(&clock->timer_lock); > > > > @@ -161,10 +182,10 @@ void bch2_io_timers_to_text(struct printbuf *out, struct io_clock *clock) > > spin_lock(&clock->timer_lock); > > now = atomic64_read(&clock->now); > > > > - for (i = 0; i < clock->timers.used; i++) > > + for (i = 0; i < clock->timers.heap.nr; i++) > > prt_printf(out, "%ps:\t%li\n", > > - clock->timers.data[i]->fn, > > - clock->timers.data[i]->expire - now); > > + min_heap_peek(&clock->timers)[i]->fn, > > + min_heap_peek(&clock->timers)[i]->expire - now); > > spin_unlock(&clock->timer_lock); > > --out->atomic; > > } > > diff --git a/fs/bcachefs/clock_types.h b/fs/bcachefs/clock_types.h > > index 5fae0012d808..b02b24b9d74f 100644 > > --- a/fs/bcachefs/clock_types.h > > +++ b/fs/bcachefs/clock_types.h > > @@ -23,7 +23,7 @@ struct io_timer { > > /* Amount to buffer up on a percpu counter */ > > #define IO_CLOCK_PCPU_SECTORS 128 > > > > -typedef HEAP(struct io_timer *) io_timer_heap; > > +typedef MIN_HEAP(struct io_timer *, io_timer_heap) io_timer_heap; > > > > struct io_clock { > > atomic64_t now; > > diff --git a/fs/bcachefs/ec.c b/fs/bcachefs/ec.c > > index b98e2c2b8bf0..7b6a31237503 100644 > > --- a/fs/bcachefs/ec.c > > +++ b/fs/bcachefs/ec.c > > @@ -860,14 +860,14 @@ static int __ec_stripe_mem_alloc(struct bch_fs *c, size_t idx, gfp_t gfp) > > { > > ec_stripes_heap n, *h = &c->ec_stripes_heap; > > > > - if (idx >= h->size) { > > + if (idx >= h->heap.size) { > > if (!init_heap(&n, max(1024UL, roundup_pow_of_two(idx + 1)), gfp)) > > return -BCH_ERR_ENOMEM_ec_stripe_mem_alloc; > > > > mutex_lock(&c->ec_stripes_heap_lock); > > - if (n.size > h->size) { > > - memcpy(n.data, h->data, h->used * sizeof(h->data[0])); > > - n.used = h->used; > > + if (n.heap.size > h->heap.size) { > > + memcpy(min_heap_peek(&n), min_heap_peek(h), h->heap.nr * sizeof(*min_heap_peek(h))); > > + n.heap.nr = h->heap.nr; > > swap(*h, n); > > } > > mutex_unlock(&c->ec_stripes_heap_lock); > > @@ -958,20 +958,21 @@ static u64 stripe_idx_to_delete(struct bch_fs *c) > > > > lockdep_assert_held(&c->ec_stripes_heap_lock); > > > > - if (h->used && > > - h->data[0].blocks_nonempty == 0 && > > - !bch2_stripe_is_open(c, h->data[0].idx)) > > - return h->data[0].idx; > > + if (h->heap.nr && > > + min_heap_peek(h)->blocks_nonempty == 0 && > > + !bch2_stripe_is_open(c, min_heap_peek(h)->idx)) > > + return min_heap_peek(h)->idx; > > > > return 0; > > } > > > > -static inline int ec_stripes_heap_cmp(ec_stripes_heap *h, > > - struct ec_stripe_heap_entry l, > > - struct ec_stripe_heap_entry r) > > +static inline bool ec_stripes_heap_cmp(const void *l, const void *r, void *args) > > { > > - return ((l.blocks_nonempty > r.blocks_nonempty) - > > - (l.blocks_nonempty < r.blocks_nonempty)); > > + struct ec_stripe_heap_entry *_l = (struct ec_stripe_heap_entry *)l; > > + struct ec_stripe_heap_entry *_r = (struct ec_stripe_heap_entry *)r; > > + > > + return ((_l->blocks_nonempty > _r->blocks_nonempty) >= > > + (_l->blocks_nonempty < _r->blocks_nonempty)); > > } > > > > static inline void ec_stripes_heap_set_backpointer(ec_stripes_heap *h, > > @@ -979,7 +980,21 @@ static inline void ec_stripes_heap_set_backpointer(ec_stripes_heap *h, > > { > > struct bch_fs *c = container_of(h, struct bch_fs, ec_stripes_heap); > > > > - genradix_ptr(&c->stripes, h->data[i].idx)->heap_idx = i; > > + genradix_ptr(&c->stripes, min_heap_peek(h)[i].idx)->heap_idx = i; > > +} > > + > > +static inline void ec_stripes_heap_swap(void *l, void *r, void *h) > > +{ > > + struct ec_stripe_heap_entry *_l = (struct ec_stripe_heap_entry *)l; > > + struct ec_stripe_heap_entry *_r = (struct ec_stripe_heap_entry *)r; > > + ec_stripes_heap *_h = (ec_stripes_heap *)h; > > + size_t i = _l - min_heap_peek(_h); > > + size_t j = _r - min_heap_peek(_h); > > + > > + ec_stripes_heap_set_backpointer(_h, i); > > + ec_stripes_heap_set_backpointer(_h, j); > > + > > + swap(*_l, *_r); > > } > > > > static void heap_verify_backpointer(struct bch_fs *c, size_t idx) > > @@ -987,34 +1002,43 @@ static void heap_verify_backpointer(struct bch_fs *c, size_t idx) > > ec_stripes_heap *h = &c->ec_stripes_heap; > > struct stripe *m = genradix_ptr(&c->stripes, idx); > > > > - BUG_ON(m->heap_idx >= h->used); > > - BUG_ON(h->data[m->heap_idx].idx != idx); > > + BUG_ON(m->heap_idx >= h->heap.nr); > > + BUG_ON(min_heap_peek(h)[m->heap_idx].idx != idx); > > } > > > > void bch2_stripes_heap_del(struct bch_fs *c, > > struct stripe *m, size_t idx) > > { > > + const struct min_heap_callbacks callbacks = { > > + .less = ec_stripes_heap_cmp, > > + .swp = ec_stripes_heap_swap, > > + }; > > + > > mutex_lock(&c->ec_stripes_heap_lock); > > heap_verify_backpointer(c, idx); > > > > - heap_del(&c->ec_stripes_heap, m->heap_idx, > > - ec_stripes_heap_cmp, > > - ec_stripes_heap_set_backpointer); > > + min_heap_del(&c->ec_stripes_heap, m->heap_idx, &callbacks, &c->ec_stripes_heap); > > mutex_unlock(&c->ec_stripes_heap_lock); > > } > > > > void bch2_stripes_heap_insert(struct bch_fs *c, > > struct stripe *m, size_t idx) > > { > > + const struct min_heap_callbacks callbacks = { > > + .less = ec_stripes_heap_cmp, > > + .swp = ec_stripes_heap_swap, > > + }; > > + > > mutex_lock(&c->ec_stripes_heap_lock); > > - BUG_ON(heap_full(&c->ec_stripes_heap)); > > + BUG_ON(min_heap_full(&c->ec_stripes_heap)); > > > > - heap_add(&c->ec_stripes_heap, ((struct ec_stripe_heap_entry) { > > + genradix_ptr(&c->stripes, idx)->heap_idx = c->ec_stripes_heap.heap.nr; > > + min_heap_push(&c->ec_stripes_heap, &((struct ec_stripe_heap_entry) { > > .idx = idx, > > .blocks_nonempty = m->blocks_nonempty, > > }), > > - ec_stripes_heap_cmp, > > - ec_stripes_heap_set_backpointer); > > + &callbacks, > > + &c->ec_stripes_heap); > > > > heap_verify_backpointer(c, idx); > > mutex_unlock(&c->ec_stripes_heap_lock); > > @@ -1026,17 +1050,20 @@ void bch2_stripes_heap_update(struct bch_fs *c, > > ec_stripes_heap *h = &c->ec_stripes_heap; > > bool do_deletes; > > size_t i; > > + const struct min_heap_callbacks callbacks = { > > + .less = ec_stripes_heap_cmp, > > + .swp = ec_stripes_heap_swap, > > + }; > > > > mutex_lock(&c->ec_stripes_heap_lock); > > heap_verify_backpointer(c, idx); > > > > - h->data[m->heap_idx].blocks_nonempty = m->blocks_nonempty; > > + min_heap_peek(h)[m->heap_idx].blocks_nonempty = m->blocks_nonempty; > > > > i = m->heap_idx; > > - heap_sift_up(h, i, ec_stripes_heap_cmp, > > - ec_stripes_heap_set_backpointer); > > - heap_sift_down(h, i, ec_stripes_heap_cmp, > > - ec_stripes_heap_set_backpointer); > > + > > + min_heap_sift_up(h, i, &callbacks, &c->ec_stripes_heap); > > + min_heapify(h, i, &callbacks, &c->ec_stripes_heap); > > > > heap_verify_backpointer(c, idx); > > > > @@ -1828,12 +1855,12 @@ static s64 get_existing_stripe(struct bch_fs *c, > > return -1; > > > > mutex_lock(&c->ec_stripes_heap_lock); > > - for (heap_idx = 0; heap_idx < h->used; heap_idx++) { > > + for (heap_idx = 0; heap_idx < h->heap.nr; heap_idx++) { > > /* No blocks worth reusing, stripe will just be deleted: */ > > - if (!h->data[heap_idx].blocks_nonempty) > > + if (!min_heap_peek(h)[heap_idx].blocks_nonempty) > > continue; > > > > - stripe_idx = h->data[heap_idx].idx; > > + stripe_idx = min_heap_peek(h)[heap_idx].idx; > > > > m = genradix_ptr(&c->stripes, stripe_idx); > > > > @@ -2159,14 +2186,14 @@ void bch2_stripes_heap_to_text(struct printbuf *out, struct bch_fs *c) > > size_t i; > > > > mutex_lock(&c->ec_stripes_heap_lock); > > - for (i = 0; i < min_t(size_t, h->used, 50); i++) { > > - m = genradix_ptr(&c->stripes, h->data[i].idx); > > + for (i = 0; i < min_t(size_t, h->heap.nr, 50); i++) { > > + m = genradix_ptr(&c->stripes, min_heap_peek(h)[i].idx); > > > > - prt_printf(out, "%zu %u/%u+%u", h->data[i].idx, > > - h->data[i].blocks_nonempty, > > + prt_printf(out, "%zu %u/%u+%u", min_heap_peek(h)[i].idx, > > + min_heap_peek(h)[i].blocks_nonempty, > > m->nr_blocks - m->nr_redundant, > > m->nr_redundant); > > - if (bch2_stripe_is_open(c, h->data[i].idx)) > > + if (bch2_stripe_is_open(c, min_heap_peek(h)[i].idx)) > > prt_str(out, " open"); > > prt_newline(out); > > } > > diff --git a/fs/bcachefs/ec_types.h b/fs/bcachefs/ec_types.h > > index 976426da3a12..2ed67431a81c 100644 > > --- a/fs/bcachefs/ec_types.h > > +++ b/fs/bcachefs/ec_types.h > > @@ -36,6 +36,6 @@ struct ec_stripe_heap_entry { > > unsigned blocks_nonempty; > > }; > > > > -typedef HEAP(struct ec_stripe_heap_entry) ec_stripes_heap; > > +typedef MIN_HEAP(struct ec_stripe_heap_entry, ec_stripes_heap) ec_stripes_heap; > > > > #endif /* _BCACHEFS_EC_TYPES_H */ > > diff --git a/fs/bcachefs/util.h b/fs/bcachefs/util.h > > index 7ffbddb80400..c599dc5276ac 100644 > > --- a/fs/bcachefs/util.h > > +++ b/fs/bcachefs/util.h > > @@ -8,6 +8,7 @@ > > #include > > #include > > #include > > +#include > > #include > > #include > > #include > > @@ -54,134 +55,20 @@ static inline size_t buf_pages(void *p, size_t len) > > PAGE_SIZE); > > } > > > > -#define HEAP(type) \ > > -struct { \ > > - size_t size, used; \ > > - type *data; \ > > -} > > - > > -#define DECLARE_HEAP(type, name) HEAP(type) name > > - > > #define init_heap(heap, _size, gfp) \ > > ({ \ > > - (heap)->used = 0; \ > > - (heap)->size = (_size); \ > > - (heap)->data = kvmalloc((heap)->size * sizeof((heap)->data[0]),\ > > - (gfp)); \ > > -}) > > - > > -#define free_heap(heap) \ > > -do { \ > > - kvfree((heap)->data); \ > > - (heap)->data = NULL; \ > > -} while (0) > > - > > -#define heap_set_backpointer(h, i, _fn) \ > > -do { \ > > - void (*fn)(typeof(h), size_t) = _fn; \ > > - if (fn) \ > > - fn(h, i); \ > > -} while (0) > > - > > -#define heap_swap(h, i, j, set_backpointer) \ > > -do { \ > > - swap((h)->data[i], (h)->data[j]); \ > > - heap_set_backpointer(h, i, set_backpointer); \ > > - heap_set_backpointer(h, j, set_backpointer); \ > > -} while (0) > > - > > -#define heap_peek(h) \ > > -({ \ > > - EBUG_ON(!(h)->used); \ > > - (h)->data[0]; \ > > -}) > > - > > -#define heap_full(h) ((h)->used == (h)->size) > > - > > -#define heap_sift_down(h, i, cmp, set_backpointer) \ > > -do { \ > > - size_t _c, _j = i; \ > > - \ > > - for (; _j * 2 + 1 < (h)->used; _j = _c) { \ > > - _c = _j * 2 + 1; \ > > - if (_c + 1 < (h)->used && \ > > - cmp(h, (h)->data[_c], (h)->data[_c + 1]) >= 0) \ > > - _c++; \ > > - \ > > - if (cmp(h, (h)->data[_c], (h)->data[_j]) >= 0) \ > > - break; \ > > - heap_swap(h, _c, _j, set_backpointer); \ > > - } \ > > -} while (0) > > - > > -#define heap_sift_up(h, i, cmp, set_backpointer) \ > > -do { \ > > - while (i) { \ > > - size_t p = (i - 1) / 2; \ > > - if (cmp(h, (h)->data[i], (h)->data[p]) >= 0) \ > > - break; \ > > - heap_swap(h, i, p, set_backpointer); \ > > - i = p; \ > > - } \ > > -} while (0) > > - > > -#define __heap_add(h, d, cmp, set_backpointer) \ > > -({ \ > > - size_t _i = (h)->used++; \ > > - (h)->data[_i] = d; \ > > - heap_set_backpointer(h, _i, set_backpointer); \ > > - \ > > - heap_sift_up(h, _i, cmp, set_backpointer); \ > > - _i; \ > > -}) > > - > > -#define heap_add(h, d, cmp, set_backpointer) \ > > -({ \ > > - bool _r = !heap_full(h); \ > > - if (_r) \ > > - __heap_add(h, d, cmp, set_backpointer); \ > > - _r; \ > > + void *data = kvmalloc(_size * sizeof(*min_heap_peek(heap)), (gfp));\ > > + min_heap_init(heap, data, _size); \ > > + min_heap_peek(heap); \ > > }) > > > > -#define heap_add_or_replace(h, new, cmp, set_backpointer) \ > > -do { \ > > - if (!heap_add(h, new, cmp, set_backpointer) && \ > > - cmp(h, new, heap_peek(h)) >= 0) { \ > > - (h)->data[0] = new; \ > > - heap_set_backpointer(h, 0, set_backpointer); \ > > - heap_sift_down(h, 0, cmp, set_backpointer); \ > > - } \ > > -} while (0) > > > > -#define heap_del(h, i, cmp, set_backpointer) \ > > +#define free_heap(_heap) \ > > do { \ > > - size_t _i = (i); \ > > - \ > > - BUG_ON(_i >= (h)->used); \ > > - (h)->used--; \ > > - if ((_i) < (h)->used) { \ > > - heap_swap(h, _i, (h)->used, set_backpointer); \ > > - heap_sift_up(h, _i, cmp, set_backpointer); \ > > - heap_sift_down(h, _i, cmp, set_backpointer); \ > > - } \ > > + kvfree((_heap)->heap.data); \ > > + (_heap)->heap.data = NULL; \ > > } while (0) > > > > -#define heap_pop(h, d, cmp, set_backpointer) \ > > -({ \ > > - bool _r = (h)->used; \ > > - if (_r) { \ > > - (d) = (h)->data[0]; \ > > - heap_del(h, 0, cmp, set_backpointer); \ > > - } \ > > - _r; \ > > -}) > > - > > -#define heap_resort(heap, cmp, set_backpointer) \ > > -do { \ > > - ssize_t _i; \ > > - for (_i = (ssize_t) (heap)->used / 2 - 1; _i >= 0; --_i) \ > > - heap_sift_down(heap, _i, cmp, set_backpointer); \ > > -} while (0) > > > > #define ANYSINT_MAX(t) \ > > ((((t) 1 << (sizeof(t) * 8 - 2)) - (t) 1) * (t) 2 + (t) 1) > > -- > > 2.34.1 > >