1 /*
2 * PROJECT: ReactOS Kernel
3 * LICENSE: GPL - See COPYING in the top level directory
4 * FILE: lib/cmlib/cmkeydel.c
5 * PURPOSE: Configuration Manager Library - Key Body Deletion
6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
7 */
8
9 /* INCLUDES ******************************************************************/
10
11 #include "cmlib.h"
12 #define NDEBUG
13 #include <debug.h>
14
15 /* FUNCTIONS *****************************************************************/
16
17 BOOLEAN
18 NTAPI
CmpMarkKeyDirty(IN PHHIVE Hive,IN HCELL_INDEX Cell,IN BOOLEAN CheckNoSubkeys)19 CmpMarkKeyDirty(IN PHHIVE Hive,
20 IN HCELL_INDEX Cell,
21 IN BOOLEAN CheckNoSubkeys)
22 {
23 PCM_KEY_NODE CellData;
24 PCM_KEY_SECURITY SecurityData;
25 PCELL_DATA ListData, ValueData;
26 ULONG i;
27
28 /* Get the cell data for our target */
29 CellData = (PCM_KEY_NODE)HvGetCell(Hive, Cell);
30 if (!CellData) return FALSE;
31
32 /* Check if sanity checks requested */
33 if (CheckNoSubkeys)
34 {
35 /* Do them */
36 ASSERT(CellData->SubKeyCounts[Stable] == 0);
37 ASSERT(CellData->SubKeyCounts[Volatile] == 0);
38 }
39
40 /* If this is an exit node, there's nothing to do */
41 if (CellData->Flags & KEY_HIVE_EXIT)
42 {
43 /* Release the cell and get out */
44 HvReleaseCell(Hive, Cell);
45 return TRUE;
46 }
47
48 /* Otherwise, mark it dirty and release it */
49 HvMarkCellDirty(Hive, Cell, FALSE);
50 HvReleaseCell(Hive, Cell);
51
52 /* Check if we have a class */
53 if (CellData->Class != HCELL_NIL)
54 {
55 /* Mark it dirty */
56 HvMarkCellDirty(Hive, CellData->Class, FALSE);
57 }
58
59 /* Check if we have security */
60 if (CellData->Security != HCELL_NIL)
61 {
62 /* Mark it dirty */
63 HvMarkCellDirty(Hive, CellData->Security, FALSE);
64
65 /* Get the security data and release it */
66 SecurityData = (PCM_KEY_SECURITY)HvGetCell(Hive, CellData->Security);
67 ASSERT(SecurityData);
68 HvReleaseCell(Hive, CellData->Security);
69
70 /* Mark the security links dirty too */
71 HvMarkCellDirty(Hive, SecurityData->Flink, FALSE);
72 HvMarkCellDirty(Hive, SecurityData->Blink, FALSE);
73 }
74
75 // TODO: Handle predefined keys (Flags: KEY_PREDEF_HANDLE)
76 /* Check if we have any values */
77 if (CellData->ValueList.Count > 0)
78 {
79 /* Dirty the value list */
80 HvMarkCellDirty(Hive, CellData->ValueList.List, FALSE);
81
82 /* Get the list data itself, and release it */
83 ListData = HvGetCell(Hive, CellData->ValueList.List);
84 ASSERT(ListData);
85 HvReleaseCell(Hive, CellData->ValueList.List);
86
87 /* Loop all values */
88 for (i = 0; i < CellData->ValueList.Count; i++)
89 {
90 /* Dirty each value */
91 HvMarkCellDirty(Hive, ListData->u.KeyList[i], FALSE);
92
93 /* Get the value data and release it */
94 ValueData = HvGetCell(Hive, ListData->u.KeyList[i]);
95 ASSERT(ValueData);
96 HvReleaseCell(Hive,ListData->u.KeyList[i]);
97
98 /* Mark the value data dirty too */
99 if (!CmpMarkValueDataDirty(Hive, &ValueData->u.KeyValue))
100 {
101 /* Failure */
102 return FALSE;
103 }
104 }
105 }
106
107 /* If this is an entry node, we're done */
108 if (CellData->Flags & KEY_HIVE_ENTRY) return TRUE;
109
110 /* Otherwise mark the index dirty too */
111 if (!CmpMarkIndexDirty(Hive, CellData->Parent, Cell))
112 {
113 /* Failure */
114 return FALSE;
115 }
116
117 /* Finally, mark the parent dirty */
118 HvMarkCellDirty(Hive, CellData->Parent, FALSE);
119 return TRUE;
120 }
121
122 BOOLEAN
123 NTAPI
CmpFreeKeyBody(IN PHHIVE Hive,IN HCELL_INDEX Cell)124 CmpFreeKeyBody(IN PHHIVE Hive,
125 IN HCELL_INDEX Cell)
126 {
127 PCM_KEY_NODE CellData;
128
129 /* Get the key node */
130 CellData = (PCM_KEY_NODE)HvGetCell(Hive, Cell);
131 ASSERT(CellData);
132
133 /* Check if we can delete the child cells */
134 if (!(CellData->Flags & KEY_HIVE_EXIT))
135 {
136 /* Check if we have a security cell */
137 if (CellData->Security != HCELL_NIL)
138 {
139 /* Free the security cell */
140 HvFreeCell(Hive, CellData->Security);
141 }
142
143 /* Check if we have a class */
144 if (CellData->ClassLength > 0)
145 {
146 /* Free it */
147 HvFreeCell(Hive, CellData->Class);
148 }
149 }
150
151 /* Release and free the cell */
152 HvReleaseCell(Hive, Cell);
153 HvFreeCell(Hive, Cell);
154 return TRUE;
155 }
156
157 NTSTATUS
158 NTAPI
CmpFreeKeyByCell(IN PHHIVE Hive,IN HCELL_INDEX Cell,IN BOOLEAN Unlink)159 CmpFreeKeyByCell(IN PHHIVE Hive,
160 IN HCELL_INDEX Cell,
161 IN BOOLEAN Unlink)
162 {
163 PCM_KEY_NODE CellData, ParentData;
164 PCELL_DATA ListData;
165 ULONG i;
166 BOOLEAN Result;
167
168 /* Mark the entire key dirty */
169 CmpMarkKeyDirty(Hive, Cell, TRUE);
170
171 /* Get the target node and release it */
172 CellData = (PCM_KEY_NODE)HvGetCell(Hive, Cell);
173 ASSERT(CellData);
174 HvReleaseCell(Hive, Cell);
175
176 /* Make sure we don't have subkeys */
177 ASSERT(CellData->SubKeyCounts[Stable] + CellData->SubKeyCounts[Volatile] == 0);
178
179 /* Check if we have to unlink */
180 if (Unlink)
181 {
182 /* Remove the subkey */
183 Result = CmpRemoveSubKey(Hive, CellData->Parent, Cell);
184 if (!Result) return STATUS_INSUFFICIENT_RESOURCES;
185
186 /* Get the parent node and release it */
187 ParentData = (PCM_KEY_NODE)HvGetCell(Hive, CellData->Parent);
188 ASSERT(ParentData);
189 HvReleaseCell(Hive, CellData->Parent);
190
191 /* Check if the parent node has no more subkeys */
192 if (ParentData->SubKeyCounts[Stable] + ParentData->SubKeyCounts[Volatile] == 0)
193 {
194 /* Then free the cached name/class lengths */
195 ParentData->MaxNameLen = 0;
196 ParentData->MaxClassLen = 0;
197 }
198 }
199
200 // TODO: Handle predefined keys (Flags: KEY_PREDEF_HANDLE)
201 /* If this is an exit node, we don't have values */
202 if (!(CellData->Flags & KEY_HIVE_EXIT))
203 {
204 /* Check if we have any values */
205 if (CellData->ValueList.Count > 0)
206 {
207 /* Get the value list and release it */
208 ListData = HvGetCell(Hive, CellData->ValueList.List);
209 ASSERT(ListData);
210 HvReleaseCell(Hive, CellData->ValueList.List);
211
212 /* Loop every value */
213 for (i = 0; i < CellData->ValueList.Count; i++)
214 {
215 /* Free it */
216 Result = CmpFreeValue(Hive, ListData->u.KeyList[i]);
217 ASSERT(Result);
218 }
219
220 /* Free the value list */
221 HvFreeCell(Hive, CellData->ValueList.List);
222 }
223
224 /* Free the key security descriptor */
225 CmpFreeSecurityDescriptor(Hive, Cell);
226 }
227
228 /* Free the key body itself, and then return our status */
229 if (!CmpFreeKeyBody(Hive, Cell)) return STATUS_INSUFFICIENT_RESOURCES;
230 return STATUS_SUCCESS;
231 }
232