1dd123e62SHenning Schild /* SPDX-License-Identifier: GPL-2.0 */
2dd123e62SHenning Schild /*
3dd123e62SHenning Schild  * Siemens SIMATIC IPC drivers
4dd123e62SHenning Schild  *
51b3aa970SHenning Schild  * Copyright (c) Siemens AG, 2018-2023
6dd123e62SHenning Schild  *
7dd123e62SHenning Schild  * Authors:
8dd123e62SHenning Schild  *  Henning Schild <[email protected]>
9dd123e62SHenning Schild  *  Gerd Haeussler <[email protected]>
10dd123e62SHenning Schild  */
11dd123e62SHenning Schild 
12dd123e62SHenning Schild #ifndef __PLATFORM_DATA_X86_SIMATIC_IPC_H
13dd123e62SHenning Schild #define __PLATFORM_DATA_X86_SIMATIC_IPC_H
14dd123e62SHenning Schild 
15dd123e62SHenning Schild #include <linux/dmi.h>
16dd123e62SHenning Schild #include <linux/platform_data/x86/simatic-ipc-base.h>
17dd123e62SHenning Schild 
18dd123e62SHenning Schild #define SIMATIC_IPC_DMI_ENTRY_OEM	129
19dd123e62SHenning Schild /* binary type */
20dd123e62SHenning Schild #define SIMATIC_IPC_DMI_TYPE		0xff
21dd123e62SHenning Schild #define SIMATIC_IPC_DMI_GROUP		0x05
22dd123e62SHenning Schild #define SIMATIC_IPC_DMI_ENTRY		0x02
23dd123e62SHenning Schild #define SIMATIC_IPC_DMI_TID		0x02
24dd123e62SHenning Schild 
25dd123e62SHenning Schild enum simatic_ipc_station_ids {
26dd123e62SHenning Schild 	SIMATIC_IPC_INVALID_STATION_ID = 0,
27dd123e62SHenning Schild 	SIMATIC_IPC_IPC227D = 0x00000501,
28dd123e62SHenning Schild 	SIMATIC_IPC_IPC427D = 0x00000701,
29dd123e62SHenning Schild 	SIMATIC_IPC_IPC227E = 0x00000901,
30dd123e62SHenning Schild 	SIMATIC_IPC_IPC277E = 0x00000902,
31dd123e62SHenning Schild 	SIMATIC_IPC_IPC427E = 0x00000A01,
32dd123e62SHenning Schild 	SIMATIC_IPC_IPC477E = 0x00000A02,
33dd123e62SHenning Schild 	SIMATIC_IPC_IPC127E = 0x00000D01,
34a9712626SHenning Schild 	SIMATIC_IPC_IPC227G = 0x00000F01,
358529673aSHenning Schild 	SIMATIC_IPC_IPC277G = 0x00000F02,
36ed058eabSHenning Schild 	SIMATIC_IPC_IPCBX_39A = 0x00001001,
37d348b1d7SHenning Schild 	SIMATIC_IPC_IPCPX_39A = 0x00001002,
381b3aa970SHenning Schild 	SIMATIC_IPC_IPCBX_21A = 0x00001101,
39*b8af7795Sxingtong.wu 	SIMATIC_IPC_IPCBX_56A = 0x00001201,
40*b8af7795Sxingtong.wu 	SIMATIC_IPC_IPCBX_59A = 0x00001202,
41dd123e62SHenning Schild };
42dd123e62SHenning Schild 
simatic_ipc_get_station_id(u8 * data,int max_len)43dd123e62SHenning Schild static inline u32 simatic_ipc_get_station_id(u8 *data, int max_len)
44dd123e62SHenning Schild {
45dd123e62SHenning Schild 	struct {
46dd123e62SHenning Schild 		u8	type;		/* type (0xff = binary) */
47dd123e62SHenning Schild 		u8	len;		/* len of data entry */
48dd123e62SHenning Schild 		u8	group;
49dd123e62SHenning Schild 		u8	entry;
50dd123e62SHenning Schild 		u8	tid;
51dd123e62SHenning Schild 		__le32	station_id;	/* station id (LE) */
52dd123e62SHenning Schild 	} __packed * data_entry = (void *)data + sizeof(struct dmi_header);
53dd123e62SHenning Schild 
54dd123e62SHenning Schild 	while ((u8 *)data_entry < data + max_len) {
55dd123e62SHenning Schild 		if (data_entry->type == SIMATIC_IPC_DMI_TYPE &&
56dd123e62SHenning Schild 		    data_entry->len == sizeof(*data_entry) &&
57dd123e62SHenning Schild 		    data_entry->group == SIMATIC_IPC_DMI_GROUP &&
58dd123e62SHenning Schild 		    data_entry->entry == SIMATIC_IPC_DMI_ENTRY &&
59dd123e62SHenning Schild 		    data_entry->tid == SIMATIC_IPC_DMI_TID) {
60dd123e62SHenning Schild 			return le32_to_cpu(data_entry->station_id);
61dd123e62SHenning Schild 		}
62dd123e62SHenning Schild 		data_entry = (void *)((u8 *)(data_entry) + data_entry->len);
63dd123e62SHenning Schild 	}
64dd123e62SHenning Schild 
65dd123e62SHenning Schild 	return SIMATIC_IPC_INVALID_STATION_ID;
66dd123e62SHenning Schild }
67dd123e62SHenning Schild 
68dd123e62SHenning Schild static inline void
simatic_ipc_find_dmi_entry_helper(const struct dmi_header * dh,void * _data)69dd123e62SHenning Schild simatic_ipc_find_dmi_entry_helper(const struct dmi_header *dh, void *_data)
70dd123e62SHenning Schild {
71dd123e62SHenning Schild 	u32 *id = _data;
72dd123e62SHenning Schild 
73dd123e62SHenning Schild 	if (dh->type != SIMATIC_IPC_DMI_ENTRY_OEM)
74dd123e62SHenning Schild 		return;
75dd123e62SHenning Schild 
76dd123e62SHenning Schild 	*id = simatic_ipc_get_station_id((u8 *)dh, dh->length);
77dd123e62SHenning Schild }
78dd123e62SHenning Schild 
79dd123e62SHenning Schild #endif /* __PLATFORM_DATA_X86_SIMATIC_IPC_H */
80