blob: 9dbd18639b2e36aadd6748434c92ac2f770fc212 [file] [log] [blame]
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001/*
Fred Drake270e19b2000-06-29 16:14:14 +00002 _winreg.c
Guido van Rossum9f3712c2000-03-28 20:37:15 +00003
4 Windows Registry access module for Python.
5
6 * Simple registry access written by Mark Hammond in win32api
7 module circa 1995.
8 * Bill Tutt expanded the support significantly not long after.
9 * Numerous other people have submitted patches since then.
10 * Ripped from win32api module 03-Feb-2000 by Mark Hammond, and
11 basic Unicode support added.
12
13*/
14
Guido van Rossum9f3712c2000-03-28 20:37:15 +000015#include "Python.h"
16#include "structmember.h"
17#include "malloc.h" /* for alloca */
Guido van Rossume7ba4952007-06-06 23:52:48 +000018#include "windows.h"
Guido van Rossum9f3712c2000-03-28 20:37:15 +000019
20static BOOL PyHKEY_AsHKEY(PyObject *ob, HKEY *pRes, BOOL bNoneOK);
21static PyObject *PyHKEY_FromHKEY(HKEY h);
22static BOOL PyHKEY_Close(PyObject *obHandle);
23
24static char errNotAHandle[] = "Object is not a handle";
25
26/* The win32api module reports the function name that failed,
27 but this concept is not in the Python core.
28 Hopefully it will one day, and in the meantime I dont
29 want to lose this info...
30*/
31#define PyErr_SetFromWindowsErrWithFunction(rc, fnname) \
32 PyErr_SetFromWindowsErr(rc)
33
34/* Forward declares */
35
36/* Doc strings */
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000037PyDoc_STRVAR(module_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +000038"This module provides access to the Windows registry API.\n"
39"\n"
40"Functions:\n"
41"\n"
42"CloseKey() - Closes a registry key.\n"
43"ConnectRegistry() - Establishes a connection to a predefined registry handle\n"
44" on another computer.\n"
45"CreateKey() - Creates the specified key, or opens it if it already exists.\n"
46"DeleteKey() - Deletes the specified key.\n"
47"DeleteValue() - Removes a named value from the specified registry key.\n"
48"EnumKey() - Enumerates subkeys of the specified open registry key.\n"
49"EnumValue() - Enumerates values of the specified open registry key.\n"
50"FlushKey() - Writes all the attributes of the specified key to the registry.\n"
51"LoadKey() - Creates a subkey under HKEY_USER or HKEY_LOCAL_MACHINE and stores\n"
52" registration information from a specified file into that subkey.\n"
53"OpenKey() - Alias for <om win32api.RegOpenKeyEx>\n"
54"OpenKeyEx() - Opens the specified key.\n"
55"QueryValue() - Retrieves the value associated with the unnamed value for a\n"
56" specified key in the registry.\n"
57"QueryValueEx() - Retrieves the type and data for a specified value name\n"
58" associated with an open registry key.\n"
59"QueryInfoKey() - Returns information about the specified key.\n"
60"SaveKey() - Saves the specified key, and all its subkeys a file.\n"
61"SetValue() - Associates a value with a specified key.\n"
62"SetValueEx() - Stores data in the value field of an open registry key.\n"
63"\n"
64"Special objects:\n"
65"\n"
66"HKEYType -- type object for HKEY objects\n"
67"error -- exception raised for Win32 errors\n"
68"\n"
69"Integer constants:\n"
70"Many constants are defined - see the documentation for each function\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000071"to see what constants are used, and where.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +000072
73
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000074PyDoc_STRVAR(CloseKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +000075"CloseKey(hkey) - Closes a previously opened registry key.\n"
76"\n"
77"The hkey argument specifies a previously opened key.\n"
78"\n"
79"Note that if the key is not closed using this method, it will be\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000080"closed when the hkey object is destroyed by Python.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +000081
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000082PyDoc_STRVAR(ConnectRegistry_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +000083"key = ConnectRegistry(computer_name, key) - "
84"Establishes a connection to a predefined registry handle on another computer.\n"
85"\n"
86"computer_name is the name of the remote computer, of the form \\\\computername.\n"
87" If None, the local computer is used.\n"
88"key is the predefined handle to connect to.\n"
89"\n"
90"The return value is the handle of the opened key.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000091"If the function fails, an EnvironmentError exception is raised.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +000092
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +000093PyDoc_STRVAR(CreateKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +000094"key = CreateKey(key, sub_key) - Creates or opens the specified key.\n"
95"\n"
96"key is an already open key, or one of the predefined HKEY_* constants\n"
97"sub_key is a string that names the key this method opens or creates.\n"
98" If key is one of the predefined keys, sub_key may be None. In that case,\n"
99" the handle returned is the same key handle passed in to the function.\n"
100"\n"
101"If the key already exists, this function opens the existing key\n"
102"\n"
103"The return value is the handle of the opened key.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000104"If the function fails, an exception is raised.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000105
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000106PyDoc_STRVAR(DeleteKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000107"DeleteKey(key, sub_key) - Deletes the specified key.\n"
108"\n"
109"key is an already open key, or any one of the predefined HKEY_* constants.\n"
110"sub_key is a string that must be a subkey of the key identified by the key parameter.\n"
111" This value must not be None, and the key may not have subkeys.\n"
112"\n"
113"This method can not delete keys with subkeys.\n"
114"\n"
115"If the method succeeds, the entire key, including all of its values,\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000116"is removed. If the method fails, an EnvironmentError exception is raised.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000117
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000118PyDoc_STRVAR(DeleteValue_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000119"DeleteValue(key, value) - Removes a named value from a registry key.\n"
120"\n"
121"key is an already open key, or any one of the predefined HKEY_* constants.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000122"value is a string that identifies the value to remove.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000123
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000124PyDoc_STRVAR(EnumKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000125"string = EnumKey(key, index) - Enumerates subkeys of an open registry key.\n"
126"\n"
127"key is an already open key, or any one of the predefined HKEY_* constants.\n"
128"index is an integer that identifies the index of the key to retrieve.\n"
129"\n"
130"The function retrieves the name of one subkey each time it is called.\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000131"It is typically called repeatedly until an EnvironmentError exception is\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000132"raised, indicating no more values are available.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000133
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000134PyDoc_STRVAR(EnumValue_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000135"tuple = EnumValue(key, index) - Enumerates values of an open registry key.\n"
136"key is an already open key, or any one of the predefined HKEY_* constants.\n"
137"index is an integer that identifies the index of the value to retrieve.\n"
138"\n"
139"The function retrieves the name of one subkey each time it is called.\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000140"It is typically called repeatedly, until an EnvironmentError exception\n"
141"is raised, indicating no more values.\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000142"\n"
143"The result is a tuple of 3 items:\n"
144"value_name is a string that identifies the value.\n"
145"value_data is an object that holds the value data, and whose type depends\n"
146" on the underlying registry type.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000147"data_type is an integer that identifies the type of the value data.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000148
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000149PyDoc_STRVAR(FlushKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000150"FlushKey(key) - Writes all the attributes of a key to the registry.\n"
151"\n"
152"key is an already open key, or any one of the predefined HKEY_* constants.\n"
153"\n"
154"It is not necessary to call RegFlushKey to change a key.\n"
155"Registry changes are flushed to disk by the registry using its lazy flusher.\n"
156"Registry changes are also flushed to disk at system shutdown.\n"
157"Unlike CloseKey(), the FlushKey() method returns only when all the data has\n"
158"been written to the registry.\n"
159"An application should only call FlushKey() if it requires absolute certainty that registry changes are on disk.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000160"If you don't know whether a FlushKey() call is required, it probably isn't.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000161
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000162PyDoc_STRVAR(LoadKey_doc,
Mark Hammondb422f952000-06-09 06:01:47 +0000163"LoadKey(key, sub_key, file_name) - Creates a subkey under the specified key\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000164"and stores registration information from a specified file into that subkey.\n"
165"\n"
166"key is an already open key, or any one of the predefined HKEY_* constants.\n"
167"sub_key is a string that identifies the sub_key to load\n"
168"file_name is the name of the file to load registry data from.\n"
169" This file must have been created with the SaveKey() function.\n"
170" Under the file allocation table (FAT) file system, the filename may not\n"
171"have an extension.\n"
172"\n"
173"A call to LoadKey() fails if the calling process does not have the\n"
174"SE_RESTORE_PRIVILEGE privilege.\n"
175"\n"
176"If key is a handle returned by ConnectRegistry(), then the path specified\n"
177"in fileName is relative to the remote computer.\n"
178"\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000179"The docs imply key must be in the HKEY_USER or HKEY_LOCAL_MACHINE tree");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000180
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000181PyDoc_STRVAR(OpenKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000182"key = OpenKey(key, sub_key, res = 0, sam = KEY_READ) - Opens the specified key.\n"
183"\n"
184"key is an already open key, or any one of the predefined HKEY_* constants.\n"
185"sub_key is a string that identifies the sub_key to open\n"
186"res is a reserved integer, and must be zero. Default is zero.\n"
187"sam is an integer that specifies an access mask that describes the desired\n"
188" security access for the key. Default is KEY_READ\n"
189"\n"
190"The result is a new handle to the specified key\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000191"If the function fails, an EnvironmentError exception is raised.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000192
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000193PyDoc_STRVAR(OpenKeyEx_doc, "See OpenKey()");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000194
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000195PyDoc_STRVAR(QueryInfoKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000196"tuple = QueryInfoKey(key) - Returns information about a key.\n"
197"\n"
198"key is an already open key, or any one of the predefined HKEY_* constants.\n"
199"\n"
200"The result is a tuple of 3 items:"
201"An integer that identifies the number of sub keys this key has.\n"
202"An integer that identifies the number of values this key has.\n"
203"A long integer that identifies when the key was last modified (if available)\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000204" as 100's of nanoseconds since Jan 1, 1600.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000205
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000206PyDoc_STRVAR(QueryValue_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000207"string = QueryValue(key, sub_key) - retrieves the unnamed value for a key.\n"
208"\n"
209"key is an already open key, or any one of the predefined HKEY_* constants.\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000210"sub_key is a string that holds the name of the subkey with which the value\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000211" is associated. If this parameter is None or empty, the function retrieves\n"
212" the value set by the SetValue() method for the key identified by key."
213"\n"
214"Values in the registry have name, type, and data components. This method\n"
215"retrieves the data for a key's first value that has a NULL name.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000216"But the underlying API call doesn't return the type, Lame Lame Lame, DONT USE THIS!!!");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000217
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000218PyDoc_STRVAR(QueryValueEx_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000219"value,type_id = QueryValueEx(key, value_name) - Retrieves the type and data for a specified value name associated with an open registry key.\n"
220"\n"
221"key is an already open key, or any one of the predefined HKEY_* constants.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000222"value_name is a string indicating the value to query");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000223
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000224PyDoc_STRVAR(SaveKey_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000225"SaveKey(key, file_name) - Saves the specified key, and all its subkeys to the specified file.\n"
226"\n"
227"key is an already open key, or any one of the predefined HKEY_* constants.\n"
228"file_name is the name of the file to save registry data to.\n"
229" This file cannot already exist. If this filename includes an extension,\n"
230" it cannot be used on file allocation table (FAT) file systems by the\n"
231" LoadKey(), ReplaceKey() or RestoreKey() methods.\n"
232"\n"
233"If key represents a key on a remote computer, the path described by\n"
234"file_name is relative to the remote computer.\n"
235"The caller of this method must possess the SeBackupPrivilege security privilege.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000236"This function passes NULL for security_attributes to the API.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000237
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000238PyDoc_STRVAR(SetValue_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000239"SetValue(key, sub_key, type, value) - Associates a value with a specified key.\n"
240"\n"
241"key is an already open key, or any one of the predefined HKEY_* constants.\n"
242"sub_key is a string that names the subkey with which the value is associated.\n"
243"type is an integer that specifies the type of the data. Currently this\n"
244" must be REG_SZ, meaning only strings are supported.\n"
245"value is a string that specifies the new value.\n"
246"\n"
247"If the key specified by the sub_key parameter does not exist, the SetValue\n"
248"function creates it.\n"
249"\n"
250"Value lengths are limited by available memory. Long values (more than\n"
251"2048 bytes) should be stored as files with the filenames stored in \n"
252"the configuration registry. This helps the registry perform efficiently.\n"
253"\n"
254"The key identified by the key parameter must have been opened with\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000255"KEY_SET_VALUE access.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000256
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000257PyDoc_STRVAR(SetValueEx_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000258"SetValueEx(key, value_name, reserved, type, value) - Stores data in the value field of an open registry key.\n"
259"\n"
260"key is an already open key, or any one of the predefined HKEY_* constants.\n"
Mark Hammondc9083b62003-01-15 23:38:15 +0000261"value_name is a string containing the name of the value to set, or None\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000262"type is an integer that specifies the type of the data. This should be one of:\n"
263" REG_BINARY -- Binary data in any form.\n"
264" REG_DWORD -- A 32-bit number.\n"
265" REG_DWORD_LITTLE_ENDIAN -- A 32-bit number in little-endian format.\n"
266" REG_DWORD_BIG_ENDIAN -- A 32-bit number in big-endian format.\n"
267" REG_EXPAND_SZ -- A null-terminated string that contains unexpanded references\n"
268" to environment variables (for example, %PATH%).\n"
269" REG_LINK -- A Unicode symbolic link.\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000270" REG_MULTI_SZ -- An sequence of null-terminated strings, terminated by\n"
271" two null characters. Note that Python handles this\n"
272" termination automatically.\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000273" REG_NONE -- No defined value type.\n"
274" REG_RESOURCE_LIST -- A device-driver resource list.\n"
275" REG_SZ -- A null-terminated string.\n"
276"reserved can be anything - zero is always passed to the API.\n"
277"value is a string that specifies the new value.\n"
278"\n"
279"This method can also set additional value and type information for the\n"
280"specified key. The key identified by the key parameter must have been\n"
281"opened with KEY_SET_VALUE access.\n"
282"\n"
283"To open the key, use the CreateKeyEx() or OpenKeyEx() methods.\n"
284"\n"
285"Value lengths are limited by available memory. Long values (more than\n"
286"2048 bytes) should be stored as files with the filenames stored in \n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000287"the configuration registry. This helps the registry perform efficiently.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000288
289/* PyHKEY docstrings */
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000290PyDoc_STRVAR(PyHKEY_doc,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000291"PyHKEY Object - A Python object, representing a win32 registry key.\n"
292"\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000293"This object wraps a Windows HKEY object, automatically closing it when\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000294"the object is destroyed. To guarantee cleanup, you can call either\n"
295"the Close() method on the PyHKEY, or the CloseKey() method.\n"
296"\n"
297"All functions which accept a handle object also accept an integer - \n"
298"however, use of the handle object is encouraged.\n"
299"\n"
300"Functions:\n"
301"Close() - Closes the underlying handle.\n"
302"Detach() - Returns the integer Win32 handle, detaching it from the object\n"
303"\n"
304"Properties:\n"
305"handle - The integer Win32 handle.\n"
306"\n"
307"Operations:\n"
Jack Diederich4dafcc42006-11-28 19:15:13 +0000308"__bool__ - Handles with an open object return true, otherwise false.\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000309"__int__ - Converting a handle to an integer returns the Win32 handle.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000310"__cmp__ - Handle objects are compared using the handle value.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000311
312
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000313PyDoc_STRVAR(PyHKEY_Close_doc,
Mark Hammondb422f952000-06-09 06:01:47 +0000314"key.Close() - Closes the underlying Windows handle.\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000315"\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000316"If the handle is already closed, no error is raised.");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000317
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000318PyDoc_STRVAR(PyHKEY_Detach_doc,
Mark Hammondb422f952000-06-09 06:01:47 +0000319"int = key.Detach() - Detaches the Windows handle from the handle object.\n"
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000320"\n"
321"The result is the value of the handle before it is detached. If the\n"
322"handle is already detached, this will return zero.\n"
323"\n"
324"After calling this function, the handle is effectively invalidated,\n"
325"but the handle is not closed. You would call this function when you\n"
326"need the underlying win32 handle to exist beyond the lifetime of the\n"
Mark Hammondb422f952000-06-09 06:01:47 +0000327"handle object.\n"
Martin v. Löwis14f8b4c2002-06-13 20:33:02 +0000328"On 64 bit windows, the result of this function is a long integer");
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000329
330
331/************************************************************************
332
333 The PyHKEY object definition
334
335************************************************************************/
336typedef struct {
337 PyObject_VAR_HEAD
338 HKEY hkey;
339} PyHKEYObject;
340
341#define PyHKEY_Check(op) ((op)->ob_type == &PyHKEY_Type)
342
343static char *failMsg = "bad operand type";
344
345static PyObject *
346PyHKEY_unaryFailureFunc(PyObject *ob)
347{
348 PyErr_SetString(PyExc_TypeError, failMsg);
349 return NULL;
350}
351static PyObject *
352PyHKEY_binaryFailureFunc(PyObject *ob1, PyObject *ob2)
353{
354 PyErr_SetString(PyExc_TypeError, failMsg);
355 return NULL;
356}
357static PyObject *
358PyHKEY_ternaryFailureFunc(PyObject *ob1, PyObject *ob2, PyObject *ob3)
359{
360 PyErr_SetString(PyExc_TypeError, failMsg);
361 return NULL;
362}
363
364static void
365PyHKEY_deallocFunc(PyObject *ob)
366{
367 /* Can not call PyHKEY_Close, as the ob->tp_type
368 has already been cleared, thus causing the type
369 check to fail!
370 */
371 PyHKEYObject *obkey = (PyHKEYObject *)ob;
372 if (obkey->hkey)
373 RegCloseKey((HKEY)obkey->hkey);
Guido van Rossumb18618d2000-05-03 23:44:39 +0000374 PyObject_DEL(ob);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000375}
376
377static int
Jack Diederich4dafcc42006-11-28 19:15:13 +0000378PyHKEY_boolFunc(PyObject *ob)
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000379{
380 return ((PyHKEYObject *)ob)->hkey != 0;
381}
382
383static PyObject *
384PyHKEY_intFunc(PyObject *ob)
385{
386 PyHKEYObject *pyhkey = (PyHKEYObject *)ob;
387 return PyLong_FromVoidPtr(pyhkey->hkey);
388}
389
390static int
391PyHKEY_printFunc(PyObject *ob, FILE *fp, int flags)
392{
393 PyHKEYObject *pyhkey = (PyHKEYObject *)ob;
394 char resBuf[160];
395 wsprintf(resBuf, "<PyHKEY at %p (%p)>",
396 ob, pyhkey->hkey);
397 fputs(resBuf, fp);
398 return 0;
399}
400
401static PyObject *
402PyHKEY_strFunc(PyObject *ob)
403{
404 PyHKEYObject *pyhkey = (PyHKEYObject *)ob;
405 char resBuf[160];
406 wsprintf(resBuf, "<PyHKEY:%p>", pyhkey->hkey);
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000407 return PyUnicode_FromString(resBuf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000408}
409
410static int
411PyHKEY_compareFunc(PyObject *ob1, PyObject *ob2)
412{
413 PyHKEYObject *pyhkey1 = (PyHKEYObject *)ob1;
414 PyHKEYObject *pyhkey2 = (PyHKEYObject *)ob2;
415 return pyhkey1 == pyhkey2 ? 0 :
416 (pyhkey1 < pyhkey2 ? -1 : 1);
417}
418
419static long
420PyHKEY_hashFunc(PyObject *ob)
421{
422 /* Just use the address.
423 XXX - should we use the handle value?
424 */
Fred Drake13634cf2000-06-29 19:17:04 +0000425 return _Py_HashPointer(ob);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000426}
427
428
429static PyNumberMethods PyHKEY_NumberMethods =
430{
431 PyHKEY_binaryFailureFunc, /* nb_add */
432 PyHKEY_binaryFailureFunc, /* nb_subtract */
433 PyHKEY_binaryFailureFunc, /* nb_multiply */
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000434 PyHKEY_binaryFailureFunc, /* nb_remainder */
435 PyHKEY_binaryFailureFunc, /* nb_divmod */
436 PyHKEY_ternaryFailureFunc, /* nb_power */
437 PyHKEY_unaryFailureFunc, /* nb_negative */
438 PyHKEY_unaryFailureFunc, /* nb_positive */
439 PyHKEY_unaryFailureFunc, /* nb_absolute */
Jack Diederich4dafcc42006-11-28 19:15:13 +0000440 PyHKEY_boolFunc, /* nb_bool */
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000441 PyHKEY_unaryFailureFunc, /* nb_invert */
442 PyHKEY_binaryFailureFunc, /* nb_lshift */
443 PyHKEY_binaryFailureFunc, /* nb_rshift */
444 PyHKEY_binaryFailureFunc, /* nb_and */
445 PyHKEY_binaryFailureFunc, /* nb_xor */
446 PyHKEY_binaryFailureFunc, /* nb_or */
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000447 NULL, /* nb_coerce */
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000448 PyHKEY_intFunc, /* nb_int */
449 PyHKEY_unaryFailureFunc, /* nb_long */
450 PyHKEY_unaryFailureFunc, /* nb_float */
451 PyHKEY_unaryFailureFunc, /* nb_oct */
452 PyHKEY_unaryFailureFunc, /* nb_hex */
453};
454
455
456/* fwd declare __getattr__ */
Tim Petersc3d12ac2005-12-24 06:03:06 +0000457static PyObject *PyHKEY_getattr(PyObject *self, const char *name);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000458
459/* The type itself */
460PyTypeObject PyHKEY_Type =
461{
462 PyObject_HEAD_INIT(0) /* fill in type at module init */
463 0,
464 "PyHKEY",
465 sizeof(PyHKEYObject),
466 0,
467 PyHKEY_deallocFunc, /* tp_dealloc */
468 PyHKEY_printFunc, /* tp_print */
469 PyHKEY_getattr, /* tp_getattr */
470 0, /* tp_setattr */
471 PyHKEY_compareFunc, /* tp_compare */
472 0, /* tp_repr */
473 &PyHKEY_NumberMethods, /* tp_as_number */
474 0, /* tp_as_sequence */
475 0, /* tp_as_mapping */
476 PyHKEY_hashFunc, /* tp_hash */
477 0, /* tp_call */
478 PyHKEY_strFunc, /* tp_str */
479 0, /* tp_getattro */
480 0, /* tp_setattro */
481 0, /* tp_as_buffer */
482 0, /* tp_flags */
483 PyHKEY_doc, /* tp_doc */
484};
485
486#define OFF(e) offsetof(PyHKEYObject, e)
487
488static struct memberlist PyHKEY_memberlist[] = {
489 {"handle", T_INT, OFF(hkey)},
490 {NULL} /* Sentinel */
491};
492
493/************************************************************************
494
495 The PyHKEY object methods
496
497************************************************************************/
498static PyObject *
499PyHKEY_CloseMethod(PyObject *self, PyObject *args)
500{
501 if (!PyArg_ParseTuple(args, ":Close"))
502 return NULL;
503 if (!PyHKEY_Close(self))
504 return NULL;
505 Py_INCREF(Py_None);
506 return Py_None;
507}
508
509static PyObject *
510PyHKEY_DetachMethod(PyObject *self, PyObject *args)
511{
512 void* ret;
513 PyHKEYObject *pThis = (PyHKEYObject *)self;
514 if (!PyArg_ParseTuple(args, ":Detach"))
515 return NULL;
516 ret = (void*)pThis->hkey;
517 pThis->hkey = 0;
518 return PyLong_FromVoidPtr(ret);
519}
520
521static struct PyMethodDef PyHKEY_methods[] = {
Neal Norwitz031829d2002-03-31 14:37:44 +0000522 {"Close", PyHKEY_CloseMethod, METH_VARARGS, PyHKEY_Close_doc},
523 {"Detach", PyHKEY_DetachMethod, METH_VARARGS, PyHKEY_Detach_doc},
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000524 {NULL}
525};
526
527/*static*/ PyObject *
Tim Petersc3d12ac2005-12-24 06:03:06 +0000528PyHKEY_getattr(PyObject *self, const char *name)
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000529{
530 PyObject *res;
531
532 res = Py_FindMethod(PyHKEY_methods, self, name);
533 if (res != NULL)
534 return res;
535 PyErr_Clear();
536 if (strcmp(name, "handle") == 0)
537 return PyLong_FromVoidPtr(((PyHKEYObject *)self)->hkey);
538 return PyMember_Get((char *)self, PyHKEY_memberlist, name);
539}
540
541/************************************************************************
542 The public PyHKEY API (well, not public yet :-)
543************************************************************************/
544PyObject *
545PyHKEY_New(HKEY hInit)
546{
547 PyHKEYObject *key = PyObject_NEW(PyHKEYObject, &PyHKEY_Type);
548 if (key)
549 key->hkey = hInit;
550 return (PyObject *)key;
551}
552
553BOOL
554PyHKEY_Close(PyObject *ob_handle)
555{
556 LONG rc;
557 PyHKEYObject *key;
558
559 if (!PyHKEY_Check(ob_handle)) {
560 PyErr_SetString(PyExc_TypeError, "bad operand type");
561 return FALSE;
562 }
563 key = (PyHKEYObject *)ob_handle;
564 rc = key->hkey ? RegCloseKey((HKEY)key->hkey) : ERROR_SUCCESS;
565 key->hkey = 0;
566 if (rc != ERROR_SUCCESS)
567 PyErr_SetFromWindowsErrWithFunction(rc, "RegCloseKey");
568 return rc == ERROR_SUCCESS;
569}
570
571BOOL
572PyHKEY_AsHKEY(PyObject *ob, HKEY *pHANDLE, BOOL bNoneOK)
573{
574 if (ob == Py_None) {
575 if (!bNoneOK) {
576 PyErr_SetString(
577 PyExc_TypeError,
578 "None is not a valid HKEY in this context");
579 return FALSE;
580 }
581 *pHANDLE = (HKEY)0;
582 }
583 else if (PyHKEY_Check(ob)) {
584 PyHKEYObject *pH = (PyHKEYObject *)ob;
585 *pHANDLE = pH->hkey;
586 }
587 else if (PyInt_Check(ob) || PyLong_Check(ob)) {
588 /* We also support integers */
589 PyErr_Clear();
590 *pHANDLE = (HKEY)PyLong_AsVoidPtr(ob);
591 if (PyErr_Occurred())
592 return FALSE;
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000593 }
594 else {
595 PyErr_SetString(
596 PyExc_TypeError,
597 "The object is not a PyHKEY object");
598 return FALSE;
599 }
600 return TRUE;
601}
602
603PyObject *
604PyHKEY_FromHKEY(HKEY h)
605{
Guido van Rossumb18618d2000-05-03 23:44:39 +0000606 PyHKEYObject *op;
607
Guido van Rossume3a8e7e2002-08-19 19:26:42 +0000608 /* Inline PyObject_New */
Guido van Rossumb18618d2000-05-03 23:44:39 +0000609 op = (PyHKEYObject *) PyObject_MALLOC(sizeof(PyHKEYObject));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000610 if (op == NULL)
611 return PyErr_NoMemory();
Guido van Rossumb18618d2000-05-03 23:44:39 +0000612 PyObject_INIT(op, &PyHKEY_Type);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000613 op->hkey = h;
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000614 return (PyObject *)op;
615}
616
617
618/************************************************************************
619 The module methods
620************************************************************************/
621BOOL
622PyWinObject_CloseHKEY(PyObject *obHandle)
623{
624 BOOL ok;
625 if (PyHKEY_Check(obHandle)) {
626 ok = PyHKEY_Close(obHandle);
627 }
Fred Drake25e17262000-06-30 17:48:51 +0000628#if SIZEOF_LONG >= SIZEOF_HKEY
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000629 else if (PyInt_Check(obHandle)) {
630 long rc = RegCloseKey((HKEY)PyInt_AsLong(obHandle));
631 ok = (rc == ERROR_SUCCESS);
632 if (!ok)
633 PyErr_SetFromWindowsErrWithFunction(rc, "RegCloseKey");
634 }
Fred Drake25e17262000-06-30 17:48:51 +0000635#else
636 else if (PyLong_Check(obHandle)) {
637 long rc = RegCloseKey((HKEY)PyLong_AsVoidPtr(obHandle));
638 ok = (rc == ERROR_SUCCESS);
639 if (!ok)
640 PyErr_SetFromWindowsErrWithFunction(rc, "RegCloseKey");
641 }
642#endif
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000643 else {
644 PyErr_SetString(
645 PyExc_TypeError,
646 "A handle must be a HKEY object or an integer");
647 return FALSE;
648 }
649 return ok;
650}
651
652
653/*
654 Private Helper functions for the registry interfaces
655
656** Note that fixupMultiSZ and countString have both had changes
657** made to support "incorrect strings". The registry specification
658** calls for strings to be terminated with 2 null bytes. It seems
Thomas Wouters7e474022000-07-16 12:04:32 +0000659** some commercial packages install strings which dont conform,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000660** causing this code to fail - however, "regedit" etc still work
661** with these strings (ie only we dont!).
662*/
663static void
664fixupMultiSZ(char **str, char *data, int len)
665{
666 char *P;
667 int i;
668 char *Q;
669
670 Q = data + len;
671 for (P = data, i = 0; P < Q && *P != '\0'; P++, i++) {
672 str[i] = P;
673 for(; *P != '\0'; P++)
674 ;
675 }
676}
677
678static int
679countStrings(char *data, int len)
680{
681 int strings;
682 char *P;
683 char *Q = data + len;
684
685 for (P = data, strings = 0; P < Q && *P != '\0'; P++, strings++)
686 for (; P < Q && *P != '\0'; P++)
687 ;
688 return strings;
689}
690
691/* Convert PyObject into Registry data.
692 Allocates space as needed. */
693static BOOL
694Py2Reg(PyObject *value, DWORD typ, BYTE **retDataBuf, DWORD *retDataSize)
695{
696 int i,j;
Guido van Rossum7eaf8222007-06-18 17:58:50 +0000697 DWORD d;
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000698 switch (typ) {
699 case REG_DWORD:
Guido van Rossum7eaf8222007-06-18 17:58:50 +0000700 if (value != Py_None && !PyLong_Check(value))
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000701 return FALSE;
Guido van Rossumd8faa362007-04-27 19:54:29 +0000702 *retDataBuf = (BYTE *)PyMem_NEW(DWORD, 1);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000703 if (*retDataBuf==NULL){
704 PyErr_NoMemory();
705 return FALSE;
706 }
707 *retDataSize = sizeof(DWORD);
708 if (value == Py_None) {
709 DWORD zero = 0;
710 memcpy(*retDataBuf, &zero, sizeof(DWORD));
711 }
Guido van Rossum7eaf8222007-06-18 17:58:50 +0000712 else {
713 d = PyLong_AsLong(value);
714 memcpy(*retDataBuf, &d, sizeof(DWORD));
715 }
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000716 break;
717 case REG_SZ:
718 case REG_EXPAND_SZ:
719 {
720 int need_decref = 0;
721 if (value == Py_None)
722 *retDataSize = 1;
723 else {
724 if (PyUnicode_Check(value)) {
725 value = PyUnicode_AsEncodedString(
726 value,
727 "mbcs",
728 NULL);
729 if (value==NULL)
730 return FALSE;
731 need_decref = 1;
732 }
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000733 if (!PyBytes_Check(value))
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000734 return FALSE;
735 *retDataSize = 1 + strlen(
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000736 PyBytes_AS_STRING(value));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000737 }
738 *retDataBuf = (BYTE *)PyMem_NEW(DWORD, *retDataSize);
739 if (*retDataBuf==NULL){
740 PyErr_NoMemory();
741 return FALSE;
742 }
743 if (value == Py_None)
744 strcpy((char *)*retDataBuf, "");
745 else
746 strcpy((char *)*retDataBuf,
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000747 PyBytes_AS_STRING(value));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000748 if (need_decref)
749 Py_DECREF(value);
750 break;
751 }
752 case REG_MULTI_SZ:
753 {
754 DWORD size = 0;
755 char *P;
756 PyObject **obs = NULL;
757
758 if (value == Py_None)
759 i = 0;
760 else {
761 if (!PyList_Check(value))
762 return FALSE;
763 i = PyList_Size(value);
764 }
765 obs = malloc(sizeof(PyObject *) * i);
766 memset(obs, 0, sizeof(PyObject *) * i);
767 for (j = 0; j < i; j++)
768 {
769 PyObject *t;
770 t = PyList_GET_ITEM(
771 (PyListObject *)value,j);
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000772 if (PyBytes_Check(t)) {
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000773 obs[j] = t;
774 Py_INCREF(t);
775 } else if (PyUnicode_Check(t)) {
776 obs[j] = PyUnicode_AsEncodedString(
777 t,
778 "mbcs",
779 NULL);
780 if (obs[j]==NULL)
781 goto reg_multi_fail;
782 } else
783 goto reg_multi_fail;
784 size += 1 + strlen(
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000785 PyBytes_AS_STRING(obs[j]));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000786 }
787
788 *retDataSize = size + 1;
789 *retDataBuf = (BYTE *)PyMem_NEW(char,
790 *retDataSize);
791 if (*retDataBuf==NULL){
792 PyErr_NoMemory();
793 goto reg_multi_fail;
794 }
795 P = (char *)*retDataBuf;
796
797 for (j = 0; j < i; j++)
798 {
799 PyObject *t;
800 t = obs[j];
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000801 strcpy(P, PyBytes_AS_STRING(t));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000802 P += 1 + strlen(
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000803 PyBytes_AS_STRING(t));
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000804 Py_DECREF(obs[j]);
805 }
806 /* And doubly-terminate the list... */
807 *P = '\0';
808 free(obs);
809 break;
810 reg_multi_fail:
811 if (obs) {
812 for (j = 0; j < i; j++)
813 Py_XDECREF(obs[j]);
814
815 free(obs);
816 }
817 return FALSE;
818 }
819 case REG_BINARY:
820 /* ALSO handle ALL unknown data types here. Even if we can't
821 support it natively, we should handle the bits. */
822 default:
823 if (value == Py_None)
824 *retDataSize = 0;
825 else {
Mark Hammond4e80bb52000-07-28 03:44:41 +0000826 void *src_buf;
827 PyBufferProcs *pb = value->ob_type->tp_as_buffer;
828 if (pb==NULL) {
Tim Peters313fcd42006-02-19 04:05:39 +0000829 PyErr_Format(PyExc_TypeError,
Mark Hammond4e80bb52000-07-28 03:44:41 +0000830 "Objects of type '%s' can not "
Tim Peters313fcd42006-02-19 04:05:39 +0000831 "be used as binary registry values",
Mark Hammond4e80bb52000-07-28 03:44:41 +0000832 value->ob_type->tp_name);
833 return FALSE;
834 }
835 *retDataSize = (*pb->bf_getreadbuffer)(value, 0, &src_buf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000836 *retDataBuf = (BYTE *)PyMem_NEW(char,
837 *retDataSize);
838 if (*retDataBuf==NULL){
839 PyErr_NoMemory();
840 return FALSE;
841 }
Mark Hammond4e80bb52000-07-28 03:44:41 +0000842 memcpy(*retDataBuf, src_buf, *retDataSize);
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000843 }
844 break;
845 }
846 return TRUE;
847}
848
849/* Convert Registry data into PyObject*/
850static PyObject *
851Reg2Py(char *retDataBuf, DWORD retDataSize, DWORD typ)
852{
853 PyObject *obData;
854
855 switch (typ) {
856 case REG_DWORD:
857 if (retDataSize == 0)
858 obData = Py_BuildValue("i", 0);
859 else
860 obData = Py_BuildValue("i",
861 *(int *)retDataBuf);
862 break;
863 case REG_SZ:
864 case REG_EXPAND_SZ:
865 /* retDataBuf may or may not have a trailing NULL in
866 the buffer. */
867 if (retDataSize && retDataBuf[retDataSize-1] == '\0')
868 --retDataSize;
869 if (retDataSize ==0)
870 retDataBuf = "";
871 obData = PyUnicode_DecodeMBCS(retDataBuf,
872 retDataSize,
873 NULL);
874 break;
875 case REG_MULTI_SZ:
876 if (retDataSize == 0)
877 obData = PyList_New(0);
878 else
879 {
880 int index = 0;
881 int s = countStrings(retDataBuf, retDataSize);
882 char **str = (char **)malloc(sizeof(char *)*s);
883 if (str == NULL)
884 return PyErr_NoMemory();
885
886 fixupMultiSZ(str, retDataBuf, retDataSize);
887 obData = PyList_New(s);
Fred Drake25e17262000-06-30 17:48:51 +0000888 if (obData == NULL)
889 return NULL;
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000890 for (index = 0; index < s; index++)
891 {
Fred Drake25e17262000-06-30 17:48:51 +0000892 size_t len = _mbstrlen(str[index]);
893 if (len > INT_MAX) {
894 PyErr_SetString(PyExc_OverflowError,
895 "registry string is too long for a Python string");
896 Py_DECREF(obData);
897 return NULL;
898 }
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000899 PyList_SetItem(obData,
900 index,
901 PyUnicode_DecodeMBCS(
902 (const char *)str[index],
Fred Drake25e17262000-06-30 17:48:51 +0000903 (int)len,
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000904 NULL)
905 );
906 }
907 free(str);
908
909 break;
910 }
911 case REG_BINARY:
912 /* ALSO handle ALL unknown data types here. Even if we can't
913 support it natively, we should handle the bits. */
914 default:
915 if (retDataSize == 0) {
916 Py_INCREF(Py_None);
917 obData = Py_None;
918 }
919 else
Guido van Rossuma8c360e2007-07-17 20:50:43 +0000920 obData = Py_BuildValue("y#",
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000921 (char *)retDataBuf,
922 retDataSize);
923 break;
924 }
925 if (obData == NULL)
926 return NULL;
927 else
928 return obData;
929}
930
931/* The Python methods */
932
933static PyObject *
934PyCloseKey(PyObject *self, PyObject *args)
935{
936 PyObject *obKey;
937 if (!PyArg_ParseTuple(args, "O:CloseKey", &obKey))
938 return NULL;
939 if (!PyHKEY_Close(obKey))
940 return NULL;
941 Py_INCREF(Py_None);
942 return Py_None;
943}
944
945static PyObject *
946PyConnectRegistry(PyObject *self, PyObject *args)
947{
948 HKEY hKey;
949 PyObject *obKey;
950 char *szCompName = NULL;
951 HKEY retKey;
952 long rc;
953 if (!PyArg_ParseTuple(args, "zO:ConnectRegistry", &szCompName, &obKey))
954 return NULL;
955 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
956 return NULL;
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000957 Py_BEGIN_ALLOW_THREADS
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000958 rc = RegConnectRegistry(szCompName, hKey, &retKey);
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000959 Py_END_ALLOW_THREADS
Guido van Rossum9f3712c2000-03-28 20:37:15 +0000960 if (rc != ERROR_SUCCESS)
961 return PyErr_SetFromWindowsErrWithFunction(rc,
962 "ConnectRegistry");
963 return PyHKEY_FromHKEY(retKey);
964}
965
966static PyObject *
967PyCreateKey(PyObject *self, PyObject *args)
968{
969 HKEY hKey;
970 PyObject *obKey;
971 char *subKey;
972 HKEY retKey;
973 long rc;
974 if (!PyArg_ParseTuple(args, "Oz:CreateKey", &obKey, &subKey))
975 return NULL;
976 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
977 return NULL;
978 rc = RegCreateKey(hKey, subKey, &retKey);
979 if (rc != ERROR_SUCCESS)
980 return PyErr_SetFromWindowsErrWithFunction(rc, "CreateKey");
981 return PyHKEY_FromHKEY(retKey);
982}
983
984static PyObject *
985PyDeleteKey(PyObject *self, PyObject *args)
986{
987 HKEY hKey;
988 PyObject *obKey;
989 char *subKey;
990 long rc;
991 if (!PyArg_ParseTuple(args, "Os:DeleteKey", &obKey, &subKey))
992 return NULL;
993 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
994 return NULL;
995 rc = RegDeleteKey(hKey, subKey );
996 if (rc != ERROR_SUCCESS)
997 return PyErr_SetFromWindowsErrWithFunction(rc, "RegDeleteKey");
998 Py_INCREF(Py_None);
999 return Py_None;
1000}
1001
1002static PyObject *
1003PyDeleteValue(PyObject *self, PyObject *args)
1004{
1005 HKEY hKey;
1006 PyObject *obKey;
1007 char *subKey;
1008 long rc;
1009 if (!PyArg_ParseTuple(args, "Oz:DeleteValue", &obKey, &subKey))
1010 return NULL;
1011 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1012 return NULL;
1013 Py_BEGIN_ALLOW_THREADS
1014 rc = RegDeleteValue(hKey, subKey);
1015 Py_END_ALLOW_THREADS
1016 if (rc !=ERROR_SUCCESS)
1017 return PyErr_SetFromWindowsErrWithFunction(rc,
1018 "RegDeleteValue");
1019 Py_INCREF(Py_None);
1020 return Py_None;
1021}
1022
1023static PyObject *
1024PyEnumKey(PyObject *self, PyObject *args)
1025{
1026 HKEY hKey;
1027 PyObject *obKey;
1028 int index;
1029 long rc;
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001030 PyObject *retStr;
Georg Brandl9a928e72006-02-18 23:35:11 +00001031 char tmpbuf[256]; /* max key name length is 255 */
Georg Brandlb2699b22006-02-18 23:44:24 +00001032 DWORD len = sizeof(tmpbuf); /* includes NULL terminator */
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001033
1034 if (!PyArg_ParseTuple(args, "Oi:EnumKey", &obKey, &index))
1035 return NULL;
1036 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1037 return NULL;
Tim Peters313fcd42006-02-19 04:05:39 +00001038
Georg Brandl9a928e72006-02-18 23:35:11 +00001039 Py_BEGIN_ALLOW_THREADS
1040 rc = RegEnumKeyEx(hKey, index, tmpbuf, &len, NULL, NULL, NULL, NULL);
1041 Py_END_ALLOW_THREADS
1042 if (rc != ERROR_SUCCESS)
1043 return PyErr_SetFromWindowsErrWithFunction(rc, "RegEnumKeyEx");
Tim Peters313fcd42006-02-19 04:05:39 +00001044
Guido van Rossuma8c360e2007-07-17 20:50:43 +00001045 retStr = PyUnicode_FromStringAndSize(tmpbuf, len);
Georg Brandl9a928e72006-02-18 23:35:11 +00001046 return retStr; /* can be NULL */
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001047}
1048
1049static PyObject *
1050PyEnumValue(PyObject *self, PyObject *args)
1051{
1052 HKEY hKey;
1053 PyObject *obKey;
1054 int index;
1055 long rc;
1056 char *retValueBuf;
1057 char *retDataBuf;
1058 DWORD retValueSize;
1059 DWORD retDataSize;
1060 DWORD typ;
1061 PyObject *obData;
1062 PyObject *retVal;
1063
1064 if (!PyArg_ParseTuple(args, "Oi:EnumValue", &obKey, &index))
1065 return NULL;
1066 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1067 return NULL;
1068
1069 if ((rc = RegQueryInfoKey(hKey, NULL, NULL, NULL, NULL, NULL, NULL,
1070 NULL,
1071 &retValueSize, &retDataSize, NULL, NULL))
1072 != ERROR_SUCCESS)
1073 return PyErr_SetFromWindowsErrWithFunction(rc,
1074 "RegQueryInfoKey");
1075 ++retValueSize; /* include null terminators */
1076 ++retDataSize;
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001077 retValueBuf = (char *)PyMem_Malloc(retValueSize);
1078 if (retValueBuf == NULL)
1079 return PyErr_NoMemory();
1080 retDataBuf = (char *)PyMem_Malloc(retDataSize);
1081 if (retDataBuf == NULL) {
1082 PyMem_Free(retValueBuf);
1083 return PyErr_NoMemory();
1084 }
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001085
1086 Py_BEGIN_ALLOW_THREADS
1087 rc = RegEnumValue(hKey,
1088 index,
1089 retValueBuf,
1090 &retValueSize,
1091 NULL,
1092 &typ,
1093 (BYTE *)retDataBuf,
1094 &retDataSize);
1095 Py_END_ALLOW_THREADS
1096
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001097 if (rc != ERROR_SUCCESS) {
1098 retVal = PyErr_SetFromWindowsErrWithFunction(rc,
1099 "PyRegEnumValue");
1100 goto fail;
1101 }
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001102 obData = Reg2Py(retDataBuf, retDataSize, typ);
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001103 if (obData == NULL) {
1104 retVal = NULL;
1105 goto fail;
1106 }
Guido van Rossuma8c360e2007-07-17 20:50:43 +00001107 retVal = Py_BuildValue("UOi", retValueBuf, obData, typ);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001108 Py_DECREF(obData);
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001109 fail:
1110 PyMem_Free(retValueBuf);
1111 PyMem_Free(retDataBuf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001112 return retVal;
1113}
1114
1115static PyObject *
1116PyFlushKey(PyObject *self, PyObject *args)
1117{
1118 HKEY hKey;
1119 PyObject *obKey;
1120 long rc;
1121 if (!PyArg_ParseTuple(args, "O:FlushKey", &obKey))
1122 return NULL;
1123 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1124 return NULL;
1125 Py_BEGIN_ALLOW_THREADS
1126 rc = RegFlushKey(hKey);
1127 Py_END_ALLOW_THREADS
1128 if (rc != ERROR_SUCCESS)
1129 return PyErr_SetFromWindowsErrWithFunction(rc, "RegFlushKey");
1130 Py_INCREF(Py_None);
1131 return Py_None;
1132}
1133static PyObject *
1134PyLoadKey(PyObject *self, PyObject *args)
1135{
1136 HKEY hKey;
1137 PyObject *obKey;
1138 char *subKey;
1139 char *fileName;
1140
1141 long rc;
1142 if (!PyArg_ParseTuple(args, "Oss:LoadKey", &obKey, &subKey, &fileName))
1143 return NULL;
1144 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1145 return NULL;
1146 Py_BEGIN_ALLOW_THREADS
1147 rc = RegLoadKey(hKey, subKey, fileName );
1148 Py_END_ALLOW_THREADS
1149 if (rc != ERROR_SUCCESS)
1150 return PyErr_SetFromWindowsErrWithFunction(rc, "RegLoadKey");
1151 Py_INCREF(Py_None);
1152 return Py_None;
1153}
1154
1155static PyObject *
1156PyOpenKey(PyObject *self, PyObject *args)
1157{
1158 HKEY hKey;
1159 PyObject *obKey;
1160
1161 char *subKey;
1162 int res = 0;
1163 HKEY retKey;
1164 long rc;
1165 REGSAM sam = KEY_READ;
1166 if (!PyArg_ParseTuple(args, "Oz|ii:OpenKey", &obKey, &subKey,
1167 &res, &sam))
1168 return NULL;
1169 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1170 return NULL;
1171
1172 Py_BEGIN_ALLOW_THREADS
1173 rc = RegOpenKeyEx(hKey, subKey, res, sam, &retKey);
1174 Py_END_ALLOW_THREADS
1175 if (rc != ERROR_SUCCESS)
1176 return PyErr_SetFromWindowsErrWithFunction(rc, "RegOpenKeyEx");
1177 return PyHKEY_FromHKEY(retKey);
1178}
1179
1180
1181static PyObject *
1182PyQueryInfoKey(PyObject *self, PyObject *args)
1183{
1184 HKEY hKey;
1185 PyObject *obKey;
1186 long rc;
1187 DWORD nSubKeys, nValues;
1188 FILETIME ft;
1189 LARGE_INTEGER li;
1190 PyObject *l;
1191 PyObject *ret;
1192 if (!PyArg_ParseTuple(args, "O:QueryInfoKey", &obKey))
1193 return NULL;
1194 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1195 return NULL;
1196 if ((rc = RegQueryInfoKey(hKey, NULL, NULL, 0, &nSubKeys, NULL, NULL,
1197 &nValues, NULL, NULL, NULL, &ft))
1198 != ERROR_SUCCESS)
1199 return PyErr_SetFromWindowsErrWithFunction(rc, "RegQueryInfoKey");
1200 li.LowPart = ft.dwLowDateTime;
1201 li.HighPart = ft.dwHighDateTime;
1202 l = PyLong_FromLongLong(li.QuadPart);
1203 if (l == NULL)
1204 return NULL;
1205 ret = Py_BuildValue("iiO", nSubKeys, nValues, l);
1206 Py_DECREF(l);
1207 return ret;
1208}
1209
1210static PyObject *
1211PyQueryValue(PyObject *self, PyObject *args)
1212{
1213 HKEY hKey;
1214 PyObject *obKey;
1215 char *subKey;
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001216 long rc;
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001217 PyObject *retStr;
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001218 char *retBuf;
1219 long bufSize = 0;
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001220
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001221 if (!PyArg_ParseTuple(args, "Oz:QueryValue", &obKey, &subKey))
1222 return NULL;
1223
1224 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1225 return NULL;
1226 if ((rc = RegQueryValue(hKey, subKey, NULL, &bufSize))
1227 != ERROR_SUCCESS)
1228 return PyErr_SetFromWindowsErrWithFunction(rc,
1229 "RegQueryValue");
Guido van Rossuma8c360e2007-07-17 20:50:43 +00001230 retBuf = (char *)PyMem_Malloc(bufSize);
1231 if (retBuf == NULL)
1232 return PyErr_NoMemory();
1233
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001234 if ((rc = RegQueryValue(hKey, subKey, retBuf, &bufSize))
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001235 != ERROR_SUCCESS) {
Guido van Rossuma8c360e2007-07-17 20:50:43 +00001236 PyMem_Free(retBuf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001237 return PyErr_SetFromWindowsErrWithFunction(rc,
1238 "RegQueryValue");
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001239 }
Guido van Rossuma8c360e2007-07-17 20:50:43 +00001240
1241 retStr = PyUnicode_DecodeMBCS(retBuf, strlen(retBuf), NULL);
1242 PyMem_Free(retBuf);
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001243 return retStr;
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001244}
1245
1246static PyObject *
1247PyQueryValueEx(PyObject *self, PyObject *args)
1248{
1249 HKEY hKey;
1250 PyObject *obKey;
1251 char *valueName;
1252
1253 long rc;
1254 char *retBuf;
1255 DWORD bufSize = 0;
1256 DWORD typ;
1257 PyObject *obData;
1258 PyObject *result;
1259
1260 if (!PyArg_ParseTuple(args, "Oz:QueryValueEx", &obKey, &valueName))
1261 return NULL;
1262
1263 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1264 return NULL;
1265 if ((rc = RegQueryValueEx(hKey, valueName,
1266 NULL, NULL, NULL,
1267 &bufSize))
1268 != ERROR_SUCCESS)
1269 return PyErr_SetFromWindowsErrWithFunction(rc,
1270 "RegQueryValueEx");
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001271 retBuf = (char *)PyMem_Malloc(bufSize);
1272 if (retBuf == NULL)
1273 return PyErr_NoMemory();
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001274 if ((rc = RegQueryValueEx(hKey, valueName, NULL,
1275 &typ, (BYTE *)retBuf, &bufSize))
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001276 != ERROR_SUCCESS) {
1277 PyMem_Free(retBuf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001278 return PyErr_SetFromWindowsErrWithFunction(rc,
1279 "RegQueryValueEx");
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001280 }
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001281 obData = Reg2Py(retBuf, bufSize, typ);
Guido van Rossuma6a38ad2003-11-30 22:01:43 +00001282 PyMem_Free((void *)retBuf);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001283 if (obData == NULL)
1284 return NULL;
1285 result = Py_BuildValue("Oi", obData, typ);
1286 Py_DECREF(obData);
1287 return result;
1288}
1289
1290
1291static PyObject *
1292PySaveKey(PyObject *self, PyObject *args)
1293{
1294 HKEY hKey;
1295 PyObject *obKey;
1296 char *fileName;
1297 LPSECURITY_ATTRIBUTES pSA = NULL;
1298
1299 long rc;
1300 if (!PyArg_ParseTuple(args, "Os:SaveKey", &obKey, &fileName))
1301 return NULL;
1302 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1303 return NULL;
1304/* One day we may get security into the core?
1305 if (!PyWinObject_AsSECURITY_ATTRIBUTES(obSA, &pSA, TRUE))
1306 return NULL;
1307*/
1308 Py_BEGIN_ALLOW_THREADS
1309 rc = RegSaveKey(hKey, fileName, pSA );
1310 Py_END_ALLOW_THREADS
1311 if (rc != ERROR_SUCCESS)
1312 return PyErr_SetFromWindowsErrWithFunction(rc, "RegSaveKey");
1313 Py_INCREF(Py_None);
1314 return Py_None;
1315}
1316
1317static PyObject *
1318PySetValue(PyObject *self, PyObject *args)
1319{
1320 HKEY hKey;
1321 PyObject *obKey;
1322 char *subKey;
1323 char *str;
1324 DWORD typ;
1325 DWORD len;
1326 long rc;
1327 PyObject *obStrVal;
1328 PyObject *obSubKey;
1329 if (!PyArg_ParseTuple(args, "OOiO:SetValue",
1330 &obKey,
1331 &obSubKey,
1332 &typ,
1333 &obStrVal))
1334 return NULL;
1335 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1336 return NULL;
1337 if (typ != REG_SZ) {
1338 PyErr_SetString(PyExc_TypeError,
Thomas Hellere1d18f52002-12-20 20:13:35 +00001339 "Type must be _winreg.REG_SZ");
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001340 return NULL;
1341 }
1342 /* XXX - need Unicode support */
1343 str = PyString_AsString(obStrVal);
1344 if (str == NULL)
1345 return NULL;
1346 len = PyString_Size(obStrVal);
1347 if (obSubKey == Py_None)
1348 subKey = NULL;
1349 else {
1350 subKey = PyString_AsString(obSubKey);
1351 if (subKey == NULL)
1352 return NULL;
1353 }
1354 Py_BEGIN_ALLOW_THREADS
1355 rc = RegSetValue(hKey, subKey, REG_SZ, str, len+1);
1356 Py_END_ALLOW_THREADS
1357 if (rc != ERROR_SUCCESS)
1358 return PyErr_SetFromWindowsErrWithFunction(rc, "RegSetValue");
1359 Py_INCREF(Py_None);
1360 return Py_None;
1361}
1362
1363static PyObject *
1364PySetValueEx(PyObject *self, PyObject *args)
1365{
1366 HKEY hKey;
1367 PyObject *obKey;
1368 char *valueName;
1369 PyObject *obRes;
1370 PyObject *value;
1371 BYTE *data;
1372 DWORD len;
1373 DWORD typ;
1374
1375 LONG rc;
1376
1377 if (!PyArg_ParseTuple(args, "OzOiO:SetValueEx",
1378 &obKey,
1379 &valueName,
1380 &obRes,
1381 &typ,
1382 &value))
1383 return NULL;
1384 if (!PyHKEY_AsHKEY(obKey, &hKey, FALSE))
1385 return NULL;
1386 if (!Py2Reg(value, typ, &data, &len))
1387 {
1388 if (!PyErr_Occurred())
1389 PyErr_SetString(PyExc_ValueError,
1390 "Could not convert the data to the specified type.");
1391 return NULL;
1392 }
1393 Py_BEGIN_ALLOW_THREADS
1394 rc = RegSetValueEx(hKey, valueName, 0, typ, data, len);
1395 Py_END_ALLOW_THREADS
Guido van Rossumb18618d2000-05-03 23:44:39 +00001396 PyMem_DEL(data);
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001397 if (rc != ERROR_SUCCESS)
1398 return PyErr_SetFromWindowsErrWithFunction(rc,
1399 "RegSetValueEx");
1400 Py_INCREF(Py_None);
1401 return Py_None;
1402}
1403
1404static struct PyMethodDef winreg_methods[] = {
Neal Norwitz031829d2002-03-31 14:37:44 +00001405 {"CloseKey", PyCloseKey, METH_VARARGS, CloseKey_doc},
1406 {"ConnectRegistry", PyConnectRegistry, METH_VARARGS, ConnectRegistry_doc},
1407 {"CreateKey", PyCreateKey, METH_VARARGS, CreateKey_doc},
1408 {"DeleteKey", PyDeleteKey, METH_VARARGS, DeleteKey_doc},
1409 {"DeleteValue", PyDeleteValue, METH_VARARGS, DeleteValue_doc},
1410 {"EnumKey", PyEnumKey, METH_VARARGS, EnumKey_doc},
1411 {"EnumValue", PyEnumValue, METH_VARARGS, EnumValue_doc},
1412 {"FlushKey", PyFlushKey, METH_VARARGS, FlushKey_doc},
1413 {"LoadKey", PyLoadKey, METH_VARARGS, LoadKey_doc},
1414 {"OpenKey", PyOpenKey, METH_VARARGS, OpenKey_doc},
1415 {"OpenKeyEx", PyOpenKey, METH_VARARGS, OpenKeyEx_doc},
1416 {"QueryValue", PyQueryValue, METH_VARARGS, QueryValue_doc},
1417 {"QueryValueEx", PyQueryValueEx, METH_VARARGS, QueryValueEx_doc},
1418 {"QueryInfoKey", PyQueryInfoKey, METH_VARARGS, QueryInfoKey_doc},
1419 {"SaveKey", PySaveKey, METH_VARARGS, SaveKey_doc},
1420 {"SetValue", PySetValue, METH_VARARGS, SetValue_doc},
1421 {"SetValueEx", PySetValueEx, METH_VARARGS, SetValueEx_doc},
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001422 NULL,
1423};
1424
1425static void
1426insint(PyObject * d, char * name, long value)
1427{
1428 PyObject *v = PyInt_FromLong(value);
1429 if (!v || PyDict_SetItemString(d, name, v))
1430 PyErr_Clear();
1431 Py_XDECREF(v);
1432}
1433
1434#define ADD_INT(val) insint(d, #val, val)
1435
1436static void
1437inskey(PyObject * d, char * name, HKEY key)
1438{
1439 PyObject *v = PyLong_FromVoidPtr(key);
1440 if (!v || PyDict_SetItemString(d, name, v))
1441 PyErr_Clear();
1442 Py_XDECREF(v);
1443}
1444
1445#define ADD_KEY(val) inskey(d, #val, val)
1446
Mark Hammond8235ea12002-07-19 06:55:41 +00001447PyMODINIT_FUNC init_winreg(void)
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001448{
1449 PyObject *m, *d;
Fred Drake270e19b2000-06-29 16:14:14 +00001450 m = Py_InitModule3("_winreg", winreg_methods, module_doc);
Neal Norwitz1ac754f2006-01-19 06:09:39 +00001451 if (m == NULL)
1452 return;
Guido van Rossum9f3712c2000-03-28 20:37:15 +00001453 d = PyModule_GetDict(m);
1454 PyHKEY_Type.ob_type = &PyType_Type;
1455 PyHKEY_Type.tp_doc = PyHKEY_doc;
1456 Py_INCREF(&PyHKEY_Type);
1457 if (PyDict_SetItemString(d, "HKEYType",
1458 (PyObject *)&PyHKEY_Type) != 0)
1459 return;
1460 Py_INCREF(PyExc_WindowsError);
1461 if (PyDict_SetItemString(d, "error",
1462 PyExc_WindowsError) != 0)
1463 return;
1464
1465 /* Add the relevant constants */
1466 ADD_KEY(HKEY_CLASSES_ROOT);
1467 ADD_KEY(HKEY_CURRENT_USER);
1468 ADD_KEY(HKEY_LOCAL_MACHINE);
1469 ADD_KEY(HKEY_USERS);
1470 ADD_KEY(HKEY_PERFORMANCE_DATA);
1471#ifdef HKEY_CURRENT_CONFIG
1472 ADD_KEY(HKEY_CURRENT_CONFIG);
1473#endif
1474#ifdef HKEY_DYN_DATA
1475 ADD_KEY(HKEY_DYN_DATA);
1476#endif
1477 ADD_INT(KEY_QUERY_VALUE);
1478 ADD_INT(KEY_SET_VALUE);
1479 ADD_INT(KEY_CREATE_SUB_KEY);
1480 ADD_INT(KEY_ENUMERATE_SUB_KEYS);
1481 ADD_INT(KEY_NOTIFY);
1482 ADD_INT(KEY_CREATE_LINK);
1483 ADD_INT(KEY_READ);
1484 ADD_INT(KEY_WRITE);
1485 ADD_INT(KEY_EXECUTE);
1486 ADD_INT(KEY_ALL_ACCESS);
1487 ADD_INT(REG_OPTION_RESERVED);
1488 ADD_INT(REG_OPTION_NON_VOLATILE);
1489 ADD_INT(REG_OPTION_VOLATILE);
1490 ADD_INT(REG_OPTION_CREATE_LINK);
1491 ADD_INT(REG_OPTION_BACKUP_RESTORE);
1492 ADD_INT(REG_OPTION_OPEN_LINK);
1493 ADD_INT(REG_LEGAL_OPTION);
1494 ADD_INT(REG_CREATED_NEW_KEY);
1495 ADD_INT(REG_OPENED_EXISTING_KEY);
1496 ADD_INT(REG_WHOLE_HIVE_VOLATILE);
1497 ADD_INT(REG_REFRESH_HIVE);
1498 ADD_INT(REG_NO_LAZY_FLUSH);
1499 ADD_INT(REG_NOTIFY_CHANGE_NAME);
1500 ADD_INT(REG_NOTIFY_CHANGE_ATTRIBUTES);
1501 ADD_INT(REG_NOTIFY_CHANGE_LAST_SET);
1502 ADD_INT(REG_NOTIFY_CHANGE_SECURITY);
1503 ADD_INT(REG_LEGAL_CHANGE_FILTER);
1504 ADD_INT(REG_NONE);
1505 ADD_INT(REG_SZ);
1506 ADD_INT(REG_EXPAND_SZ);
1507 ADD_INT(REG_BINARY);
1508 ADD_INT(REG_DWORD);
1509 ADD_INT(REG_DWORD_LITTLE_ENDIAN);
1510 ADD_INT(REG_DWORD_BIG_ENDIAN);
1511 ADD_INT(REG_LINK);
1512 ADD_INT(REG_MULTI_SZ);
1513 ADD_INT(REG_RESOURCE_LIST);
1514 ADD_INT(REG_FULL_RESOURCE_DESCRIPTOR);
1515 ADD_INT(REG_RESOURCE_REQUIREMENTS_LIST);
1516}
1517