mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-04 04:02:26 +00:00
c6c3187d66
There exist card implementations with a CDAT table using a fixed size buffer, but with entries filled in that do not fill the whole table length size. Then, the last entry in the CDAT table may not mark the end of the CDAT table buffer specified by the length field in the CDAT header. It can be shorter with trailing unused (zero'ed) data. The actual table length is determined while reading all CDAT entries of the table with DOE. If the table is greater than expected (containing zero'ed trailing data), the CDAT parser fails with: [ 48.691717] Malformed DSMAS table length: (24:0) [ 48.702084] [CDAT:0x00] Invalid zero length [ 48.711460] cxl_port endpoint1: Failed to parse CDAT: -22 In addition, a check of the table buffer length is missing to prevent an out-of-bound access then parsing the CDAT table. Hardening code against device returning borked table. Fix that by providing an optional buffer length argument to acpi_parse_entries_array() that can be used by cdat_table_parse() to propagate the buffer size down to its users to check the buffer length. This also prevents a possible out-of-bound access mentioned. Add a check to warn about a malformed CDAT table length. Cc: Rafael J. Wysocki <rafael@kernel.org> Cc: Len Brown <lenb@kernel.org> Reviewed-by: Dave Jiang <dave.jiang@intel.com> Signed-off-by: Robert Richter <rrichter@amd.com> Reviewed-by: Jonathan Cameron <Jonathan.Cameron@huawei.com> Link: https://lore.kernel.org/r/ZdEnopFO0Tl3t2O1@rric.localdomain Signed-off-by: Dan Williams <dan.j.williams@intel.com>
62 lines
1.8 KiB
C
62 lines
1.8 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* fw_tables.h - Parsing support for ACPI and ACPI-like tables provided by
|
|
* platform or device firmware
|
|
*
|
|
* Copyright (C) 2001 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
|
|
* Copyright (C) 2023 Intel Corp.
|
|
*/
|
|
#ifndef _FW_TABLE_H_
|
|
#define _FW_TABLE_H_
|
|
|
|
union acpi_subtable_headers;
|
|
|
|
typedef int (*acpi_tbl_entry_handler)(union acpi_subtable_headers *header,
|
|
const unsigned long end);
|
|
|
|
typedef int (*acpi_tbl_entry_handler_arg)(union acpi_subtable_headers *header,
|
|
void *arg, const unsigned long end);
|
|
|
|
struct acpi_subtable_proc {
|
|
int id;
|
|
acpi_tbl_entry_handler handler;
|
|
acpi_tbl_entry_handler_arg handler_arg;
|
|
void *arg;
|
|
int count;
|
|
};
|
|
|
|
union fw_table_header {
|
|
struct acpi_table_header acpi;
|
|
struct acpi_table_cdat cdat;
|
|
};
|
|
|
|
union acpi_subtable_headers {
|
|
struct acpi_subtable_header common;
|
|
struct acpi_hmat_structure hmat;
|
|
struct acpi_prmt_module_header prmt;
|
|
struct acpi_cedt_header cedt;
|
|
struct acpi_cdat_header cdat;
|
|
};
|
|
|
|
int acpi_parse_entries_array(char *id, unsigned long table_size,
|
|
union fw_table_header *table_header,
|
|
unsigned long max_length,
|
|
struct acpi_subtable_proc *proc,
|
|
int proc_num, unsigned int max_entries);
|
|
|
|
int cdat_table_parse(enum acpi_cdat_type type,
|
|
acpi_tbl_entry_handler_arg handler_arg, void *arg,
|
|
struct acpi_table_cdat *table_header,
|
|
unsigned long length);
|
|
|
|
/* CXL is the only non-ACPI consumer of the FIRMWARE_TABLE library */
|
|
#if IS_ENABLED(CONFIG_ACPI) && !IS_ENABLED(CONFIG_CXL_BUS)
|
|
#define EXPORT_SYMBOL_FWTBL_LIB(x) EXPORT_SYMBOL_ACPI_LIB(x)
|
|
#define __init_or_fwtbl_lib __init_or_acpilib
|
|
#else
|
|
#define EXPORT_SYMBOL_FWTBL_LIB(x) EXPORT_SYMBOL_NS_GPL(x, CXL)
|
|
#define __init_or_fwtbl_lib
|
|
#endif
|
|
|
|
#endif
|