Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 1 | /* |
| 2 | * QEMU TPM Backend |
| 3 | * |
| 4 | * Copyright IBM, Corp. 2013 |
| 5 | * |
| 6 | * Authors: |
| 7 | * Stefan Berger <stefanb@us.ibm.com> |
| 8 | * |
| 9 | * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| 10 | * See the COPYING file in the top-level directory. |
| 11 | */ |
| 12 | |
Markus Armbruster | 121d071 | 2016-06-29 10:12:57 +0200 | [diff] [blame] | 13 | #ifndef TPM_BACKEND_H |
| 14 | #define TPM_BACKEND_H |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 15 | |
| 16 | #include "qom/object.h" |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 17 | #include "qemu/option.h" |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 18 | #include "sysemu/tpm.h" |
Marc-André Lureau | 6a8a235 | 2018-01-29 19:33:06 +0100 | [diff] [blame] | 19 | #include "qapi/error.h" |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 20 | |
Stefan Berger | e542b71 | 2021-06-15 16:21:20 +0200 | [diff] [blame] | 21 | #ifdef CONFIG_TPM |
| 22 | |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 23 | #define TYPE_TPM_BACKEND "tpm-backend" |
Eduardo Habkost | c821774a | 2020-08-31 17:07:37 -0400 | [diff] [blame] | 24 | OBJECT_DECLARE_TYPE(TPMBackend, TPMBackendClass, |
Eduardo Habkost | 30b5707 | 2020-09-16 14:25:17 -0400 | [diff] [blame] | 25 | TPM_BACKEND) |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 26 | |
Marc-André Lureau | d31076b | 2017-10-10 00:55:49 +0200 | [diff] [blame] | 27 | |
Marc-André Lureau | 0e43b7e | 2017-10-10 00:55:55 +0200 | [diff] [blame] | 28 | typedef struct TPMBackendCmd { |
| 29 | uint8_t locty; |
| 30 | const uint8_t *in; |
| 31 | uint32_t in_len; |
| 32 | uint8_t *out; |
| 33 | uint32_t out_len; |
| 34 | bool selftest_done; |
| 35 | } TPMBackendCmd; |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 36 | |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 37 | struct TPMBackend { |
| 38 | Object parent; |
| 39 | |
| 40 | /*< protected >*/ |
Marc-André Lureau | 8a89c9a | 2017-11-06 19:39:00 +0100 | [diff] [blame] | 41 | TPMIf *tpmif; |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 42 | bool opened; |
Amarnath Valluri | 93330cf | 2017-09-29 14:10:16 +0300 | [diff] [blame] | 43 | bool had_startup_error; |
Marc-André Lureau | c4fb856 | 2018-01-29 19:33:05 +0100 | [diff] [blame] | 44 | TPMBackendCmd *cmd; |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 45 | |
Amarnath Valluri | f59864b | 2017-09-29 14:10:17 +0300 | [diff] [blame] | 46 | /* <public> */ |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 47 | char *id; |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 48 | |
| 49 | QLIST_ENTRY(TPMBackend) list; |
| 50 | }; |
| 51 | |
Amarnath Valluri | b19a5ee | 2017-09-29 14:10:14 +0300 | [diff] [blame] | 52 | struct TPMBackendClass { |
| 53 | ObjectClass parent_class; |
| 54 | |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 55 | enum TpmType type; |
Stefan Berger | bb71623 | 2013-04-22 10:41:39 -0400 | [diff] [blame] | 56 | const QemuOptDesc *opts; |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 57 | /* get a descriptive text of the backend to display to the user */ |
Amarnath Valluri | 93330cf | 2017-09-29 14:10:16 +0300 | [diff] [blame] | 58 | const char *desc; |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 59 | |
Marc-André Lureau | 9f7c0ef | 2017-11-06 19:39:09 +0100 | [diff] [blame] | 60 | TPMBackend *(*create)(QemuOpts *opts); |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 61 | |
Marc-André Lureau | ebca2df | 2017-11-06 19:39:07 +0100 | [diff] [blame] | 62 | /* start up the TPM on the backend - optional */ |
Stefan Berger | 9375c44 | 2017-11-04 19:57:15 -0400 | [diff] [blame] | 63 | int (*startup_tpm)(TPMBackend *t, size_t buffersize); |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 64 | |
Marc-André Lureau | ebca2df | 2017-11-06 19:39:07 +0100 | [diff] [blame] | 65 | /* optional */ |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 66 | void (*reset)(TPMBackend *t); |
| 67 | |
| 68 | void (*cancel_cmd)(TPMBackend *t); |
| 69 | |
Marc-André Lureau | ebca2df | 2017-11-06 19:39:07 +0100 | [diff] [blame] | 70 | /* optional */ |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 71 | bool (*get_tpm_established_flag)(TPMBackend *t); |
Stefan Berger | 116694c | 2015-05-26 16:51:05 -0400 | [diff] [blame] | 72 | |
Marc-André Lureau | ebca2df | 2017-11-06 19:39:07 +0100 | [diff] [blame] | 73 | /* optional */ |
Stefan Berger | 116694c | 2015-05-26 16:51:05 -0400 | [diff] [blame] | 74 | int (*reset_tpm_established_flag)(TPMBackend *t, uint8_t locty); |
| 75 | |
| 76 | TPMVersion (*get_tpm_version)(TPMBackend *t); |
Amarnath Valluri | f59864b | 2017-09-29 14:10:17 +0300 | [diff] [blame] | 77 | |
Stefan Berger | b21e6aa | 2017-11-03 18:10:01 -0400 | [diff] [blame] | 78 | size_t (*get_buffer_size)(TPMBackend *t); |
| 79 | |
Amarnath Valluri | f59864b | 2017-09-29 14:10:17 +0300 | [diff] [blame] | 80 | TpmTypeOptions *(*get_tpm_options)(TPMBackend *t); |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 81 | |
Marc-André Lureau | 6a8a235 | 2018-01-29 19:33:06 +0100 | [diff] [blame] | 82 | void (*handle_request)(TPMBackend *s, TPMBackendCmd *cmd, Error **errp); |
Marc-André Lureau | d31076b | 2017-10-10 00:55:49 +0200 | [diff] [blame] | 83 | }; |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 84 | |
| 85 | /** |
| 86 | * tpm_backend_get_type: |
| 87 | * @s: the backend |
| 88 | * |
| 89 | * Returns the TpmType of the backend. |
| 90 | */ |
| 91 | enum TpmType tpm_backend_get_type(TPMBackend *s); |
| 92 | |
| 93 | /** |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 94 | * tpm_backend_init: |
| 95 | * @s: the backend to initialized |
Marc-André Lureau | 8a89c9a | 2017-11-06 19:39:00 +0100 | [diff] [blame] | 96 | * @tpmif: TPM interface |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 97 | * @datacb: callback for sending data to frontend |
Marc-André Lureau | 0bd6c8a | 2017-11-06 19:39:03 +0100 | [diff] [blame] | 98 | * @errp: a pointer to return the #Error object if an error occurs. |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 99 | * |
| 100 | * Initialize the backend with the given variables. |
| 101 | * |
| 102 | * Returns 0 on success. |
| 103 | */ |
Marc-André Lureau | 0bd6c8a | 2017-11-06 19:39:03 +0100 | [diff] [blame] | 104 | int tpm_backend_init(TPMBackend *s, TPMIf *tpmif, Error **errp); |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 105 | |
| 106 | /** |
| 107 | * tpm_backend_startup_tpm: |
| 108 | * @s: the backend whose TPM support is to be started |
Stefan Berger | 9375c44 | 2017-11-04 19:57:15 -0400 | [diff] [blame] | 109 | * @buffersize: the buffer size the TPM is supposed to use, |
| 110 | * 0 to leave it as-is |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 111 | * |
| 112 | * Returns 0 on success. |
| 113 | */ |
Stefan Berger | 9375c44 | 2017-11-04 19:57:15 -0400 | [diff] [blame] | 114 | int tpm_backend_startup_tpm(TPMBackend *s, size_t buffersize); |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 115 | |
| 116 | /** |
| 117 | * tpm_backend_had_startup_error: |
| 118 | * @s: the backend to query for a statup error |
| 119 | * |
| 120 | * Check whether the backend had an error during startup. Returns |
| 121 | * false if no error occurred and the backend can be used, true |
| 122 | * otherwise. |
| 123 | */ |
| 124 | bool tpm_backend_had_startup_error(TPMBackend *s); |
| 125 | |
| 126 | /** |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 127 | * tpm_backend_deliver_request: |
| 128 | * @s: the backend to send the request to |
Marc-André Lureau | 0e43b7e | 2017-10-10 00:55:55 +0200 | [diff] [blame] | 129 | * @cmd: the command to deliver |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 130 | * |
| 131 | * Send a request to the backend. The backend will then send the request |
| 132 | * to the TPM implementation. |
| 133 | */ |
Marc-André Lureau | 0e43b7e | 2017-10-10 00:55:55 +0200 | [diff] [blame] | 134 | void tpm_backend_deliver_request(TPMBackend *s, TPMBackendCmd *cmd); |
Stefan Berger | 8f0605c | 2013-03-28 07:26:21 -0400 | [diff] [blame] | 135 | |
| 136 | /** |
| 137 | * tpm_backend_reset: |
| 138 | * @s: the backend to reset |
| 139 | * |
| 140 | * Reset the backend into a well defined state with all previous errors |
| 141 | * reset. |
| 142 | */ |
| 143 | void tpm_backend_reset(TPMBackend *s); |
| 144 | |
| 145 | /** |
| 146 | * tpm_backend_cancel_cmd: |
| 147 | * @s: the backend |
| 148 | * |
| 149 | * Cancel any ongoing command being processed by the TPM implementation |
| 150 | * on behalf of the QEMU guest. |
| 151 | */ |
| 152 | void tpm_backend_cancel_cmd(TPMBackend *s); |
| 153 | |
| 154 | /** |
| 155 | * tpm_backend_get_tpm_established_flag: |
| 156 | * @s: the backend |
| 157 | * |
| 158 | * Get the TPM establishment flag. This function may be called very |
| 159 | * frequently by the frontend since for example in the TIS implementation |
| 160 | * this flag is part of a register. |
| 161 | */ |
| 162 | bool tpm_backend_get_tpm_established_flag(TPMBackend *s); |
| 163 | |
| 164 | /** |
Stefan Berger | 116694c | 2015-05-26 16:51:05 -0400 | [diff] [blame] | 165 | * tpm_backend_reset_tpm_established_flag: |
| 166 | * @s: the backend |
| 167 | * @locty: the locality number |
| 168 | * |
| 169 | * Reset the TPM establishment flag. |
| 170 | */ |
| 171 | int tpm_backend_reset_tpm_established_flag(TPMBackend *s, uint8_t locty); |
| 172 | |
| 173 | /** |
Stefan Berger | 116694c | 2015-05-26 16:51:05 -0400 | [diff] [blame] | 174 | * tpm_backend_get_tpm_version: |
| 175 | * @s: the backend to call into |
| 176 | * |
| 177 | * Get the TPM Version that is emulated at the backend. |
| 178 | * |
| 179 | * Returns TPMVersion. |
| 180 | */ |
| 181 | TPMVersion tpm_backend_get_tpm_version(TPMBackend *s); |
| 182 | |
Amarnath Valluri | f59864b | 2017-09-29 14:10:17 +0300 | [diff] [blame] | 183 | /** |
Stefan Berger | b21e6aa | 2017-11-03 18:10:01 -0400 | [diff] [blame] | 184 | * tpm_backend_get_buffer_size: |
| 185 | * @s: the backend to call into |
| 186 | * |
| 187 | * Get the TPM's buffer size. |
| 188 | * |
| 189 | * Returns buffer size. |
| 190 | */ |
| 191 | size_t tpm_backend_get_buffer_size(TPMBackend *s); |
| 192 | |
| 193 | /** |
Marc-André Lureau | c4fb856 | 2018-01-29 19:33:05 +0100 | [diff] [blame] | 194 | * tpm_backend_finish_sync: |
| 195 | * @s: the backend to call into |
| 196 | * |
| 197 | * Finish the pending command synchronously (this will call aio_poll() |
| 198 | * on qemu main AIOContext until it ends) |
| 199 | */ |
| 200 | void tpm_backend_finish_sync(TPMBackend *s); |
| 201 | |
| 202 | /** |
Amarnath Valluri | f59864b | 2017-09-29 14:10:17 +0300 | [diff] [blame] | 203 | * tpm_backend_query_tpm: |
| 204 | * @s: the backend |
| 205 | * |
| 206 | * Query backend tpm info |
| 207 | * |
| 208 | * Returns newly allocated TPMInfo |
| 209 | */ |
| 210 | TPMInfo *tpm_backend_query_tpm(TPMBackend *s); |
| 211 | |
Marc-André Lureau | d36e7db | 2017-11-06 19:39:14 +0100 | [diff] [blame] | 212 | TPMBackend *qemu_find_tpm_be(const char *id); |
Paolo Bonzini | bdee56f | 2013-04-02 18:28:41 +0200 | [diff] [blame] | 213 | |
Stefan Berger | e542b71 | 2021-06-15 16:21:20 +0200 | [diff] [blame] | 214 | #endif /* CONFIG_TPM */ |
| 215 | |
| 216 | #endif /* TPM_BACKEND_H */ |