Greg Kroah-Hartman | b244131 | 2017-11-01 15:07:57 +0100 | [diff] [blame] | 1 | // SPDX-License-Identifier: GPL-2.0 |
Jiri Olsa | 696e245 | 2017-10-11 17:01:24 +0200 | [diff] [blame] | 2 | #include <linux/compiler.h> |
| 3 | |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 4 | static struct ins_ops *s390__associate_ins_ops(struct arch *arch, const char *name) |
| 5 | { |
| 6 | struct ins_ops *ops = NULL; |
| 7 | |
| 8 | /* catch all kind of jumps */ |
| 9 | if (strchr(name, 'j') || |
| 10 | !strncmp(name, "bct", 3) || |
| 11 | !strncmp(name, "br", 2)) |
| 12 | ops = &jump_ops; |
| 13 | /* override call/returns */ |
| 14 | if (!strcmp(name, "bras") || |
| 15 | !strcmp(name, "brasl") || |
| 16 | !strcmp(name, "basr")) |
| 17 | ops = &call_ops; |
| 18 | if (!strcmp(name, "br")) |
| 19 | ops = &ret_ops; |
| 20 | |
Thomas Richter | 36c2636 | 2017-11-24 10:46:37 +0100 | [diff] [blame] | 21 | if (ops) |
| 22 | arch__associate_ins_ops(arch, name, ops); |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 23 | return ops; |
| 24 | } |
| 25 | |
Thomas Richter | c59124f | 2018-02-13 16:14:17 +0100 | [diff] [blame^] | 26 | static int s390__cpuid_parse(struct arch *arch, char *cpuid) |
| 27 | { |
| 28 | unsigned int family; |
| 29 | char model[16], model_c[16], cpumf_v[16], cpumf_a[16]; |
| 30 | int ret; |
| 31 | |
| 32 | /* |
| 33 | * cpuid string format: |
| 34 | * "IBM,family,model-capacity,model[,cpum_cf-version,cpum_cf-authorization]" |
| 35 | */ |
| 36 | ret = sscanf(cpuid, "%*[^,],%u,%[^,],%[^,],%[^,],%s", &family, model_c, |
| 37 | model, cpumf_v, cpumf_a); |
| 38 | if (ret >= 2) { |
| 39 | arch->family = family; |
| 40 | arch->model = 0; |
| 41 | return 0; |
| 42 | } |
| 43 | |
| 44 | return -1; |
| 45 | } |
| 46 | |
Jiri Olsa | 696e245 | 2017-10-11 17:01:24 +0200 | [diff] [blame] | 47 | static int s390__annotate_init(struct arch *arch, char *cpuid __maybe_unused) |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 48 | { |
Thomas Richter | c59124f | 2018-02-13 16:14:17 +0100 | [diff] [blame^] | 49 | int err = 0; |
| 50 | |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 51 | if (!arch->initialized) { |
| 52 | arch->initialized = true; |
| 53 | arch->associate_instruction_ops = s390__associate_ins_ops; |
Thomas Richter | c59124f | 2018-02-13 16:14:17 +0100 | [diff] [blame^] | 54 | if (cpuid) |
| 55 | err = s390__cpuid_parse(arch, cpuid); |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 56 | } |
| 57 | |
Thomas Richter | c59124f | 2018-02-13 16:14:17 +0100 | [diff] [blame^] | 58 | return err; |
Christian Borntraeger | d9f8dfa | 2017-04-06 09:51:52 +0200 | [diff] [blame] | 59 | } |