/* SPDX-License-Identifier: GPL-2.0 */ /* * Siemens SIMATIC IPC drivers * * Copyright (c) Siemens AG, 2018-2023 * * Authors: * Henning Schild * Gerd Haeussler */ #ifndef __PLATFORM_DATA_X86_SIMATIC_IPC_H #define __PLATFORM_DATA_X86_SIMATIC_IPC_H #include #include #define SIMATIC_IPC_DMI_ENTRY_OEM 129 /* binary type */ #define SIMATIC_IPC_DMI_TYPE 0xff #define SIMATIC_IPC_DMI_GROUP 0x05 #define SIMATIC_IPC_DMI_ENTRY 0x02 #define SIMATIC_IPC_DMI_TID 0x02 enum simatic_ipc_station_ids { SIMATIC_IPC_INVALID_STATION_ID = 0, SIMATIC_IPC_IPC227D = 0x00000501, SIMATIC_IPC_IPC427D = 0x00000701, SIMATIC_IPC_IPC227E = 0x00000901, SIMATIC_IPC_IPC277E = 0x00000902, SIMATIC_IPC_IPC427E = 0x00000A01, SIMATIC_IPC_IPC477E = 0x00000A02, SIMATIC_IPC_IPC127E = 0x00000D01, SIMATIC_IPC_IPC227G = 0x00000F01, SIMATIC_IPC_IPC277G = 0x00000F02, SIMATIC_IPC_IPCBX_39A = 0x00001001, SIMATIC_IPC_IPCPX_39A = 0x00001002, SIMATIC_IPC_IPCBX_21A = 0x00001101, SIMATIC_IPC_IPCBX_56A = 0x00001201, SIMATIC_IPC_IPCBX_59A = 0x00001202, }; static inline u32 simatic_ipc_get_station_id(u8 *data, int max_len) { struct { u8 type; /* type (0xff = binary) */ u8 len; /* len of data entry */ u8 group; u8 entry; u8 tid; __le32 station_id; /* station id (LE) */ } __packed * data_entry = (void *)data + sizeof(struct dmi_header); while ((u8 *)data_entry < data + max_len) { if (data_entry->type == SIMATIC_IPC_DMI_TYPE && data_entry->len == sizeof(*data_entry) && data_entry->group == SIMATIC_IPC_DMI_GROUP && data_entry->entry == SIMATIC_IPC_DMI_ENTRY && data_entry->tid == SIMATIC_IPC_DMI_TID) { return le32_to_cpu(data_entry->station_id); } data_entry = (void *)((u8 *)(data_entry) + data_entry->len); } return SIMATIC_IPC_INVALID_STATION_ID; } static inline void simatic_ipc_find_dmi_entry_helper(const struct dmi_header *dh, void *_data) { u32 *id = _data; if (dh->type != SIMATIC_IPC_DMI_ENTRY_OEM) return; *id = simatic_ipc_get_station_id((u8 *)dh, dh->length); } #endif /* __PLATFORM_DATA_X86_SIMATIC_IPC_H */