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 gabe.freedesktop.org (gabe.freedesktop.org [131.252.210.177]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.lore.kernel.org (Postfix) with ESMTPS id E6D59105F7AC for ; Fri, 13 Mar 2026 13:24:17 +0000 (UTC) Received: from gabe.freedesktop.org (localhost [127.0.0.1]) by gabe.freedesktop.org (Postfix) with ESMTP id 2C9CF10EC0D; Fri, 13 Mar 2026 13:24:17 +0000 (UTC) Authentication-Results: gabe.freedesktop.org; dkim=pass (2048-bit key; unprotected) header.d=gmail.com header.i=@gmail.com header.b="AGqe1FGw"; dkim-atps=neutral Received: from mail-oi1-f181.google.com (mail-oi1-f181.google.com [209.85.167.181]) by gabe.freedesktop.org (Postfix) with ESMTPS id B0CF310EC06 for ; Fri, 13 Mar 2026 13:24:03 +0000 (UTC) Received: by mail-oi1-f181.google.com with SMTP id 5614622812f47-466ebbf7ff7so801934b6e.1 for ; Fri, 13 Mar 2026 06:24:03 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20230601; t=1773408243; x=1774013043; darn=lists.freedesktop.org; h=content-transfer-encoding:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:from:to:cc:subject:date :message-id:reply-to; bh=1lq5jaJPqLqnk3f4baOzmy3HWsNUAD6m3P658X9bd80=; b=AGqe1FGw7lbep3ljCAlRAytG+Qzamz5ZM2Rg9SLSHr3aB911wU52NVTuAPVSH+ncnZ Hj2407vdxpl/SdKIjRJm6ga9TFzxWQdmGf72gSQYDcfqv8aIOOSUT1hoXzpeKyLFYU3H 9eGVuX/BXXvJTqsHzDugDz9iDIAAdDNysSojeMH5rgvKZvAZ0Mk4nOtqpHc5zlcDapbz tkfTPTEb8IhxNEALadCb1MbRdR8YPzRqVXAxoVonBR9S10JNz8WktRK0yu29fl8yF879 CRBXiZY6IDIbNr9j/KsHyzCH5oE+O/+Y1ezgKlMNNi7Zuw/FOYRLrpR+o8uuTISLlUzF nmDw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20251104; t=1773408243; x=1774013043; h=content-transfer-encoding:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:x-gm-gg:x-gm-message-state:from :to:cc:subject:date:message-id:reply-to; bh=1lq5jaJPqLqnk3f4baOzmy3HWsNUAD6m3P658X9bd80=; b=Tu4HTrYGMZDcZcQ6u7TtYqGWT8/PzFqzDrw5JNRsXRT88j/ZfQVGqaJjETfkXlhuit 1uM9NFBa/goGkhQ40ElDCtGfeezyPyASKM1VIv30AzhgFp1wo11wKV4jLUnbT+RY0yA2 lARZ83Cpi4UimBuROIFD8D5VQoEg3K4h5mSZVwQ2z4w1bHwctjoRO3FTbtj1sRXpFzQ5 wPv9OsoreWbQXLvf+7evQhTYQ2Y49lYDBM1s0Xv1wRFoeXJzfBJOWT58EaZHDragBmAt MzrrUz0ZSm3MAZDbgqej/341qF+K69ZrctPti410qpgnFMhGstJojZAOtdjsBlu4c552 rPAA== X-Forwarded-Encrypted: i=1; AJvYcCVSthoz7QDHNUSC1cPK9qQ5wS77+jr4zTFPaFL0bBxRqCR/McrJ5LApBZ/CrRkgcOBSWFUUqNdN3mQ=@lists.freedesktop.org X-Gm-Message-State: AOJu0YwOeUzD0JbOxNE/wI+LcJp7x0OdWwuf6CGOL3gBNlPovQfDTN5o u8HrccULmFO9RQDjQy1lClnGWe/LaUolsgrCckXiD2ToOGOvgFyK1x5n X-Gm-Gg: ATEYQzylOvxYEVXrs9E1crRm67S4H1fEBogDOiPqtJFk+99JExWEhlbfZfnkmrjmD44 sMT+YA7GZUgsnmOLdSfCmHv3tWhhe1MtTaxTZCFQN8TPQ3+pul/ONUuvFb82DSHHYQzeYMYN4Cf NdPs6KZNzzItUuOCr+x0p3VGUKZjBjxqgNqk1AFb4XFn9BhdbiiJ0aHs4ivVSNe3pOMhkQIAqy6 iB5HgjtdyX42oEJnkUmGf3eY9/snP2Z0Jse1vYprjROpZAKSeEjiZbm8QDwmL0oh+PL3jetuiIv Cp/N3QyGKSea3NVX7OVhcbN9RlgVKE7t8l5qFRp/fQLhNpRfix+32JKsEw0TGt3nxCfBf4rtj1b dwnpJ19LDP88bjO95scBLLCw3EhY5x7SkAnd4mzZX8U+OA5JL/ySkQrHlhZsnZ8kFm6EvR22RlL kF7RvnsyIeqmuy18Gg9wSisPLpyoM+RTG5O+9222xaChA2/8jY X-Received: by 2002:a05:6808:1592:b0:467:42ab:2f69 with SMTP id 5614622812f47-467572f2c8emr1586112b6e.34.1773408242720; Fri, 13 Mar 2026 06:24:02 -0700 (PDT) Received: from frodo (c-98-38-17-99.hsd1.co.comcast.net. [98.38.17.99]) by smtp.googlemail.com with ESMTPSA id 5614622812f47-4673434a35csm4795987b6e.20.2026.03.13.06.24.01 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 13 Mar 2026 06:24:02 -0700 (PDT) From: Jim Cromie To: airlied@gmail.com, simona@ffwll.ch, jbaron@akamai.com, gregkh@linuxfoundation.org, Jim Cromie , Andrew Morton , linux-kernel@vger.kernel.org Cc: mripard@kernel.org, tzimmermann@suse.de, maarten.lankhorst@linux.intel.com, jani.nikula@intel.com, ville.syrjala@linux.intel.com, christian.koenig@amd.com, matthew.auld@intel.com, arunpravin.paneerselvam@amd.com, louis.chauvet@bootlin.com, skhan@linuxfoundation.org, pmladek@suse.com, ukaszb@chromium.org, dri-devel@lists.freedesktop.org, intel-gfx@lists.freedesktop.org, amd-gfx@lists.freedesktop.org Subject: [PATCH v11 30/65] dyndbg: resolve "protection" of class'd pr_debug Date: Fri, 13 Mar 2026 07:19:55 -0600 Message-ID: <20260313132103.2529746-31-jim.cromie@gmail.com> X-Mailer: git-send-email 2.53.0 In-Reply-To: <20260313132103.2529746-1-jim.cromie@gmail.com> References: <20260313132103.2529746-1-jim.cromie@gmail.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-BeenThere: dri-devel@lists.freedesktop.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: Direct Rendering Infrastructure - Development List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: dri-devel-bounces@lists.freedesktop.org Sender: "dri-devel" classmap-v1 code protected class'd pr_debugs from unintended changes by unclassed/_DFLT queries: # - to declutter examples: alias ddcmd='echo $* > /proc/dynamic_debug/control' # IOW, this should NOT alter drm.debug settings ddcmd -p # Instead, you must name the class to change it. # Protective but tedious ddcmd class DRM_UT_CORE +p # Or do it the (old school) subsystem way # This is ABI !! echo 1 > /sys/module/drm/parameters/debug Since the debug sysfs-node is ABI, if dyndbg is going to implement it, it must also honor its settings; it must at least protect against accidental changes to its classes from legacy queries. The protection allows all previously conceived queries to work the way they always have; ie select the same set of pr_debugs, despite the inclusion of whole new classes of pr_debugs. But that choice has 2 downsides: 1. "name the class to change it" makes a tedious long-winded interface, needing many commands to set DRM_UT_* one at a time. 2. It makes the class keyword special in some sense; the other keywords skip only on query mismatch, otherwise the code falls thru to adjust the pr-debug site. Jason Baron didn't like v1 on point 2. Louis Chauvet didn't like recent rev on point 1 tedium. But that said: /sys/ is ABI, so this must be reliable: #> echo 0x1f > /sys/module/drm/parameters/debug It 'just works' without dyndbg underneath; we must deliver that same stability. Convenience is secondary. The new resolution: If ABI is the blocking issue, then no ABI means no blocking issue. IOW, if the classmap has no presence under /sys/*, ie no PARAM, there is no ABI to guard, and no reason to enforce a tedious interface. In the future, if DRM wants to alter this protection, that is practical, but I think default-on is the correct mode. So atm classes without a PARAM are unprotected at >control, allowing admins their shortcuts. I think this could satisfy all viewpoints. That said, theres also a possibility of wildcard classes: #> ddcmd class '*' +p Currently, the query-class is exact-matched against each module's classmaps.names. This gives precise behavior, a good basis. But class wildcards are possible, they just did'nt appear useful for DRM, whose classmap names are a flat DRM_UT_* namespace. IOW, theres no useful selectivity there: #> ddcmd class "DRM_*" +p # these enable every DRM_* class #> ddcmd class "DRM_UT_*" +p #> ddcmd class "DRM_UT_V*" +p # finally select just 1: DRM_UT_VBL #> ddcmd class "DRM_UT_D*" +p # but this gets 3 #> ddcmd class "D*V*" +p # here be dragons But there is debatable utility in the feature. #> ddcmd class __DEFAULT__ -p # what about this ? #> ddcmd -p # thats what this does. automatically Anyway, this patch does: 1. adds link field from _ddebug_class_map to the .controlling_param 2. sets it in ddebug_match_apply_kparam(), during modprobe/init, when options like drm.debug=VAL are handled. 3. ddebug_class_has_param() now checks .controlling_param 4. ddebug_class_wants_protection() macro renames 3. this frames it as a separable policy decision 5. ddebug_match_desc() gets the most attention: a. move classmap consideration to the bottom this insures all other constraints act 1st. allows simpler 'final' decisions. b. split class choices cleanly on query: class FOO vs none, and class'd vs _DPRINTK_CLASS_DFLT site. c. calls 4 when applying a class-less query to a class'd pr_debug here we need a new fn to find the classmap with this .class_id d. calls new ddebug_find_classmap_by_class_id(). when class-less query looks at a class'd pr_debug. finds classmap, which can then decide, currently by PARAM existence. NOTES: protection is only against class-less queries, explicit "class FOO" adjustments are allowed (that is the mechanism). The drm.debug sysfs-node heavily under-specifies the class'd pr_debugs it controls; none of the +mfls prefixing flags have any effect, and each callsite remains individually controllable. drm.debug just toggles the +p flag for all the modules' class'd pr_debugs. Signed-off-by: Jim Cromie --- include/linux/dynamic_debug.h | 14 ++-- lib/dynamic_debug.c | 128 +++++++++++++++++++++++++++------- 2 files changed, 111 insertions(+), 31 deletions(-) diff --git a/include/linux/dynamic_debug.h b/include/linux/dynamic_debug.h index 4c36c2a4801a..a1454db1bcb0 100644 --- a/include/linux/dynamic_debug.h +++ b/include/linux/dynamic_debug.h @@ -76,6 +76,7 @@ enum ddebug_class_map_type { * map @class_names 0..N to consecutive constants starting at @base. */ struct _ddebug_class_map { + struct _ddebug_class_param *controlling_param; const struct module *mod; /* NULL for builtins */ const char *mod_name; const char **class_names; @@ -259,7 +260,12 @@ struct _ddebug_class_param { * * Creates a sysfs-param to control the classes defined by the * exported classmap, with bits 0..N-1 mapped to the classes named. - * This version keeps class-state in a private long int. + * + * Since sysfs-params are ABI, this also protects the classmap'd + * pr_debugs from un-class'd `echo -p > /proc/dynamic_debug/control` + * changes. + * + * This keeps class-state in a private long int. */ #define DYNAMIC_DEBUG_CLASSMAP_PARAM(_name, _var, _flags) \ static unsigned long _name##_bvec; \ @@ -272,10 +278,8 @@ struct _ddebug_class_param { * @_var: name of the (exported) classmap var defining the classes/bits * @_flags: flags to be toggled, typically just 'p' * - * Creates a sysfs-param to control the classes defined by the - * exported clasmap, with bits 0..N-1 mapped to the classes named. - * This version keeps class-state in user @_bits. This lets drm check - * __drm_debug elsewhere too. + * Like DYNAMIC_DEBUG_CLASSMAP_PARAM, but maintains param-state in + * extern @_bits. This lets DRM check __drm_debug elsewhere too. */ #define DYNAMIC_DEBUG_CLASSMAP_PARAM_REF(_name, _bits, _var, _flags) \ __DYNAMIC_DEBUG_CLASSMAP_PARAM(_name, _bits, _var, _flags) diff --git a/lib/dynamic_debug.c b/lib/dynamic_debug.c index a1c3141372d1..e9fa0c8868db 100644 --- a/lib/dynamic_debug.c +++ b/lib/dynamic_debug.c @@ -206,6 +206,50 @@ ddebug_find_valid_class(struct _ddebug_info const *di, const char *query_class, return NULL; } +static bool ddebug_class_in_range(const int class_id, const struct _ddebug_class_map *map) +{ + return (class_id >= map->base && + class_id < map->base + map->length); +} + +static struct _ddebug_class_map * +ddebug_find_map_by_class_id(struct _ddebug_info *di, int class_id) +{ + struct _ddebug_class_map *map; + struct _ddebug_class_user *cli; + int i; + + for_subvec(i, map, di, maps) + if (ddebug_class_in_range(class_id, map)) + return map; + + for_subvec(i, cli, di, users) + if (ddebug_class_in_range(class_id, cli->map)) + return cli->map; + + return NULL; +} + +/* + * classmaps-V1 protected classes from changes by legacy commands + * (those selecting _DPRINTK_CLASS_DFLT by omission). This had the + * downside that saying "class FOO" for every change can get tedious. + * + * V2 is smarter, it protects class-maps if the defining module also + * calls DYNAMIC_DEBUG_CLASSMAP_PARAM to create a sysfs parameter. + * Since the author wants the knob, we should assume they intend to + * use it (in preference to "class FOO +p" >control), and want to + * trust its settings. This gives protection when its useful, and not + * when its just tedious. + */ +static inline bool ddebug_class_has_param(const struct _ddebug_class_map *map) +{ + return !!(map->controlling_param); +} + +/* re-framed as a policy choice */ +#define ddebug_class_wants_protection(map) (ddebug_class_has_param(map)) + /* * Search the tables for _ddebug's which match the given `query' and * apply the `flags' and `mask' to them. Returns number of matching @@ -214,11 +258,10 @@ ddebug_find_valid_class(struct _ddebug_info const *di, const char *query_class, */ static bool ddebug_match_desc(const struct ddebug_query *query, struct _ddebug *dp, - int valid_class) + struct _ddebug_info *di, + int selected_class) { - /* match site against query-class */ - if (dp->class_id != valid_class) - return false; + struct _ddebug_class_map *site_map; /* match against the source filename */ if (query->filename && @@ -255,7 +298,29 @@ static bool ddebug_match_desc(const struct ddebug_query *query, dp->lineno > query->last_lineno) return false; - return true; + /* + * above are all satisfied, so we can make final decisions: + * 1- class FOO or implied class __DEFAULT__ + * 2- site.is_classed or not + */ + if (query->class_string) { + /* class FOO given, exact match required */ + return (dp->class_id == selected_class); + } + /* query class __DEFAULT__ by omission. */ + if (dp->class_id == _DPRINTK_CLASS_DFLT) { + /* un-classed site */ + return true; + } + /* site is class'd */ + site_map = ddebug_find_map_by_class_id(di, dp->class_id); + if (!site_map) { + WARN_ONCE(1, "unknown class_id %d, check %s's CLASSMAP definitions", + dp->class_id, di->mod_name); + return false; + } + /* module(-param) decides protection */ + return !ddebug_class_wants_protection(site_map); } static int ddebug_change(const struct ddebug_query *query, struct flag_settings *modifiers) @@ -265,33 +330,31 @@ static int ddebug_change(const struct ddebug_query *query, struct flag_settings unsigned int newflags; unsigned int nfound = 0; struct flagsbuf fbuf, nbuf; - struct _ddebug_class_map *map = NULL; - int valid_class; + int selected_class; /* search for matching ddebugs */ mutex_lock(&ddebug_lock); list_for_each_entry(dt, &ddebug_tables, link) { struct _ddebug_info *di = &dt->info; + struct _ddebug_class_map *mods_map; /* match against the module name */ if (query->module && !match_wildcard(query->module, di->mod_name)) continue; + selected_class = _DPRINTK_CLASS_DFLT; if (query->class_string) { - map = ddebug_find_valid_class(&dt->info, query->class_string, - &valid_class); - if (!map) + mods_map = ddebug_find_valid_class(di, query->class_string, + &selected_class); + if (!mods_map) continue; - } else { - /* constrain query, do not touch class'd callsites */ - valid_class = _DPRINTK_CLASS_DFLT; } for (i = 0; i < di->descs.len; i++) { struct _ddebug *dp = &di->descs.start[i]; - if (!ddebug_match_desc(query, dp, valid_class)) + if (!ddebug_match_desc(query, dp, di, selected_class)) continue; nfound++; @@ -656,6 +719,7 @@ static int ddebug_exec_queries(char *query, const char *modname) } /* apply a new class-param setting */ + static int ddebug_apply_class_bitmap(const struct _ddebug_class_param *dcp, const unsigned long *new_bits, const unsigned long old_bits, @@ -1107,12 +1171,6 @@ static void *ddebug_proc_next(struct seq_file *m, void *p, loff_t *pos) return dp; } -static bool ddebug_class_in_range(const int class_id, const struct _ddebug_class_map *map) -{ - return (class_id >= map->base && - class_id < map->base + map->length); -} - static const char *ddebug_class_name(struct _ddebug_info *di, struct _ddebug *dp) { struct _ddebug_class_map *map; @@ -1237,25 +1295,36 @@ static void ddebug_sync_classbits(const struct kernel_param *kp, const char *mod } } -static void ddebug_match_apply_kparam(const struct kernel_param *kp, - const struct _ddebug_class_map *map, - const char *mod_name) +static struct _ddebug_class_param * +ddebug_get_classmap_kparam(const struct kernel_param *kp, + const struct _ddebug_class_map *map) { struct _ddebug_class_param *dcp; if (kp->ops != ¶m_ops_dyndbg_classes) - return; + return NULL; dcp = (struct _ddebug_class_param *)kp->arg; - if (map == dcp->map) { + return (map == dcp->map) + ? dcp : (struct _ddebug_class_param *)NULL; +} + +static void ddebug_match_apply_kparam(const struct kernel_param *kp, + struct _ddebug_class_map *map, + const char *mod_name) +{ + struct _ddebug_class_param *dcp = ddebug_get_classmap_kparam(kp, map); + + if (dcp) { + map->controlling_param = dcp; v2pr_info(" kp:%s.%s =0x%lx", mod_name, kp->name, *dcp->bits); vpr_cm_info(map, " %s maps ", mod_name); ddebug_sync_classbits(kp, mod_name); } } -static void ddebug_apply_params(const struct _ddebug_class_map *cm, const char *mod_name) +static void ddebug_apply_params(struct _ddebug_class_map *cm, const char *mod_name) { const struct kernel_param *kp; #if IS_ENABLED(CONFIG_MODULES) @@ -1275,6 +1344,13 @@ static void ddebug_apply_params(const struct _ddebug_class_map *cm, const char * } } +/* + * called from add_module, ie early. it can find controlling kparams, + * which can/does? enable protection of this classmap from class-less + * queries, on the grounds that the user created the kparam, means to + * use it, and expects it to reflect reality. We should oblige him, + * and protect those classmaps from classless "-p" changes. + */ static void ddebug_apply_class_maps(const struct _ddebug_info *di) { struct _ddebug_class_map *cm; -- 2.53.0