uuid.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  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 u8 uuid_le_index[16] = {3,2,1,0,5,4,7,6,8,9,10,11,12,13,14,15};
  23. EXPORT_SYMBOL(uuid_le_index);
  24. const u8 uuid_be_index[16] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15};
  25. EXPORT_SYMBOL(uuid_be_index);
  26. /***************************************************************
  27. * Random UUID interface
  28. *
  29. * Used here for a Boot ID, but can be useful for other kernel
  30. * drivers.
  31. ***************************************************************/
  32. /*
  33. * Generate random UUID
  34. */
  35. void generate_random_uuid(unsigned char uuid[16])
  36. {
  37. get_random_bytes(uuid, 16);
  38. /* Set UUID version to 4 --- truly random generation */
  39. uuid[6] = (uuid[6] & 0x0F) | 0x40;
  40. /* Set the UUID variant to DCE */
  41. uuid[8] = (uuid[8] & 0x3F) | 0x80;
  42. }
  43. EXPORT_SYMBOL(generate_random_uuid);
  44. static void __uuid_gen_common(__u8 b[16])
  45. {
  46. prandom_bytes(b, 16);
  47. /* reversion 0b10 */
  48. b[8] = (b[8] & 0x3F) | 0x80;
  49. }
  50. void uuid_le_gen(uuid_le *lu)
  51. {
  52. __uuid_gen_common(lu->b);
  53. /* version 4 : random generation */
  54. lu->b[7] = (lu->b[7] & 0x0F) | 0x40;
  55. }
  56. EXPORT_SYMBOL_GPL(uuid_le_gen);
  57. void uuid_be_gen(uuid_be *bu)
  58. {
  59. __uuid_gen_common(bu->b);
  60. /* version 4 : random generation */
  61. bu->b[6] = (bu->b[6] & 0x0F) | 0x40;
  62. }
  63. EXPORT_SYMBOL_GPL(uuid_be_gen);
  64. /**
  65. * uuid_is_valid - checks if UUID string valid
  66. * @uuid: UUID string to check
  67. *
  68. * Description:
  69. * It checks if the UUID string is following the format:
  70. * xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
  71. * where x is a hex digit.
  72. *
  73. * Return: true if input is valid UUID string.
  74. */
  75. bool uuid_is_valid(const char *uuid)
  76. {
  77. unsigned int i;
  78. for (i = 0; i < UUID_STRING_LEN; i++) {
  79. if (i == 8 || i == 13 || i == 18 || i == 23) {
  80. if (uuid[i] != '-')
  81. return false;
  82. } else if (!isxdigit(uuid[i])) {
  83. return false;
  84. }
  85. }
  86. return true;
  87. }
  88. EXPORT_SYMBOL(uuid_is_valid);
  89. static int __uuid_to_bin(const char *uuid, __u8 b[16], const u8 ei[16])
  90. {
  91. static const u8 si[16] = {0,2,4,6,9,11,14,16,19,21,24,26,28,30,32,34};
  92. unsigned int i;
  93. if (!uuid_is_valid(uuid))
  94. return -EINVAL;
  95. for (i = 0; i < 16; i++) {
  96. int hi = hex_to_bin(uuid[si[i] + 0]);
  97. int lo = hex_to_bin(uuid[si[i] + 1]);
  98. b[ei[i]] = (hi << 4) | lo;
  99. }
  100. return 0;
  101. }
  102. int uuid_le_to_bin(const char *uuid, uuid_le *u)
  103. {
  104. return __uuid_to_bin(uuid, u->b, uuid_le_index);
  105. }
  106. EXPORT_SYMBOL(uuid_le_to_bin);
  107. int uuid_be_to_bin(const char *uuid, uuid_be *u)
  108. {
  109. return __uuid_to_bin(uuid, u->b, uuid_be_index);
  110. }
  111. EXPORT_SYMBOL(uuid_be_to_bin);