uuid.c 3.0 KB

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