/* * Copyright 2022 Esme Povirk * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #define COBJMACROS #include "windef.h" #include "winbase.h" #include "objbase.h" #include "rpc.h" #include "wine/mscvpdb.h" #include "wine/debug.h" #include "wine/heap.h" #include "diasymreader_private.h" WINE_DEFAULT_DEBUG_CHANNEL(diasymreader); typedef struct SymWriter { ISymUnmanagedWriter5 iface; IPdbWriter IPdbWriter_iface; LONG ref; CRITICAL_SECTION lock; GUID pdb_guid; DWORD pdb_timestamp; DWORD pdb_age; WCHAR pdb_filename[MAX_PATH]; } SymWriter; typedef struct SymDocumentWriter { ISymUnmanagedDocumentWriter iface; LONG ref; } SymDocumentWriter; static inline SymWriter *impl_from_ISymUnmanagedWriter5(ISymUnmanagedWriter5 *iface) { return CONTAINING_RECORD(iface, SymWriter, iface); } static inline SymWriter *impl_from_IPdbWriter(IPdbWriter *iface) { return CONTAINING_RECORD(iface, SymWriter, IPdbWriter_iface); } static inline SymDocumentWriter *impl_from_ISymUnmanagedDocumentWriter(ISymUnmanagedDocumentWriter *iface) { return CONTAINING_RECORD(iface, SymDocumentWriter, iface); } static HRESULT WINAPI SymDocumentWriter_QueryInterface(ISymUnmanagedDocumentWriter *iface, REFIID iid, void **ppv) { SymDocumentWriter *This = impl_from_ISymUnmanagedDocumentWriter(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_ISymUnmanagedDocumentWriter, iid)) { *ppv = &This->iface; } else { WARN("unknown interface %s\n", debugstr_guid(iid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI SymDocumentWriter_AddRef(ISymUnmanagedDocumentWriter *iface) { SymDocumentWriter *This = impl_from_ISymUnmanagedDocumentWriter(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p) refcount=%lu\n", iface, ref); return ref; } static ULONG WINAPI SymDocumentWriter_Release(ISymUnmanagedDocumentWriter *iface) { SymDocumentWriter *This = impl_from_ISymUnmanagedDocumentWriter(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p) refcount=%lu\n", iface, ref); if (ref == 0) { heap_free(This); } return ref; } static HRESULT WINAPI SymDocumentWriter_SetSource(ISymUnmanagedDocumentWriter *iface, ULONG32 sourceSize, BYTE source[]) { FIXME("(%p,%u,%p)\n", iface, sourceSize, source); return S_OK; } static HRESULT WINAPI SymDocumentWriter_SetChecksum(ISymUnmanagedDocumentWriter *iface, GUID algorithmID, ULONG32 checkSumSize, BYTE checkSum[]) { FIXME("(%p,%s,%u,%p)\n", iface, debugstr_guid(&algorithmID), checkSumSize, checkSum); return S_OK; } static const ISymUnmanagedDocumentWriterVtbl SymDocumentWriter_Vtbl = { SymDocumentWriter_QueryInterface, SymDocumentWriter_AddRef, SymDocumentWriter_Release, SymDocumentWriter_SetSource, SymDocumentWriter_SetChecksum }; static HRESULT WINAPI SymWriter_QueryInterface(ISymUnmanagedWriter5 *iface, REFIID iid, void **ppv) { SymWriter *This = impl_from_ISymUnmanagedWriter5(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_ISymUnmanagedWriter, iid) || IsEqualIID(&IID_ISymUnmanagedWriter2, iid) || IsEqualIID(&IID_ISymUnmanagedWriter3, iid) || IsEqualIID(&IID_ISymUnmanagedWriter4, iid) || IsEqualIID(&IID_ISymUnmanagedWriter5, iid)) { *ppv = &This->iface; } else if (IsEqualIID(&IID_IPdbWriter, iid)) { *ppv = &This->IPdbWriter_iface; } else { WARN("unknown interface %s\n", debugstr_guid(iid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI SymWriter_AddRef(ISymUnmanagedWriter5 *iface) { SymWriter *This = impl_from_ISymUnmanagedWriter5(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p) refcount=%lu\n", iface, ref); return ref; } static ULONG WINAPI SymWriter_Release(ISymUnmanagedWriter5 *iface) { SymWriter *This = impl_from_ISymUnmanagedWriter5(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p) refcount=%lu\n", iface, ref); if (ref == 0) { This->lock.DebugInfo->Spare[0] = 0; DeleteCriticalSection(&This->lock); heap_free(This); } return ref; } static HRESULT WINAPI SymWriter_DefineDocument(ISymUnmanagedWriter5 *iface, const WCHAR *url, const GUID *language, const GUID *languageVendor, const GUID *documentType, ISymUnmanagedDocumentWriter** pRetVal) { SymDocumentWriter *result; FIXME("(%p,%s,%s,%s,%s,%p)\n", iface, debugstr_w(url), debugstr_guid(language), debugstr_guid(languageVendor), debugstr_guid(documentType), pRetVal); if (!pRetVal) return E_POINTER; result = heap_alloc(sizeof(*result)); if (!result) return E_OUTOFMEMORY; result->iface.lpVtbl = &SymDocumentWriter_Vtbl; result->ref = 1; *pRetVal = &result->iface; return S_OK; } static HRESULT WINAPI SymWriter_SetUserEntryPoint(ISymUnmanagedWriter5 *iface, mdMethodDef entryMethod) { FIXME("(%p,0x%x)\n", iface, entryMethod); return S_OK; } static HRESULT WINAPI SymWriter_OpenMethod(ISymUnmanagedWriter5 *iface, mdMethodDef method) { FIXME("(%p,0x%x)\n", iface, method); return S_OK; } static HRESULT WINAPI SymWriter_CloseMethod(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_OpenScope(ISymUnmanagedWriter5 *iface, ULONG32 startOffset, ULONG32 *pRetVal) { FIXME("(%p,%u,%p)\n", iface, startOffset, pRetVal); *pRetVal = 0xdeadbeef; return S_OK; } static HRESULT WINAPI SymWriter_CloseScope(ISymUnmanagedWriter5 *iface, ULONG32 endOffset) { FIXME("(%p,%u)\n", iface, endOffset); return S_OK; } static HRESULT WINAPI SymWriter_SetScopeRange(ISymUnmanagedWriter5 *iface, ULONG32 scopeID, ULONG32 startOffset, ULONG32 endOffset) { FIXME("(%p,%u,%u,%u)\n", iface, scopeID, startOffset, endOffset); return S_OK; } static HRESULT WINAPI SymWriter_DefineLocalVariable(ISymUnmanagedWriter5 *iface, const WCHAR *name, ULONG32 attributes, ULONG32 cSig, unsigned char signature[], ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3, ULONG32 startOffset, ULONG32 endOffset) { FIXME("(%p,%s,0x%x,%u,%u)\n", iface, debugstr_w(name), attributes, cSig, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_DefineParameter(ISymUnmanagedWriter5 *iface, const WCHAR *name, ULONG32 attributes, ULONG32 sequence, ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3) { FIXME("(%p,%s,0x%x,%u,%u)\n", iface, debugstr_w(name), attributes, sequence, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_DefineField(ISymUnmanagedWriter5 *iface, mdTypeDef parent, const WCHAR *name, ULONG32 attributes, ULONG32 cSig, unsigned char signature[], ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3) { FIXME("(%p,0x%x,%s,0x%x,%u,%u)\n", iface, parent, debugstr_w(name), attributes, cSig, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_DefineGlobalVariable(ISymUnmanagedWriter5 *iface, const WCHAR *name, ULONG32 attributes, ULONG32 cSig, unsigned char signature[], ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3) { FIXME("(%p,%s,0x%x,%u,%u)\n", iface, debugstr_w(name), attributes, cSig, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_Close(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_SetSymAttributes(ISymUnmanagedWriter5 *iface, mdToken parent, const WCHAR *name, ULONG32 cData, unsigned char data[]) { FIXME("(%p,0x%x,%s,%u)\n", iface, parent, debugstr_w(name), cData); return S_OK; } static HRESULT WINAPI SymWriter_OpenNamespace(ISymUnmanagedWriter5 *iface, const WCHAR *name) { FIXME("(%p,%s)\n", iface, debugstr_w(name)); return S_OK; } static HRESULT WINAPI SymWriter_CloseNamespace(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_UsingNamespace(ISymUnmanagedWriter5 *iface, const WCHAR *fullName) { FIXME("(%p,%s)\n", iface, debugstr_w(fullName)); return S_OK; } static HRESULT WINAPI SymWriter_SetMethodSourceRange(ISymUnmanagedWriter5 *iface, ISymUnmanagedDocumentWriter *startDoc, ULONG32 startLine, ULONG32 startColumn, ISymUnmanagedDocumentWriter *endDoc, ULONG32 endLine, ULONG32 endColumn) { FIXME("(%p,%p,%u,%u,%p,%u,%u)\n", iface, startDoc, startLine, startColumn, endDoc, endLine, endColumn); return S_OK; } static HRESULT WINAPI SymWriter_Initialize(ISymUnmanagedWriter5 *iface, IUnknown *emitter, const WCHAR *filename, IStream *pIStream, BOOL fFullBuild) { SymWriter *This = impl_from_ISymUnmanagedWriter5(iface); FIXME("(%p,%p,%s,%p,%u)\n", iface, emitter, debugstr_w(filename), pIStream, fFullBuild); EnterCriticalSection(&This->lock); if (filename) wcsncpy_s(This->pdb_filename, MAX_PATH, filename, _TRUNCATE); This->pdb_timestamp = _time32(NULL); LeaveCriticalSection(&This->lock); return S_OK; } static HRESULT WINAPI SymWriter_GetDebugInfo(ISymUnmanagedWriter5 *iface, IMAGE_DEBUG_DIRECTORY *pIDD, DWORD cData, DWORD *pcData, BYTE data[]) { SymWriter *This = impl_from_ISymUnmanagedWriter5(iface); DWORD name_length, data_size; OMFSignatureRSDS *rsds_data = (OMFSignatureRSDS*)data; TRACE("(%p,%p,%lu,%p,%p)\n", iface, pIDD, cData, pcData, data); EnterCriticalSection(&This->lock); name_length = WideCharToMultiByte(CP_UTF8, 0, This->pdb_filename, -1, NULL, 0, NULL, NULL); data_size = FIELD_OFFSET(OMFSignatureRSDS, name) + name_length; if (pcData) *pcData = data_size; if (pIDD) { pIDD->Characteristics = 0; pIDD->MajorVersion = 0; pIDD->MinorVersion = 0; pIDD->Type = IMAGE_DEBUG_TYPE_CODEVIEW; pIDD->SizeOfData = data_size; } if (data) { if (data_size > cData) { LeaveCriticalSection(&This->lock); return HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER); } memcpy(rsds_data->Signature, "RSDS", 4); rsds_data->guid = This->pdb_guid; rsds_data->age = This->pdb_age; WideCharToMultiByte(CP_UTF8, 0, This->pdb_filename, -1, rsds_data->name, name_length, NULL, NULL); } LeaveCriticalSection(&This->lock); return S_OK; } static HRESULT WINAPI SymWriter_DefineSequencePoints(ISymUnmanagedWriter5 *iface, ISymUnmanagedDocumentWriter *document, ULONG32 spCount, ULONG32 offsets[], ULONG32 lines[], ULONG32 columns[], ULONG32 endLines[], ULONG32 endColumns[]) { FIXME("(%p,%p,%u)\n", iface, document, spCount); return S_OK; } static HRESULT WINAPI SymWriter_RemapToken(ISymUnmanagedWriter5 *iface, mdToken oldToken, mdToken newToken) { FIXME("(%p,0x%x,0x%x)\n", iface, oldToken, newToken); return S_OK; } static HRESULT WINAPI SymWriter_Initialize2(ISymUnmanagedWriter5 *iface, IUnknown *emitter, const WCHAR *tempFilename, IStream *pIStream, BOOL fFullBuild, const WCHAR *finalFilename) { FIXME("(%p,%p,%s,%p,%u,%s)\n", iface, emitter, debugstr_w(tempFilename), pIStream, fFullBuild, debugstr_w(finalFilename)); return S_OK; } static HRESULT WINAPI SymWriter_DefineConstant(ISymUnmanagedWriter5 *iface, const WCHAR *name, VARIANT value, ULONG32 cSig, unsigned char signature[]) { FIXME("(%p,%s,%s,%u,%p)\n", iface, debugstr_w(name), debugstr_variant(&value), cSig, signature); return S_OK; } static HRESULT WINAPI SymWriter_Abort(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_DefineLocalVariable2(ISymUnmanagedWriter5 *iface, const WCHAR *name, ULONG32 attributes, mdSignature sigToken, ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3, ULONG32 startOffset, ULONG32 endOffset) { FIXME("(%p,%s,0x%x,0x%x,%u)\n", iface, debugstr_w(name), attributes, sigToken, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_DefineGlobalVariable2(ISymUnmanagedWriter5 *iface, const WCHAR *name, ULONG32 attributes, mdSignature sigToken, ULONG32 addrKind, ULONG32 addr1, ULONG32 addr2, ULONG32 addr3) { FIXME("(%p,%s,0x%x,0x%x,%u)\n", iface, debugstr_w(name), attributes, sigToken, addrKind); return S_OK; } static HRESULT WINAPI SymWriter_DefineConstant2(ISymUnmanagedWriter5 *iface, const WCHAR *name, VARIANT value, mdSignature sigToken) { FIXME("(%p,%s,%s,0x%x)\n", iface, debugstr_w(name), debugstr_variant(&value), sigToken); return S_OK; } static HRESULT WINAPI SymWriter_OpenMethod2(ISymUnmanagedWriter5 *iface, mdMethodDef method, ULONG32 isect, ULONG32 offset) { FIXME("(%p,0x%x,%u,%u)\n", iface, method, isect, offset); return S_OK; } static HRESULT WINAPI SymWriter_Commit(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_GetDebugInfoWithPadding(ISymUnmanagedWriter5 *iface, IMAGE_DEBUG_DIRECTORY *pIDD, DWORD cbData, DWORD* pcData, BYTE data[]) { FIXME("(%p,%p,%lu,%p,%p)\n", iface, pIDD, cbData, pcData, data); return E_NOTIMPL; } static HRESULT WINAPI SymWriter_OpenMapTokensToSourceSpans(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_CloseMapTokensToSourceSpans(ISymUnmanagedWriter5 *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_MapTokenToSourceSpan(ISymUnmanagedWriter5 *iface, mdToken token, ISymUnmanagedDocumentWriter* document, ULONG32 line, ULONG32 column, ULONG32 endLine, ULONG32 endColumn) { FIXME("(%p,%x,%p,%u,%u,%u,%u)\n", iface, token, document, line, column, endLine, endColumn); return S_OK; } static const ISymUnmanagedWriter5Vtbl SymWriter_Vtbl = { SymWriter_QueryInterface, SymWriter_AddRef, SymWriter_Release, SymWriter_DefineDocument, SymWriter_SetUserEntryPoint, SymWriter_OpenMethod, SymWriter_CloseMethod, SymWriter_OpenScope, SymWriter_CloseScope, SymWriter_SetScopeRange, SymWriter_DefineLocalVariable, SymWriter_DefineParameter, SymWriter_DefineField, SymWriter_DefineGlobalVariable, SymWriter_Close, SymWriter_SetSymAttributes, SymWriter_OpenNamespace, SymWriter_CloseNamespace, SymWriter_UsingNamespace, SymWriter_SetMethodSourceRange, SymWriter_Initialize, SymWriter_GetDebugInfo, SymWriter_DefineSequencePoints, SymWriter_RemapToken, SymWriter_Initialize2, SymWriter_DefineConstant, SymWriter_Abort, SymWriter_DefineLocalVariable2, SymWriter_DefineGlobalVariable2, SymWriter_DefineConstant2, SymWriter_OpenMethod2, SymWriter_Commit, SymWriter_GetDebugInfoWithPadding, SymWriter_OpenMapTokensToSourceSpans, SymWriter_CloseMapTokensToSourceSpans, SymWriter_MapTokenToSourceSpan }; static HRESULT WINAPI SymWriter_PdbWriter_QueryInterface(IPdbWriter *iface, REFIID iid, void **ppv) { SymWriter *This = impl_from_IPdbWriter(iface); return ISymUnmanagedWriter5_QueryInterface(&This->iface, iid, ppv); } static ULONG WINAPI SymWriter_PdbWriter_AddRef(IPdbWriter *iface) { SymWriter *This = impl_from_IPdbWriter(iface); return ISymUnmanagedWriter5_AddRef(&This->iface); } static ULONG WINAPI SymWriter_PdbWriter_Release(IPdbWriter *iface) { SymWriter *This = impl_from_IPdbWriter(iface); return ISymUnmanagedWriter5_Release(&This->iface); } static HRESULT WINAPI SymWriter_SetPath(IPdbWriter *iface, const WCHAR *fullpath, IStream *stream, BOOL fullbuild) { FIXME("(%p,%s,%p,%u)\n", iface, debugstr_w(fullpath), stream, fullbuild); return E_NOTIMPL; } static HRESULT WINAPI SymWriter_OpenMod(IPdbWriter *iface, const WCHAR *modulename, const WCHAR *fullpath) { FIXME("(%p,%s,%s)\n", iface, debugstr_w(modulename), debugstr_w(fullpath)); return E_NOTIMPL; } static HRESULT WINAPI SymWriter_CloseMod(IPdbWriter *iface) { FIXME("(%p)\n", iface); return S_OK; } static HRESULT WINAPI SymWriter_GetPath(IPdbWriter *iface, DWORD ccData, DWORD *pccData, WCHAR *path) { FIXME("(%p,%lu,%p,%p)\n", iface, ccData, pccData, path); return E_NOTIMPL; } static HRESULT WINAPI SymWriter_GetSignatureAge(IPdbWriter *iface, DWORD *timestamp, DWORD *age) { SymWriter *This = impl_from_IPdbWriter(iface); TRACE("(%p,%p,%p)\n", This, timestamp, age); EnterCriticalSection(&This->lock); if (timestamp) *timestamp = This->pdb_timestamp; if (age) *age = This->pdb_age; LeaveCriticalSection(&This->lock); return S_OK; } static const IPdbWriterVtbl SymWriter_PdbWriter_Vtbl = { SymWriter_PdbWriter_QueryInterface, SymWriter_PdbWriter_AddRef, SymWriter_PdbWriter_Release, SymWriter_SetPath, SymWriter_OpenMod, SymWriter_CloseMod, SymWriter_GetPath, SymWriter_GetSignatureAge }; HRESULT SymWriter_CreateInstance(REFIID iid, void **ppv) { SymWriter *This; HRESULT hr; This = heap_alloc(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->iface.lpVtbl = &SymWriter_Vtbl; This->IPdbWriter_iface.lpVtbl = &SymWriter_PdbWriter_Vtbl; This->ref = 1; InitializeCriticalSection(&This->lock); This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": SymWriter.lock"); UuidCreate(&This->pdb_guid); This->pdb_age = 1; This->pdb_filename[0] = 0; hr = IUnknown_QueryInterface(&This->iface, iid, ppv); ISymUnmanagedWriter5_Release(&This->iface); return hr; }