tpm-common.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2013 The Chromium OS Authors.
  4. * Coypright (c) 2013 Guntermann & Drunck GmbH
  5. */
  6. #include <common.h>
  7. #include <dm.h>
  8. #include <asm/unaligned.h>
  9. #include <tpm-common.h>
  10. #include "tpm-utils.h"
  11. int pack_byte_string(u8 *str, size_t size, const char *format, ...)
  12. {
  13. va_list args;
  14. size_t offset = 0, length = 0;
  15. u8 *data = NULL;
  16. u32 value = 0;
  17. va_start(args, format);
  18. for (; *format; format++) {
  19. switch (*format) {
  20. case 'b':
  21. offset = va_arg(args, size_t);
  22. value = va_arg(args, int);
  23. length = 1;
  24. break;
  25. case 'w':
  26. offset = va_arg(args, size_t);
  27. value = va_arg(args, int);
  28. length = 2;
  29. break;
  30. case 'd':
  31. offset = va_arg(args, size_t);
  32. value = va_arg(args, u32);
  33. length = 4;
  34. break;
  35. case 's':
  36. offset = va_arg(args, size_t);
  37. data = va_arg(args, u8 *);
  38. length = va_arg(args, u32);
  39. break;
  40. default:
  41. debug("Couldn't recognize format string\n");
  42. va_end(args);
  43. return -1;
  44. }
  45. if (offset + length > size) {
  46. va_end(args);
  47. return -1;
  48. }
  49. switch (*format) {
  50. case 'b':
  51. str[offset] = value;
  52. break;
  53. case 'w':
  54. put_unaligned_be16(value, str + offset);
  55. break;
  56. case 'd':
  57. put_unaligned_be32(value, str + offset);
  58. break;
  59. case 's':
  60. memcpy(str + offset, data, length);
  61. break;
  62. }
  63. }
  64. va_end(args);
  65. return 0;
  66. }
  67. int unpack_byte_string(const u8 *str, size_t size, const char *format, ...)
  68. {
  69. va_list args;
  70. size_t offset = 0, length = 0;
  71. u8 *ptr8 = NULL;
  72. u16 *ptr16 = NULL;
  73. u32 *ptr32 = NULL;
  74. va_start(args, format);
  75. for (; *format; format++) {
  76. switch (*format) {
  77. case 'b':
  78. offset = va_arg(args, size_t);
  79. ptr8 = va_arg(args, u8 *);
  80. length = 1;
  81. break;
  82. case 'w':
  83. offset = va_arg(args, size_t);
  84. ptr16 = va_arg(args, u16 *);
  85. length = 2;
  86. break;
  87. case 'd':
  88. offset = va_arg(args, size_t);
  89. ptr32 = va_arg(args, u32 *);
  90. length = 4;
  91. break;
  92. case 's':
  93. offset = va_arg(args, size_t);
  94. ptr8 = va_arg(args, u8 *);
  95. length = va_arg(args, u32);
  96. break;
  97. default:
  98. va_end(args);
  99. debug("Couldn't recognize format string\n");
  100. return -1;
  101. }
  102. if (offset + length > size) {
  103. va_end(args);
  104. return -1;
  105. }
  106. switch (*format) {
  107. case 'b':
  108. *ptr8 = str[offset];
  109. break;
  110. case 'w':
  111. *ptr16 = get_unaligned_be16(str + offset);
  112. break;
  113. case 'd':
  114. *ptr32 = get_unaligned_be32(str + offset);
  115. break;
  116. case 's':
  117. memcpy(ptr8, str + offset, length);
  118. break;
  119. }
  120. }
  121. va_end(args);
  122. return 0;
  123. }
  124. u32 tpm_command_size(const void *command)
  125. {
  126. const size_t command_size_offset = 2;
  127. return get_unaligned_be32(command + command_size_offset);
  128. }
  129. u32 tpm_return_code(const void *response)
  130. {
  131. const size_t return_code_offset = 6;
  132. return get_unaligned_be32(response + return_code_offset);
  133. }
  134. u32 tpm_sendrecv_command(const void *command, void *response, size_t *size_ptr)
  135. {
  136. struct udevice *dev;
  137. int err, ret;
  138. u8 response_buffer[COMMAND_BUFFER_SIZE];
  139. size_t response_length;
  140. if (response) {
  141. response_length = *size_ptr;
  142. } else {
  143. response = response_buffer;
  144. response_length = sizeof(response_buffer);
  145. }
  146. ret = uclass_first_device_err(UCLASS_TPM, &dev);
  147. if (ret)
  148. return ret;
  149. err = tpm_xfer(dev, command, tpm_command_size(command),
  150. response, &response_length);
  151. if (err < 0)
  152. return TPM_LIB_ERROR;
  153. if (size_ptr)
  154. *size_ptr = response_length;
  155. return tpm_return_code(response);
  156. }
  157. int tpm_init(void)
  158. {
  159. struct udevice *dev;
  160. int err;
  161. err = uclass_first_device_err(UCLASS_TPM, &dev);
  162. if (err)
  163. return err;
  164. return tpm_open(dev);
  165. }