//+------------------------------------------------------------------------- // // Microsoft Windows // // Copyright (C) Microsoft Corporation, 1998 - 1999 // // File: ddraw7obj.cpp // //-------------------------------------------------------------------------- // dDrawObj.cpp : Implementation of CDirectApp and DLL registration. #include "stdafx.h" #include "Direct.h" #include "dms.h" #include "dDraw7Obj.h" #include "ddClipperObj.h" #include "ddSurface7Obj.h" #include "ddPaletteObj.h" #include "ddEnumModesObj.h" #include "ddEnumSurfacesObj.h" #include "d3d7Obj.h" #include "ddIdentifierObj.h" extern BOOL is4Bit; extern HRESULT CopyInDDSurfaceDesc2(DDSURFACEDESC2 *,DDSurfaceDesc2*); extern HRESULT CopyOutDDSurfaceDesc2(DDSurfaceDesc2*,DDSURFACEDESC2 *); /////////////////////////////////////////////////////////////////// // InternalAddRef /////////////////////////////////////////////////////////////////// DWORD C_dxj_DirectDraw7Object::InternalAddRef(){ DWORD i; i=CComObjectRoot::InternalAddRef(); DPF2(1,"DDraw7 [%d] AddRef %d \n",creationid,i); return i; } /////////////////////////////////////////////////////////////////// // InternalRelease /////////////////////////////////////////////////////////////////// DWORD C_dxj_DirectDraw7Object::InternalRelease(){ DWORD i; i=CComObjectRoot::InternalRelease(); DPF2(1,"DDraw4 [%d] Release %d \n",creationid,i); return i; } /////////////////////////////////////////////////////////////////// // C_dxj_DirectDraw7Object /////////////////////////////////////////////////////////////////// C_dxj_DirectDraw7Object::C_dxj_DirectDraw7Object(){ DPF1(1,"Constructor Creation DirectDraw7Object[%d] \n ",g_creationcount); m__dxj_DirectDraw7= NULL; parent = NULL; pinterface = NULL; nextobj = g_dxj_DirectDraw7; creationid = ++g_creationcount; g_dxj_DirectDraw7 = (void *)this; m_hwnd=NULL; } /////////////////////////////////////////////////////////////////// // ~C_dxj_DirectDraw7Object /////////////////////////////////////////////////////////////////// C_dxj_DirectDraw7Object::~C_dxj_DirectDraw7Object() { DPF(1,"Entering ~DirectDraw7Object destructor \n"); C_dxj_DirectDraw7Object *prev=NULL; for(C_dxj_DirectDraw7Object *ptr=(C_dxj_DirectDraw7Object *)g_dxj_DirectDraw7; ptr; ptr=(C_dxj_DirectDraw7Object *)ptr->nextobj) { if(ptr == this) { if(prev) prev->nextobj = ptr->nextobj; else g_dxj_DirectDraw7 = (void*)ptr->nextobj; DPF(1,"DirectDraw7Object found in g_dxj list now removed\n"); break; } prev = ptr; } if(m__dxj_DirectDraw7){ int count = IUNK(m__dxj_DirectDraw7)->Release(); #ifdef DEBUG char buffer[256]; wsprintf(buffer,"DirectX IDirectDraw7 Ref count [%d] \n",count); #endif if(count==0) m__dxj_DirectDraw7 = NULL; } if(parent) IUNK(parent)->Release(); } /////////////////////////////////////////////////////////////////// // InternalGetObject // InternalSetObject // restoreDisplayMode // flipToGDISurface // setDisplayMode /////////////////////////////////////////////////////////////////// GETSET_OBJECT(_dxj_DirectDraw7); PASS_THROUGH_R(_dxj_DirectDraw7, restoreDisplayMode, RestoreDisplayMode) PASS_THROUGH_R(_dxj_DirectDraw7, flipToGDISurface, FlipToGDISurface) PASS_THROUGH5_R(_dxj_DirectDraw7, setDisplayMode, SetDisplayMode, long,long,long,long,long) /////////////////////////////////////////////////////////////////// // getMonitorFrequency /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getMonitorFrequency(long *ret) { HRESULT hr; hr=m__dxj_DirectDraw7->GetMonitorFrequency((DWORD*)ret); return hr; } /////////////////////////////////////////////////////////////////// // getGDISurface /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getGDISurface(I_dxj_DirectDrawSurface7 **rv) { LPDIRECTDRAWSURFACE7 lp4=NULL; if ( is4Bit ) return E_FAIL; *rv = NULL; HRESULT hr = DD_OK; if( ( hr=m__dxj_DirectDraw7->GetGDISurface(&lp4) ) != DD_OK) return hr; INTERNAL_CREATE(_dxj_DirectDrawSurface7, lp4, rv); return hr; } /////////////////////////////////////////////////////////////////// // getVerticalBlankStatus /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getVerticalBlankStatus( long *status) { if ( is4Bit ) return E_FAIL; return m__dxj_DirectDraw7->GetVerticalBlankStatus((int *)status); } /////////////////////////////////////////////////////////////////// // setCooperativeLevel /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::setCooperativeLevel( HWnd hwn, long flags) { if ( is4Bit ) return E_FAIL; m_hwnd = (HWND)hwn; return m__dxj_DirectDraw7->SetCooperativeLevel((HWND)hwn, (DWORD)flags); } /////////////////////////////////////////////////////////////////// // waitForVerticalBlank /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::waitForVerticalBlank(long flags,long handle, long *status) { if ( is4Bit ) return E_FAIL; *status = m__dxj_DirectDraw7->WaitForVerticalBlank(flags, (void *)handle); return S_OK; } /////////////////////////////////////////////////////////////////// // createClipper /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::createClipper(long flags, I_dxj_DirectDrawClipper **val) { if ( is4Bit ) return E_FAIL; DPF1(1,"enter DDraw4[%d]::createClipper ",creationid); // // need to create one of MY surfaces! // LPDIRECTDRAWCLIPPER ddc; HRESULT hr = DD_OK; if( (hr=m__dxj_DirectDraw7->CreateClipper( flags, &ddc, NULL)) != DD_OK ) return hr; INTERNAL_CREATE(_dxj_DirectDrawClipper, ddc, val); DPF1(1,"exit DDraw4[%d]::createClipper ",creationid); return hr; } /////////////////////////////////////////////////////////////////// // createPalette /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::createPalette(long flags, SAFEARRAY **pe, I_dxj_DirectDrawPalette **val) { LPPALETTEENTRY ppe; if ( is4Bit ) return E_FAIL; if (!ISSAFEARRAY1D(pe,(DWORD)256)) return E_INVALIDARG; ppe = (LPPALETTEENTRY)((SAFEARRAY*)*pe)->pvData; LPDIRECTDRAWPALETTE ddp; HRESULT hr = DD_OK; *val = NULL; if( (hr=m__dxj_DirectDraw7->CreatePalette( flags, (LPPALETTEENTRY)ppe, &ddp, NULL)) == DD_OK ) { INTERNAL_CREATE( _dxj_DirectDrawPalette, ddp, val); } return hr; } /////////////////////////////////////////////////////////////////// // createSurface /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::createSurface(DDSurfaceDesc2 *dd, I_dxj_DirectDrawSurface7 **retval) { HRESULT retv; LPDIRECTDRAWSURFACE7 dds7; // DirectX object pointer DDSURFACEDESC2 ddsd; DPF1(1,"enter DDraw7[%d]::createSurface ",creationid); if ( is4Bit ) return E_FAIL; if(! (dd && retval) ) return E_POINTER; CopyInDDSurfaceDesc2(&ddsd,dd); //docdoc: CreateSurface returns error if 'punk' is anything but NULL retv = m__dxj_DirectDraw7->CreateSurface( &ddsd, &dds7, NULL); if FAILED(retv) return retv; INTERNAL_CREATE(_dxj_DirectDrawSurface7, dds7, retval); dd->lpSurface = NULL; DPF1(1,"exit DDraw7[%d]::createSurface ",creationid); return S_OK; } /////////////////////////////////////////////////////////////////// // duplicateSurface /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::duplicateSurface(I_dxj_DirectDrawSurface7 *ddIn, I_dxj_DirectDrawSurface7 **ddOut) { HRESULT retval; if ( is4Bit ) return E_FAIL; // // need to create one of MY surfaces! // LPDIRECTDRAWSURFACE7 lpddout7=NULL; DO_GETOBJECT_NOTNULL( LPDIRECTDRAWSURFACE7, lpddin, ddIn); if( (retval = m__dxj_DirectDraw7->DuplicateSurface(lpddin, &lpddout7)) != DD_OK ) return retval; INTERNAL_CREATE( _dxj_DirectDrawSurface7, lpddout7, ddOut); return S_OK; } /////////////////////////////////////////////////////////////////// // getCaps /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getCaps(DDCaps *driverCaps, DDCaps *HELcaps) { if ( is4Bit ) return E_FAIL; if (!driverCaps) return E_INVALIDARG; if (!HELcaps) return E_INVALIDARG; ((DDCAPS*)driverCaps)->dwSize=sizeof(DDCAPS); ((DDCAPS*)HELcaps)->dwSize=sizeof(DDCAPS); HRESULT hr = m__dxj_DirectDraw7->GetCaps((DDCAPS*)driverCaps, (DDCAPS*)HELcaps); return hr; } /////////////////////////////////////////////////////////////////// // getDisplayMode /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getDisplayMode(DDSurfaceDesc2 *desc) { HRESULT retval; DDSURFACEDESC2 ddsd; if (!desc) return E_INVALIDARG; CopyInDDSurfaceDesc2(&ddsd,desc); retval = m__dxj_DirectDraw7->GetDisplayMode(&ddsd); if( retval != S_OK) return retval; CopyOutDDSurfaceDesc2(desc,&ddsd); desc->lpSurface = NULL; return S_OK; } /////////////////////////////////////////////////////////////////// // getAvailableTotalMem /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getAvailableTotalMem(DDSCaps2 *ddsCaps, long *m) { return m__dxj_DirectDraw7->GetAvailableVidMem((LPDDSCAPS2)ddsCaps, (unsigned long *)m, NULL); } /////////////////////////////////////////////////////////////////// // getFreeMem /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getFreeMem(DDSCaps2 *ddsCaps, long *m) { return m__dxj_DirectDraw7->GetAvailableVidMem((LPDDSCAPS2)ddsCaps, NULL, (unsigned long *)m); } /////////////////////////////////////////////////////////////////// // getDirect3D /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getDirect3D(I_dxj_Direct3d7 **retval) { LPDIRECT3D7 lpD3D; HRESULT hr = DD_OK; if ( is4Bit ) return E_FAIL; if( (hr=m__dxj_DirectDraw7->QueryInterface(IID_IDirect3D7, (void**) &lpD3D)) != DD_OK) return hr; INTERNAL_CREATE(_dxj_Direct3d7, lpD3D, retval); return hr; } /////////////////////////////////////////////////////////////////// // getNumFourCCCodes /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getNumFourCCCodes(long *retval) { return m__dxj_DirectDraw7->GetFourCCCodes((DWORD*)retval, NULL); } /////////////////////////////////////////////////////////////////// // getScanLine /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getScanLine(long *lines, long *status) { *status = (long)m__dxj_DirectDraw7->GetScanLine((DWORD*)lines); return S_OK; } /////////////////////////////////////////////////////////////////// // loadPaletteFromBitmap /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::loadPaletteFromBitmap(BSTR bName, I_dxj_DirectDrawPalette **retval) { USES_CONVERSION; IDirectDrawPalette* ddpal; int i; int n; int fh; HRSRC h; LPBITMAPINFOHEADER lpbi; PALETTEENTRY ape[256]; RGBQUAD * prgb; HRESULT hr=S_OK; if ( is4Bit ) return E_FAIL; LPCTSTR szBitmap = W2T(bName); for (i=0; i<256; i++) // build a 332 palette as the default { ape[i].peRed = (BYTE)(((i >> 5) & 0x07) * 255 / 7); ape[i].peGreen = (BYTE)(((i >> 2) & 0x07) * 255 / 7); ape[i].peBlue = (BYTE)(((i >> 0) & 0x03) * 255 / 3); ape[i].peFlags = (BYTE)0; } // // get a pointer to the bitmap resource. // if (szBitmap && (h = FindResource(NULL, szBitmap, RT_BITMAP))) { lpbi = (LPBITMAPINFOHEADER)LockResource(LoadResource(NULL, h)); if (!lpbi){ DPF(1,"lock resource failed\n"); return E_OUTOFMEMORY; // error } prgb = (RGBQUAD*)((BYTE*)lpbi + lpbi->biSize); if (lpbi == NULL || lpbi->biSize < sizeof(BITMAPINFOHEADER)) n = 0; else if (lpbi->biBitCount > 8) n = 0; else if (lpbi->biClrUsed == 0) n = 1 << lpbi->biBitCount; else n = lpbi->biClrUsed; // // a DIB color table has its colors stored BGR not RGB // so flip them around. // for(i=0; i 8) n = 0; else if (bi.biClrUsed == 0) n = 1 << bi.biBitCount; else n = bi.biClrUsed; // // a DIB color table has its colors stored BGR not RGB // so flip them around. // for(i=0; iCreatePalette(DDPCAPS_8BIT, ape, &ddpal, NULL); if( ddpal ) { INTERNAL_CREATE(_dxj_DirectDrawPalette, ddpal, retval); } else { // // no object, set the return value to NULL as well. // *retval = NULL; hr = E_FAIL; } return hr; } /////////////////////////////////////////////////////////////////// // createSurfaceFromFile /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::createSurfaceFromFile(BSTR file, DDSurfaceDesc2 *desc, I_dxj_DirectDrawSurface7 **surf) { DPF1(1,"enter DDraw7[%d]::createSurfaceFromFile ",creationid); HDC hdc; HDC hdcImage; BITMAP bm; HRESULT hr; HBITMAP hbm; HRESULT retv; LPDIRECTDRAWSURFACE7 dds7; // DirectX object pointer LPSTR szFileName=NULL; int width=0; int height=0; if ( is4Bit ) return E_FAIL; if(! (desc && surf) ) return E_POINTER; USES_CONVERSION; szFileName=W2T(file); //If width and height are zero then we will generate our own width and //height from the bitmap. //The LoadImage api however doesnt work propery without size params //Consider there must be a way to make it work. if ((desc->lWidth!=0)&&(desc->lHeight!=0)&&(desc->lFlags & DDSD_WIDTH)&&(desc->lFlags & DDSD_HEIGHT)) { width=desc->lWidth ; height=desc->lHeight; } if (desc->lFlags==0) { desc->lFlags=DDSD_CAPS; ((DDSURFACEDESC*)desc)->ddsCaps.dwCaps=DDSCAPS_OFFSCREENPLAIN; } if (!szFileName) return CTL_E_FILENOTFOUND; hbm = (HBITMAP)LoadImage((HINSTANCE)NULL, szFileName, IMAGE_BITMAP, width, height, LR_LOADFROMFILE|LR_CREATEDIBSECTION); DWORD dwErr=GetLastError(); if (!hbm){ //often users pass in width and height in twips and not pixels //loadimage compails and returns ERROR_NOT_ENOUGH_MEMORY if (dwErr==ERROR_NOT_ENOUGH_MEMORY) { return E_OUTOFMEMORY; } else if (dwErr==ERROR_INVALID_PARAMETER) { return E_INVALIDARG; } else { return CTL_E_FILENOTFOUND; } } // get size of the bitmap // GetObject(hbm, sizeof(bm), &bm); // get size of bitmap width=bm.bmWidth; height=bm.bmHeight; desc->lFlags = desc->lFlags | DDSD_WIDTH | DDSD_HEIGHT; if ((desc->lWidth==0)||(desc->lHeight==0)) { desc->lWidth =width; desc->lHeight =height; } DDSURFACEDESC2 ddsd; CopyInDDSurfaceDesc2(&ddsd,desc); if( (retv = m__dxj_DirectDraw7->CreateSurface(&ddsd, &dds7, NULL)) != DD_OK ) return retv; CopyOutDDSurfaceDesc2(desc,&ddsd); INTERNAL_CREATE(_dxj_DirectDrawSurface7, dds7, surf); desc->lpSurface = NULL; // // make sure this surface is restored. // dds7->Restore(); // // select bitmap into a memoryDC so we can use it. // hdcImage = CreateCompatibleDC(NULL); SelectObject(hdcImage, hbm); if (!hdcImage){ DeleteObject(hbm); return E_FAIL; } if ((hr = dds7->GetDC(&hdc)) == DD_OK) { StretchBlt(hdc, 0, 0, desc->lWidth , desc->lHeight, hdcImage, 0, 0, width, height, SRCCOPY); dds7->ReleaseDC(hdc); } DeleteDC(hdcImage); if (hbm) DeleteObject(hbm); DPF1(buffer,"exit DDraw7[%d]::createSurfaceFromFile",creationid); return S_OK; } /////////////////////////////////////////////////////////////////// // createSurfaceFromResource /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::createSurfaceFromResource(BSTR resFile, BSTR resourceName, DDSurfaceDesc2 *desc, I_dxj_DirectDrawSurface7 **surf) { DPF1(1,"enter DDraw4[%d]::createSurfaceFromResource ",creationid); if ( is4Bit ) return E_FAIL; if(! (desc && surf) ) return E_POINTER; HRESULT hr; HRSRC hres=NULL; HGLOBAL hglob=NULL; HDC hdc; HDC hdcImage; BITMAP bm; HBITMAP hbm; HRESULT retv; LPDIRECTDRAWSURFACE7 dds7; // DirectX object pointer LPSTR szResName=NULL; if (!resourceName) return E_INVALIDARG; if (!surf) return E_INVALIDARG; HMODULE hMod=NULL; USES_CONVERSION; if ((resFile) &&(resFile[0]!=0)){ // NOTE: // seems that GetModuleHandleW is // always returning 0 on w98 // so we must convert to ansi first LPCTSTR pszName = W2T(resFile); hMod= GetModuleHandle(pszName); } else { hMod= GetModuleHandle(NULL); } LPCTSTR pszName2 = W2T(resourceName); //hbm = (HBITMAP)LoadImageW((HINSTANCE)hMod, resourceName, // IMAGE_BITMAP, // 0, 0, // LR_CREATEDIBSECTION); hbm = (HBITMAP)LoadImage((HINSTANCE)hMod, pszName2, IMAGE_BITMAP, 0, 0, LR_CREATEDIBSECTION); if (!hbm){ //MessageBox(NULL,"FAILED ON LOAD IMAGE","TEST",MB_OK); return E_FAIL; } // get size of the bitmap // GetObject(hbm, sizeof(bm), &bm); // get size of bitmap DWORD width=bm.bmWidth; DWORD height=bm.bmHeight; desc->lFlags = desc->lFlags | DDSD_WIDTH | DDSD_HEIGHT; if ((desc->lWidth==0)||(desc->lHeight==0)) { desc->lWidth =width; desc->lHeight =height; } DDSURFACEDESC2 ddsd; CopyInDDSurfaceDesc2(&ddsd,desc); if( (retv = m__dxj_DirectDraw7->CreateSurface(&ddsd, &dds7, NULL)) != DD_OK ){ if (hbm) DeleteObject(hbm); return retv; } CopyOutDDSurfaceDesc2(desc,&ddsd); INTERNAL_CREATE(_dxj_DirectDrawSurface7, dds7, surf); desc->lpSurface = NULL; // // make sure this surface is restored. // dds7->Restore(); // // select bitmap into a memoryDC so we can use it. // hdcImage = CreateCompatibleDC(NULL); if (!hdcImage){ DeleteObject(hbm); return E_OUTOFMEMORY; } SelectObject(hdcImage, hbm); if ((hr = dds7->GetDC(&hdc)) == DD_OK) { StretchBlt(hdc, 0, 0, desc->lWidth , desc->lHeight, hdcImage, 0, 0, width, height, SRCCOPY); dds7->ReleaseDC(hdc); } DeleteDC(hdcImage); if (hbm) DeleteObject(hbm); DPF1(1r,"exit DDraw4[%d]::createSurfaceFromFile",creationid); return S_OK; } /////////////////////////////////////////////////////////////////// // getFourCCCodes /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getFourCCCodes(SAFEARRAY **ppsa) { DWORD count= ((SAFEARRAY*)*ppsa)->rgsabound[0].cElements; if ( ((SAFEARRAY*)*ppsa)->cDims!=1) return E_INVALIDARG; return m__dxj_DirectDraw7->GetFourCCCodes(&count,(DWORD*)((SAFEARRAY*)*ppsa)->pvData); } /////////////////////////////////////////////////////////////////// // getDisplayModesEnum /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::getDisplayModesEnum( /* [in] */ long flags, /* [in] */ DDSurfaceDesc2 *ddsd, /* [retval][out] */ I_dxj_DirectDrawEnumModes __RPC_FAR *__RPC_FAR *retval) { HRESULT hr; hr=C_dxj_DirectDrawEnumModesObject::create(m__dxj_DirectDraw7,flags, ddsd, retval); return hr; } /////////////////////////////////////////////////////////////////// // testCooperativeLevel /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::testCooperativeLevel( /* [in,out] */ long *status) { HRESULT hr; hr=m__dxj_DirectDraw7->TestCooperativeLevel(); *status=(long)hr; return S_OK; } /////////////////////////////////////////////////////////////////// // restoreAllSurfaces /////////////////////////////////////////////////////////////////// STDMETHODIMP C_dxj_DirectDraw7Object::restoreAllSurfaces() { HRESULT hr; hr=m__dxj_DirectDraw7->RestoreAllSurfaces(); return hr; } STDMETHODIMP C_dxj_DirectDraw7Object::getSurfaceFromDC(long hdc, I_dxj_DirectDrawSurface7 **ret) { HRESULT hr; LPDIRECTDRAWSURFACE7 pDDS=NULL; hr=m__dxj_DirectDraw7->GetSurfaceFromDC((HDC)hdc,&pDDS); if FAILED(hr) return hr; INTERNAL_CREATE(_dxj_DirectDrawSurface7,pDDS,ret); return hr; } STDMETHODIMP C_dxj_DirectDraw7Object::getSurfacesEnum( /* [in] */ long flags, /* [in] */ DDSurfaceDesc2 __RPC_FAR *desc, /* [retval][out] */ I_dxj_DirectDrawEnumSurfaces __RPC_FAR *__RPC_FAR *ret) { HRESULT hr=C_dxj_DirectDrawEnumSurfacesObject::create((I_dxj_DirectDraw7*)this , flags, desc,ret); return hr; } STDMETHODIMP C_dxj_DirectDraw7Object::getDeviceIdentifier( long flags,I_dxj_DirectDrawIdentifier **ret) { HRESULT hr; hr=C_dxj_DirectDrawIdentifierObject::Create(m__dxj_DirectDraw7,(DWORD)flags,ret); return hr; }