uuid.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Unified UUID/GUID definition
  4. *
  5. * Copyright (C) 2009, 2016 Intel Corp.
  6. * Huang Ying <ying.huang@intel.com>
  7. */
  8. #include <linux/kernel.h>
  9. #include <linux/ctype.h>
  10. #include <linux/errno.h>
  11. #include <linux/export.h>
  12. #include <linux/hex.h>
  13. #include <linux/uuid.h>
  14. #include <linux/random.h>
  15. const guid_t guid_null;
  16. EXPORT_SYMBOL(guid_null);
  17. const uuid_t uuid_null;
  18. EXPORT_SYMBOL(uuid_null);
  19. const u8 guid_index[16] = {3,2,1,0,5,4,7,6,8,9,10,11,12,13,14,15};
  20. const u8 uuid_index[16] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15};
  21. /**
  22. * generate_random_uuid - generate a random UUID
  23. * @uuid: where to put the generated UUID
  24. *
  25. * Random UUID interface
  26. *
  27. * Used to create a Boot ID or a filesystem UUID/GUID, but can be
  28. * useful for other kernel drivers.
  29. */
  30. void generate_random_uuid(unsigned char uuid[16])
  31. {
  32. get_random_bytes(uuid, 16);
  33. /* Set UUID version to 4 --- truly random generation */
  34. uuid[6] = (uuid[6] & 0x0F) | 0x40;
  35. /* Set the UUID variant to DCE */
  36. uuid[8] = (uuid[8] & 0x3F) | 0x80;
  37. }
  38. EXPORT_SYMBOL(generate_random_uuid);
  39. void generate_random_guid(unsigned char guid[16])
  40. {
  41. get_random_bytes(guid, 16);
  42. /* Set GUID version to 4 --- truly random generation */
  43. guid[7] = (guid[7] & 0x0F) | 0x40;
  44. /* Set the GUID variant to DCE */
  45. guid[8] = (guid[8] & 0x3F) | 0x80;
  46. }
  47. EXPORT_SYMBOL(generate_random_guid);
  48. static void __uuid_gen_common(__u8 b[16])
  49. {
  50. get_random_bytes(b, 16);
  51. /* reversion 0b10 */
  52. b[8] = (b[8] & 0x3F) | 0x80;
  53. }
  54. void guid_gen(guid_t *lu)
  55. {
  56. __uuid_gen_common(lu->b);
  57. /* version 4 : random generation */
  58. lu->b[7] = (lu->b[7] & 0x0F) | 0x40;
  59. }
  60. EXPORT_SYMBOL_GPL(guid_gen);
  61. void uuid_gen(uuid_t *bu)
  62. {
  63. __uuid_gen_common(bu->b);
  64. /* version 4 : random generation */
  65. bu->b[6] = (bu->b[6] & 0x0F) | 0x40;
  66. }
  67. EXPORT_SYMBOL_GPL(uuid_gen);
  68. /**
  69. * uuid_is_valid - checks if a UUID string is valid
  70. * @uuid: UUID string to check
  71. *
  72. * Description:
  73. * It checks if the UUID string is following the format:
  74. * xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
  75. *
  76. * where x is a hex digit.
  77. *
  78. * Return: true if input is valid UUID string.
  79. */
  80. bool uuid_is_valid(const char *uuid)
  81. {
  82. unsigned int i;
  83. for (i = 0; i < UUID_STRING_LEN; i++) {
  84. if (i == 8 || i == 13 || i == 18 || i == 23) {
  85. if (uuid[i] != '-')
  86. return false;
  87. } else if (!isxdigit(uuid[i])) {
  88. return false;
  89. }
  90. }
  91. return true;
  92. }
  93. EXPORT_SYMBOL(uuid_is_valid);
  94. static int __uuid_parse(const char *uuid, __u8 b[16], const u8 ei[16])
  95. {
  96. static const u8 si[16] = {0,2,4,6,9,11,14,16,19,21,24,26,28,30,32,34};
  97. unsigned int i;
  98. if (!uuid_is_valid(uuid))
  99. return -EINVAL;
  100. for (i = 0; i < 16; i++) {
  101. int hi = hex_to_bin(uuid[si[i] + 0]);
  102. int lo = hex_to_bin(uuid[si[i] + 1]);
  103. b[ei[i]] = (hi << 4) | lo;
  104. }
  105. return 0;
  106. }
  107. int guid_parse(const char *uuid, guid_t *u)
  108. {
  109. return __uuid_parse(uuid, u->b, guid_index);
  110. }
  111. EXPORT_SYMBOL(guid_parse);
  112. int uuid_parse(const char *uuid, uuid_t *u)
  113. {
  114. return __uuid_parse(uuid, u->b, uuid_index);
  115. }
  116. EXPORT_SYMBOL(uuid_parse);