]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
perf pmu: Fix perf_pmu__parse_scale/unit() OOB access on empty sysfs file
authorArnaldo Carvalho de Melo <acme@redhat.com>
Mon, 8 Jun 2026 00:03:13 +0000 (21:03 -0300)
committerArnaldo Carvalho de Melo <acme@redhat.com>
Wed, 10 Jun 2026 21:56:01 +0000 (18:56 -0300)
perf_pmu__parse_scale() reads a PMU scale file then accesses
scale[sret - 1] to strip a trailing newline.  Only sret < 0 is
guarded, so an empty file (sret == 0) causes scale[-1] — a stack
buffer underflow that reads and potentially writes out of bounds.

perf_pmu__parse_unit() has the same pattern: alias->unit[sret - 1]
with sret == 0 accesses the byte before the struct member, which
may corrupt the adjacent pmu_name pointer field.

Change both guards from sret < 0 to sret <= 0 so that empty files
are treated as read errors.

Fixes: 410136f5dd96b601 ("tools/perf/stat: Add event unit and scale support")
Reported-by: sashiko-bot <sashiko-bot@kernel.org>
Cc: Stephane Eranian <eranian@google.com>
Assisted-by: Claude:claude-opus-4.6
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
tools/perf/util/pmu.c

index f588cce601941a2270ac55e17d91806792978c1e..a550f030b85df39e5eeb66b3c0837cd19815b242 100644 (file)
@@ -314,7 +314,7 @@ static int perf_pmu__parse_scale(struct perf_pmu *pmu, struct perf_pmu_alias *al
                goto error;
 
        sret = read(fd, scale, sizeof(scale)-1);
-       if (sret < 0)
+       if (sret <= 0)
                goto error;
 
        if (scale[sret - 1] == '\n')
@@ -346,7 +346,7 @@ static int perf_pmu__parse_unit(struct perf_pmu *pmu, struct perf_pmu_alias *ali
                return -1;
 
        sret = read(fd, alias->unit, UNIT_MAX_LEN);
-       if (sret < 0)
+       if (sret <= 0)
                goto error;
 
        close(fd);