// CodecExports.cpp #include "StdAfx.h" #include "../../../C/CpuArch.h" #include "../../Common/ComTry.h" #include "../../Common/MyCom.h" #include "../../Windows/Defs.h" #include "../ICoder.h" #include "../Common/RegisterCodec.h" extern unsigned g_NumCodecs; extern const CCodecInfo *g_Codecs[]; extern unsigned g_NumHashers; extern const CHasherInfo *g_Hashers[]; static void SetPropFromAscii(const char *s, PROPVARIANT *prop) throw() { UINT len = (UINT)strlen(s); BSTR dest = ::SysAllocStringLen(NULL, len); if (dest) { for (UINT i = 0; i <= len; i++) dest[i] = (Byte)s[i]; prop->bstrVal = dest; prop->vt = VT_BSTR; } } static inline HRESULT SetPropGUID(const GUID &guid, PROPVARIANT *value) throw() { if ((value->bstrVal = ::SysAllocStringByteLen((const char *)&guid, sizeof(guid))) != NULL) value->vt = VT_BSTR; return S_OK; } static HRESULT MethodToClassID(UInt16 typeId, CMethodId id, PROPVARIANT *value) throw() { GUID clsId; clsId.Data1 = k_7zip_GUID_Data1; clsId.Data2 = k_7zip_GUID_Data2; clsId.Data3 = typeId; SetUi64(clsId.Data4, id); return SetPropGUID(clsId, value); } static HRESULT FindCodecClassId(const GUID *clsid, bool isCoder2, bool isFilter, bool &encode, int &index) throw() { index = -1; if (clsid->Data1 != k_7zip_GUID_Data1 || clsid->Data2 != k_7zip_GUID_Data2) return S_OK; encode = true; if (clsid->Data3 == k_7zip_GUID_Data3_Decoder) encode = false; else if (clsid->Data3 != k_7zip_GUID_Data3_Encoder) return S_OK; UInt64 id = GetUi64(clsid->Data4); for (unsigned i = 0; i < g_NumCodecs; i++) { const CCodecInfo &codec = *g_Codecs[i]; if (id != codec.Id || (encode ? !codec.CreateEncoder : !codec.CreateDecoder) || (isFilter ? !codec.IsFilter : codec.IsFilter)) continue; if (codec.NumStreams == 1 ? isCoder2 : !isCoder2) return E_NOINTERFACE; index = i; return S_OK; } return S_OK; } static HRESULT CreateCoderMain(unsigned index, bool encode, void **coder) { COM_TRY_BEGIN const CCodecInfo &codec = *g_Codecs[index]; void *c; if (encode) c = codec.CreateEncoder(); else c = codec.CreateDecoder(); if (c) { IUnknown *unk; if (codec.IsFilter) unk = (IUnknown *)(ICompressFilter *)c; else if (codec.NumStreams != 1) unk = (IUnknown *)(ICompressCoder2 *)c; else unk = (IUnknown *)(ICompressCoder *)c; unk->AddRef(); *coder = c; } return S_OK; COM_TRY_END } static HRESULT CreateCoder2(bool encode, UInt32 index, const GUID *iid, void **outObject) { *outObject = NULL; const CCodecInfo &codec = *g_Codecs[index]; if (encode ? !codec.CreateEncoder : !codec.CreateDecoder) return CLASS_E_CLASSNOTAVAILABLE; if (codec.IsFilter) { if (*iid != IID_ICompressFilter) return E_NOINTERFACE; } else if (codec.NumStreams != 1) { if (*iid != IID_ICompressCoder2) return E_NOINTERFACE; } else { if (*iid != IID_ICompressCoder) return E_NOINTERFACE; } return CreateCoderMain(index, encode, outObject); } STDAPI CreateDecoder(UInt32 index, const GUID *iid, void **outObject) { return CreateCoder2(false, index, iid, outObject); } STDAPI CreateEncoder(UInt32 index, const GUID *iid, void **outObject) { return CreateCoder2(true, index, iid, outObject); } STDAPI CreateCoder(const GUID *clsid, const GUID *iid, void **outObject) { *outObject = NULL; bool isFilter = false; bool isCoder2 = false; bool isCoder = (*iid == IID_ICompressCoder) != 0; if (!isCoder) { isFilter = (*iid == IID_ICompressFilter) != 0; if (!isFilter) { isCoder2 = (*iid == IID_ICompressCoder2) != 0; if (!isCoder2) return E_NOINTERFACE; } } bool encode; int codecIndex; HRESULT res = FindCodecClassId(clsid, isCoder2, isFilter, encode, codecIndex); if (res != S_OK) return res; if (codecIndex < 0) return CLASS_E_CLASSNOTAVAILABLE; return CreateCoderMain(codecIndex, encode, outObject); } STDAPI GetMethodProperty(UInt32 codecIndex, PROPID propID, PROPVARIANT *value) { ::VariantClear((VARIANTARG *)value); const CCodecInfo &codec = *g_Codecs[codecIndex]; switch (propID) { case NMethodPropID::kID: value->uhVal.QuadPart = (UInt64)codec.Id; value->vt = VT_UI8; break; case NMethodPropID::kName: SetPropFromAscii(codec.Name, value); break; case NMethodPropID::kDecoder: if (codec.CreateDecoder) return MethodToClassID(k_7zip_GUID_Data3_Decoder, codec.Id, value); break; case NMethodPropID::kEncoder: if (codec.CreateEncoder) return MethodToClassID(k_7zip_GUID_Data3_Encoder, codec.Id, value); break; case NMethodPropID::kDecoderIsAssigned: value->vt = VT_BOOL; value->boolVal = BoolToVARIANT_BOOL(codec.CreateDecoder != NULL); break; case NMethodPropID::kEncoderIsAssigned: value->vt = VT_BOOL; value->boolVal = BoolToVARIANT_BOOL(codec.CreateEncoder != NULL); break; case NMethodPropID::kPackStreams: if (codec.NumStreams != 1) { value->vt = VT_UI4; value->ulVal = (ULONG)codec.NumStreams; } break; /* case NMethodPropID::kIsFilter: // if (codec.IsFilter) { value->vt = VT_BOOL; value->boolVal = BoolToVARIANT_BOOL(codec.IsFilter); } break; */ /* case NMethodPropID::kDecoderFlags: { value->vt = VT_UI4; value->ulVal = (ULONG)codec.DecoderFlags; } break; case NMethodPropID::kEncoderFlags: { value->vt = VT_UI4; value->ulVal = (ULONG)codec.EncoderFlags; } break; */ } return S_OK; } STDAPI GetNumberOfMethods(UINT32 *numCodecs) { *numCodecs = g_NumCodecs; return S_OK; } // ---------- Hashers ---------- static int FindHasherClassId(const GUID *clsid) throw() { if (clsid->Data1 != k_7zip_GUID_Data1 || clsid->Data2 != k_7zip_GUID_Data2 || clsid->Data3 != k_7zip_GUID_Data3_Hasher) return -1; UInt64 id = GetUi64(clsid->Data4); for (unsigned i = 0; i < g_NumCodecs; i++) if (id == g_Hashers[i]->Id) return i; return -1; } static HRESULT CreateHasher2(UInt32 index, IHasher **hasher) { COM_TRY_BEGIN *hasher = g_Hashers[index]->CreateHasher(); if (*hasher) (*hasher)->AddRef(); return S_OK; COM_TRY_END } STDAPI CreateHasher(const GUID *clsid, IHasher **outObject) { COM_TRY_BEGIN *outObject = 0; int index = FindHasherClassId(clsid); if (index < 0) return CLASS_E_CLASSNOTAVAILABLE; return CreateHasher2(index, outObject); COM_TRY_END } STDAPI GetHasherProp(UInt32 codecIndex, PROPID propID, PROPVARIANT *value) { ::VariantClear((VARIANTARG *)value); const CHasherInfo &codec = *g_Hashers[codecIndex]; switch (propID) { case NMethodPropID::kID: value->uhVal.QuadPart = (UInt64)codec.Id; value->vt = VT_UI8; break; case NMethodPropID::kName: SetPropFromAscii(codec.Name, value); break; case NMethodPropID::kEncoder: if (codec.CreateHasher) return MethodToClassID(k_7zip_GUID_Data3_Hasher, codec.Id, value); break; case NMethodPropID::kDigestSize: value->ulVal = (ULONG)codec.DigestSize; value->vt = VT_UI4; break; } return S_OK; } class CHashers: public IHashers, public CMyUnknownImp { public: MY_UNKNOWN_IMP1(IHashers) STDMETHOD_(UInt32, GetNumHashers)(); STDMETHOD(GetHasherProp)(UInt32 index, PROPID propID, PROPVARIANT *value); STDMETHOD(CreateHasher)(UInt32 index, IHasher **hasher); }; STDAPI GetHashers(IHashers **hashers) { COM_TRY_BEGIN *hashers = new CHashers; if (*hashers) (*hashers)->AddRef(); return S_OK; COM_TRY_END } STDMETHODIMP_(UInt32) CHashers::GetNumHashers() { return g_NumHashers; } STDMETHODIMP CHashers::GetHasherProp(UInt32 index, PROPID propID, PROPVARIANT *value) { return ::GetHasherProp(index, propID, value); } STDMETHODIMP CHashers::CreateHasher(UInt32 index, IHasher **hasher) { return ::CreateHasher2(index, hasher); }