xref: /openbmc/linux/arch/s390/hypfs/hypfs_diag.c (revision 83f95671943e6394eda4d20fa9458d4a2ae13c5c)
1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
224bbb1faSMichael Holzheu /*
324bbb1faSMichael Holzheu  *    Hypervisor filesystem for Linux on s390. Diag 204 and 224
424bbb1faSMichael Holzheu  *    implementation.
524bbb1faSMichael Holzheu  *
6f55495baSMichael Holzheu  *    Copyright IBM Corp. 2006, 2008
724bbb1faSMichael Holzheu  *    Author(s): Michael Holzheu <holzheu@de.ibm.com>
824bbb1faSMichael Holzheu  */
924bbb1faSMichael Holzheu 
10f55495baSMichael Holzheu #define KMSG_COMPONENT "hypfs"
11f55495baSMichael Holzheu #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
12f55495baSMichael Holzheu 
1324bbb1faSMichael Holzheu #include <linux/types.h>
1424bbb1faSMichael Holzheu #include <linux/errno.h>
1533b26d79SHeiko Carstens #include <linux/slab.h>
1624bbb1faSMichael Holzheu #include <linux/string.h>
1724bbb1faSMichael Holzheu #include <linux/vmalloc.h>
1857b28f66SMichael Holzheu #include <linux/mm.h>
191ec2772eSMartin Schwidefsky #include <asm/diag.h>
2024bbb1faSMichael Holzheu #include <asm/ebcdic.h>
2124bbb1faSMichael Holzheu #include "hypfs.h"
2224bbb1faSMichael Holzheu 
2324bbb1faSMichael Holzheu #define TMP_SIZE 64		/* size of temporary buffers */
2424bbb1faSMichael Holzheu 
2557b28f66SMichael Holzheu #define DBFS_D204_HDR_VERSION	0
2657b28f66SMichael Holzheu 
2724bbb1faSMichael Holzheu static char *diag224_cpu_names;			/* diag 224 name table */
2824bbb1faSMichael Holzheu static enum diag204_sc diag204_store_sc;	/* used subcode for store */
2924bbb1faSMichael Holzheu static enum diag204_format diag204_info_type;	/* used diag 204 data format */
3024bbb1faSMichael Holzheu 
3124bbb1faSMichael Holzheu static void *diag204_buf;		/* 4K aligned buffer for diag204 data */
3224bbb1faSMichael Holzheu static int diag204_buf_pages;		/* number of pages for diag204 data */
3324bbb1faSMichael Holzheu 
3457b28f66SMichael Holzheu static struct dentry *dbfs_d204_file;
3557b28f66SMichael Holzheu 
3624bbb1faSMichael Holzheu /*
37e65f30e0SJanosch Frank  * DIAG 204 member access functions.
3824bbb1faSMichael Holzheu  *
3924bbb1faSMichael Holzheu  * Since we have two different diag 204 data formats for old and new s390
4024bbb1faSMichael Holzheu  * machines, we do not access the structs directly, but use getter functions for
4124bbb1faSMichael Holzheu  * each struct member instead. This should make the code more readable.
4224bbb1faSMichael Holzheu  */
4324bbb1faSMichael Holzheu 
4424bbb1faSMichael Holzheu /* Time information block */
4524bbb1faSMichael Holzheu 
4624bbb1faSMichael Holzheu static inline int info_blk_hdr__size(enum diag204_format type)
4724bbb1faSMichael Holzheu {
48e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
49e65f30e0SJanosch Frank 		return sizeof(struct diag204_info_blk_hdr);
50e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
51e65f30e0SJanosch Frank 		return sizeof(struct diag204_x_info_blk_hdr);
5224bbb1faSMichael Holzheu }
5324bbb1faSMichael Holzheu 
5424bbb1faSMichael Holzheu static inline __u8 info_blk_hdr__npar(enum diag204_format type, void *hdr)
5524bbb1faSMichael Holzheu {
56e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
57e65f30e0SJanosch Frank 		return ((struct diag204_info_blk_hdr *)hdr)->npar;
58e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
59e65f30e0SJanosch Frank 		return ((struct diag204_x_info_blk_hdr *)hdr)->npar;
6024bbb1faSMichael Holzheu }
6124bbb1faSMichael Holzheu 
6224bbb1faSMichael Holzheu static inline __u8 info_blk_hdr__flags(enum diag204_format type, void *hdr)
6324bbb1faSMichael Holzheu {
64e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
65e65f30e0SJanosch Frank 		return ((struct diag204_info_blk_hdr *)hdr)->flags;
66e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
67e65f30e0SJanosch Frank 		return ((struct diag204_x_info_blk_hdr *)hdr)->flags;
6824bbb1faSMichael Holzheu }
6924bbb1faSMichael Holzheu 
7024bbb1faSMichael Holzheu /* Partition header */
7124bbb1faSMichael Holzheu 
7224bbb1faSMichael Holzheu static inline int part_hdr__size(enum diag204_format type)
7324bbb1faSMichael Holzheu {
74e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
75e65f30e0SJanosch Frank 		return sizeof(struct diag204_part_hdr);
76e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
77e65f30e0SJanosch Frank 		return sizeof(struct diag204_x_part_hdr);
7824bbb1faSMichael Holzheu }
7924bbb1faSMichael Holzheu 
8024bbb1faSMichael Holzheu static inline __u8 part_hdr__rcpus(enum diag204_format type, void *hdr)
8124bbb1faSMichael Holzheu {
82e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
83e65f30e0SJanosch Frank 		return ((struct diag204_part_hdr *)hdr)->cpus;
84e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
85e65f30e0SJanosch Frank 		return ((struct diag204_x_part_hdr *)hdr)->rcpus;
8624bbb1faSMichael Holzheu }
8724bbb1faSMichael Holzheu 
8824bbb1faSMichael Holzheu static inline void part_hdr__part_name(enum diag204_format type, void *hdr,
8924bbb1faSMichael Holzheu 				       char *name)
9024bbb1faSMichael Holzheu {
91e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
92e65f30e0SJanosch Frank 		memcpy(name, ((struct diag204_part_hdr *)hdr)->part_name,
93e65f30e0SJanosch Frank 		       DIAG204_LPAR_NAME_LEN);
94e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
95e65f30e0SJanosch Frank 		memcpy(name, ((struct diag204_x_part_hdr *)hdr)->part_name,
96e65f30e0SJanosch Frank 		       DIAG204_LPAR_NAME_LEN);
97e65f30e0SJanosch Frank 	EBCASC(name, DIAG204_LPAR_NAME_LEN);
98e65f30e0SJanosch Frank 	name[DIAG204_LPAR_NAME_LEN] = 0;
991d802e24SHeiko Carstens 	strim(name);
10024bbb1faSMichael Holzheu }
10124bbb1faSMichael Holzheu 
10224bbb1faSMichael Holzheu /* CPU info block */
10324bbb1faSMichael Holzheu 
10424bbb1faSMichael Holzheu static inline int cpu_info__size(enum diag204_format type)
10524bbb1faSMichael Holzheu {
106e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
107e65f30e0SJanosch Frank 		return sizeof(struct diag204_cpu_info);
108e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
109e65f30e0SJanosch Frank 		return sizeof(struct diag204_x_cpu_info);
11024bbb1faSMichael Holzheu }
11124bbb1faSMichael Holzheu 
11224bbb1faSMichael Holzheu static inline __u8 cpu_info__ctidx(enum diag204_format type, void *hdr)
11324bbb1faSMichael Holzheu {
114e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
115e65f30e0SJanosch Frank 		return ((struct diag204_cpu_info *)hdr)->ctidx;
116e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
117e65f30e0SJanosch Frank 		return ((struct diag204_x_cpu_info *)hdr)->ctidx;
11824bbb1faSMichael Holzheu }
11924bbb1faSMichael Holzheu 
12024bbb1faSMichael Holzheu static inline __u16 cpu_info__cpu_addr(enum diag204_format type, void *hdr)
12124bbb1faSMichael Holzheu {
122e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
123e65f30e0SJanosch Frank 		return ((struct diag204_cpu_info *)hdr)->cpu_addr;
124e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
125e65f30e0SJanosch Frank 		return ((struct diag204_x_cpu_info *)hdr)->cpu_addr;
12624bbb1faSMichael Holzheu }
12724bbb1faSMichael Holzheu 
12824bbb1faSMichael Holzheu static inline __u64 cpu_info__acc_time(enum diag204_format type, void *hdr)
12924bbb1faSMichael Holzheu {
130e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
131e65f30e0SJanosch Frank 		return ((struct diag204_cpu_info *)hdr)->acc_time;
132e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
133e65f30e0SJanosch Frank 		return ((struct diag204_x_cpu_info *)hdr)->acc_time;
13424bbb1faSMichael Holzheu }
13524bbb1faSMichael Holzheu 
13624bbb1faSMichael Holzheu static inline __u64 cpu_info__lp_time(enum diag204_format type, void *hdr)
13724bbb1faSMichael Holzheu {
138e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
139e65f30e0SJanosch Frank 		return ((struct diag204_cpu_info *)hdr)->lp_time;
140e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
141e65f30e0SJanosch Frank 		return ((struct diag204_x_cpu_info *)hdr)->lp_time;
14224bbb1faSMichael Holzheu }
14324bbb1faSMichael Holzheu 
14424bbb1faSMichael Holzheu static inline __u64 cpu_info__online_time(enum diag204_format type, void *hdr)
14524bbb1faSMichael Holzheu {
146e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
14724bbb1faSMichael Holzheu 		return 0;	/* online_time not available in simple info */
148e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
149e65f30e0SJanosch Frank 		return ((struct diag204_x_cpu_info *)hdr)->online_time;
15024bbb1faSMichael Holzheu }
15124bbb1faSMichael Holzheu 
15224bbb1faSMichael Holzheu /* Physical header */
15324bbb1faSMichael Holzheu 
15424bbb1faSMichael Holzheu static inline int phys_hdr__size(enum diag204_format type)
15524bbb1faSMichael Holzheu {
156e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
157e65f30e0SJanosch Frank 		return sizeof(struct diag204_phys_hdr);
158e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
159e65f30e0SJanosch Frank 		return sizeof(struct diag204_x_phys_hdr);
16024bbb1faSMichael Holzheu }
16124bbb1faSMichael Holzheu 
16224bbb1faSMichael Holzheu static inline __u8 phys_hdr__cpus(enum diag204_format type, void *hdr)
16324bbb1faSMichael Holzheu {
164e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
165e65f30e0SJanosch Frank 		return ((struct diag204_phys_hdr *)hdr)->cpus;
166e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
167e65f30e0SJanosch Frank 		return ((struct diag204_x_phys_hdr *)hdr)->cpus;
16824bbb1faSMichael Holzheu }
16924bbb1faSMichael Holzheu 
17024bbb1faSMichael Holzheu /* Physical CPU info block */
17124bbb1faSMichael Holzheu 
17224bbb1faSMichael Holzheu static inline int phys_cpu__size(enum diag204_format type)
17324bbb1faSMichael Holzheu {
174e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
175e65f30e0SJanosch Frank 		return sizeof(struct diag204_phys_cpu);
176e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
177e65f30e0SJanosch Frank 		return sizeof(struct diag204_x_phys_cpu);
17824bbb1faSMichael Holzheu }
17924bbb1faSMichael Holzheu 
18024bbb1faSMichael Holzheu static inline __u16 phys_cpu__cpu_addr(enum diag204_format type, void *hdr)
18124bbb1faSMichael Holzheu {
182e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
183e65f30e0SJanosch Frank 		return ((struct diag204_phys_cpu *)hdr)->cpu_addr;
184e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
185e65f30e0SJanosch Frank 		return ((struct diag204_x_phys_cpu *)hdr)->cpu_addr;
18624bbb1faSMichael Holzheu }
18724bbb1faSMichael Holzheu 
18824bbb1faSMichael Holzheu static inline __u64 phys_cpu__mgm_time(enum diag204_format type, void *hdr)
18924bbb1faSMichael Holzheu {
190e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
191e65f30e0SJanosch Frank 		return ((struct diag204_phys_cpu *)hdr)->mgm_time;
192e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
193e65f30e0SJanosch Frank 		return ((struct diag204_x_phys_cpu *)hdr)->mgm_time;
19424bbb1faSMichael Holzheu }
19524bbb1faSMichael Holzheu 
19624bbb1faSMichael Holzheu static inline __u64 phys_cpu__ctidx(enum diag204_format type, void *hdr)
19724bbb1faSMichael Holzheu {
198e65f30e0SJanosch Frank 	if (type == DIAG204_INFO_SIMPLE)
199e65f30e0SJanosch Frank 		return ((struct diag204_phys_cpu *)hdr)->ctidx;
200e65f30e0SJanosch Frank 	else /* DIAG204_INFO_EXT */
201e65f30e0SJanosch Frank 		return ((struct diag204_x_phys_cpu *)hdr)->ctidx;
20224bbb1faSMichael Holzheu }
20324bbb1faSMichael Holzheu 
20424bbb1faSMichael Holzheu /* Diagnose 204 functions */
20524bbb1faSMichael Holzheu /*
20624bbb1faSMichael Holzheu  * For the old diag subcode 4 with simple data format we have to use real
20724bbb1faSMichael Holzheu  * memory. If we use subcode 6 or 7 with extended data format, we can (and
20824bbb1faSMichael Holzheu  * should) use vmalloc, since we need a lot of memory in that case. Currently
20924bbb1faSMichael Holzheu  * up to 93 pages!
21024bbb1faSMichael Holzheu  */
21124bbb1faSMichael Holzheu 
21224bbb1faSMichael Holzheu static void diag204_free_buffer(void)
21324bbb1faSMichael Holzheu {
214*83f95671SHeiko Carstens 	vfree(diag204_buf);
21524bbb1faSMichael Holzheu 	diag204_buf = NULL;
21624bbb1faSMichael Holzheu }
21724bbb1faSMichael Holzheu 
21857b28f66SMichael Holzheu static void *page_align_ptr(void *ptr)
21957b28f66SMichael Holzheu {
22057b28f66SMichael Holzheu 	return (void *) PAGE_ALIGN((unsigned long) ptr);
22157b28f66SMichael Holzheu }
22257b28f66SMichael Holzheu 
22324bbb1faSMichael Holzheu static void *diag204_get_buffer(enum diag204_format fmt, int *pages)
22424bbb1faSMichael Holzheu {
22524bbb1faSMichael Holzheu 	if (diag204_buf) {
22624bbb1faSMichael Holzheu 		*pages = diag204_buf_pages;
22724bbb1faSMichael Holzheu 		return diag204_buf;
22824bbb1faSMichael Holzheu 	}
229e65f30e0SJanosch Frank 	if (fmt == DIAG204_INFO_SIMPLE) {
23024bbb1faSMichael Holzheu 		*pages = 1;
231e65f30e0SJanosch Frank 	} else {/* DIAG204_INFO_EXT */
232e65f30e0SJanosch Frank 		*pages = diag204((unsigned long)DIAG204_SUBC_RSI |
233e65f30e0SJanosch Frank 				 (unsigned long)DIAG204_INFO_EXT, 0, NULL);
23424bbb1faSMichael Holzheu 		if (*pages <= 0)
23524bbb1faSMichael Holzheu 			return ERR_PTR(-ENOSYS);
23624bbb1faSMichael Holzheu 	}
237*83f95671SHeiko Carstens 	diag204_buf = __vmalloc_node(array_size(*pages, PAGE_SIZE),
238*83f95671SHeiko Carstens 				     PAGE_SIZE, GFP_KERNEL, NUMA_NO_NODE,
239*83f95671SHeiko Carstens 				     __builtin_return_address(0));
240*83f95671SHeiko Carstens 	if (!diag204_buf)
241*83f95671SHeiko Carstens 		return ERR_PTR(-ENOMEM);
242*83f95671SHeiko Carstens 	diag204_buf_pages = *pages;
243*83f95671SHeiko Carstens 	return diag204_buf;
24424bbb1faSMichael Holzheu }
24524bbb1faSMichael Holzheu 
24624bbb1faSMichael Holzheu /*
24724bbb1faSMichael Holzheu  * diag204_probe() has to find out, which type of diagnose 204 implementation
24824bbb1faSMichael Holzheu  * we have on our machine. Currently there are three possible scanarios:
24924bbb1faSMichael Holzheu  *   - subcode 4   + simple data format (only one page)
25024bbb1faSMichael Holzheu  *   - subcode 4-6 + extended data format
25124bbb1faSMichael Holzheu  *   - subcode 4-7 + extended data format
25224bbb1faSMichael Holzheu  *
25324bbb1faSMichael Holzheu  * Subcode 5 is used to retrieve the size of the data, provided by subcodes
25424bbb1faSMichael Holzheu  * 6 and 7. Subcode 7 basically has the same function as subcode 6. In addition
25524bbb1faSMichael Holzheu  * to subcode 6 it provides also information about secondary cpus.
25624bbb1faSMichael Holzheu  * In order to get as much information as possible, we first try
25724bbb1faSMichael Holzheu  * subcode 7, then 6 and if both fail, we use subcode 4.
25824bbb1faSMichael Holzheu  */
25924bbb1faSMichael Holzheu 
26024bbb1faSMichael Holzheu static int diag204_probe(void)
26124bbb1faSMichael Holzheu {
26224bbb1faSMichael Holzheu 	void *buf;
26324bbb1faSMichael Holzheu 	int pages, rc;
26424bbb1faSMichael Holzheu 
265e65f30e0SJanosch Frank 	buf = diag204_get_buffer(DIAG204_INFO_EXT, &pages);
26624bbb1faSMichael Holzheu 	if (!IS_ERR(buf)) {
267e65f30e0SJanosch Frank 		if (diag204((unsigned long)DIAG204_SUBC_STIB7 |
268e65f30e0SJanosch Frank 			    (unsigned long)DIAG204_INFO_EXT, pages, buf) >= 0) {
269e65f30e0SJanosch Frank 			diag204_store_sc = DIAG204_SUBC_STIB7;
270e65f30e0SJanosch Frank 			diag204_info_type = DIAG204_INFO_EXT;
27124bbb1faSMichael Holzheu 			goto out;
27224bbb1faSMichael Holzheu 		}
273e65f30e0SJanosch Frank 		if (diag204((unsigned long)DIAG204_SUBC_STIB6 |
274e65f30e0SJanosch Frank 			    (unsigned long)DIAG204_INFO_EXT, pages, buf) >= 0) {
275e65f30e0SJanosch Frank 			diag204_store_sc = DIAG204_SUBC_STIB6;
276e65f30e0SJanosch Frank 			diag204_info_type = DIAG204_INFO_EXT;
27724bbb1faSMichael Holzheu 			goto out;
27824bbb1faSMichael Holzheu 		}
27924bbb1faSMichael Holzheu 		diag204_free_buffer();
28024bbb1faSMichael Holzheu 	}
28124bbb1faSMichael Holzheu 
28224bbb1faSMichael Holzheu 	/* subcodes 6 and 7 failed, now try subcode 4 */
28324bbb1faSMichael Holzheu 
284e65f30e0SJanosch Frank 	buf = diag204_get_buffer(DIAG204_INFO_SIMPLE, &pages);
28524bbb1faSMichael Holzheu 	if (IS_ERR(buf)) {
28624bbb1faSMichael Holzheu 		rc = PTR_ERR(buf);
28724bbb1faSMichael Holzheu 		goto fail_alloc;
28824bbb1faSMichael Holzheu 	}
289e65f30e0SJanosch Frank 	if (diag204((unsigned long)DIAG204_SUBC_STIB4 |
290e65f30e0SJanosch Frank 		    (unsigned long)DIAG204_INFO_SIMPLE, pages, buf) >= 0) {
291e65f30e0SJanosch Frank 		diag204_store_sc = DIAG204_SUBC_STIB4;
292e65f30e0SJanosch Frank 		diag204_info_type = DIAG204_INFO_SIMPLE;
29324bbb1faSMichael Holzheu 		goto out;
29424bbb1faSMichael Holzheu 	} else {
29524bbb1faSMichael Holzheu 		rc = -ENOSYS;
29624bbb1faSMichael Holzheu 		goto fail_store;
29724bbb1faSMichael Holzheu 	}
29824bbb1faSMichael Holzheu out:
29924bbb1faSMichael Holzheu 	rc = 0;
30024bbb1faSMichael Holzheu fail_store:
30124bbb1faSMichael Holzheu 	diag204_free_buffer();
30224bbb1faSMichael Holzheu fail_alloc:
30324bbb1faSMichael Holzheu 	return rc;
30424bbb1faSMichael Holzheu }
30524bbb1faSMichael Holzheu 
30657b28f66SMichael Holzheu static int diag204_do_store(void *buf, int pages)
30757b28f66SMichael Holzheu {
30857b28f66SMichael Holzheu 	int rc;
30957b28f66SMichael Holzheu 
31057b28f66SMichael Holzheu 	rc = diag204((unsigned long) diag204_store_sc |
31157b28f66SMichael Holzheu 		     (unsigned long) diag204_info_type, pages, buf);
31257b28f66SMichael Holzheu 	return rc < 0 ? -ENOSYS : 0;
31357b28f66SMichael Holzheu }
31457b28f66SMichael Holzheu 
31524bbb1faSMichael Holzheu static void *diag204_store(void)
31624bbb1faSMichael Holzheu {
31724bbb1faSMichael Holzheu 	void *buf;
31857b28f66SMichael Holzheu 	int pages, rc;
31924bbb1faSMichael Holzheu 
32024bbb1faSMichael Holzheu 	buf = diag204_get_buffer(diag204_info_type, &pages);
32124bbb1faSMichael Holzheu 	if (IS_ERR(buf))
32224bbb1faSMichael Holzheu 		goto out;
32357b28f66SMichael Holzheu 	rc = diag204_do_store(buf, pages);
32457b28f66SMichael Holzheu 	if (rc)
32557b28f66SMichael Holzheu 		return ERR_PTR(rc);
32624bbb1faSMichael Holzheu out:
32724bbb1faSMichael Holzheu 	return buf;
32824bbb1faSMichael Holzheu }
32924bbb1faSMichael Holzheu 
33024bbb1faSMichael Holzheu /* Diagnose 224 functions */
33124bbb1faSMichael Holzheu 
33224bbb1faSMichael Holzheu static int diag224_get_name_table(void)
33324bbb1faSMichael Holzheu {
33424bbb1faSMichael Holzheu 	/* memory must be below 2GB */
335237d6e68SMichael Holzheu 	diag224_cpu_names = (char *) __get_free_page(GFP_KERNEL | GFP_DMA);
33624bbb1faSMichael Holzheu 	if (!diag224_cpu_names)
33724bbb1faSMichael Holzheu 		return -ENOMEM;
338c41d4e3eSMichael Holzheu 	if (diag224(diag224_cpu_names)) {
339237d6e68SMichael Holzheu 		free_page((unsigned long) diag224_cpu_names);
340b8e660b8SHeiko Carstens 		return -EOPNOTSUPP;
341c41d4e3eSMichael Holzheu 	}
34224bbb1faSMichael Holzheu 	EBCASC(diag224_cpu_names + 16, (*diag224_cpu_names + 1) * 16);
34324bbb1faSMichael Holzheu 	return 0;
34424bbb1faSMichael Holzheu }
34524bbb1faSMichael Holzheu 
34624bbb1faSMichael Holzheu static void diag224_delete_name_table(void)
34724bbb1faSMichael Holzheu {
348237d6e68SMichael Holzheu 	free_page((unsigned long) diag224_cpu_names);
34924bbb1faSMichael Holzheu }
35024bbb1faSMichael Holzheu 
35124bbb1faSMichael Holzheu static int diag224_idx2name(int index, char *name)
35224bbb1faSMichael Holzheu {
353e65f30e0SJanosch Frank 	memcpy(name, diag224_cpu_names + ((index + 1) * DIAG204_CPU_NAME_LEN),
354e65f30e0SJanosch Frank 	       DIAG204_CPU_NAME_LEN);
355e65f30e0SJanosch Frank 	name[DIAG204_CPU_NAME_LEN] = 0;
3561d802e24SHeiko Carstens 	strim(name);
35724bbb1faSMichael Holzheu 	return 0;
35824bbb1faSMichael Holzheu }
35924bbb1faSMichael Holzheu 
36057b28f66SMichael Holzheu struct dbfs_d204_hdr {
36157b28f66SMichael Holzheu 	u64	len;		/* Length of d204 buffer without header */
36257b28f66SMichael Holzheu 	u16	version;	/* Version of header */
36357b28f66SMichael Holzheu 	u8	sc;		/* Used subcode */
36457b28f66SMichael Holzheu 	char	reserved[53];
36557b28f66SMichael Holzheu } __attribute__ ((packed));
36657b28f66SMichael Holzheu 
36757b28f66SMichael Holzheu struct dbfs_d204 {
36857b28f66SMichael Holzheu 	struct dbfs_d204_hdr	hdr;	/* 64 byte header */
36957b28f66SMichael Holzheu 	char			buf[];	/* d204 buffer */
37057b28f66SMichael Holzheu } __attribute__ ((packed));
37157b28f66SMichael Holzheu 
3722fcb3686SMichael Holzheu static int dbfs_d204_create(void **data, void **data_free_ptr, size_t *size)
37357b28f66SMichael Holzheu {
37457b28f66SMichael Holzheu 	struct dbfs_d204 *d204;
37557b28f66SMichael Holzheu 	int rc, buf_size;
3762fcb3686SMichael Holzheu 	void *base;
37757b28f66SMichael Holzheu 
37857b28f66SMichael Holzheu 	buf_size = PAGE_SIZE * (diag204_buf_pages + 1) + sizeof(d204->hdr);
379dc8a5c99SJoe Perches 	base = vzalloc(buf_size);
3802fcb3686SMichael Holzheu 	if (!base)
3812fcb3686SMichael Holzheu 		return -ENOMEM;
3822fcb3686SMichael Holzheu 	d204 = page_align_ptr(base + sizeof(d204->hdr)) - sizeof(d204->hdr);
3832fcb3686SMichael Holzheu 	rc = diag204_do_store(d204->buf, diag204_buf_pages);
3842fcb3686SMichael Holzheu 	if (rc) {
3852fcb3686SMichael Holzheu 		vfree(base);
3862fcb3686SMichael Holzheu 		return rc;
38757b28f66SMichael Holzheu 	}
38857b28f66SMichael Holzheu 	d204->hdr.version = DBFS_D204_HDR_VERSION;
38957b28f66SMichael Holzheu 	d204->hdr.len = PAGE_SIZE * diag204_buf_pages;
39057b28f66SMichael Holzheu 	d204->hdr.sc = diag204_store_sc;
3912fcb3686SMichael Holzheu 	*data = d204;
3922fcb3686SMichael Holzheu 	*data_free_ptr = base;
3932fcb3686SMichael Holzheu 	*size = d204->hdr.len + sizeof(struct dbfs_d204_hdr);
39457b28f66SMichael Holzheu 	return 0;
39557b28f66SMichael Holzheu }
39657b28f66SMichael Holzheu 
3972fcb3686SMichael Holzheu static struct hypfs_dbfs_file dbfs_file_d204 = {
3982fcb3686SMichael Holzheu 	.name		= "diag_204",
3992fcb3686SMichael Holzheu 	.data_create	= dbfs_d204_create,
4002fcb3686SMichael Holzheu 	.data_free	= vfree,
40157b28f66SMichael Holzheu };
40257b28f66SMichael Holzheu 
40324bbb1faSMichael Holzheu __init int hypfs_diag_init(void)
40424bbb1faSMichael Holzheu {
40524bbb1faSMichael Holzheu 	int rc;
40624bbb1faSMichael Holzheu 
40724bbb1faSMichael Holzheu 	if (diag204_probe()) {
4087b6670b0SJuergen Gross 		pr_info("The hardware system does not support hypfs\n");
40924bbb1faSMichael Holzheu 		return -ENODATA;
41024bbb1faSMichael Holzheu 	}
411f36108c4SGreg Kroah-Hartman 
412f36108c4SGreg Kroah-Hartman 	if (diag204_info_type == DIAG204_INFO_EXT)
413f36108c4SGreg Kroah-Hartman 		hypfs_dbfs_create_file(&dbfs_file_d204);
414f36108c4SGreg Kroah-Hartman 
4153c8ebca0SMichael Holzheu 	if (MACHINE_IS_LPAR) {
4163c8ebca0SMichael Holzheu 		rc = diag224_get_name_table();
4173c8ebca0SMichael Holzheu 		if (rc) {
4183c8ebca0SMichael Holzheu 			pr_err("The hardware system does not provide all "
4193c8ebca0SMichael Holzheu 			       "functions required by hypfs\n");
4203c8ebca0SMichael Holzheu 			debugfs_remove(dbfs_d204_file);
4213c8ebca0SMichael Holzheu 			return rc;
4223c8ebca0SMichael Holzheu 		}
4233c8ebca0SMichael Holzheu 	}
4243c8ebca0SMichael Holzheu 	return 0;
4253c8ebca0SMichael Holzheu }
42624bbb1faSMichael Holzheu 
4271375fc1fSHeiko Carstens void hypfs_diag_exit(void)
42824bbb1faSMichael Holzheu {
42957b28f66SMichael Holzheu 	debugfs_remove(dbfs_d204_file);
43024bbb1faSMichael Holzheu 	diag224_delete_name_table();
43124bbb1faSMichael Holzheu 	diag204_free_buffer();
4322fcb3686SMichael Holzheu 	hypfs_dbfs_remove_file(&dbfs_file_d204);
43324bbb1faSMichael Holzheu }
43424bbb1faSMichael Holzheu 
43524bbb1faSMichael Holzheu /*
43624bbb1faSMichael Holzheu  * Functions to create the directory structure
43724bbb1faSMichael Holzheu  * *******************************************
43824bbb1faSMichael Holzheu  */
43924bbb1faSMichael Holzheu 
440e334cf4fSAl Viro static int hypfs_create_cpu_files(struct dentry *cpus_dir, void *cpu_info)
44124bbb1faSMichael Holzheu {
44224bbb1faSMichael Holzheu 	struct dentry *cpu_dir;
44324bbb1faSMichael Holzheu 	char buffer[TMP_SIZE];
44424bbb1faSMichael Holzheu 	void *rc;
44524bbb1faSMichael Holzheu 
44624bbb1faSMichael Holzheu 	snprintf(buffer, TMP_SIZE, "%d", cpu_info__cpu_addr(diag204_info_type,
44724bbb1faSMichael Holzheu 							    cpu_info));
448a118cfdfSAl Viro 	cpu_dir = hypfs_mkdir(cpus_dir, buffer);
449a118cfdfSAl Viro 	rc = hypfs_create_u64(cpu_dir, "mgmtime",
45024bbb1faSMichael Holzheu 			      cpu_info__acc_time(diag204_info_type, cpu_info) -
45124bbb1faSMichael Holzheu 			      cpu_info__lp_time(diag204_info_type, cpu_info));
45224bbb1faSMichael Holzheu 	if (IS_ERR(rc))
45324bbb1faSMichael Holzheu 		return PTR_ERR(rc);
454a118cfdfSAl Viro 	rc = hypfs_create_u64(cpu_dir, "cputime",
45524bbb1faSMichael Holzheu 			      cpu_info__lp_time(diag204_info_type, cpu_info));
45624bbb1faSMichael Holzheu 	if (IS_ERR(rc))
45724bbb1faSMichael Holzheu 		return PTR_ERR(rc);
458e65f30e0SJanosch Frank 	if (diag204_info_type == DIAG204_INFO_EXT) {
459a118cfdfSAl Viro 		rc = hypfs_create_u64(cpu_dir, "onlinetime",
46024bbb1faSMichael Holzheu 				      cpu_info__online_time(diag204_info_type,
46124bbb1faSMichael Holzheu 							    cpu_info));
46224bbb1faSMichael Holzheu 		if (IS_ERR(rc))
46324bbb1faSMichael Holzheu 			return PTR_ERR(rc);
46424bbb1faSMichael Holzheu 	}
46524bbb1faSMichael Holzheu 	diag224_idx2name(cpu_info__ctidx(diag204_info_type, cpu_info), buffer);
466a118cfdfSAl Viro 	rc = hypfs_create_str(cpu_dir, "type", buffer);
4679fa6a659SGustavo A. R. Silva 	return PTR_ERR_OR_ZERO(rc);
46824bbb1faSMichael Holzheu }
46924bbb1faSMichael Holzheu 
470e334cf4fSAl Viro static void *hypfs_create_lpar_files(struct dentry *systems_dir, void *part_hdr)
47124bbb1faSMichael Holzheu {
47224bbb1faSMichael Holzheu 	struct dentry *cpus_dir;
47324bbb1faSMichael Holzheu 	struct dentry *lpar_dir;
474e65f30e0SJanosch Frank 	char lpar_name[DIAG204_LPAR_NAME_LEN + 1];
47524bbb1faSMichael Holzheu 	void *cpu_info;
47624bbb1faSMichael Holzheu 	int i;
47724bbb1faSMichael Holzheu 
47824bbb1faSMichael Holzheu 	part_hdr__part_name(diag204_info_type, part_hdr, lpar_name);
479e65f30e0SJanosch Frank 	lpar_name[DIAG204_LPAR_NAME_LEN] = 0;
480a118cfdfSAl Viro 	lpar_dir = hypfs_mkdir(systems_dir, lpar_name);
48124bbb1faSMichael Holzheu 	if (IS_ERR(lpar_dir))
48224bbb1faSMichael Holzheu 		return lpar_dir;
483a118cfdfSAl Viro 	cpus_dir = hypfs_mkdir(lpar_dir, "cpus");
48424bbb1faSMichael Holzheu 	if (IS_ERR(cpus_dir))
48524bbb1faSMichael Holzheu 		return cpus_dir;
48624bbb1faSMichael Holzheu 	cpu_info = part_hdr + part_hdr__size(diag204_info_type);
48724bbb1faSMichael Holzheu 	for (i = 0; i < part_hdr__rcpus(diag204_info_type, part_hdr); i++) {
48824bbb1faSMichael Holzheu 		int rc;
489e334cf4fSAl Viro 		rc = hypfs_create_cpu_files(cpus_dir, cpu_info);
49024bbb1faSMichael Holzheu 		if (rc)
49124bbb1faSMichael Holzheu 			return ERR_PTR(rc);
49224bbb1faSMichael Holzheu 		cpu_info += cpu_info__size(diag204_info_type);
49324bbb1faSMichael Holzheu 	}
49424bbb1faSMichael Holzheu 	return cpu_info;
49524bbb1faSMichael Holzheu }
49624bbb1faSMichael Holzheu 
497e334cf4fSAl Viro static int hypfs_create_phys_cpu_files(struct dentry *cpus_dir, void *cpu_info)
49824bbb1faSMichael Holzheu {
49924bbb1faSMichael Holzheu 	struct dentry *cpu_dir;
50024bbb1faSMichael Holzheu 	char buffer[TMP_SIZE];
50124bbb1faSMichael Holzheu 	void *rc;
50224bbb1faSMichael Holzheu 
50324bbb1faSMichael Holzheu 	snprintf(buffer, TMP_SIZE, "%i", phys_cpu__cpu_addr(diag204_info_type,
50424bbb1faSMichael Holzheu 							    cpu_info));
505a118cfdfSAl Viro 	cpu_dir = hypfs_mkdir(cpus_dir, buffer);
50624bbb1faSMichael Holzheu 	if (IS_ERR(cpu_dir))
50724bbb1faSMichael Holzheu 		return PTR_ERR(cpu_dir);
508a118cfdfSAl Viro 	rc = hypfs_create_u64(cpu_dir, "mgmtime",
50924bbb1faSMichael Holzheu 			      phys_cpu__mgm_time(diag204_info_type, cpu_info));
51024bbb1faSMichael Holzheu 	if (IS_ERR(rc))
51124bbb1faSMichael Holzheu 		return PTR_ERR(rc);
51224bbb1faSMichael Holzheu 	diag224_idx2name(phys_cpu__ctidx(diag204_info_type, cpu_info), buffer);
513a118cfdfSAl Viro 	rc = hypfs_create_str(cpu_dir, "type", buffer);
5149fa6a659SGustavo A. R. Silva 	return PTR_ERR_OR_ZERO(rc);
51524bbb1faSMichael Holzheu }
51624bbb1faSMichael Holzheu 
517e334cf4fSAl Viro static void *hypfs_create_phys_files(struct dentry *parent_dir, void *phys_hdr)
51824bbb1faSMichael Holzheu {
51924bbb1faSMichael Holzheu 	int i;
52024bbb1faSMichael Holzheu 	void *cpu_info;
52124bbb1faSMichael Holzheu 	struct dentry *cpus_dir;
52224bbb1faSMichael Holzheu 
523a118cfdfSAl Viro 	cpus_dir = hypfs_mkdir(parent_dir, "cpus");
52424bbb1faSMichael Holzheu 	if (IS_ERR(cpus_dir))
52524bbb1faSMichael Holzheu 		return cpus_dir;
52624bbb1faSMichael Holzheu 	cpu_info = phys_hdr + phys_hdr__size(diag204_info_type);
52724bbb1faSMichael Holzheu 	for (i = 0; i < phys_hdr__cpus(diag204_info_type, phys_hdr); i++) {
52824bbb1faSMichael Holzheu 		int rc;
529e334cf4fSAl Viro 		rc = hypfs_create_phys_cpu_files(cpus_dir, cpu_info);
53024bbb1faSMichael Holzheu 		if (rc)
53124bbb1faSMichael Holzheu 			return ERR_PTR(rc);
53224bbb1faSMichael Holzheu 		cpu_info += phys_cpu__size(diag204_info_type);
53324bbb1faSMichael Holzheu 	}
53424bbb1faSMichael Holzheu 	return cpu_info;
53524bbb1faSMichael Holzheu }
53624bbb1faSMichael Holzheu 
537e334cf4fSAl Viro int hypfs_diag_create_files(struct dentry *root)
53824bbb1faSMichael Holzheu {
53924bbb1faSMichael Holzheu 	struct dentry *systems_dir, *hyp_dir;
54024bbb1faSMichael Holzheu 	void *time_hdr, *part_hdr;
54124bbb1faSMichael Holzheu 	int i, rc;
54224bbb1faSMichael Holzheu 	void *buffer, *ptr;
54324bbb1faSMichael Holzheu 
54424bbb1faSMichael Holzheu 	buffer = diag204_store();
54524bbb1faSMichael Holzheu 	if (IS_ERR(buffer))
54624bbb1faSMichael Holzheu 		return PTR_ERR(buffer);
54724bbb1faSMichael Holzheu 
548a118cfdfSAl Viro 	systems_dir = hypfs_mkdir(root, "systems");
54924bbb1faSMichael Holzheu 	if (IS_ERR(systems_dir)) {
55024bbb1faSMichael Holzheu 		rc = PTR_ERR(systems_dir);
55124bbb1faSMichael Holzheu 		goto err_out;
55224bbb1faSMichael Holzheu 	}
55324bbb1faSMichael Holzheu 	time_hdr = (struct x_info_blk_hdr *)buffer;
55424bbb1faSMichael Holzheu 	part_hdr = time_hdr + info_blk_hdr__size(diag204_info_type);
55524bbb1faSMichael Holzheu 	for (i = 0; i < info_blk_hdr__npar(diag204_info_type, time_hdr); i++) {
556e334cf4fSAl Viro 		part_hdr = hypfs_create_lpar_files(systems_dir, part_hdr);
55724bbb1faSMichael Holzheu 		if (IS_ERR(part_hdr)) {
55824bbb1faSMichael Holzheu 			rc = PTR_ERR(part_hdr);
55924bbb1faSMichael Holzheu 			goto err_out;
56024bbb1faSMichael Holzheu 		}
56124bbb1faSMichael Holzheu 	}
562e65f30e0SJanosch Frank 	if (info_blk_hdr__flags(diag204_info_type, time_hdr) &
563e65f30e0SJanosch Frank 	    DIAG204_LPAR_PHYS_FLG) {
564e334cf4fSAl Viro 		ptr = hypfs_create_phys_files(root, part_hdr);
56524bbb1faSMichael Holzheu 		if (IS_ERR(ptr)) {
56624bbb1faSMichael Holzheu 			rc = PTR_ERR(ptr);
56724bbb1faSMichael Holzheu 			goto err_out;
56824bbb1faSMichael Holzheu 		}
56924bbb1faSMichael Holzheu 	}
570a118cfdfSAl Viro 	hyp_dir = hypfs_mkdir(root, "hyp");
57124bbb1faSMichael Holzheu 	if (IS_ERR(hyp_dir)) {
57224bbb1faSMichael Holzheu 		rc = PTR_ERR(hyp_dir);
57324bbb1faSMichael Holzheu 		goto err_out;
57424bbb1faSMichael Holzheu 	}
575a118cfdfSAl Viro 	ptr = hypfs_create_str(hyp_dir, "type", "LPAR Hypervisor");
57624bbb1faSMichael Holzheu 	if (IS_ERR(ptr)) {
57724bbb1faSMichael Holzheu 		rc = PTR_ERR(ptr);
57824bbb1faSMichael Holzheu 		goto err_out;
57924bbb1faSMichael Holzheu 	}
58024bbb1faSMichael Holzheu 	rc = 0;
58124bbb1faSMichael Holzheu 
58224bbb1faSMichael Holzheu err_out:
58324bbb1faSMichael Holzheu 	return rc;
58424bbb1faSMichael Holzheu }
585