cpu_ucode.c revision 1.7 1 1.7 christos /* $NetBSD: cpu_ucode.c,v 1.7 2018/03/17 20:02:32 christos Exp $ */
2 1.1 cegger /*
3 1.1 cegger * Copyright (c) 2012 The NetBSD Foundation, Inc.
4 1.1 cegger * All rights reserved.
5 1.1 cegger *
6 1.1 cegger * This code is derived from software contributed to The NetBSD Foundation
7 1.1 cegger * by Christoph Egger.
8 1.1 cegger *
9 1.1 cegger * Redistribution and use in source and binary forms, with or without
10 1.1 cegger * modification, are permitted provided that the following conditions
11 1.1 cegger * are met:
12 1.1 cegger * 1. Redistributions of source code must retain the above copyright
13 1.1 cegger * notice, this list of conditions and the following disclaimer.
14 1.1 cegger * 2. Redistributions in binary form must reproduce the above copyright
15 1.1 cegger * notice, this list of conditions and the following disclaimer in the
16 1.1 cegger * documentation and/or other materials provided with the distribution.
17 1.1 cegger *
18 1.1 cegger * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
19 1.1 cegger * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
20 1.1 cegger * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21 1.1 cegger * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
22 1.1 cegger * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23 1.1 cegger * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24 1.1 cegger * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25 1.1 cegger * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26 1.1 cegger * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27 1.1 cegger * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28 1.1 cegger * POSSIBILITY OF SUCH DAMAGE.
29 1.1 cegger */
30 1.1 cegger
31 1.1 cegger #include <sys/cdefs.h>
32 1.7 christos __KERNEL_RCSID(0, "$NetBSD: cpu_ucode.c,v 1.7 2018/03/17 20:02:32 christos Exp $");
33 1.1 cegger
34 1.1 cegger #include "opt_cpu_ucode.h"
35 1.7 christos #include "opt_xen.h"
36 1.7 christos
37 1.3 drochner #include "opt_compat_netbsd.h"
38 1.1 cegger
39 1.1 cegger #include <sys/param.h>
40 1.1 cegger #include <sys/cpuio.h>
41 1.1 cegger #include <sys/cpu.h>
42 1.1 cegger
43 1.1 cegger #include <dev/firmload.h>
44 1.1 cegger
45 1.1 cegger #include <machine/cpuvar.h>
46 1.1 cegger #include <machine/cputypes.h>
47 1.1 cegger
48 1.1 cegger #include <x86/cpu_ucode.h>
49 1.1 cegger
50 1.7 christos #ifdef XEN
51 1.7 christos #include <xen/xen-public/xen.h>
52 1.7 christos #include <xen/hypervisor.h>
53 1.7 christos #endif
54 1.7 christos
55 1.1 cegger static struct cpu_ucode_softc ucode_softc;
56 1.1 cegger
57 1.1 cegger int
58 1.2 drochner cpu_ucode_get_version(struct cpu_ucode_version *data)
59 1.1 cegger {
60 1.6 christos union {
61 1.6 christos struct cpu_ucode_version_amd a;
62 1.6 christos struct cpu_ucode_version_intel1 i;
63 1.6 christos } v;
64 1.6 christos size_t l;
65 1.6 christos int error;
66 1.6 christos
67 1.6 christos if (!data->data)
68 1.6 christos return 0;
69 1.1 cegger
70 1.1 cegger switch (cpu_vendor) {
71 1.1 cegger case CPUVENDOR_AMD:
72 1.6 christos error = cpu_ucode_amd_get_version(data, &v, l = sizeof(v.a));
73 1.2 drochner case CPUVENDOR_INTEL:
74 1.6 christos error = cpu_ucode_intel_get_version(data, &v, l = sizeof(v.i));
75 1.1 cegger default:
76 1.1 cegger return EOPNOTSUPP;
77 1.1 cegger }
78 1.1 cegger
79 1.6 christos if (error)
80 1.6 christos return error;
81 1.2 drochner
82 1.6 christos return copyout(&v, data->data, l);
83 1.2 drochner }
84 1.2 drochner
85 1.2 drochner int
86 1.2 drochner cpu_ucode_md_open(firmware_handle_t *fwh, int loader_version, const char *fwname)
87 1.1 cegger {
88 1.1 cegger switch (cpu_vendor) {
89 1.1 cegger case CPUVENDOR_AMD:
90 1.1 cegger return cpu_ucode_amd_firmware_open(fwh, fwname);
91 1.1 cegger case CPUVENDOR_INTEL:
92 1.2 drochner return cpu_ucode_intel_firmware_open(fwh, fwname);
93 1.1 cegger default:
94 1.1 cegger return EOPNOTSUPP;
95 1.1 cegger }
96 1.1 cegger }
97 1.1 cegger
98 1.7 christos #ifndef XEN
99 1.1 cegger int
100 1.2 drochner cpu_ucode_apply(const struct cpu_ucode *data)
101 1.1 cegger {
102 1.1 cegger struct cpu_ucode_softc *sc = &ucode_softc;
103 1.1 cegger int error;
104 1.1 cegger
105 1.2 drochner sc->loader_version = data->loader_version;
106 1.2 drochner
107 1.2 drochner error = cpu_ucode_load(sc, data->fwname);
108 1.1 cegger if (error)
109 1.1 cegger return error;
110 1.1 cegger
111 1.1 cegger switch (cpu_vendor) {
112 1.1 cegger case CPUVENDOR_AMD:
113 1.2 drochner error = cpu_ucode_amd_apply(sc, data->cpu_nr);
114 1.2 drochner break;
115 1.2 drochner case CPUVENDOR_INTEL:
116 1.2 drochner error = cpu_ucode_intel_apply(sc, data->cpu_nr);
117 1.1 cegger break;
118 1.1 cegger default:
119 1.7 christos error = EOPNOTSUPP;
120 1.1 cegger }
121 1.1 cegger
122 1.1 cegger if (sc->sc_blob != NULL)
123 1.5 ozaki firmware_free(sc->sc_blob, sc->sc_blobsize);
124 1.1 cegger sc->sc_blob = NULL;
125 1.1 cegger sc->sc_blobsize = 0;
126 1.1 cegger return error;
127 1.1 cegger }
128 1.7 christos #else
129 1.7 christos int
130 1.7 christos cpu_ucode_apply(const struct cpu_ucode *data)
131 1.7 christos {
132 1.7 christos struct cpu_ucode_softc *sc = &ucode_softc;
133 1.7 christos struct xen_platform_op op;
134 1.7 christos int error;
135 1.7 christos
136 1.7 christos /* Xen updates all??? */
137 1.7 christos if (data->cpu_nr != CPU_UCODE_ALL_CPUS)
138 1.7 christos return EOPNOTSUPP;
139 1.7 christos
140 1.7 christos sc->loader_version = data->loader_version;
141 1.7 christos error = cpu_ucode_load(sc, data->fwname);
142 1.7 christos if (error)
143 1.7 christos return error;
144 1.7 christos
145 1.7 christos op.cmd = XENPF_microcode_update;
146 1.7 christos set_xen_guest_handle(op.u.microcode.data, sc->sc_blob);
147 1.7 christos op.u.microcode.length = sc->sc_blobsize;
148 1.7 christos
149 1.7 christos error = -HYPERVISOR_platform_op(&op);
150 1.7 christos
151 1.7 christos if (sc->sc_blob)
152 1.7 christos firmware_free(sc->sc_blob, sc->sc_blobsize);
153 1.7 christos sc->sc_blob = NULL;
154 1.7 christos sc->sc_blobsize = 0;
155 1.7 christos return error;
156 1.7 christos }
157 1.7 christos #endif
158 1.2 drochner
159 1.3 drochner #ifdef COMPAT_60
160 1.2 drochner int
161 1.6 christos compat6_cpu_ucode_get_version(struct compat6_cpu_ucode *data)
162 1.6 christos {
163 1.6 christos struct cpu_ucode_version ndata;
164 1.6 christos
165 1.6 christos switch (cpu_vendor) {
166 1.6 christos case CPUVENDOR_AMD:
167 1.6 christos ndata.loader_version = CPU_UCODE_LOADER_AMD;
168 1.6 christos return cpu_ucode_amd_get_version(&ndata, &data->version,
169 1.6 christos sizeof(data->version));
170 1.6 christos default:
171 1.6 christos return EOPNOTSUPP;
172 1.6 christos }
173 1.6 christos }
174 1.6 christos
175 1.6 christos int
176 1.7 christos compat6_cpu_ucode_apply(const struct compat6_cpu_ucode *data6)
177 1.2 drochner {
178 1.2 drochner
179 1.2 drochner if (cpu_vendor != CPUVENDOR_AMD)
180 1.2 drochner return EOPNOTSUPP;
181 1.2 drochner
182 1.7 christos struct cpu_ucode data;
183 1.2 drochner
184 1.7 christos data.loader_version = CPU_UCODE_LOADER_AMD;
185 1.7 christos data.cpu_nr = CPU_UCODE_ALL_CPUS;
186 1.7 christos strcpy(data.fwname, data6->fwname);
187 1.2 drochner
188 1.7 christos return cpu_ucode_apply(&data);
189 1.2 drochner }
190 1.4 gdt #endif /* COMPAT60 */
191