* PURPOSE: Sdb low level glue layer
* COPYRIGHT: Copyright 2011 André Hentschel
* Copyright 2013 Mislav Blaževic
- * Copyright 2015-2018 Mark Jansen (mark.jansen@reactos.org)
+ * Copyright 2015-2019 Mark Jansen (mark.jansen@reactos.org)
*/
#include "ntndk.h"
ASSERT(pdb->for_write);
Status = NtWriteFile(pdb->file, NULL, NULL, NULL, &io,
pdb->data, pdb->write_iter, NULL, NULL);
- if( !NT_SUCCESS(Status))
+ if (!NT_SUCCESS(Status))
SHIM_WARN("failed with 0x%lx\n", Status);
}
return SdbpCheckTagType(tag, type);
}
-PDB SdbpOpenDatabase(LPCWSTR path, PATH_TYPE type, PDWORD major, PDWORD minor)
+PDB SdbpOpenDatabase(LPCWSTR path, PATH_TYPE type)
{
IO_STATUS_BLOCK io;
FILE_STANDARD_INFORMATION fsi;
return NULL;
}
- *major = *(DWORD*)&header[0];
- *minor = *(DWORD*)&header[4];
+ pdb->major = *(DWORD*)&header[0];
+ pdb->minor = *(DWORD*)&header[4];
return pdb;
}
PDB WINAPI SdbOpenDatabase(LPCWSTR path, PATH_TYPE type)
{
PDB pdb;
- DWORD major, minor;
+ TAGID root, name;
- pdb = SdbpOpenDatabase(path, type, &major, &minor);
+ pdb = SdbpOpenDatabase(path, type);
if (!pdb)
return NULL;
- if (major != 2 && major != 3)
+ if (pdb->major != 2 && pdb->major != 3)
{
SdbCloseDatabase(pdb);
SHIM_ERR("Invalid shim database version\n");
}
pdb->stringtable = SdbFindFirstTag(pdb, TAGID_ROOT, TAG_STRINGTABLE);
- if(!SdbGetDatabaseID(pdb, &pdb->database_id))
+ if (!SdbGetDatabaseID(pdb, &pdb->database_id))
{
SHIM_INFO("Failed to get the database id\n");
}
+
+ root = SdbFindFirstTag(pdb, TAGID_ROOT, TAG_DATABASE);
+ if (root != TAGID_NULL)
+ {
+ name = SdbFindFirstTag(pdb, root, TAG_NAME);
+ if (name != TAGID_NULL)
+ {
+ pdb->database_name = SdbGetStringTagPtr(pdb, name);
+ }
+ }
+ if (!pdb->database_name)
+ {
+ SHIM_INFO("Failed to get the database name\n");
+ }
+
return pdb;
}
BOOL WINAPI SdbGUIDToString(CONST GUID *Guid, PWSTR GuidString, SIZE_T Length)
{
UNICODE_STRING GuidString_u;
- if(NT_SUCCESS(RtlStringFromGUID(Guid, &GuidString_u)))
+ if (NT_SUCCESS(RtlStringFromGUID(Guid, &GuidString_u)))
{
HRESULT hr = StringCchCopyNW(GuidString, Length, GuidString_u.Buffer, GuidString_u.Length / 2);
RtlFreeUnicodeString(&GuidString_u);
SHIM_ERR("Cannot obtain database guid for databases other than main\n");
return FALSE;
}
- if(Guid)
+ if (Guid)
{
memcpy(Guid, copy_from, sizeof(GUID));
}
{
PDB pdb;
- pdb = SdbpOpenDatabase(database, DOS_PATH, VersionHi, VersionLo);
+ pdb = SdbpOpenDatabase(database, DOS_PATH);
if (pdb)
+ {
+ *VersionHi = pdb->major;
+ *VersionLo = pdb->minor;
SdbCloseDatabase(pdb);
+ }
return TRUE;
}
+/**
+ * @name SdbGetDatabaseInformation
+ * Get information about the database
+ *
+ * @param pdb The database
+ * @param information The returned information
+ * @return TRUE on success
+ */
+BOOL WINAPI SdbGetDatabaseInformation(PDB pdb, PDB_INFORMATION information)
+{
+ if (pdb && information)
+ {
+ information->dwFlags = 0;
+ information->dwMajor = pdb->major;
+ information->dwMinor = pdb->minor;
+ information->Description = pdb->database_name;
+ if (!SdbIsNullGUID(&pdb->database_id))
+ {
+ information->dwFlags |= DB_INFO_FLAGS_VALID_GUID;
+ information->Id = pdb->database_id;
+ }
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+/**
+ * @name SdbFreeDatabaseInformation
+ * Free up resources allocated in SdbGetDatabaseInformation
+ *
+ * @param information The information retrieved from SdbGetDatabaseInformation
+ */
+VOID WINAPI SdbFreeDatabaseInformation(PDB_INFORMATION information)
+{
+ // No-op
+}
+
/**
* Find the first named child tag.
* PURPOSE: Tests for shim-database api's
* COPYRIGHT: Copyright 2012 Detlef Riekenberg
* Copyright 2013 Mislav Blažević
- * Copyright 2015-2018 Mark Jansen (mark.jansen@reactos.org)
+ * Copyright 2015-2019 Mark Jansen (mark.jansen@reactos.org)
*/
#include <ntstatus.h>
#define TAG_DATABASE_ID (0x7 | TAG_TYPE_BINARY)
+typedef struct _DB_INFORMATION
+{
+ DWORD dwSomething;
+ DWORD dwMajor;
+ DWORD dwMinor;
+ LPCWSTR Description;
+ GUID Id;
+ /* Win10+ has an extra field here */
+} DB_INFORMATION, *PDB_INFORMATION;
+
static HMODULE hdll;
static LPCWSTR (WINAPI *pSdbTagToString)(TAG);
static DWORD (WINAPI* pSdbQueryData)(HSDB hsdb, TAGREF trWhich, LPCWSTR lpszDataName, LPDWORD lpdwDataType, LPVOID lpBuffer, LPDWORD lpcbBufferSize);
static DWORD (WINAPI* pSdbQueryDataEx)(HSDB hsdb, TAGREF trWhich, LPCWSTR lpszDataName, LPDWORD lpdwDataType, LPVOID lpBuffer, LPDWORD lpcbBufferSize, TAGREF *ptrData);
static DWORD (WINAPI* pSdbQueryDataExTagID)(PDB pdb, TAGID tiExe, LPCWSTR lpszDataName, LPDWORD lpdwDataType, LPVOID lpBuffer, LPDWORD lpcbBufferSize, TAGID *ptiData);
+static BOOL (WINAPI* pSdbGetDatabaseInformation)(PDB pdb, PDB_INFORMATION information);
+static VOID (WINAPI* pSdbFreeDatabaseInformation)(PDB_INFORMATION information);
DEFINE_GUID(GUID_DATABASE_TEST, 0xe39b0eb0, 0x55db, 0x450b, 0x9b, 0xd4, 0xd2, 0x0c, 0x94, 0x84, 0x26, 0x0f);
DEFINE_GUID(GUID_MAIN_DATABASE, 0x11111111, 0x1111, 0x1111, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11);
WriteFile(file, buffer, size, &dwWritten, NULL);
}
+static void test_GetDatabaseInformationEmpty(PDB pdb)
+{
+ PDB_INFORMATION pInfo;
+ BOOL fResult;
+
+ if (!pSdbGetDatabaseInformation || !pSdbFreeDatabaseInformation)
+ {
+ skip("GetDatabaseInformation or SdbFreeDatabaseInformation not found\n");
+ return;
+ }
+
+ pInfo = (PDB_INFORMATION)malloc(sizeof(*pInfo) * 4);
+ memset(pInfo, 0xDE, sizeof(*pInfo) * 2);
+
+ fResult = pSdbGetDatabaseInformation(pdb, pInfo);
+ ok(fResult, "SdbGetDatabaseInformation failed\n");
+ if (fResult)
+ {
+ ok_int(pInfo->dwSomething, 0);
+ ok(IsEqualGUID(GUID_NULL, pInfo->Id), "expected guid to be empty(%s)\n", wine_dbgstr_guid(&pInfo->Id));
+ ok(pInfo->Description == NULL, "Expected pInfo->Description to be NULL, was %s\n", wine_dbgstr_w(pInfo->Description));
+
+ /* Struct is slightly bigger on some Win10, and the DB version nr is different on all */
+ if (g_WinVersion >= WINVER_WIN10)
+ {
+ ok(pInfo->dwMajor == 3, "Expected pInfo->dwMajor to be 3, was: %d\n", pInfo->dwMajor);
+ ok(pInfo->dwMinor == 0, "Expected pInfo->dwMinor to be 0, was: %d\n", pInfo->dwMinor);
+
+ ok(pInfo[1].dwSomething == 0 || pInfo[1].dwSomething == 0xdededede, "Something amiss: 0x%x\n", pInfo[1].dwSomething);
+ ok(pInfo[1].dwMajor == 0xdededede, "Cookie2 corrupt: 0x%x\n", pInfo[1].dwMajor);
+ }
+ else
+ {
+ ok(pInfo->dwMajor == 2, "Expected pInfo->dwMajor to be 2, was: %d\n", pInfo->dwMajor);
+ ok(pInfo->dwMinor == 1, "Expected pInfo->dwMinor to be 1, was: %d\n", pInfo->dwMinor);
+
+ ok(pInfo[1].dwSomething == 0xdededede, "Cookie1 corrupt: 0x%x\n", pInfo[1].dwSomething);
+ ok(pInfo[1].dwMajor == 0xdededede, "Cookie2 corrupt: 0x%x\n", pInfo[1].dwMajor);
+ }
+
+ }
+ free(pInfo);
+}
+
+
static void test_Sdb(void)
{
static const WCHAR temp[] = L"temp";
word = pSdbReadWORDTag(pdb, tagid, 0);
ok(word == 0xACE, "unexpected value 0x%x, expected 0x%x\n", word, 0xACE);
+ test_GetDatabaseInformationEmpty(pdb);
+
pSdbCloseDatabase(pdb);
}
DeleteFileW(path1);
ok(num == 2, "Expected to find 2 layer tags, found: %d\n", num);
}
+static void test_GetDatabaseInformation(PDB pdb)
+{
+ PDB_INFORMATION pInfo;
+ BOOL fResult;
+
+ if (!pSdbGetDatabaseInformation || !pSdbFreeDatabaseInformation)
+ {
+ skip("GetDatabaseInformation or SdbFreeDatabaseInformation not found\n");
+ return;
+ }
+
+ _SEH2_TRY
+ {
+ pSdbFreeDatabaseInformation(NULL);
+ }
+ _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
+ {
+ ok(0, "SdbFreeDatabaseInformation did not handle a NULL pointer very gracefully.\n");
+ }
+ _SEH2_END;
+
+
+ pInfo = (PDB_INFORMATION)malloc(sizeof(*pInfo) * 4);
+ memset(pInfo, 0xDE, sizeof(*pInfo) * 2);
+
+ fResult = pSdbGetDatabaseInformation(pdb, pInfo);
+ ok(fResult, "SdbGetDatabaseInformation failed\n");
+ if (fResult)
+ {
+ ok_int(pInfo->dwSomething, 1);
+ ok(IsEqualGUID(GUID_DATABASE_TEST, pInfo->Id), "expected guids to be equal(%s:%s)\n",
+ wine_dbgstr_guid(&GUID_DATABASE_TEST), wine_dbgstr_guid(&pInfo->Id));
+ ok(wcscmp(pInfo->Description, L"apphelp_test1") == 0,
+ "Expected pInfo->Description to be 'apphelp_test1', was %s\n", wine_dbgstr_w(pInfo->Description));
+
+ /* Struct is slightly bigger on some Win10, and the DB version nr is different on all */
+ if (g_WinVersion >= WINVER_WIN10)
+ {
+ ok(pInfo->dwMajor == 3, "Expected pInfo->dwMajor to be 3, was: %d\n", pInfo->dwMajor);
+ ok(pInfo->dwMinor == 0, "Expected pInfo->dwMinor to be 0, was: %d\n", pInfo->dwMinor);
+
+ ok(pInfo[1].dwSomething == 4 || pInfo[1].dwSomething == 0xdededede, "Something amiss: 0x%x\n", pInfo[1].dwSomething);
+ ok(pInfo[1].dwMajor == 0xdededede, "Cookie2 corrupt: 0x%x\n", pInfo[1].dwMajor);
+ }
+ else
+ {
+ ok(pInfo->dwMajor == 2, "Expected pInfo->dwMajor to be 2, was: %d\n", pInfo->dwMajor);
+ ok(pInfo->dwMinor == 1, "Expected pInfo->dwMinor to be 1, was: %d\n", pInfo->dwMinor);
+
+ ok(pInfo[1].dwSomething == 0xdededede, "Cookie1 corrupt: 0x%x\n", pInfo[1].dwSomething);
+ ok(pInfo[1].dwMajor == 0xdededede, "Cookie2 corrupt: 0x%x\n", pInfo[1].dwMajor);
+ }
+
+ }
+ free(pInfo);
+}
+
static void test_CheckDatabaseManually(void)
{
static const WCHAR path[] = {'t','e','s','t','_','d','b','.','s','d','b',0};
check_db_exes(pdb, root);
check_db_apphelp(pdb, root);
}
+ test_GetDatabaseInformation(pdb);
pSdbCloseDatabase(pdb);
DeleteFileA("test_db.sdb");
*(void**)&pSdbQueryDataEx = (void *)GetProcAddress(hdll, "SdbQueryDataEx");
*(void**)&pSdbQueryDataExTagID = (void *)GetProcAddress(hdll, "SdbQueryDataExTagID");
*(void**)&pSdbGetLayerTagRef = (void *)GetProcAddress(hdll, "SdbGetLayerTagRef");
+ *(void**)&pSdbGetDatabaseInformation = (void *)GetProcAddress(hdll, "SdbGetDatabaseInformation");
+ *(void**)&pSdbFreeDatabaseInformation = (void *)GetProcAddress(hdll, "SdbFreeDatabaseInformation");
test_Sdb();
test_write_ex();