From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from kanga.kvack.org (kanga.kvack.org [205.233.56.17]) by smtp.lore.kernel.org (Postfix) with ESMTP id C62A5C3ABBC for ; Fri, 9 May 2025 05:53:37 +0000 (UTC) Received: by kanga.kvack.org (Postfix) id 5BFC86B0089; Fri, 9 May 2025 01:53:35 -0400 (EDT) Received: by kanga.kvack.org (Postfix, from userid 40) id 56F266B009E; Fri, 9 May 2025 01:53:35 -0400 (EDT) X-Delivered-To: int-list-linux-mm@kvack.org Received: by kanga.kvack.org (Postfix, from userid 63042) id 45D026B009F; Fri, 9 May 2025 01:53:35 -0400 (EDT) X-Delivered-To: linux-mm@kvack.org Received: from relay.hostedemail.com (smtprelay0013.hostedemail.com [216.40.44.13]) by kanga.kvack.org (Postfix) with ESMTP id 2E4C56B0089 for ; Fri, 9 May 2025 01:53:35 -0400 (EDT) Received: from smtpin08.hostedemail.com (a10.router.float.18 [10.200.18.1]) by unirelay07.hostedemail.com (Postfix) with ESMTP id 8CF73160E9B for ; Fri, 9 May 2025 05:53:36 +0000 (UTC) X-FDA: 83422302432.08.D34E072 Received: from m16.mail.163.com (m16.mail.163.com [220.197.31.2]) by imf06.hostedemail.com (Postfix) with ESMTP id B71CA18000A for ; Fri, 9 May 2025 05:53:33 +0000 (UTC) Authentication-Results: imf06.hostedemail.com; dkim=pass header.d=163.com header.s=s110527 header.b=pKxExUwq; spf=pass (imf06.hostedemail.com: domain of 00107082@163.com designates 220.197.31.2 as permitted sender) smtp.mailfrom=00107082@163.com; dmarc=pass (policy=none) header.from=163.com ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=hostedemail.com; s=arc-20220608; t=1746770014; h=from:from:sender:reply-to:subject:subject:date:date: message-id:message-id:to:to:cc:cc:mime-version:mime-version: content-type:content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references:dkim-signature; bh=uoPTW7D7y9aEdAcDnoAxcHgWKo3sfKuT8ai8wcx1ncU=; b=BtoeJtknPs0p7drUBVuwd56Dyn6YmMoNs/k7CaeDNYtaYn10WN6QlsQ4TdWtncgxZJfeW5 INP0uhy7qxbIDZO6mk5YgYe1tvkue+SjfcoyaebrcDo6TvgoKcOHHmeRl85STuCP9/XPEw Ti+GY7jv5FpejZDZRm8asVoIerfYB5k= ARC-Authentication-Results: i=1; imf06.hostedemail.com; dkim=pass header.d=163.com header.s=s110527 header.b=pKxExUwq; spf=pass (imf06.hostedemail.com: domain of 00107082@163.com designates 220.197.31.2 as permitted sender) smtp.mailfrom=00107082@163.com; dmarc=pass (policy=none) header.from=163.com ARC-Seal: i=1; s=arc-20220608; d=hostedemail.com; t=1746770014; a=rsa-sha256; cv=none; b=Tnrvo44ot0z1XXig2q0e6aNSrV+xQGCr2U5iPcjw7QPDKic/TH+qBsEzedMCli8hTDpmAa JdCAN9VU79w7dWYT9SSp6icL4f14AsJ8gSPsX2LrqJLx0N3fclDcEkzYw+UV9gE06FkPMm woWWe6/JIOQuUg2Y+TOcDj/EfWgxS1g= DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=163.com; s=s110527; h=From:To:Subject:Date:Message-Id:MIME-Version; bh=uo PTW7D7y9aEdAcDnoAxcHgWKo3sfKuT8ai8wcx1ncU=; b=pKxExUwqNpLOlxIDqh 3bQJMibGdgzQTeCDQByD3Py2nU247yED53sO/mYARKq+E4p5xLF8BXiEMdzO58ta Ah8mdKgOCxEXjOoSaSNe2YKPg2PqncXO2LA3cj+Aar90A7LtYgOybl18yVCnOBWR g0Z2uByv5H+4x+2/SOTv1MDeo= Received: from localhost.localdomain (unknown []) by gzga-smtp-mtada-g1-0 (Coremail) with SMTP id _____wDHzktLmB1ot4qaAA--.13910S4; Fri, 09 May 2025 13:53:22 +0800 (CST) From: David Wang <00107082@163.com> To: surenb@google.com, kent.overstreet@linux.dev, akpm@linux-foundation.org Cc: linux-kernel@vger.kernel.org, linux-mm@kvack.org, David Wang <00107082@163.com> Subject: [PATCH 2/2] alloc_tag: keep codetag iterator cross read() calls Date: Fri, 9 May 2025 13:53:13 +0800 Message-Id: <20250509055313.922707-1-00107082@163.com> X-Mailer: git-send-email 2.39.2 In-Reply-To: <20250507175500.204569-1-00107082@163.com> References: <20250507175500.204569-1-00107082@163.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-CM-TRANSID:_____wDHzktLmB1ot4qaAA--.13910S4 X-Coremail-Antispam: 1Uf129KBjvJXoWxGFWUWw1UXF18ZrWDXF47Jwb_yoWruF1Upa y3ua4YkF4rAr1UuF4rJr4IqFW3ta4ktF48XF42qr4FvF1Yvrs8uF98Jryj9FW3AFy8Kanx Zan0k34UAr9FvaDanT9S1TB71UUUUU7qnTZGkaVYY2UrUUUUjbIjqfuFe4nvWSU5nxnvy2 9KBjDUYxBIdaVFxhVjvjDU0xZFpf9x0p__-BDUUUUU= X-Originating-IP: [111.35.191.17] X-CM-SenderInfo: qqqrilqqysqiywtou0bp/1tbiqApIqmgdj33+6wABsT X-Stat-Signature: yb11mqwuj3dg4azpuq45bc89nfec95ts X-Rspamd-Server: rspam01 X-Rspamd-Queue-Id: B71CA18000A X-Rspam-User: X-HE-Tag: 1746770013-534721 X-HE-Meta: U2FsdGVkX19RofaZ/tHyhdPJVKHbn/ojxWEHkzpNlv0Pou5E4ejFbZMb34lZ0vw3/Ua0iYrS506vioXtmo8getZWH4Oe/vcC8bhr7M02NV9BbCZU3Cm0bhCRY9lbTiFL+M0cWon68h+Cv0Kix+vJM1bG1aIuh00nxbCAW71XGBiZCvdK0jRJpDvcwQPhV4sEDUtT2JeN/OqgPAf6YR6opRqT/qHEma/Y5NEotinGOrzNafnUaXg/gER3E0fcgZYh1280gq8VwdBB7c6g1fUU5Dyh6Z9Qdlb91ssMhgP9+7dsL4BQiP7JaTfY3cdTVWk1W26n8DF325l/ymefV0RppvxUeu14MWv0BrLt4vr11Pa3i7WBq+z4ayj5touCuIyq0P5P0JO0fO5M6WO6x0Ukwadun5wFN/EyvUhJUDlXN74Pjspxlu6KmP0QVJh6EB0+z/O+x4q4HRXr/pZhnQEMKtoL96bh12SZLcvioVIb9xcXSVNROs552hi6NgIprpTdzwezLyz/NB70B3e/JkcKFFAPr80DV8pD54mb0d6T7L3vfxmYViYtt6+kMioOOMBP+QBXeL026DoqK1gHVNpH+Zi5N6bqK5OOi2XvKYsqxPQzjuQKrLPk9FHCR0WDA18wqRKJQYMhHLoeOc4shSitngnzsqLm6l08x5Mg549xe+VIv4QAg4cKS7elHj+MC9nNbDnwRwlC3eFICpwnj9bt2r0E4RLVv2f5cPIQfGw3sQqLZMTmQBnZlWz/8yzNa2Zn/7cnLvT3DFlOA4kfjzqXLo0eszex6HHZ2QsQY/8Clm84cfrcLXKlgPWGJALtXqTgwDbLLSA0m5Xpn6XooIbSiDwENtOtLKE3Qu6LLJsWPfNgt5dXCp4HfC/EcIVR9w5NOB4yK0Kks+4giDWJh8q8wu0x0F1Uax0Y1/XffJSbPOMysifFU66ujNUmMmhco1++AGkuIo6c/wNSqykt4eu /gnAuXTt fDKMYDWcuNSaMyRWxbYkxekCVl41YDMhVF7R2IqXrjaviN//Xv3m4ogZwMPH2ugcyRKxyWCAqrPZi97gSW6Pm8jzzWIP/k91orCWzYVBwfZkDLkZu5gcp8Pi/XQ6PcJfZyS/C8rjS+Jj+FX9fHSafOQRoD2tj4EuiaU0uE3qFpDuSYNQu+r87kJk7eT42pDtP5EJ9d9qsK4HmClLowTxq7R42s00Tif1U39AJxydlSSLjVgfr9sZDKd7I2TE88TzOPRZylk0TDwntLPHygBYndE7V4s/FtiihoJNJ3DdoxbZS06jcj4/YFrFMXOb6hw11ozZmLf/S4lzeMS5C7+wGpd73ZW65f9M3o0xTVvnaCx1bkZpsHcABKGqU7zNSkwmnBvvWTDPHXcXWsu+0v4ESY2mQhB/qCwoh20a7Rt2cLk664HE= X-Bogosity: Ham, tests=bogofilter, spamicity=0.000000, version=1.2.4 Sender: owner-linux-mm@kvack.org Precedence: bulk X-Loop: owner-majordomo@kvack.org List-ID: List-Subscribe: List-Unsubscribe: When reading /proc/allocinfo, for each read syscall, seq_file would invoke start/stop callbacks. In start callback, a memory is alloced to store iterator and the iterator would start from beginning to walk to current read position. seq_file read() takes at most 4096 bytes, even if read with a larger user space buffer, meaning read out all of /proc/allocinfo, tens of read syscalls are needed. For example, a 306036 bytes allocinfo files need 76 reads: $ sudo cat /proc/allocinfo | wc 3964 16678 306036 $ sudo strace -T -e read cat /proc/allocinfo ... read(3, " 4096 1 arch/x86/k"..., 131072) = 4063 <0.000062> ... read(3, " 0 0 sound/core"..., 131072) = 4021 <0.000150> ... For those n=3964 lines, each read takes about m=3964/76=52 lines, since iterator restart from beginning for each read(), it would move forward m steps on 1st read 2*m steps on 2nd read 3*m steps on 3rd read ... n steps on last read As read() along, more iterator steps make read() calls slower and slower. Adding those up, codetag iterator moves about O(n*n/m) steps, making data structure traversal take significant part of the whole reading. Profiling when stress reading /proc/allocinfo confirms it: vfs_read(99.959% 1677299/1677995) proc_reg_read_iter(99.856% 1674881/1677299) seq_read_iter(99.959% 1674191/1674881) allocinfo_start(75.664% 1266755/1674191) codetag_next_ct(79.217% 1003487/1266755) <--- srso_return_thunk(1.264% 16011/1266755) __kmalloc_cache_noprof(0.102% 1296/1266755) ... allocinfo_show(21.287% 356378/1674191) allocinfo_next(1.530% 25621/1674191) allocinfo_start() takes about 75% of seq_read(). A private data alloced at open() time can be used to carry iterator alive across read() calls, and avoid the memory allocation and iterator reset for each read(). This way, only O(1) memory allocation and O(n) steps iterating, and `time` shows performance improvement from ~7ms to ~4ms. Profiling with the change: vfs_read(99.865% 1581073/1583214) proc_reg_read_iter(99.485% 1572934/1581073) seq_read_iter(99.846% 1570519/1572934) allocinfo_show(87.428% 1373074/1570519) seq_buf_printf(83.695% 1149196/1373074) seq_buf_putc(1.917% 26321/1373074) _find_next_bit(1.531% 21023/1373074) ... codetag_to_text(0.490% 6727/1373074) ... allocinfo_next(6.275% 98543/1570519) ... allocinfo_start(0.369% 5790/1570519) ... allocinfo_start taks less than 1%. Signed-off-by: David Wang <00107082@163.com> --- lib/alloc_tag.c | 29 ++++++++++------------------- 1 file changed, 10 insertions(+), 19 deletions(-) diff --git a/lib/alloc_tag.c b/lib/alloc_tag.c index 25ecc1334b67..fdd5887769a6 100644 --- a/lib/alloc_tag.c +++ b/lib/alloc_tag.c @@ -45,21 +45,16 @@ struct allocinfo_private { static void *allocinfo_start(struct seq_file *m, loff_t *pos) { struct allocinfo_private *priv; - struct codetag *ct; loff_t node = *pos; - priv = kzalloc(sizeof(*priv), GFP_KERNEL); - m->private = priv; - if (!priv) - return NULL; - - priv->print_header = (node == 0); + priv = (struct allocinfo_private *)m->private; codetag_lock_module_list(alloc_tag_cttype, true); - priv->iter = codetag_get_ct_iter(alloc_tag_cttype); - while ((ct = codetag_next_ct(&priv->iter)) != NULL && node) - node--; - - return ct ? priv : NULL; + if (node == 0) { + priv->print_header = true; + priv->iter = codetag_get_ct_iter(alloc_tag_cttype); + codetag_next_ct(&priv->iter); + } + return priv->iter.ct ? priv : NULL; } static void *allocinfo_next(struct seq_file *m, void *arg, loff_t *pos) @@ -76,12 +71,7 @@ static void *allocinfo_next(struct seq_file *m, void *arg, loff_t *pos) static void allocinfo_stop(struct seq_file *m, void *arg) { - struct allocinfo_private *priv = (struct allocinfo_private *)m->private; - - if (priv) { - codetag_lock_module_list(alloc_tag_cttype, false); - kfree(priv); - } + codetag_lock_module_list(alloc_tag_cttype, false); } static void print_allocinfo_header(struct seq_buf *buf) @@ -249,7 +239,8 @@ static void __init procfs_init(void) if (!mem_profiling_support) return; - if (!proc_create_seq(ALLOCINFO_FILE_NAME, 0400, NULL, &allocinfo_seq_op)) { + if (!proc_create_seq_private(ALLOCINFO_FILE_NAME, 0400, NULL, &allocinfo_seq_op, + sizeof(struct allocinfo_private), NULL)) { pr_err("Failed to create %s file\n", ALLOCINFO_FILE_NAME); shutdown_mem_profiling(false); } -- 2.39.2