Message ID | 20221208052115.800170-2-ira.weiny@intel.com |
---|---|
State | Superseded |
Headers | show |
Series | CXL: Process event logs | expand |
ira.weiny@ wrote: > From: Ira Weiny <ira.weiny@intel.com> > > CXL devices have multiple event logs which can be queried for CXL event > records. Devices are required to support the storage of at least one > event record in each event log type. > > Devices track event log overflow by incrementing a counter and tracking > the time of the first and last overflow event seen. > > Software queries events via the Get Event Record mailbox command; CXL > rev 3.0 section 8.2.9.2.2 and clears events via CXL rev 3.0 section > 8.2.9.2.3 Clear Event Records mailbox command. > > CXL _OSC Error Reporting Control is used by the OS to determine if > Firmware has control of various error reporting capabilities including > the event logs. > > Expose the result of negotiating CXL Error Reporting Control in struct > pci_host_bridge for consumption by the CXL drivers. If support is > controlled by the OS read and clear all event logs on driver load. > > Ensure a clean slate of events by reading and clearing the events on > driver load. The operation is performed twice to ensure that any prior > partial readings are completed and a fresh read from the start is done > at least one time. This is done even if rogue reads cause clear errors. > > The status register is not used because a device may continue to trigger > events and the only requirement is to empty the log at least once. This > allows for the required transition from empty to non-empty for interrupt > generation. Handling of interrupts is in a follow on patch. > > The device can return up to 1MB worth of event records per query. > Allocate a shared large buffer to handle the max number of records based > on the mailbox payload size. > > This patch traces a raw event record and leaves specific event record > type tracing to subsequent patches. Macros are created to aid in > tracing the common CXL Event header fields. > > Each record is cleared explicitly. A clear all bit is specified but is > only valid when the log overflows. > > Signed-off-by: Ira Weiny <ira.weiny@intel.com> > > --- > Changes from V2: > Rebased on 6.3 pending changes > Move cxl_mem_alloc_event_buf() to pci.c > Define and use CXLDEV_EVENT_STATUS_ALL > Fix error flow on clear failure > Remove tags > Jonathan/Dan > Add in OSC Error Reporting Control check > Dan (Jonathan in previous version) > Squash Clear events and the driver load patch into this one. > Dan > Make event device status a separate structure > Move tracing to within cxl core > Reduce clear double loop to a single loop > Pass struct device to trace points > Adjust to new cxl_internal_send_cmd() > Query error logs in order of severity fatal -> Info > Remove uapi defines entirely > pass total via get_pl > fix 'Clearning' spelling > Better clarify event_buf singular allocation > Use decimal for command payload array sizes > Remove trace_*_enabled() optimization > Put GET/CLEAR macros at the end of the user enum to > preserve compatibility > Add Get/Clear Events to kernel exclusive commands > Remove cxl_event_log_type_str() outside of tracing > Add cond_resched() to event log processing > Jonathan > s/event_buf_lock/event_log_lock > Read through all logs two times to ensure partial reads are > covered. > Pass buffer to cxl_mem_free_event_buffer() > kdoc for event buf > Account for cxlds->payload_size limiting the max handles > Don't use min_t as it was used incorrectly > > Changes from V1: > Clear Event Record allows for u8 handles while Get Event Record > allows for u16 records to be returned. Based on Jonathan's > feedback; allow for all event records to be handled in this > clear. Which means a double loop with potentially multiple > Clear Event payloads being sent to clear all events sent. > > Changes from RFC: > Jonathan > Clean up init of payload and use return code. > Also report any error to clear the event. > s/v3.0/rev 3.0 This is a lot of text to skip over. The cover letter has the summary of the changes and a link to v2 and that has the link to v1. Why does each patch need the full history of changes all over again? > > squash: make event device state a separate structure. > --- > drivers/acpi/pci_root.c | 3 + > drivers/cxl/core/mbox.c | 138 +++++++++++++++++++++++++++++++++++++++ > drivers/cxl/core/trace.h | 120 ++++++++++++++++++++++++++++++++++ > drivers/cxl/cxl.h | 12 ++++ > drivers/cxl/cxlmem.h | 84 ++++++++++++++++++++++++ > drivers/cxl/pci.c | 42 ++++++++++++ > include/linux/pci.h | 1 + > 7 files changed, 400 insertions(+) > > diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c > index b3c202d2a433..cee8f56fb63a 100644 > --- a/drivers/acpi/pci_root.c > +++ b/drivers/acpi/pci_root.c I do appreciate that this patch has a full thought about error handling implementing the full cycle of retrieve and clear, but I think these drivers/acpi/ changes can stand on their own as a lead-in patch if only so that Bjorn and Rafael do not need to dig through the CXL details. > @@ -1047,6 +1047,9 @@ struct pci_bus *acpi_pci_root_create(struct acpi_pci_root *root, > if (!(root->osc_control_set & OSC_PCI_EXPRESS_DPC_CONTROL)) > host_bridge->native_dpc = 0; > > + if (root->osc_ext_control_set & OSC_CXL_ERROR_REPORTING_CONTROL) > + host_bridge->native_cxl_error = 1; > + Copy the existing style and init the flag to 1 in pci_init_host_bridge() and clear it in acpi_pci_root_create() upon fail to take control. > /* > * Evaluate the "PCI Boot Configuration" _DSM Function. If it > * exists and returns 0, we must preserve any PCI resource > diff --git a/drivers/cxl/core/mbox.c b/drivers/cxl/core/mbox.c > index b03fba212799..815da3aac081 100644 > --- a/drivers/cxl/core/mbox.c > +++ b/drivers/cxl/core/mbox.c > @@ -8,6 +8,7 @@ > #include <cxl.h> > > #include "core.h" > +#include "trace.h" > > static bool cxl_raw_allow_all; > > @@ -717,6 +718,142 @@ int cxl_enumerate_cmds(struct cxl_dev_state *cxlds) > } > EXPORT_SYMBOL_NS_GPL(cxl_enumerate_cmds, CXL); > > +static int cxl_clear_event_record(struct cxl_dev_state *cxlds, > + enum cxl_event_log_type log, > + struct cxl_get_event_payload *get_pl) > +{ > + struct cxl_mbox_clear_event_payload payload = { > + .event_log = log, > + }; > + u16 total = le16_to_cpu(get_pl->record_count); > + u8 max_handles = CXL_CLEAR_EVENT_MAX_HANDLES; > + size_t pl_size = sizeof(payload); > + struct cxl_mbox_cmd mbox_cmd; > + u16 cnt; > + int rc; > + int i; > + > + /* Payload size may limit the max handles */ > + if (pl_size > cxlds->payload_size) { > + max_handles = CXL_CLEAR_EVENT_LIMIT_HANDLES(cxlds->payload_size); > + pl_size = cxlds->payload_size; > + } > + > + mbox_cmd = (struct cxl_mbox_cmd) { > + .opcode = CXL_MBOX_OP_CLEAR_EVENT_RECORD, > + .payload_in = &payload, > + .size_in = pl_size, > + }; > + > + /* > + * Clear Event Records uses u8 for the handle cnt while Get Event > + * Record can return up to 0xffff records. > + */ > + i = 0; > + for (cnt = 0; cnt < total; cnt++) { > + payload.handle[i++] = get_pl->records[cnt].hdr.handle; > + dev_dbg(cxlds->dev, "Event log '%d': Clearing %u\n", > + log, le16_to_cpu(payload.handle[i])); > + > + if (i == max_handles) { > + payload.nr_recs = i; > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > + if (rc) > + return rc; > + i = 0; > + } > + } > + > + /* Clear what is left if any */ > + if (i) { > + payload.nr_recs = i; > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > + if (rc) > + return rc; > + } > + > + return 0; > +} > + > +static void cxl_mem_get_records_log(struct cxl_dev_state *cxlds, > + enum cxl_event_log_type type) > +{ > + struct cxl_get_event_payload *payload; > + struct cxl_mbox_cmd mbox_cmd; > + u8 log_type = type; > + u16 nr_rec; > + > + mutex_lock(&cxlds->event.log_lock); > + payload = cxlds->event.buf; > + > + mbox_cmd = (struct cxl_mbox_cmd) { > + .opcode = CXL_MBOX_OP_GET_EVENT_RECORD, > + .payload_in = &log_type, > + .size_in = sizeof(log_type), > + .payload_out = payload, > + .size_out = cxlds->payload_size, > + .min_out = struct_size(payload, records, 0), > + }; > + > + do { > + int rc; > + > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > + if (rc) { > + dev_err(cxlds->dev, "Event log '%d': Failed to query event records : %d", > + type, rc); > + goto unlock_buffer; > + } > + > + nr_rec = le16_to_cpu(payload->record_count); > + if (nr_rec > 0) { Can save some indentation here and just do: if (!nr_rec) break; ...and then out-indent this below: > + int i; > + > + for (i = 0; i < nr_rec; i++) > + trace_cxl_generic_event(cxlds->dev, type, > + &payload->records[i]); > + > + rc = cxl_clear_event_record(cxlds, type, payload); > + if (rc) { > + dev_err(cxlds->dev, "Event log '%d': Failed to clear events : %d", > + type, rc); This and the other dev_err() above should be dev_err_ratelimited() because if these ever fire they'll probably start firing in bunches. > + goto unlock_buffer; Nit, but how about just "break;" here? No need for a label. > + } > + } > + > + if (payload->flags & CXL_GET_EVENT_FLAG_OVERFLOW) > + trace_cxl_overflow(cxlds->dev, type, payload); I'm worried about this potentially looping and causing softlockups without a cond_resched(), but lets wait and see if more complexity is needed. > + } while (nr_rec); > + > +unlock_buffer: > + mutex_unlock(&cxlds->event.log_lock); > +} > + > +/** > + * cxl_mem_get_event_records - Get Event Records from the device > + * @cxlds: The device data for the operation > + * > + * Retrieve all event records available on the device, report them as trace > + * events, and clear them. > + * > + * See CXL rev 3.0 @8.2.9.2.2 Get Event Records > + * See CXL rev 3.0 @8.2.9.2.3 Clear Event Records > + */ > +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status) > +{ > + dev_dbg(cxlds->dev, "Reading event logs: %x\n", status); > + > + if (status & CXLDEV_EVENT_STATUS_FATAL) > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FATAL); > + if (status & CXLDEV_EVENT_STATUS_FAIL) > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FAIL); > + if (status & CXLDEV_EVENT_STATUS_WARN) > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_WARN); > + if (status & CXLDEV_EVENT_STATUS_INFO) > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_INFO); > +} > +EXPORT_SYMBOL_NS_GPL(cxl_mem_get_event_records, CXL); > + > /** > * cxl_mem_get_partition_info - Get partition info > * @cxlds: The device data for the operation > @@ -868,6 +1005,7 @@ struct cxl_dev_state *cxl_dev_state_create(struct device *dev) > } > > mutex_init(&cxlds->mbox_mutex); > + mutex_init(&cxlds->event.log_lock); > cxlds->dev = dev; > > return cxlds; > diff --git a/drivers/cxl/core/trace.h b/drivers/cxl/core/trace.h > index 20ca2fe2ca8e..24eef6909f13 100644 > --- a/drivers/cxl/core/trace.h > +++ b/drivers/cxl/core/trace.h > @@ -6,7 +6,9 @@ > #if !defined(_CXL_EVENTS_H) || defined(TRACE_HEADER_MULTI_READ) > #define _CXL_EVENTS_H > > +#include <asm-generic/unaligned.h> > #include <cxl.h> > +#include <cxlmem.h> > #include <linux/tracepoint.h> > > #define CXL_RAS_UC_CACHE_DATA_PARITY BIT(0) > @@ -103,6 +105,124 @@ TRACE_EVENT(cxl_aer_correctable_error, > ) > ); > > +#include <linux/tracepoint.h> > + > +#define cxl_event_log_type_str(type) \ > + __print_symbolic(type, \ > + { CXL_EVENT_TYPE_INFO, "Informational" }, \ > + { CXL_EVENT_TYPE_WARN, "Warning" }, \ > + { CXL_EVENT_TYPE_FAIL, "Failure" }, \ > + { CXL_EVENT_TYPE_FATAL, "Fatal" }) > + > +TRACE_EVENT(cxl_overflow, > + > + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, > + struct cxl_get_event_payload *payload), > + > + TP_ARGS(dev, log, payload), > + > + TP_STRUCT__entry( > + __string(dev_name, dev_name(dev)) > + __field(int, log) > + __field(u64, first_ts) > + __field(u64, last_ts) > + __field(u16, count) > + ), > + > + TP_fast_assign( > + __assign_str(dev_name, dev_name(dev)); > + __entry->log = log; > + __entry->count = le16_to_cpu(payload->overflow_err_count); > + __entry->first_ts = le64_to_cpu(payload->first_overflow_timestamp); > + __entry->last_ts = le64_to_cpu(payload->last_overflow_timestamp); > + ), > + > + TP_printk("%s: EVENT LOG OVERFLOW log=%s : %u records from %llu to %llu", The tracepoint is already called "cxl_overflow", seems redundant to also print "EVENT LOG OVERFLOW". > + __get_str(dev_name), cxl_event_log_type_str(__entry->log), > + __entry->count, __entry->first_ts, __entry->last_ts) > + > +); > + > +/* > + * Common Event Record Format > + * CXL 3.0 section 8.2.9.2.1; Table 8-42 > + */ > +#define CXL_EVENT_RECORD_FLAG_PERMANENT BIT(2) > +#define CXL_EVENT_RECORD_FLAG_MAINT_NEEDED BIT(3) > +#define CXL_EVENT_RECORD_FLAG_PERF_DEGRADED BIT(4) > +#define CXL_EVENT_RECORD_FLAG_HW_REPLACE BIT(5) > +#define show_hdr_flags(flags) __print_flags(flags, " | ", \ > + { CXL_EVENT_RECORD_FLAG_PERMANENT, "PERMANENT_CONDITION" }, \ > + { CXL_EVENT_RECORD_FLAG_MAINT_NEEDED, "MAINTENANCE_NEEDED" }, \ > + { CXL_EVENT_RECORD_FLAG_PERF_DEGRADED, "PERFORMANCE_DEGRADED" }, \ > + { CXL_EVENT_RECORD_FLAG_HW_REPLACE, "HARDWARE_REPLACEMENT_NEEDED" } \ Nit, I know this does not matter for parsing since tooling will use the raw TP_STRUCT__entry fields, but why are these "CAPITAL_UNDERSCORE" and other symbols are just "Capitalized term"? > +) > + > +/* > + * Define macros for the common header of each CXL event. > + * > + * Tracepoints using these macros must do 3 things: > + * > + * 1) Add CXL_EVT_TP_entry to TP_STRUCT__entry > + * 2) Use CXL_EVT_TP_fast_assign within TP_fast_assign; > + * pass the dev, log, and CXL event header > + * 3) Use CXL_EVT_TP_printk() instead of TP_printk() > + * > + * See the generic_event tracepoint as an example. > + */ > +#define CXL_EVT_TP_entry \ > + __string(dev_name, dev_name(dev)) \ > + __field(int, log) \ > + __field_struct(uuid_t, hdr_uuid) \ > + __field(u32, hdr_flags) \ > + __field(u16, hdr_handle) \ > + __field(u16, hdr_related_handle) \ > + __field(u64, hdr_timestamp) \ > + __field(u8, hdr_length) \ > + __field(u8, hdr_maint_op_class) > + > +#define CXL_EVT_TP_fast_assign(dev, l, hdr) \ > + __assign_str(dev_name, dev_name(dev)); \ > + __entry->log = (l); \ > + memcpy(&__entry->hdr_uuid, &(hdr).id, sizeof(uuid_t)); \ > + __entry->hdr_length = (hdr).length; \ > + __entry->hdr_flags = get_unaligned_le24((hdr).flags); \ > + __entry->hdr_handle = le16_to_cpu((hdr).handle); \ > + __entry->hdr_related_handle = le16_to_cpu((hdr).related_handle); \ > + __entry->hdr_timestamp = le64_to_cpu((hdr).timestamp); \ > + __entry->hdr_maint_op_class = (hdr).maint_op_class > + > +#define CXL_EVT_TP_printk(fmt, ...) \ > + TP_printk("%s log=%s : time=%llu uuid=%pUb len=%d flags='%s' " \ > + "handle=%x related_handle=%x maint_op_class=%u" \ > + " : " fmt, \ > + __get_str(dev_name), cxl_event_log_type_str(__entry->log), \ > + __entry->hdr_timestamp, &__entry->hdr_uuid, __entry->hdr_length,\ > + show_hdr_flags(__entry->hdr_flags), __entry->hdr_handle, \ > + __entry->hdr_related_handle, __entry->hdr_maint_op_class, \ > + ##__VA_ARGS__) > + > +TRACE_EVENT(cxl_generic_event, > + > + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, > + struct cxl_event_record_raw *rec), > + > + TP_ARGS(dev, log, rec), > + > + TP_STRUCT__entry( > + CXL_EVT_TP_entry > + __array(u8, data, CXL_EVENT_RECORD_DATA_LENGTH) > + ), > + > + TP_fast_assign( > + CXL_EVT_TP_fast_assign(dev, log, rec->hdr); > + memcpy(__entry->data, &rec->data, CXL_EVENT_RECORD_DATA_LENGTH); > + ), > + > + CXL_EVT_TP_printk("%s", > + __print_hex(__entry->data, CXL_EVENT_RECORD_DATA_LENGTH)) > +); > + > #endif /* _CXL_EVENTS_H */ > > #define TRACE_INCLUDE_FILE trace > diff --git a/drivers/cxl/cxl.h b/drivers/cxl/cxl.h > index aa3af3bb73b2..5974d1082210 100644 > --- a/drivers/cxl/cxl.h > +++ b/drivers/cxl/cxl.h > @@ -156,6 +156,18 @@ static inline int ways_to_eiw(unsigned int ways, u8 *eiw) > #define CXLDEV_CAP_CAP_ID_SECONDARY_MAILBOX 0x3 > #define CXLDEV_CAP_CAP_ID_MEMDEV 0x4000 > > +/* CXL 3.0 8.2.8.3.1 Event Status Register */ > +#define CXLDEV_DEV_EVENT_STATUS_OFFSET 0x00 > +#define CXLDEV_EVENT_STATUS_INFO BIT(0) > +#define CXLDEV_EVENT_STATUS_WARN BIT(1) > +#define CXLDEV_EVENT_STATUS_FAIL BIT(2) > +#define CXLDEV_EVENT_STATUS_FATAL BIT(3) > + > +#define CXLDEV_EVENT_STATUS_ALL (CXLDEV_EVENT_STATUS_INFO | \ > + CXLDEV_EVENT_STATUS_WARN | \ > + CXLDEV_EVENT_STATUS_FAIL | \ > + CXLDEV_EVENT_STATUS_FATAL) > + > /* CXL 2.0 8.2.8.4 Mailbox Registers */ > #define CXLDEV_MBOX_CAPS_OFFSET 0x00 > #define CXLDEV_MBOX_CAP_PAYLOAD_SIZE_MASK GENMASK(4, 0) > diff --git a/drivers/cxl/cxlmem.h b/drivers/cxl/cxlmem.h > index ab138004f644..dd9aa3dd738e 100644 > --- a/drivers/cxl/cxlmem.h > +++ b/drivers/cxl/cxlmem.h > @@ -4,6 +4,7 @@ > #define __CXL_MEM_H__ > #include <uapi/linux/cxl_mem.h> > #include <linux/cdev.h> > +#include <linux/uuid.h> > #include "cxl.h" > > /* CXL 2.0 8.2.8.5.1.1 Memory Device Status Register */ > @@ -193,6 +194,17 @@ struct cxl_endpoint_dvsec_info { > struct range dvsec_range[2]; > }; > > +/** > + * struct cxl_event_state - Event log driver state > + * > + * @event_buf: Buffer to receive event data > + * @event_log_lock: Serialize event_buf and log use > + */ > +struct cxl_event_state { > + struct cxl_get_event_payload *buf; > + struct mutex log_lock; > +}; > + > /** > * struct cxl_dev_state - The driver device state > * > @@ -266,12 +278,16 @@ struct cxl_dev_state { > > struct xarray doe_mbs; > > + struct cxl_event_state event; > + > int (*mbox_send)(struct cxl_dev_state *cxlds, struct cxl_mbox_cmd *cmd); > }; > > enum cxl_opcode { > CXL_MBOX_OP_INVALID = 0x0000, > CXL_MBOX_OP_RAW = CXL_MBOX_OP_INVALID, > + CXL_MBOX_OP_GET_EVENT_RECORD = 0x0100, > + CXL_MBOX_OP_CLEAR_EVENT_RECORD = 0x0101, > CXL_MBOX_OP_GET_FW_INFO = 0x0200, > CXL_MBOX_OP_ACTIVATE_FW = 0x0202, > CXL_MBOX_OP_GET_SUPPORTED_LOGS = 0x0400, > @@ -347,6 +363,73 @@ struct cxl_mbox_identify { > u8 qos_telemetry_caps; > } __packed; > > +/* > + * Common Event Record Format > + * CXL rev 3.0 section 8.2.9.2.1; Table 8-42 > + */ > +struct cxl_event_record_hdr { > + uuid_t id; > + u8 length; > + u8 flags[3]; > + __le16 handle; > + __le16 related_handle; > + __le64 timestamp; > + u8 maint_op_class; > + u8 reserved[15]; > +} __packed; > + > +#define CXL_EVENT_RECORD_DATA_LENGTH 0x50 > +struct cxl_event_record_raw { > + struct cxl_event_record_hdr hdr; > + u8 data[CXL_EVENT_RECORD_DATA_LENGTH]; > +} __packed; > + > +/* > + * Get Event Records output payload > + * CXL rev 3.0 section 8.2.9.2.2; Table 8-50 > + */ > +#define CXL_GET_EVENT_FLAG_OVERFLOW BIT(0) > +#define CXL_GET_EVENT_FLAG_MORE_RECORDS BIT(1) > +struct cxl_get_event_payload { > + u8 flags; > + u8 reserved1; > + __le16 overflow_err_count; > + __le64 first_overflow_timestamp; > + __le64 last_overflow_timestamp; > + __le16 record_count; > + u8 reserved2[10]; > + struct cxl_event_record_raw records[]; > +} __packed; > + > +/* > + * CXL rev 3.0 section 8.2.9.2.2; Table 8-49 > + */ > +enum cxl_event_log_type { > + CXL_EVENT_TYPE_INFO = 0x00, > + CXL_EVENT_TYPE_WARN, > + CXL_EVENT_TYPE_FAIL, > + CXL_EVENT_TYPE_FATAL, > + CXL_EVENT_TYPE_MAX > +}; > + > +/* > + * Clear Event Records input payload > + * CXL rev 3.0 section 8.2.9.2.3; Table 8-51 > + */ > +#define CXL_CLEAR_EVENT_MAX_HANDLES (0xff) > +struct cxl_mbox_clear_event_payload { > + u8 event_log; /* enum cxl_event_log_type */ > + u8 clear_flags; > + u8 nr_recs; > + u8 reserved[3]; > + __le16 handle[CXL_CLEAR_EVENT_MAX_HANDLES]; > +} __packed; > +#define CXL_CLEAR_EVENT_LIMIT_HANDLES(payload_size) \ > + (((payload_size) - \ > + (sizeof(struct cxl_mbox_clear_event_payload) - \ > + (sizeof(__le16) * CXL_CLEAR_EVENT_MAX_HANDLES))) / \ > + sizeof(__le16)) > + > struct cxl_mbox_get_partition_info { > __le64 active_volatile_cap; > __le64 active_persistent_cap; > @@ -441,6 +524,7 @@ int cxl_mem_create_range_info(struct cxl_dev_state *cxlds); > struct cxl_dev_state *cxl_dev_state_create(struct device *dev); > void set_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); > void clear_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); > +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status); > #ifdef CONFIG_CXL_SUSPEND > void cxl_mem_active_inc(void); > void cxl_mem_active_dec(void); > diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c > index 3a66aadb4df0..86c84611a168 100644 > --- a/drivers/cxl/pci.c > +++ b/drivers/cxl/pci.c > @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) > pci_disable_pcie_error_reporting(pdev); > } > > +static void cxl_mem_free_event_buffer(void *buf) > +{ > + kvfree(buf); > +} > + > +/* > + * There is a single buffer for reading event logs from the mailbox. All logs > + * share this buffer protected by the cxlds->event_log_lock. > + */ > +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) > +{ > + struct cxl_get_event_payload *buf; > + > + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", > + cxlds->payload_size); > + > + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); > + if (WARN_ON_ONCE(!buf)) No, why is event init so special that it behaves differently than all the other init-time allocations this driver does? > + return; return -ENOMEM; > + > + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, > + cxl_mem_free_event_buffer, buf))) > + return; ditto. > + > + cxlds->event.buf = buf; > +} > + > +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) > +{ > + /* Force read and clear of all logs */ > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > + /* Ensure prior partial reads are handled, by starting over again */ What partial reads? cxl_mem_get_event_records() reads every log until each returns an empty result. Any remaining events after this returns are events that fired during the retrieval. So I do not think cxl_clear_event_logs() needs to exist, just call cxl_mem_get_event_records(CXLDEV_EVENT_STATUS_ALL) once and that's it. > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > +} > + > static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) > { > + struct pci_host_bridge *host_bridge = pci_find_host_bridge(pdev->bus); > struct cxl_register_map map; > struct cxl_memdev *cxlmd; > struct cxl_dev_state *cxlds; > @@ -494,6 +530,12 @@ static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) > if (IS_ERR(cxlmd)) > return PTR_ERR(cxlmd); > > + if (host_bridge->native_cxl_error) { Probably deserves a small comment about why this flag matters for event init. Something like: "When BIOS maintains CXL error reporting control, it will also reap event records. Only one agent can interface with the event mechanism." > + cxl_mem_alloc_event_buf(cxlds); > + if (cxlds->event.buf) No need for this conditional here since this whole block is skipped if !native_cxl_error and cxl_mem_alloc_event_buf() will fail driver load if it fails to init. > + cxl_clear_event_logs(cxlds); > + } > + > if (cxlds->regs.ras) { > pci_enable_pcie_error_reporting(pdev); > rc = devm_add_action_or_reset(&pdev->dev, disable_aer, pdev); > diff --git a/include/linux/pci.h b/include/linux/pci.h > index 1f81807492ef..7fe3752a204e 100644 > --- a/include/linux/pci.h > +++ b/include/linux/pci.h > @@ -580,6 +580,7 @@ struct pci_host_bridge { > unsigned int preserve_config:1; /* Preserve FW resource setup */ > unsigned int size_windows:1; /* Enable root bus sizing */ > unsigned int msi_domain:1; /* Bridge wants MSI domain */ > + unsigned int native_cxl_error:1; /* OS CXL Error reporting */ I would group this with the other native_ flags and copy the comment style "OS may use CXL RAS + Events", or somesuch.
On Fri, Dec 09, 2022 at 09:56:49AM -0800, Dan Williams wrote: > ira.weiny@ wrote: > > From: Ira Weiny <ira.weiny@intel.com> > > > > CXL devices have multiple event logs which can be queried for CXL event > > records. Devices are required to support the storage of at least one > > event record in each event log type. > > > > Devices track event log overflow by incrementing a counter and tracking > > the time of the first and last overflow event seen. > > > > Software queries events via the Get Event Record mailbox command; CXL > > rev 3.0 section 8.2.9.2.2 and clears events via CXL rev 3.0 section > > 8.2.9.2.3 Clear Event Records mailbox command. > > > > CXL _OSC Error Reporting Control is used by the OS to determine if > > Firmware has control of various error reporting capabilities including > > the event logs. > > > > Expose the result of negotiating CXL Error Reporting Control in struct > > pci_host_bridge for consumption by the CXL drivers. If support is > > controlled by the OS read and clear all event logs on driver load. > > > > Ensure a clean slate of events by reading and clearing the events on > > driver load. The operation is performed twice to ensure that any prior > > partial readings are completed and a fresh read from the start is done > > at least one time. This is done even if rogue reads cause clear errors. > > > > The status register is not used because a device may continue to trigger > > events and the only requirement is to empty the log at least once. This > > allows for the required transition from empty to non-empty for interrupt > > generation. Handling of interrupts is in a follow on patch. > > > > The device can return up to 1MB worth of event records per query. > > Allocate a shared large buffer to handle the max number of records based > > on the mailbox payload size. > > > > This patch traces a raw event record and leaves specific event record > > type tracing to subsequent patches. Macros are created to aid in > > tracing the common CXL Event header fields. > > > > Each record is cleared explicitly. A clear all bit is specified but is > > only valid when the log overflows. > > > > Signed-off-by: Ira Weiny <ira.weiny@intel.com> > > > > --- > > Changes from V2: > > Rebased on 6.3 pending changes > > Move cxl_mem_alloc_event_buf() to pci.c > > Define and use CXLDEV_EVENT_STATUS_ALL > > Fix error flow on clear failure > > Remove tags > > Jonathan/Dan > > Add in OSC Error Reporting Control check > > Dan (Jonathan in previous version) > > Squash Clear events and the driver load patch into this one. > > Dan > > Make event device status a separate structure > > Move tracing to within cxl core > > Reduce clear double loop to a single loop > > Pass struct device to trace points > > Adjust to new cxl_internal_send_cmd() > > Query error logs in order of severity fatal -> Info > > Remove uapi defines entirely > > pass total via get_pl > > fix 'Clearning' spelling > > Better clarify event_buf singular allocation > > Use decimal for command payload array sizes > > Remove trace_*_enabled() optimization > > Put GET/CLEAR macros at the end of the user enum to > > preserve compatibility > > Add Get/Clear Events to kernel exclusive commands > > Remove cxl_event_log_type_str() outside of tracing > > Add cond_resched() to event log processing > > Jonathan > > s/event_buf_lock/event_log_lock > > Read through all logs two times to ensure partial reads are > > covered. > > Pass buffer to cxl_mem_free_event_buffer() > > kdoc for event buf > > Account for cxlds->payload_size limiting the max handles > > Don't use min_t as it was used incorrectly > > > > Changes from V1: > > Clear Event Record allows for u8 handles while Get Event Record > > allows for u16 records to be returned. Based on Jonathan's > > feedback; allow for all event records to be handled in this > > clear. Which means a double loop with potentially multiple > > Clear Event payloads being sent to clear all events sent. > > > > Changes from RFC: > > Jonathan > > Clean up init of payload and use return code. > > Also report any error to clear the event. > > s/v3.0/rev 3.0 > > This is a lot of text to skip over. The cover letter has the summary of > the changes and a link to v2 and that has the link to v1. Why does each > patch need the full history of changes all over again? I've seen this done elsewhere and just copied it. But I think the use of b4 send will help me in future. I agree it is unneeded. Even in the cover letter. > > > > > squash: make event device state a separate structure. > > --- > > drivers/acpi/pci_root.c | 3 + > > drivers/cxl/core/mbox.c | 138 +++++++++++++++++++++++++++++++++++++++ > > drivers/cxl/core/trace.h | 120 ++++++++++++++++++++++++++++++++++ > > drivers/cxl/cxl.h | 12 ++++ > > drivers/cxl/cxlmem.h | 84 ++++++++++++++++++++++++ > > drivers/cxl/pci.c | 42 ++++++++++++ > > include/linux/pci.h | 1 + > > 7 files changed, 400 insertions(+) > > > > diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c > > index b3c202d2a433..cee8f56fb63a 100644 > > --- a/drivers/acpi/pci_root.c > > +++ b/drivers/acpi/pci_root.c > > I do appreciate that this patch has a full thought about error handling > implementing the full cycle of retrieve and clear, but I think these > drivers/acpi/ changes can stand on their own as a lead-in patch if only > so that Bjorn and Rafael do not need to dig through the CXL details. Split it out. That was my inclination anyway. ;-) > > > > @@ -1047,6 +1047,9 @@ struct pci_bus *acpi_pci_root_create(struct acpi_pci_root *root, > > if (!(root->osc_control_set & OSC_PCI_EXPRESS_DPC_CONTROL)) > > host_bridge->native_dpc = 0; > > > > + if (root->osc_ext_control_set & OSC_CXL_ERROR_REPORTING_CONTROL) > > + host_bridge->native_cxl_error = 1; > > + > > Copy the existing style and init the flag to 1 in pci_init_host_bridge() > and clear it in acpi_pci_root_create() upon fail to take control. Ah good point. Done. > > > /* > > * Evaluate the "PCI Boot Configuration" _DSM Function. If it > > * exists and returns 0, we must preserve any PCI resource > > diff --git a/drivers/cxl/core/mbox.c b/drivers/cxl/core/mbox.c > > index b03fba212799..815da3aac081 100644 > > --- a/drivers/cxl/core/mbox.c > > +++ b/drivers/cxl/core/mbox.c > > @@ -8,6 +8,7 @@ > > #include <cxl.h> > > > > #include "core.h" > > +#include "trace.h" > > > > static bool cxl_raw_allow_all; > > > > @@ -717,6 +718,142 @@ int cxl_enumerate_cmds(struct cxl_dev_state *cxlds) > > } > > EXPORT_SYMBOL_NS_GPL(cxl_enumerate_cmds, CXL); > > > > +static int cxl_clear_event_record(struct cxl_dev_state *cxlds, > > + enum cxl_event_log_type log, > > + struct cxl_get_event_payload *get_pl) > > +{ > > + struct cxl_mbox_clear_event_payload payload = { > > + .event_log = log, > > + }; > > + u16 total = le16_to_cpu(get_pl->record_count); > > + u8 max_handles = CXL_CLEAR_EVENT_MAX_HANDLES; > > + size_t pl_size = sizeof(payload); > > + struct cxl_mbox_cmd mbox_cmd; > > + u16 cnt; > > + int rc; > > + int i; > > + > > + /* Payload size may limit the max handles */ > > + if (pl_size > cxlds->payload_size) { > > + max_handles = CXL_CLEAR_EVENT_LIMIT_HANDLES(cxlds->payload_size); > > + pl_size = cxlds->payload_size; > > + } > > + > > + mbox_cmd = (struct cxl_mbox_cmd) { > > + .opcode = CXL_MBOX_OP_CLEAR_EVENT_RECORD, > > + .payload_in = &payload, > > + .size_in = pl_size, > > + }; > > + > > + /* > > + * Clear Event Records uses u8 for the handle cnt while Get Event > > + * Record can return up to 0xffff records. > > + */ > > + i = 0; > > + for (cnt = 0; cnt < total; cnt++) { > > + payload.handle[i++] = get_pl->records[cnt].hdr.handle; > > + dev_dbg(cxlds->dev, "Event log '%d': Clearing %u\n", > > + log, le16_to_cpu(payload.handle[i])); > > + > > + if (i == max_handles) { > > + payload.nr_recs = i; > > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > > + if (rc) > > + return rc; > > + i = 0; > > + } > > + } > > + > > + /* Clear what is left if any */ > > + if (i) { > > + payload.nr_recs = i; > > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > > + if (rc) > > + return rc; > > + } > > + > > + return 0; > > +} > > + > > +static void cxl_mem_get_records_log(struct cxl_dev_state *cxlds, > > + enum cxl_event_log_type type) > > +{ > > + struct cxl_get_event_payload *payload; > > + struct cxl_mbox_cmd mbox_cmd; > > + u8 log_type = type; > > + u16 nr_rec; > > + > > + mutex_lock(&cxlds->event.log_lock); > > + payload = cxlds->event.buf; > > + > > + mbox_cmd = (struct cxl_mbox_cmd) { > > + .opcode = CXL_MBOX_OP_GET_EVENT_RECORD, > > + .payload_in = &log_type, > > + .size_in = sizeof(log_type), > > + .payload_out = payload, > > + .size_out = cxlds->payload_size, > > + .min_out = struct_size(payload, records, 0), > > + }; > > + > > + do { > > + int rc; > > + > > + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); > > + if (rc) { > > + dev_err(cxlds->dev, "Event log '%d': Failed to query event records : %d", > > + type, rc); > > + goto unlock_buffer; > > + } > > + > > + nr_rec = le16_to_cpu(payload->record_count); > > + if (nr_rec > 0) { > > Can save some indentation here and just do: > > if (!nr_rec) > break; > > ...and then out-indent this below: :-/ I suppose that does make sense. Somehow I thought the overflow flag might be set here. But that should not be the case if nr_rec == 0. > > > + int i; > > + > > + for (i = 0; i < nr_rec; i++) > > + trace_cxl_generic_event(cxlds->dev, type, > > + &payload->records[i]); > > + > > + rc = cxl_clear_event_record(cxlds, type, payload); > > + if (rc) { > > + dev_err(cxlds->dev, "Event log '%d': Failed to clear events : %d", > > + type, rc); > > This and the other dev_err() above should be dev_err_ratelimited() because if > these ever fire they'll probably start firing in bunches. Yea good catch. Previously, exited from the loop and would effectively stop. But now there is a loop on the status and this will keep printing. Effectively this may be forever because if something is wrong and the log is not clearing out the status won't clear... :-/ Thinking about this I was tempted to make this dev_err_once(). But there are some error which would correct themselves and could make progress. FWIW I know you mentioned that this error could be a trace point as well. But I opted not to do that because if no one is listening to the trace points and there is an error the admin will have no idea this device is going haywire for apparently no reason. This is especially true on start up. > > > + goto unlock_buffer; > > Nit, but how about just "break;" here? No need for a label. Just my style. The goto is more explicit. Especially with this much logic in the while loop. Also an 'unlock_buffer' label is pretty standard as the 'exception' case. > > > + } > > + } > > + > > + if (payload->flags & CXL_GET_EVENT_FLAG_OVERFLOW) > > + trace_cxl_overflow(cxlds->dev, type, payload); > > I'm worried about this potentially looping and causing softlockups > without a cond_resched(), but lets wait and see if more complexity is > needed. I did have a cond_resched() here but opted to have it at the status level rather than here. So reading of _all_ the logs will not loop without a pause. This was partly because I want this loop to drop out to read the other logs and I'm worried that if we delay here we may end up never empting the individual logs. Of course then we have other problems too. So... yea I'd like to wait and see. I feel like the OS should be able to keep up with normal errors. > > > + } while (nr_rec); > > + > > +unlock_buffer: > > + mutex_unlock(&cxlds->event.log_lock); > > +} > > + > > +/** > > + * cxl_mem_get_event_records - Get Event Records from the device > > + * @cxlds: The device data for the operation > > + * > > + * Retrieve all event records available on the device, report them as trace > > + * events, and clear them. > > + * > > + * See CXL rev 3.0 @8.2.9.2.2 Get Event Records > > + * See CXL rev 3.0 @8.2.9.2.3 Clear Event Records > > + */ > > +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status) > > +{ > > + dev_dbg(cxlds->dev, "Reading event logs: %x\n", status); > > + > > + if (status & CXLDEV_EVENT_STATUS_FATAL) > > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FATAL); > > + if (status & CXLDEV_EVENT_STATUS_FAIL) > > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FAIL); > > + if (status & CXLDEV_EVENT_STATUS_WARN) > > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_WARN); > > + if (status & CXLDEV_EVENT_STATUS_INFO) > > + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_INFO); > > +} > > +EXPORT_SYMBOL_NS_GPL(cxl_mem_get_event_records, CXL); > > + > > /** > > * cxl_mem_get_partition_info - Get partition info > > * @cxlds: The device data for the operation > > @@ -868,6 +1005,7 @@ struct cxl_dev_state *cxl_dev_state_create(struct device *dev) > > } > > > > mutex_init(&cxlds->mbox_mutex); > > + mutex_init(&cxlds->event.log_lock); > > cxlds->dev = dev; > > > > return cxlds; > > diff --git a/drivers/cxl/core/trace.h b/drivers/cxl/core/trace.h > > index 20ca2fe2ca8e..24eef6909f13 100644 > > --- a/drivers/cxl/core/trace.h > > +++ b/drivers/cxl/core/trace.h > > @@ -6,7 +6,9 @@ > > #if !defined(_CXL_EVENTS_H) || defined(TRACE_HEADER_MULTI_READ) > > #define _CXL_EVENTS_H > > > > +#include <asm-generic/unaligned.h> > > #include <cxl.h> > > +#include <cxlmem.h> > > #include <linux/tracepoint.h> > > > > #define CXL_RAS_UC_CACHE_DATA_PARITY BIT(0) > > @@ -103,6 +105,124 @@ TRACE_EVENT(cxl_aer_correctable_error, > > ) > > ); > > > > +#include <linux/tracepoint.h> > > + > > +#define cxl_event_log_type_str(type) \ > > + __print_symbolic(type, \ > > + { CXL_EVENT_TYPE_INFO, "Informational" }, \ > > + { CXL_EVENT_TYPE_WARN, "Warning" }, \ > > + { CXL_EVENT_TYPE_FAIL, "Failure" }, \ > > + { CXL_EVENT_TYPE_FATAL, "Fatal" }) > > + > > +TRACE_EVENT(cxl_overflow, > > + > > + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, > > + struct cxl_get_event_payload *payload), > > + > > + TP_ARGS(dev, log, payload), > > + > > + TP_STRUCT__entry( > > + __string(dev_name, dev_name(dev)) > > + __field(int, log) > > + __field(u64, first_ts) > > + __field(u64, last_ts) > > + __field(u16, count) > > + ), > > + > > + TP_fast_assign( > > + __assign_str(dev_name, dev_name(dev)); > > + __entry->log = log; > > + __entry->count = le16_to_cpu(payload->overflow_err_count); > > + __entry->first_ts = le64_to_cpu(payload->first_overflow_timestamp); > > + __entry->last_ts = le64_to_cpu(payload->last_overflow_timestamp); > > + ), > > + > > + TP_printk("%s: EVENT LOG OVERFLOW log=%s : %u records from %llu to %llu", > > The tracepoint is already called "cxl_overflow", seems redundant to also > print "EVENT LOG OVERFLOW". Fair enough. Deleted. > > > + __get_str(dev_name), cxl_event_log_type_str(__entry->log), > > + __entry->count, __entry->first_ts, __entry->last_ts) > > + > > +); > > + > > +/* > > + * Common Event Record Format > > + * CXL 3.0 section 8.2.9.2.1; Table 8-42 > > + */ > > +#define CXL_EVENT_RECORD_FLAG_PERMANENT BIT(2) > > +#define CXL_EVENT_RECORD_FLAG_MAINT_NEEDED BIT(3) > > +#define CXL_EVENT_RECORD_FLAG_PERF_DEGRADED BIT(4) > > +#define CXL_EVENT_RECORD_FLAG_HW_REPLACE BIT(5) > > +#define show_hdr_flags(flags) __print_flags(flags, " | ", \ > > + { CXL_EVENT_RECORD_FLAG_PERMANENT, "PERMANENT_CONDITION" }, \ > > + { CXL_EVENT_RECORD_FLAG_MAINT_NEEDED, "MAINTENANCE_NEEDED" }, \ > > + { CXL_EVENT_RECORD_FLAG_PERF_DEGRADED, "PERFORMANCE_DEGRADED" }, \ > > + { CXL_EVENT_RECORD_FLAG_HW_REPLACE, "HARDWARE_REPLACEMENT_NEEDED" } \ > > Nit, I know this does not matter for parsing since tooling will use the > raw TP_STRUCT__entry fields, but why are these "CAPITAL_UNDERSCORE" and > other symbols are just "Capitalized term"? The conclusion before was that all 'flags' values should be all caps. In addition, because they are flags (vs the other symbols being enums) I used underscores to allow for clear parsing of each flag within the field output. > > > +) > > + > > +/* > > + * Define macros for the common header of each CXL event. > > + * > > + * Tracepoints using these macros must do 3 things: > > + * > > + * 1) Add CXL_EVT_TP_entry to TP_STRUCT__entry > > + * 2) Use CXL_EVT_TP_fast_assign within TP_fast_assign; > > + * pass the dev, log, and CXL event header > > + * 3) Use CXL_EVT_TP_printk() instead of TP_printk() > > + * > > + * See the generic_event tracepoint as an example. > > + */ > > +#define CXL_EVT_TP_entry \ > > + __string(dev_name, dev_name(dev)) \ > > + __field(int, log) \ > > + __field_struct(uuid_t, hdr_uuid) \ > > + __field(u32, hdr_flags) \ > > + __field(u16, hdr_handle) \ > > + __field(u16, hdr_related_handle) \ > > + __field(u64, hdr_timestamp) \ > > + __field(u8, hdr_length) \ > > + __field(u8, hdr_maint_op_class) > > + > > +#define CXL_EVT_TP_fast_assign(dev, l, hdr) \ > > + __assign_str(dev_name, dev_name(dev)); \ > > + __entry->log = (l); \ > > + memcpy(&__entry->hdr_uuid, &(hdr).id, sizeof(uuid_t)); \ > > + __entry->hdr_length = (hdr).length; \ > > + __entry->hdr_flags = get_unaligned_le24((hdr).flags); \ > > + __entry->hdr_handle = le16_to_cpu((hdr).handle); \ > > + __entry->hdr_related_handle = le16_to_cpu((hdr).related_handle); \ > > + __entry->hdr_timestamp = le64_to_cpu((hdr).timestamp); \ > > + __entry->hdr_maint_op_class = (hdr).maint_op_class > > + > > +#define CXL_EVT_TP_printk(fmt, ...) \ > > + TP_printk("%s log=%s : time=%llu uuid=%pUb len=%d flags='%s' " \ > > + "handle=%x related_handle=%x maint_op_class=%u" \ > > + " : " fmt, \ > > + __get_str(dev_name), cxl_event_log_type_str(__entry->log), \ > > + __entry->hdr_timestamp, &__entry->hdr_uuid, __entry->hdr_length,\ > > + show_hdr_flags(__entry->hdr_flags), __entry->hdr_handle, \ > > + __entry->hdr_related_handle, __entry->hdr_maint_op_class, \ > > + ##__VA_ARGS__) > > + > > +TRACE_EVENT(cxl_generic_event, > > + > > + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, > > + struct cxl_event_record_raw *rec), > > + > > + TP_ARGS(dev, log, rec), > > + > > + TP_STRUCT__entry( > > + CXL_EVT_TP_entry > > + __array(u8, data, CXL_EVENT_RECORD_DATA_LENGTH) > > + ), > > + > > + TP_fast_assign( > > + CXL_EVT_TP_fast_assign(dev, log, rec->hdr); > > + memcpy(__entry->data, &rec->data, CXL_EVENT_RECORD_DATA_LENGTH); > > + ), > > + > > + CXL_EVT_TP_printk("%s", > > + __print_hex(__entry->data, CXL_EVENT_RECORD_DATA_LENGTH)) > > +); > > + > > #endif /* _CXL_EVENTS_H */ > > > > #define TRACE_INCLUDE_FILE trace > > diff --git a/drivers/cxl/cxl.h b/drivers/cxl/cxl.h > > index aa3af3bb73b2..5974d1082210 100644 > > --- a/drivers/cxl/cxl.h > > +++ b/drivers/cxl/cxl.h > > @@ -156,6 +156,18 @@ static inline int ways_to_eiw(unsigned int ways, u8 *eiw) > > #define CXLDEV_CAP_CAP_ID_SECONDARY_MAILBOX 0x3 > > #define CXLDEV_CAP_CAP_ID_MEMDEV 0x4000 > > > > +/* CXL 3.0 8.2.8.3.1 Event Status Register */ > > +#define CXLDEV_DEV_EVENT_STATUS_OFFSET 0x00 > > +#define CXLDEV_EVENT_STATUS_INFO BIT(0) > > +#define CXLDEV_EVENT_STATUS_WARN BIT(1) > > +#define CXLDEV_EVENT_STATUS_FAIL BIT(2) > > +#define CXLDEV_EVENT_STATUS_FATAL BIT(3) > > + > > +#define CXLDEV_EVENT_STATUS_ALL (CXLDEV_EVENT_STATUS_INFO | \ > > + CXLDEV_EVENT_STATUS_WARN | \ > > + CXLDEV_EVENT_STATUS_FAIL | \ > > + CXLDEV_EVENT_STATUS_FATAL) > > + > > /* CXL 2.0 8.2.8.4 Mailbox Registers */ > > #define CXLDEV_MBOX_CAPS_OFFSET 0x00 > > #define CXLDEV_MBOX_CAP_PAYLOAD_SIZE_MASK GENMASK(4, 0) > > diff --git a/drivers/cxl/cxlmem.h b/drivers/cxl/cxlmem.h > > index ab138004f644..dd9aa3dd738e 100644 > > --- a/drivers/cxl/cxlmem.h > > +++ b/drivers/cxl/cxlmem.h > > @@ -4,6 +4,7 @@ > > #define __CXL_MEM_H__ > > #include <uapi/linux/cxl_mem.h> > > #include <linux/cdev.h> > > +#include <linux/uuid.h> > > #include "cxl.h" > > > > /* CXL 2.0 8.2.8.5.1.1 Memory Device Status Register */ > > @@ -193,6 +194,17 @@ struct cxl_endpoint_dvsec_info { > > struct range dvsec_range[2]; > > }; > > > > +/** > > + * struct cxl_event_state - Event log driver state > > + * > > + * @event_buf: Buffer to receive event data > > + * @event_log_lock: Serialize event_buf and log use > > + */ > > +struct cxl_event_state { > > + struct cxl_get_event_payload *buf; > > + struct mutex log_lock; > > +}; > > + > > /** > > * struct cxl_dev_state - The driver device state > > * > > @@ -266,12 +278,16 @@ struct cxl_dev_state { > > > > struct xarray doe_mbs; > > > > + struct cxl_event_state event; > > + > > int (*mbox_send)(struct cxl_dev_state *cxlds, struct cxl_mbox_cmd *cmd); > > }; > > > > enum cxl_opcode { > > CXL_MBOX_OP_INVALID = 0x0000, > > CXL_MBOX_OP_RAW = CXL_MBOX_OP_INVALID, > > + CXL_MBOX_OP_GET_EVENT_RECORD = 0x0100, > > + CXL_MBOX_OP_CLEAR_EVENT_RECORD = 0x0101, > > CXL_MBOX_OP_GET_FW_INFO = 0x0200, > > CXL_MBOX_OP_ACTIVATE_FW = 0x0202, > > CXL_MBOX_OP_GET_SUPPORTED_LOGS = 0x0400, > > @@ -347,6 +363,73 @@ struct cxl_mbox_identify { > > u8 qos_telemetry_caps; > > } __packed; > > > > +/* > > + * Common Event Record Format > > + * CXL rev 3.0 section 8.2.9.2.1; Table 8-42 > > + */ > > +struct cxl_event_record_hdr { > > + uuid_t id; > > + u8 length; > > + u8 flags[3]; > > + __le16 handle; > > + __le16 related_handle; > > + __le64 timestamp; > > + u8 maint_op_class; > > + u8 reserved[15]; > > +} __packed; > > + > > +#define CXL_EVENT_RECORD_DATA_LENGTH 0x50 > > +struct cxl_event_record_raw { > > + struct cxl_event_record_hdr hdr; > > + u8 data[CXL_EVENT_RECORD_DATA_LENGTH]; > > +} __packed; > > + > > +/* > > + * Get Event Records output payload > > + * CXL rev 3.0 section 8.2.9.2.2; Table 8-50 > > + */ > > +#define CXL_GET_EVENT_FLAG_OVERFLOW BIT(0) > > +#define CXL_GET_EVENT_FLAG_MORE_RECORDS BIT(1) > > +struct cxl_get_event_payload { > > + u8 flags; > > + u8 reserved1; > > + __le16 overflow_err_count; > > + __le64 first_overflow_timestamp; > > + __le64 last_overflow_timestamp; > > + __le16 record_count; > > + u8 reserved2[10]; > > + struct cxl_event_record_raw records[]; > > +} __packed; > > + > > +/* > > + * CXL rev 3.0 section 8.2.9.2.2; Table 8-49 > > + */ > > +enum cxl_event_log_type { > > + CXL_EVENT_TYPE_INFO = 0x00, > > + CXL_EVENT_TYPE_WARN, > > + CXL_EVENT_TYPE_FAIL, > > + CXL_EVENT_TYPE_FATAL, > > + CXL_EVENT_TYPE_MAX > > +}; > > + > > +/* > > + * Clear Event Records input payload > > + * CXL rev 3.0 section 8.2.9.2.3; Table 8-51 > > + */ > > +#define CXL_CLEAR_EVENT_MAX_HANDLES (0xff) > > +struct cxl_mbox_clear_event_payload { > > + u8 event_log; /* enum cxl_event_log_type */ > > + u8 clear_flags; > > + u8 nr_recs; > > + u8 reserved[3]; > > + __le16 handle[CXL_CLEAR_EVENT_MAX_HANDLES]; > > +} __packed; > > +#define CXL_CLEAR_EVENT_LIMIT_HANDLES(payload_size) \ > > + (((payload_size) - \ > > + (sizeof(struct cxl_mbox_clear_event_payload) - \ > > + (sizeof(__le16) * CXL_CLEAR_EVENT_MAX_HANDLES))) / \ > > + sizeof(__le16)) > > + > > struct cxl_mbox_get_partition_info { > > __le64 active_volatile_cap; > > __le64 active_persistent_cap; > > @@ -441,6 +524,7 @@ int cxl_mem_create_range_info(struct cxl_dev_state *cxlds); > > struct cxl_dev_state *cxl_dev_state_create(struct device *dev); > > void set_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); > > void clear_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); > > +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status); > > #ifdef CONFIG_CXL_SUSPEND > > void cxl_mem_active_inc(void); > > void cxl_mem_active_dec(void); > > diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c > > index 3a66aadb4df0..86c84611a168 100644 > > --- a/drivers/cxl/pci.c > > +++ b/drivers/cxl/pci.c > > @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) > > pci_disable_pcie_error_reporting(pdev); > > } > > > > +static void cxl_mem_free_event_buffer(void *buf) > > +{ > > + kvfree(buf); > > +} > > + > > +/* > > + * There is a single buffer for reading event logs from the mailbox. All logs > > + * share this buffer protected by the cxlds->event_log_lock. > > + */ > > +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) > > +{ > > + struct cxl_get_event_payload *buf; > > + > > + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", > > + cxlds->payload_size); > > + > > + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); > > + if (WARN_ON_ONCE(!buf)) > > No, why is event init so special that it behaves differently than all > the other init-time allocations this driver does? Previous review agreed that a warn on once would be printed if this universal buffer was not allocated. > > > + return; > > return -ENOMEM; > > > + > > + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, > > + cxl_mem_free_event_buffer, buf))) > > + return; > > ditto. I'll change both of these with a dev_err() and bail during init. > > > + > > + cxlds->event.buf = buf; > > +} > > + > > +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) > > +{ > > + /* Force read and clear of all logs */ > > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > > + /* Ensure prior partial reads are handled, by starting over again */ > > What partial reads? cxl_mem_get_event_records() reads every log until > each returns an empty result. Any remaining events after this returns > are events that fired during the retrieval. Jonathan was concerned that something could read part of the log and because of the statefullness of the log processing this reading of the log could start in the beginning. Perhaps from a previous driver unload while reading? I guess I was also thinking the BIOS could leave things this way? But I think we should not be here if the BIOS was ever involved right? > > So I do not think cxl_clear_event_logs() needs to exist, just call > cxl_mem_get_event_records(CXLDEV_EVENT_STATUS_ALL) once and that's it. That was my inclination but Jonathan's comments got me thinking I was wrong. I'll change it back. > > > > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > > +} > > + > > static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) > > { > > + struct pci_host_bridge *host_bridge = pci_find_host_bridge(pdev->bus); > > struct cxl_register_map map; > > struct cxl_memdev *cxlmd; > > struct cxl_dev_state *cxlds; > > @@ -494,6 +530,12 @@ static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) > > if (IS_ERR(cxlmd)) > > return PTR_ERR(cxlmd); > > > > + if (host_bridge->native_cxl_error) { > > Probably deserves a small comment about why this flag matters for event > init. Something like: "When BIOS maintains CXL error reporting control, > it will also reap event records. Only one agent can interface with the > event mechanism." Sure. Done. > > > + cxl_mem_alloc_event_buf(cxlds); > > + if (cxlds->event.buf) > > No need for this conditional here since this whole block is skipped if > !native_cxl_error and cxl_mem_alloc_event_buf() will fail driver load if > it fails to init. Done. > > > + cxl_clear_event_logs(cxlds); > > + } > > + > > if (cxlds->regs.ras) { > > pci_enable_pcie_error_reporting(pdev); > > rc = devm_add_action_or_reset(&pdev->dev, disable_aer, pdev); > > diff --git a/include/linux/pci.h b/include/linux/pci.h > > index 1f81807492ef..7fe3752a204e 100644 > > --- a/include/linux/pci.h > > +++ b/include/linux/pci.h > > @@ -580,6 +580,7 @@ struct pci_host_bridge { > > unsigned int preserve_config:1; /* Preserve FW resource setup */ > > unsigned int size_windows:1; /* Enable root bus sizing */ > > unsigned int msi_domain:1; /* Bridge wants MSI domain */ > > + unsigned int native_cxl_error:1; /* OS CXL Error reporting */ > > I would group this with the other native_ flags and copy the comment > style "OS may use CXL RAS + Events", or somesuch. Done. Ira
Ira Weiny wrote: [..] > > > diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c > > > index 3a66aadb4df0..86c84611a168 100644 > > > --- a/drivers/cxl/pci.c > > > +++ b/drivers/cxl/pci.c > > > @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) > > > pci_disable_pcie_error_reporting(pdev); > > > } > > > > > > +static void cxl_mem_free_event_buffer(void *buf) > > > +{ > > > + kvfree(buf); > > > +} > > > + > > > +/* > > > + * There is a single buffer for reading event logs from the mailbox. All logs > > > + * share this buffer protected by the cxlds->event_log_lock. > > > + */ > > > +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) > > > +{ > > > + struct cxl_get_event_payload *buf; > > > + > > > + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", > > > + cxlds->payload_size); > > > + > > > + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); > > > + if (WARN_ON_ONCE(!buf)) > > > > No, why is event init so special that it behaves differently than all > > the other init-time allocations this driver does? > > Previous review agreed that a warn on once would be printed if this universal > buffer was not allocated. > > > > > > + return; > > > > return -ENOMEM; > > > > > + > > > + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, > > > + cxl_mem_free_event_buffer, buf))) > > > + return; > > > > ditto. > > I'll change both of these with a dev_err() and bail during init. No real need to dev_err() for a simple memory allocation faliure, but at least it is better than a WARN > > > > > > + > > > + cxlds->event.buf = buf; > > > +} > > > + > > > +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) > > > +{ > > > + /* Force read and clear of all logs */ > > > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > > > + /* Ensure prior partial reads are handled, by starting over again */ > > > > What partial reads? cxl_mem_get_event_records() reads every log until > > each returns an empty result. Any remaining events after this returns > > are events that fired during the retrieval. > > Jonathan was concerned that something could read part of the log and because of > the statefullness of the log processing this reading of the log could start in > the beginning. Perhaps from a previous driver unload while reading? The driver will not unload without completing any current executions of the event retrieval thread otherwise that's an irq shutdown bug. > I guess I was also thinking the BIOS could leave things this way? But I think > we should not be here if the BIOS was ever involved right? If the OS has CXL Error control and all Event irqs are steered to the OS then the driver must be allowed to assume that it has exclusive control over event retrieval and clearing. > > So I do not think cxl_clear_event_logs() needs to exist, just call > > cxl_mem_get_event_records(CXLDEV_EVENT_STATUS_ALL) once and that's it. > > That was my inclination but Jonathan's comments got me thinking I was wrong. Perhaps that was before we realized the recent CXL _OSC entanglement.
On Fri, Dec 09, 2022 at 02:33:20PM -0800, Dan Williams wrote: > Ira Weiny wrote: > [..] > > > > diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c > > > > index 3a66aadb4df0..86c84611a168 100644 > > > > --- a/drivers/cxl/pci.c > > > > +++ b/drivers/cxl/pci.c > > > > @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) > > > > pci_disable_pcie_error_reporting(pdev); > > > > } > > > > > > > > +static void cxl_mem_free_event_buffer(void *buf) > > > > +{ > > > > + kvfree(buf); > > > > +} > > > > + > > > > +/* > > > > + * There is a single buffer for reading event logs from the mailbox. All logs > > > > + * share this buffer protected by the cxlds->event_log_lock. > > > > + */ > > > > +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) > > > > +{ > > > > + struct cxl_get_event_payload *buf; > > > > + > > > > + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", > > > > + cxlds->payload_size); > > > > + > > > > + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); > > > > + if (WARN_ON_ONCE(!buf)) > > > > > > No, why is event init so special that it behaves differently than all > > > the other init-time allocations this driver does? > > > > Previous review agreed that a warn on once would be printed if this universal > > buffer was not allocated. > > > > > > > > > + return; > > > > > > return -ENOMEM; > > > > > > > + > > > > + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, > > > > + cxl_mem_free_event_buffer, buf))) > > > > + return; > > > > > > ditto. > > > > I'll change both of these with a dev_err() and bail during init. > > No real need to dev_err() for a simple memory allocation faliure, but > at least it is better than a WARN Ok no error then. > > > > > > > > > > + > > > > + cxlds->event.buf = buf; > > > > +} > > > > + > > > > +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) > > > > +{ > > > > + /* Force read and clear of all logs */ > > > > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > > > > + /* Ensure prior partial reads are handled, by starting over again */ > > > > > > What partial reads? cxl_mem_get_event_records() reads every log until > > > each returns an empty result. Any remaining events after this returns > > > are events that fired during the retrieval. > > > > Jonathan was concerned that something could read part of the log and because of > > the statefullness of the log processing this reading of the log could start in > > the beginning. Perhaps from a previous driver unload while reading? > > The driver will not unload without completing any current executions of > the event retrieval thread otherwise that's an irq shutdown bug. > > > I guess I was also thinking the BIOS could leave things this way? But I think > > we should not be here if the BIOS was ever involved right? > > If the OS has CXL Error control and all Event irqs are steered to the OS > then the driver must be allowed to assume that it has exclusive control > over event retrieval and clearing. > > > > So I do not think cxl_clear_event_logs() needs to exist, just call > > > cxl_mem_get_event_records(CXLDEV_EVENT_STATUS_ALL) once and that's it. > > > > That was my inclination but Jonathan's comments got me thinking I was wrong. > > Perhaps that was before we realized the recent CXL _OSC entanglement. Yea that could have been. I'm not clear on the order of the comments. Ok this should be good to go. Reworking the rest of the series. Thanks for the review! Ira
On Fri, 9 Dec 2022 15:34:42 -0800 Ira Weiny <ira.weiny@intel.com> wrote: > On Fri, Dec 09, 2022 at 02:33:20PM -0800, Dan Williams wrote: > > Ira Weiny wrote: > > [..] > > > > > diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c > > > > > index 3a66aadb4df0..86c84611a168 100644 > > > > > --- a/drivers/cxl/pci.c > > > > > +++ b/drivers/cxl/pci.c > > > > > @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) > > > > > pci_disable_pcie_error_reporting(pdev); > > > > > } > > > > > > > > > > +static void cxl_mem_free_event_buffer(void *buf) > > > > > +{ > > > > > + kvfree(buf); > > > > > +} > > > > > + > > > > > +/* > > > > > + * There is a single buffer for reading event logs from the mailbox. All logs > > > > > + * share this buffer protected by the cxlds->event_log_lock. > > > > > + */ > > > > > +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) > > > > > +{ > > > > > + struct cxl_get_event_payload *buf; > > > > > + > > > > > + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", > > > > > + cxlds->payload_size); > > > > > + > > > > > + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); > > > > > + if (WARN_ON_ONCE(!buf)) > > > > > > > > No, why is event init so special that it behaves differently than all > > > > the other init-time allocations this driver does? > > > > > > Previous review agreed that a warn on once would be printed if this universal > > > buffer was not allocated. > > > > > > > > > > > > + return; > > > > > > > > return -ENOMEM; > > > > > > > > > + > > > > > + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, > > > > > + cxl_mem_free_event_buffer, buf))) > > > > > + return; > > > > > > > > ditto. > > > > > > I'll change both of these with a dev_err() and bail during init. > > > > No real need to dev_err() for a simple memory allocation faliure, but > > at least it is better than a WARN > > Ok no error then. > > > > > > > > > > > > > > > + > > > > > + cxlds->event.buf = buf; > > > > > +} > > > > > + > > > > > +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) > > > > > +{ > > > > > + /* Force read and clear of all logs */ > > > > > + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); > > > > > + /* Ensure prior partial reads are handled, by starting over again */ > > > > > > > > What partial reads? cxl_mem_get_event_records() reads every log until > > > > each returns an empty result. Any remaining events after this returns > > > > are events that fired during the retrieval. > > > > > > Jonathan was concerned that something could read part of the log and because of > > > the statefullness of the log processing this reading of the log could start in > > > the beginning. Perhaps from a previous driver unload while reading? > > > > The driver will not unload without completing any current executions of > > the event retrieval thread otherwise that's an irq shutdown bug. > > > > > I guess I was also thinking the BIOS could leave things this way? But I think > > > we should not be here if the BIOS was ever involved right? > > > > If the OS has CXL Error control and all Event irqs are steered to the OS > > then the driver must be allowed to assume that it has exclusive control > > over event retrieval and clearing. The OS has control once it's asked for it ;) We have no idea what the firmware did before that. > > > > > > So I do not think cxl_clear_event_logs() needs to exist, just call > > > > cxl_mem_get_event_records(CXLDEV_EVENT_STATUS_ALL) once and that's it. > > > > > > That was my inclination but Jonathan's comments got me thinking I was wrong. > > > > Perhaps that was before we realized the recent CXL _OSC entanglement. > > Yea that could have been. I'm not clear on the order of the comments. I'm just paranoid - particularly when my excellent firmware writing colleagues are involved (or our test teams who like to simulate weird sequences of events). I'm fine crossing fingers until we know there is someone doing this sort of crazy in the wild. Jonathan > > Ok this should be good to go. Reworking the rest of the series. > > Thanks for the review! > Ira
diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c index b3c202d2a433..cee8f56fb63a 100644 --- a/drivers/acpi/pci_root.c +++ b/drivers/acpi/pci_root.c @@ -1047,6 +1047,9 @@ struct pci_bus *acpi_pci_root_create(struct acpi_pci_root *root, if (!(root->osc_control_set & OSC_PCI_EXPRESS_DPC_CONTROL)) host_bridge->native_dpc = 0; + if (root->osc_ext_control_set & OSC_CXL_ERROR_REPORTING_CONTROL) + host_bridge->native_cxl_error = 1; + /* * Evaluate the "PCI Boot Configuration" _DSM Function. If it * exists and returns 0, we must preserve any PCI resource diff --git a/drivers/cxl/core/mbox.c b/drivers/cxl/core/mbox.c index b03fba212799..815da3aac081 100644 --- a/drivers/cxl/core/mbox.c +++ b/drivers/cxl/core/mbox.c @@ -8,6 +8,7 @@ #include <cxl.h> #include "core.h" +#include "trace.h" static bool cxl_raw_allow_all; @@ -717,6 +718,142 @@ int cxl_enumerate_cmds(struct cxl_dev_state *cxlds) } EXPORT_SYMBOL_NS_GPL(cxl_enumerate_cmds, CXL); +static int cxl_clear_event_record(struct cxl_dev_state *cxlds, + enum cxl_event_log_type log, + struct cxl_get_event_payload *get_pl) +{ + struct cxl_mbox_clear_event_payload payload = { + .event_log = log, + }; + u16 total = le16_to_cpu(get_pl->record_count); + u8 max_handles = CXL_CLEAR_EVENT_MAX_HANDLES; + size_t pl_size = sizeof(payload); + struct cxl_mbox_cmd mbox_cmd; + u16 cnt; + int rc; + int i; + + /* Payload size may limit the max handles */ + if (pl_size > cxlds->payload_size) { + max_handles = CXL_CLEAR_EVENT_LIMIT_HANDLES(cxlds->payload_size); + pl_size = cxlds->payload_size; + } + + mbox_cmd = (struct cxl_mbox_cmd) { + .opcode = CXL_MBOX_OP_CLEAR_EVENT_RECORD, + .payload_in = &payload, + .size_in = pl_size, + }; + + /* + * Clear Event Records uses u8 for the handle cnt while Get Event + * Record can return up to 0xffff records. + */ + i = 0; + for (cnt = 0; cnt < total; cnt++) { + payload.handle[i++] = get_pl->records[cnt].hdr.handle; + dev_dbg(cxlds->dev, "Event log '%d': Clearing %u\n", + log, le16_to_cpu(payload.handle[i])); + + if (i == max_handles) { + payload.nr_recs = i; + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); + if (rc) + return rc; + i = 0; + } + } + + /* Clear what is left if any */ + if (i) { + payload.nr_recs = i; + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); + if (rc) + return rc; + } + + return 0; +} + +static void cxl_mem_get_records_log(struct cxl_dev_state *cxlds, + enum cxl_event_log_type type) +{ + struct cxl_get_event_payload *payload; + struct cxl_mbox_cmd mbox_cmd; + u8 log_type = type; + u16 nr_rec; + + mutex_lock(&cxlds->event.log_lock); + payload = cxlds->event.buf; + + mbox_cmd = (struct cxl_mbox_cmd) { + .opcode = CXL_MBOX_OP_GET_EVENT_RECORD, + .payload_in = &log_type, + .size_in = sizeof(log_type), + .payload_out = payload, + .size_out = cxlds->payload_size, + .min_out = struct_size(payload, records, 0), + }; + + do { + int rc; + + rc = cxl_internal_send_cmd(cxlds, &mbox_cmd); + if (rc) { + dev_err(cxlds->dev, "Event log '%d': Failed to query event records : %d", + type, rc); + goto unlock_buffer; + } + + nr_rec = le16_to_cpu(payload->record_count); + if (nr_rec > 0) { + int i; + + for (i = 0; i < nr_rec; i++) + trace_cxl_generic_event(cxlds->dev, type, + &payload->records[i]); + + rc = cxl_clear_event_record(cxlds, type, payload); + if (rc) { + dev_err(cxlds->dev, "Event log '%d': Failed to clear events : %d", + type, rc); + goto unlock_buffer; + } + } + + if (payload->flags & CXL_GET_EVENT_FLAG_OVERFLOW) + trace_cxl_overflow(cxlds->dev, type, payload); + } while (nr_rec); + +unlock_buffer: + mutex_unlock(&cxlds->event.log_lock); +} + +/** + * cxl_mem_get_event_records - Get Event Records from the device + * @cxlds: The device data for the operation + * + * Retrieve all event records available on the device, report them as trace + * events, and clear them. + * + * See CXL rev 3.0 @8.2.9.2.2 Get Event Records + * See CXL rev 3.0 @8.2.9.2.3 Clear Event Records + */ +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status) +{ + dev_dbg(cxlds->dev, "Reading event logs: %x\n", status); + + if (status & CXLDEV_EVENT_STATUS_FATAL) + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FATAL); + if (status & CXLDEV_EVENT_STATUS_FAIL) + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_FAIL); + if (status & CXLDEV_EVENT_STATUS_WARN) + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_WARN); + if (status & CXLDEV_EVENT_STATUS_INFO) + cxl_mem_get_records_log(cxlds, CXL_EVENT_TYPE_INFO); +} +EXPORT_SYMBOL_NS_GPL(cxl_mem_get_event_records, CXL); + /** * cxl_mem_get_partition_info - Get partition info * @cxlds: The device data for the operation @@ -868,6 +1005,7 @@ struct cxl_dev_state *cxl_dev_state_create(struct device *dev) } mutex_init(&cxlds->mbox_mutex); + mutex_init(&cxlds->event.log_lock); cxlds->dev = dev; return cxlds; diff --git a/drivers/cxl/core/trace.h b/drivers/cxl/core/trace.h index 20ca2fe2ca8e..24eef6909f13 100644 --- a/drivers/cxl/core/trace.h +++ b/drivers/cxl/core/trace.h @@ -6,7 +6,9 @@ #if !defined(_CXL_EVENTS_H) || defined(TRACE_HEADER_MULTI_READ) #define _CXL_EVENTS_H +#include <asm-generic/unaligned.h> #include <cxl.h> +#include <cxlmem.h> #include <linux/tracepoint.h> #define CXL_RAS_UC_CACHE_DATA_PARITY BIT(0) @@ -103,6 +105,124 @@ TRACE_EVENT(cxl_aer_correctable_error, ) ); +#include <linux/tracepoint.h> + +#define cxl_event_log_type_str(type) \ + __print_symbolic(type, \ + { CXL_EVENT_TYPE_INFO, "Informational" }, \ + { CXL_EVENT_TYPE_WARN, "Warning" }, \ + { CXL_EVENT_TYPE_FAIL, "Failure" }, \ + { CXL_EVENT_TYPE_FATAL, "Fatal" }) + +TRACE_EVENT(cxl_overflow, + + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, + struct cxl_get_event_payload *payload), + + TP_ARGS(dev, log, payload), + + TP_STRUCT__entry( + __string(dev_name, dev_name(dev)) + __field(int, log) + __field(u64, first_ts) + __field(u64, last_ts) + __field(u16, count) + ), + + TP_fast_assign( + __assign_str(dev_name, dev_name(dev)); + __entry->log = log; + __entry->count = le16_to_cpu(payload->overflow_err_count); + __entry->first_ts = le64_to_cpu(payload->first_overflow_timestamp); + __entry->last_ts = le64_to_cpu(payload->last_overflow_timestamp); + ), + + TP_printk("%s: EVENT LOG OVERFLOW log=%s : %u records from %llu to %llu", + __get_str(dev_name), cxl_event_log_type_str(__entry->log), + __entry->count, __entry->first_ts, __entry->last_ts) + +); + +/* + * Common Event Record Format + * CXL 3.0 section 8.2.9.2.1; Table 8-42 + */ +#define CXL_EVENT_RECORD_FLAG_PERMANENT BIT(2) +#define CXL_EVENT_RECORD_FLAG_MAINT_NEEDED BIT(3) +#define CXL_EVENT_RECORD_FLAG_PERF_DEGRADED BIT(4) +#define CXL_EVENT_RECORD_FLAG_HW_REPLACE BIT(5) +#define show_hdr_flags(flags) __print_flags(flags, " | ", \ + { CXL_EVENT_RECORD_FLAG_PERMANENT, "PERMANENT_CONDITION" }, \ + { CXL_EVENT_RECORD_FLAG_MAINT_NEEDED, "MAINTENANCE_NEEDED" }, \ + { CXL_EVENT_RECORD_FLAG_PERF_DEGRADED, "PERFORMANCE_DEGRADED" }, \ + { CXL_EVENT_RECORD_FLAG_HW_REPLACE, "HARDWARE_REPLACEMENT_NEEDED" } \ +) + +/* + * Define macros for the common header of each CXL event. + * + * Tracepoints using these macros must do 3 things: + * + * 1) Add CXL_EVT_TP_entry to TP_STRUCT__entry + * 2) Use CXL_EVT_TP_fast_assign within TP_fast_assign; + * pass the dev, log, and CXL event header + * 3) Use CXL_EVT_TP_printk() instead of TP_printk() + * + * See the generic_event tracepoint as an example. + */ +#define CXL_EVT_TP_entry \ + __string(dev_name, dev_name(dev)) \ + __field(int, log) \ + __field_struct(uuid_t, hdr_uuid) \ + __field(u32, hdr_flags) \ + __field(u16, hdr_handle) \ + __field(u16, hdr_related_handle) \ + __field(u64, hdr_timestamp) \ + __field(u8, hdr_length) \ + __field(u8, hdr_maint_op_class) + +#define CXL_EVT_TP_fast_assign(dev, l, hdr) \ + __assign_str(dev_name, dev_name(dev)); \ + __entry->log = (l); \ + memcpy(&__entry->hdr_uuid, &(hdr).id, sizeof(uuid_t)); \ + __entry->hdr_length = (hdr).length; \ + __entry->hdr_flags = get_unaligned_le24((hdr).flags); \ + __entry->hdr_handle = le16_to_cpu((hdr).handle); \ + __entry->hdr_related_handle = le16_to_cpu((hdr).related_handle); \ + __entry->hdr_timestamp = le64_to_cpu((hdr).timestamp); \ + __entry->hdr_maint_op_class = (hdr).maint_op_class + +#define CXL_EVT_TP_printk(fmt, ...) \ + TP_printk("%s log=%s : time=%llu uuid=%pUb len=%d flags='%s' " \ + "handle=%x related_handle=%x maint_op_class=%u" \ + " : " fmt, \ + __get_str(dev_name), cxl_event_log_type_str(__entry->log), \ + __entry->hdr_timestamp, &__entry->hdr_uuid, __entry->hdr_length,\ + show_hdr_flags(__entry->hdr_flags), __entry->hdr_handle, \ + __entry->hdr_related_handle, __entry->hdr_maint_op_class, \ + ##__VA_ARGS__) + +TRACE_EVENT(cxl_generic_event, + + TP_PROTO(const struct device *dev, enum cxl_event_log_type log, + struct cxl_event_record_raw *rec), + + TP_ARGS(dev, log, rec), + + TP_STRUCT__entry( + CXL_EVT_TP_entry + __array(u8, data, CXL_EVENT_RECORD_DATA_LENGTH) + ), + + TP_fast_assign( + CXL_EVT_TP_fast_assign(dev, log, rec->hdr); + memcpy(__entry->data, &rec->data, CXL_EVENT_RECORD_DATA_LENGTH); + ), + + CXL_EVT_TP_printk("%s", + __print_hex(__entry->data, CXL_EVENT_RECORD_DATA_LENGTH)) +); + #endif /* _CXL_EVENTS_H */ #define TRACE_INCLUDE_FILE trace diff --git a/drivers/cxl/cxl.h b/drivers/cxl/cxl.h index aa3af3bb73b2..5974d1082210 100644 --- a/drivers/cxl/cxl.h +++ b/drivers/cxl/cxl.h @@ -156,6 +156,18 @@ static inline int ways_to_eiw(unsigned int ways, u8 *eiw) #define CXLDEV_CAP_CAP_ID_SECONDARY_MAILBOX 0x3 #define CXLDEV_CAP_CAP_ID_MEMDEV 0x4000 +/* CXL 3.0 8.2.8.3.1 Event Status Register */ +#define CXLDEV_DEV_EVENT_STATUS_OFFSET 0x00 +#define CXLDEV_EVENT_STATUS_INFO BIT(0) +#define CXLDEV_EVENT_STATUS_WARN BIT(1) +#define CXLDEV_EVENT_STATUS_FAIL BIT(2) +#define CXLDEV_EVENT_STATUS_FATAL BIT(3) + +#define CXLDEV_EVENT_STATUS_ALL (CXLDEV_EVENT_STATUS_INFO | \ + CXLDEV_EVENT_STATUS_WARN | \ + CXLDEV_EVENT_STATUS_FAIL | \ + CXLDEV_EVENT_STATUS_FATAL) + /* CXL 2.0 8.2.8.4 Mailbox Registers */ #define CXLDEV_MBOX_CAPS_OFFSET 0x00 #define CXLDEV_MBOX_CAP_PAYLOAD_SIZE_MASK GENMASK(4, 0) diff --git a/drivers/cxl/cxlmem.h b/drivers/cxl/cxlmem.h index ab138004f644..dd9aa3dd738e 100644 --- a/drivers/cxl/cxlmem.h +++ b/drivers/cxl/cxlmem.h @@ -4,6 +4,7 @@ #define __CXL_MEM_H__ #include <uapi/linux/cxl_mem.h> #include <linux/cdev.h> +#include <linux/uuid.h> #include "cxl.h" /* CXL 2.0 8.2.8.5.1.1 Memory Device Status Register */ @@ -193,6 +194,17 @@ struct cxl_endpoint_dvsec_info { struct range dvsec_range[2]; }; +/** + * struct cxl_event_state - Event log driver state + * + * @event_buf: Buffer to receive event data + * @event_log_lock: Serialize event_buf and log use + */ +struct cxl_event_state { + struct cxl_get_event_payload *buf; + struct mutex log_lock; +}; + /** * struct cxl_dev_state - The driver device state * @@ -266,12 +278,16 @@ struct cxl_dev_state { struct xarray doe_mbs; + struct cxl_event_state event; + int (*mbox_send)(struct cxl_dev_state *cxlds, struct cxl_mbox_cmd *cmd); }; enum cxl_opcode { CXL_MBOX_OP_INVALID = 0x0000, CXL_MBOX_OP_RAW = CXL_MBOX_OP_INVALID, + CXL_MBOX_OP_GET_EVENT_RECORD = 0x0100, + CXL_MBOX_OP_CLEAR_EVENT_RECORD = 0x0101, CXL_MBOX_OP_GET_FW_INFO = 0x0200, CXL_MBOX_OP_ACTIVATE_FW = 0x0202, CXL_MBOX_OP_GET_SUPPORTED_LOGS = 0x0400, @@ -347,6 +363,73 @@ struct cxl_mbox_identify { u8 qos_telemetry_caps; } __packed; +/* + * Common Event Record Format + * CXL rev 3.0 section 8.2.9.2.1; Table 8-42 + */ +struct cxl_event_record_hdr { + uuid_t id; + u8 length; + u8 flags[3]; + __le16 handle; + __le16 related_handle; + __le64 timestamp; + u8 maint_op_class; + u8 reserved[15]; +} __packed; + +#define CXL_EVENT_RECORD_DATA_LENGTH 0x50 +struct cxl_event_record_raw { + struct cxl_event_record_hdr hdr; + u8 data[CXL_EVENT_RECORD_DATA_LENGTH]; +} __packed; + +/* + * Get Event Records output payload + * CXL rev 3.0 section 8.2.9.2.2; Table 8-50 + */ +#define CXL_GET_EVENT_FLAG_OVERFLOW BIT(0) +#define CXL_GET_EVENT_FLAG_MORE_RECORDS BIT(1) +struct cxl_get_event_payload { + u8 flags; + u8 reserved1; + __le16 overflow_err_count; + __le64 first_overflow_timestamp; + __le64 last_overflow_timestamp; + __le16 record_count; + u8 reserved2[10]; + struct cxl_event_record_raw records[]; +} __packed; + +/* + * CXL rev 3.0 section 8.2.9.2.2; Table 8-49 + */ +enum cxl_event_log_type { + CXL_EVENT_TYPE_INFO = 0x00, + CXL_EVENT_TYPE_WARN, + CXL_EVENT_TYPE_FAIL, + CXL_EVENT_TYPE_FATAL, + CXL_EVENT_TYPE_MAX +}; + +/* + * Clear Event Records input payload + * CXL rev 3.0 section 8.2.9.2.3; Table 8-51 + */ +#define CXL_CLEAR_EVENT_MAX_HANDLES (0xff) +struct cxl_mbox_clear_event_payload { + u8 event_log; /* enum cxl_event_log_type */ + u8 clear_flags; + u8 nr_recs; + u8 reserved[3]; + __le16 handle[CXL_CLEAR_EVENT_MAX_HANDLES]; +} __packed; +#define CXL_CLEAR_EVENT_LIMIT_HANDLES(payload_size) \ + (((payload_size) - \ + (sizeof(struct cxl_mbox_clear_event_payload) - \ + (sizeof(__le16) * CXL_CLEAR_EVENT_MAX_HANDLES))) / \ + sizeof(__le16)) + struct cxl_mbox_get_partition_info { __le64 active_volatile_cap; __le64 active_persistent_cap; @@ -441,6 +524,7 @@ int cxl_mem_create_range_info(struct cxl_dev_state *cxlds); struct cxl_dev_state *cxl_dev_state_create(struct device *dev); void set_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); void clear_exclusive_cxl_commands(struct cxl_dev_state *cxlds, unsigned long *cmds); +void cxl_mem_get_event_records(struct cxl_dev_state *cxlds, u32 status); #ifdef CONFIG_CXL_SUSPEND void cxl_mem_active_inc(void); void cxl_mem_active_dec(void); diff --git a/drivers/cxl/pci.c b/drivers/cxl/pci.c index 3a66aadb4df0..86c84611a168 100644 --- a/drivers/cxl/pci.c +++ b/drivers/cxl/pci.c @@ -417,8 +417,44 @@ static void disable_aer(void *pdev) pci_disable_pcie_error_reporting(pdev); } +static void cxl_mem_free_event_buffer(void *buf) +{ + kvfree(buf); +} + +/* + * There is a single buffer for reading event logs from the mailbox. All logs + * share this buffer protected by the cxlds->event_log_lock. + */ +static void cxl_mem_alloc_event_buf(struct cxl_dev_state *cxlds) +{ + struct cxl_get_event_payload *buf; + + dev_dbg(cxlds->dev, "Allocating event buffer size %zu\n", + cxlds->payload_size); + + buf = kvmalloc(cxlds->payload_size, GFP_KERNEL); + if (WARN_ON_ONCE(!buf)) + return; + + if (WARN_ON_ONCE(devm_add_action_or_reset(cxlds->dev, + cxl_mem_free_event_buffer, buf))) + return; + + cxlds->event.buf = buf; +} + +static void cxl_clear_event_logs(struct cxl_dev_state *cxlds) +{ + /* Force read and clear of all logs */ + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); + /* Ensure prior partial reads are handled, by starting over again */ + cxl_mem_get_event_records(cxlds, CXLDEV_EVENT_STATUS_ALL); +} + static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) { + struct pci_host_bridge *host_bridge = pci_find_host_bridge(pdev->bus); struct cxl_register_map map; struct cxl_memdev *cxlmd; struct cxl_dev_state *cxlds; @@ -494,6 +530,12 @@ static int cxl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) if (IS_ERR(cxlmd)) return PTR_ERR(cxlmd); + if (host_bridge->native_cxl_error) { + cxl_mem_alloc_event_buf(cxlds); + if (cxlds->event.buf) + cxl_clear_event_logs(cxlds); + } + if (cxlds->regs.ras) { pci_enable_pcie_error_reporting(pdev); rc = devm_add_action_or_reset(&pdev->dev, disable_aer, pdev); diff --git a/include/linux/pci.h b/include/linux/pci.h index 1f81807492ef..7fe3752a204e 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h @@ -580,6 +580,7 @@ struct pci_host_bridge { unsigned int preserve_config:1; /* Preserve FW resource setup */ unsigned int size_windows:1; /* Enable root bus sizing */ unsigned int msi_domain:1; /* Bridge wants MSI domain */ + unsigned int native_cxl_error:1; /* OS CXL Error reporting */ /* Resource alignment requirements */ resource_size_t (*align_resource)(struct pci_dev *dev,