xref: /reactos/sdk/include/ddk/storduid.h (revision 2b933529)
1 #ifndef _STORDUID_H_
2 #define _STORDUID_H_
3 
4 typedef enum _DUID_MATCH_STATUS
5 {
6   DuidExactMatch = 0,
7   DuidSubIdMatch,
8   DuidNoMatch,
9   DuidErrorGeneral = 100,
10   DuidErrorMissingDuid,
11   DuidErrorVersionMismatch,
12   DuidErrorInvalidDuid,
13   DuidErrorInvalidDeviceIdDescSize,
14   DuidErrorInvalidDeviceDescSize,
15   DuidErrorInvalidLayoutSigSize,
16   DuidErrorInvalidLayoutSigVersion,
17   DuidErrorMaximum
18 } DUID_MATCH_STATUS;
19 
20 #define DUID_VERSION_1 1
21 
22 #define DUID_HARDWARE_IDS_ONLY 0
23 #define DUID_INCLUDE_SOFTWARE_IDS 1
24 
25 #define DUID_MATCH_ERROR(_duid_status) ((_duid_status) >= DuidErrorGeneral ? TRUE : FALSE)
26 #define DUID_MATCH_SUCCESS(_duid_status) ((_duid_status) < DuidErrorGeneral ? TRUE : FALSE)
27 
28 typedef struct _STORAGE_DEVICE_UNIQUE_IDENTIFIER
29 {
30   ULONG Version;
31   ULONG Size;
32   ULONG StorageDeviceIdOffset;
33   ULONG StorageDeviceOffset;
34   ULONG DriveLayoutSignatureOffset;
35 } STORAGE_DEVICE_UNIQUE_IDENTIFIER, *PSTORAGE_DEVICE_UNIQUE_IDENTIFIER;
36 
37 typedef struct _STORAGE_DEVICE_LAYOUT_SIGNATURE
38 {
39   ULONG Version;
40   ULONG Size;
41   BOOLEAN Mbr;
42   union {
43     ULONG MbrSignature;
44     GUID GptDiskId;
45   } DeviceSpecific;
46 } STORAGE_DEVICE_LAYOUT_SIGNATURE, *PSTORAGE_DEVICE_LAYOUT_SIGNATURE;
47 
48 FORCEINLINE
49 DUID_MATCH_STATUS
50 CompareStorageDuids(
51   _In_ PSTORAGE_DEVICE_UNIQUE_IDENTIFIER Duid1,
52   _In_ PSTORAGE_DEVICE_UNIQUE_IDENTIFIER Duid2);
53 
54 FORCEINLINE
55 DUID_MATCH_STATUS
56 CompareStorageDuids(
57   _In_ PSTORAGE_DEVICE_UNIQUE_IDENTIFIER Duid1,
58   _In_ PSTORAGE_DEVICE_UNIQUE_IDENTIFIER Duid2)
59 {
60   if (!Duid1 || !Duid2)
61   {
62     return DuidErrorMissingDuid;
63   }
64 
65   if (Duid1->Size < sizeof(STORAGE_DEVICE_UNIQUE_IDENTIFIER) ||
66       Duid2->Size < sizeof(STORAGE_DEVICE_UNIQUE_IDENTIFIER))
67   {
68     return DuidErrorGeneral;
69   }
70 
71   if (Duid1->Version != DUID_VERSION_1 || Duid2->Version != DUID_VERSION_1)
72   {
73     return DuidErrorVersionMismatch;
74   }
75 
76   if (Duid1->StorageDeviceIdOffset == 0 && Duid1->StorageDeviceOffset == 0 &&
77       Duid1->DriveLayoutSignatureOffset == 0)
78   {
79     return DuidErrorInvalidDuid;
80   }
81 
82   if (Duid2->StorageDeviceIdOffset == 0 && Duid2->StorageDeviceOffset == 0 &&
83       Duid2->DriveLayoutSignatureOffset == 0)
84   {
85     return DuidErrorInvalidDuid;
86   }
87 
88   if (Duid1->Size == Duid2->Size)
89   {
90     if (memcmp(Duid1, Duid2, Duid1->Size) == 0)
91     {
92       return DuidExactMatch;
93     }
94   }
95 
96   if (Duid1->StorageDeviceIdOffset && Duid2->StorageDeviceIdOffset)
97   {
98     PSTORAGE_DEVICE_ID_DESCRIPTOR idDesc1;
99     PSTORAGE_DEVICE_ID_DESCRIPTOR idDesc2;
100 
101     PSTORAGE_IDENTIFIER ident1;
102     PSTORAGE_IDENTIFIER ident2;
103 
104     ULONG idx1;
105     ULONG idx2;
106 
107     idDesc1 = (PSTORAGE_DEVICE_ID_DESCRIPTOR)((PUCHAR)Duid1 + Duid1->StorageDeviceIdOffset);
108     idDesc2 = (PSTORAGE_DEVICE_ID_DESCRIPTOR)((PUCHAR)Duid2 + Duid2->StorageDeviceIdOffset);
109 
110     if (idDesc1->Size < sizeof(STORAGE_DEVICE_ID_DESCRIPTOR) ||
111         idDesc2->Size < sizeof(STORAGE_DEVICE_ID_DESCRIPTOR))
112     {
113       return DuidErrorInvalidDeviceIdDescSize;
114     }
115 
116     if (idDesc1->Size == idDesc2->Size)
117     {
118       if (memcmp(idDesc1, idDesc2, idDesc1->Size) == 0)
119       {
120         return DuidSubIdMatch;
121       }
122     }
123 
124     ident1 = (PSTORAGE_IDENTIFIER)(idDesc1->Identifiers);
125 
126     for (idx1 = 0; idx1 < idDesc1->NumberOfIdentifiers; idx1++)
127     {
128       if ((ident1->Type == StorageIdTypeScsiNameString || ident1->Type == StorageIdTypeFCPHName ||
129            ident1->Type == StorageIdTypeEUI64 || ident1->Type == StorageIdTypeVendorId) &&
130           (ident1->Association == StorageIdAssocPort) &&
131           (ident1->CodeSet == StorageIdCodeSetUtf8 || ident1->CodeSet == StorageIdCodeSetAscii ||
132            ident1->CodeSet == StorageIdCodeSetBinary))
133       {
134         ident2 = (PSTORAGE_IDENTIFIER)(idDesc2->Identifiers);
135 
136         for (idx2 = 0; idx2 < idDesc2->NumberOfIdentifiers; idx2++)
137         {
138           if (ident1->Type == ident2->Type && ident1->Association == ident2->Association &&
139               ident1->CodeSet == ident2->CodeSet &&
140               ident1->IdentifierSize == ident2->IdentifierSize &&
141               (memcmp(ident1->Identifier, ident2->Identifier, ident1->IdentifierSize) == 0))
142           {
143             return DuidSubIdMatch;
144           }
145 
146           ident2 = (PSTORAGE_IDENTIFIER)((PUCHAR)ident2 + ident2->NextOffset);
147         }
148       }
149 
150       ident1 = (PSTORAGE_IDENTIFIER)((PUCHAR)ident1 + ident1->NextOffset);
151     }
152   }
153 
154   if (Duid1->StorageDeviceOffset && Duid2->StorageDeviceOffset)
155   {
156     PSTORAGE_DEVICE_DESCRIPTOR desc1;
157     PSTORAGE_DEVICE_DESCRIPTOR desc2;
158 
159     desc1 = (PSTORAGE_DEVICE_DESCRIPTOR)((PUCHAR)Duid1 + Duid1->StorageDeviceOffset);
160     desc2 = (PSTORAGE_DEVICE_DESCRIPTOR)((PUCHAR)Duid2 + Duid2->StorageDeviceOffset);
161 
162     if (desc1->Size < sizeof(STORAGE_DEVICE_DESCRIPTOR) ||
163         desc2->Size < sizeof(STORAGE_DEVICE_DESCRIPTOR))
164     {
165       return DuidErrorInvalidDeviceDescSize;
166     }
167 
168     if (desc1->Size == desc2->Size)
169     {
170       if (memcmp(desc1, desc2, desc1->Size) == 0)
171       {
172         return DuidSubIdMatch;
173       }
174     }
175 
176     if (desc1->SerialNumberOffset && desc2->SerialNumberOffset)
177     {
178       const char *string1;
179       const char *string2;
180 
181       string1 = (const char *)((PUCHAR)desc1 + desc1->SerialNumberOffset);
182       string2 = (const char *)((PUCHAR)desc2 + desc2->SerialNumberOffset);
183 
184       if (strcmp(string1, string2) == 0)
185       {
186         if (desc1->VendorIdOffset && desc2->VendorIdOffset)
187         {
188           string1 = (const char *)((PUCHAR)desc1 + desc1->VendorIdOffset);
189           string2 = (const char *)((PUCHAR)desc2 + desc2->VendorIdOffset);
190 
191           if (strcmp(string1, string2) != 0)
192           {
193             return DuidNoMatch;
194           }
195         }
196 
197         if (desc1->ProductIdOffset && desc2->ProductIdOffset)
198         {
199           string1 = (const char *)((PUCHAR)desc1 + desc1->ProductIdOffset);
200           string2 = (const char *)((PUCHAR)desc2 + desc2->ProductIdOffset);
201 
202           if (strcmp(string1, string2) != 0)
203           {
204             return DuidNoMatch;
205           }
206         }
207 
208         return DuidSubIdMatch;
209       }
210     }
211   }
212 
213   if (Duid1->DriveLayoutSignatureOffset && Duid2->DriveLayoutSignatureOffset)
214   {
215     PSTORAGE_DEVICE_LAYOUT_SIGNATURE sig1;
216     PSTORAGE_DEVICE_LAYOUT_SIGNATURE sig2;
217 
218     sig1 = (PSTORAGE_DEVICE_LAYOUT_SIGNATURE)((PUCHAR)Duid1 + Duid1->DriveLayoutSignatureOffset);
219     sig2 = (PSTORAGE_DEVICE_LAYOUT_SIGNATURE)((PUCHAR)Duid2 + Duid2->DriveLayoutSignatureOffset);
220 
221     if (sig1->Version != DUID_VERSION_1 && sig2->Version != DUID_VERSION_1)
222     {
223       return DuidErrorInvalidLayoutSigVersion;
224     }
225 
226     if (sig1->Size < sizeof(STORAGE_DEVICE_LAYOUT_SIGNATURE) ||
227         sig2->Size < sizeof(STORAGE_DEVICE_LAYOUT_SIGNATURE))
228     {
229       return DuidErrorInvalidLayoutSigSize;
230     }
231 
232     if (memcmp(sig1, sig2, sizeof(STORAGE_DEVICE_LAYOUT_SIGNATURE)) == 0)
233     {
234       return DuidSubIdMatch;
235     }
236   }
237 
238   return DuidNoMatch;
239 }
240 
241 #endif /* _STORDUID_H_ */
242