windows-nt/Source/XPSP1/NT/multimedia/directx/dxvb/dx7vb/dxglob7obj.cpp
2020-09-26 16:20:57 +08:00

1367 lines
35 KiB
C++

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1998 - 1999
//
// File: dxglob7obj.cpp
//
//--------------------------------------------------------------------------
#define DIRECTINPUT_VERSION 0x0500
#define DIRECTSOUND_VERSION 0x600
// dSoundResourceObj.cpp : Implementation of CDirectApp and DLL registration.
// DHF_DS entire file
#include "windows.h"
#include "mmsystem.h"
#include "stdafx.h"
#include "Direct.h"
#include "dms.h"
#include "math.h"
#include "dxGlob7Obj.h"
#include "d3drm.h"
#include "ddraw.h"
#include "dmusici.h"
#include "dmusicf.h"
#include "ddraw4Obj.h"
#include "ddraw7Obj.h"
#include "DDEnumObj.h"
#include "d3drm3Obj.h"
#include "dsoundObj.h"
#include "dsoundCaptureObj.h"
#include "DSEnumObj.h"
#include "dplay4obj.h"
#include "dplaylobby3obj.h"
#include "DPEnumObj.h"
#include "dinput1Obj.h"
#include "dmSegmentObj.h"
#include "dmSegmentStateObj.h"
#include "dmChordMapObj.h"
#include "dmBandObj.h"
#include "dmCollectionObj.h"
#include "dmStyleObj.h"
#include "dmPerformanceObj.h"
#include "dmLoaderObj.h"
#include "dmComposerObj.h"
//#include "webObj.h"
extern HINSTANCE g_hInstD3DRMDLL;
extern HINSTANCE g_hDSoundHandle;
extern HINSTANCE g_hDPlay;
extern HINSTANCE g_hDDrawHandle;
extern HINSTANCE g_hInstDINPUTDLL;
extern HINSTANCE g_hInst;
extern HRESULT BSTRtoPPGUID(LPGUID*,BSTR);
extern HRESULT BSTRtoGUID(LPGUID,BSTR);
extern void *g_dxj_DirectMusicLoader;
extern void *g_dxj_DirectMusicComposer;
extern void *g_dxj_DirectMusicPerformance;
extern BSTR GUIDtoBSTR(LPGUID);
extern HRESULT DPLBSTRtoGUID(LPGUID pGuid,BSTR str);
extern HINSTANCE LoadDDrawDLL();
extern HINSTANCE LoadD3DXOFDLL();
extern HINSTANCE LoadDPlayDLL();
extern HINSTANCE LoadDSoundDLL();
extern HINSTANCE LoadD3DRMDLL();
extern HINSTANCE LoadDINPUTDLL();
DWORD WINAPI ThreadFunc(LPVOID param);
C_dxj_DirectX7Object::C_dxj_DirectX7Object(){
m_pDirectDrawCreate=NULL;
m_pDirectDrawCreateEx=NULL;
m_pDirectDrawCreateClipper=NULL;
m_pDirectSoundCreate=NULL;
m_pDirectSoundEnumerate=NULL;
m_pDirectSoundCaptureEnumerate=NULL;
m_pDirectSoundCaptureCreate=NULL;
m_pDirectPlayCreate=NULL;
m_pDirectPlayEnumerate=NULL;
m_pDirectPlayLobbyCreate=NULL;
m_pDirect3DRMCreate=NULL;
m_pDirectDrawEnumerate=NULL;
m_pDirectDrawEnumerateEx=NULL;
m_pEventList=NULL;
}
void C_dxj_DirectX7Object::LoadDSOUND()
{
if (!g_hDSoundHandle ) LoadDSoundDLL();
if (!m_pDirectSoundCreate) m_pDirectSoundCreate = (DSOUNDCREATE)GetProcAddress( g_hDSoundHandle, "DirectSoundCreate" );
if (!m_pDirectSoundCaptureCreate) m_pDirectSoundCaptureCreate = (DSOUNDCAPTURECREATE)GetProcAddress( g_hDSoundHandle, "DirectSoundCaptureCreate" );
if (!m_pDirectSoundEnumerate) m_pDirectSoundEnumerate = (DSOUNDENUMERATE)GetProcAddress( g_hDSoundHandle, "DirectSoundEnumerateA" );
if (!m_pDirectSoundCaptureEnumerate) m_pDirectSoundCaptureEnumerate = (DSOUNDCAPTUREENUMERATE)GetProcAddress( g_hDSoundHandle, "DirectSoundCaptureEnumerateA" );
}
void C_dxj_DirectX7Object::LoadDDRAW()
{
if (!g_hDDrawHandle ) LoadDDrawDLL();
if (!m_pDirectDrawCreate) m_pDirectDrawCreate = (DDRAWCREATE)GetProcAddress( g_hDDrawHandle, "DirectDrawCreate" );
if (!m_pDirectDrawCreateEx) m_pDirectDrawCreateEx = (DDRAWCREATEEX)GetProcAddress( g_hDDrawHandle, "DirectDrawCreateEx" );
if (!m_pDirectDrawCreateClipper) m_pDirectDrawCreateClipper = (DDCREATECLIPPER)GetProcAddress( g_hDDrawHandle, "DirectDrawCreateClipper" );
if (!m_pDirectDrawEnumerate) m_pDirectDrawEnumerate = (DDENUMERATE)GetProcAddress( g_hDDrawHandle, "DirectDrawEnumerateA" );
if (!m_pDirectDrawEnumerateEx) m_pDirectDrawEnumerateEx = (DDENUMERATEEX)GetProcAddress( g_hDDrawHandle, "DirectDrawEnumerateExA" );
}
void C_dxj_DirectX7Object::LoadDPLAY()
{
if (!g_hDPlay ) LoadDPlayDLL();
//Note use of Wide char for DirectPlayLobby Create and
//Ansi for DirectPlay
if (!m_pDirectPlayCreate) m_pDirectPlayCreate = (DIRECTPLAYCREATE)GetProcAddress( g_hDPlay, "DirectPlayCreate" );
if (!m_pDirectPlayEnumerate) m_pDirectPlayEnumerate = (DIRECTPLAYENUMERATE)GetProcAddress( g_hDPlay, "DirectPlayEnumerateW" );
if (!m_pDirectPlayLobbyCreate) m_pDirectPlayLobbyCreate = (DIRECTPLAYLOBBYCREATE)GetProcAddress( g_hDPlay, "DirectPlayLobbyCreateW" );
}
void C_dxj_DirectX7Object::LoadD3DRM()
{
if (!g_hInstD3DRMDLL) {
LoadD3DXOFDLL();
LoadD3DRMDLL();
}
//Note: always ansi no unicode version
if (!m_pDirect3DRMCreate) m_pDirect3DRMCreate =(DIRECT3DRMCREATE)GetProcAddress( g_hInstD3DRMDLL, "Direct3DRMCreate" );
}
C_dxj_DirectX7Object::~C_dxj_DirectX7Object()
{
DWORD i=1;
while (m_pEventList) {
//bugbug SUNDOWN gotcha
//vb does not have 64 bit values
//and we use the same api to release the event
//but handles are 64 bit
//note handles never get near 32 bits in win32
//but whose knows on win64
destroyEvent((long)PtrToLong(m_pEventList->hEvent));
//m_pEventList=m_pEventList->pNext; done by destroyEvent
}
}
STDMETHODIMP C_dxj_DirectX7Object::direct3dRMCreate( I_dxj_Direct3dRM3 **ret){
HRESULT hr;
LPDIRECT3DRM realrm1=NULL;
LPDIRECT3DRM3 realrm3=NULL;
LoadD3DRM();
if (!m_pDirect3DRMCreate) return E_FAIL;
hr=(m_pDirect3DRMCreate)(&realrm1);
if FAILED(hr) return hr;
hr=realrm1->QueryInterface(IID_IDirect3DRM3,(void**) &realrm3);
if FAILED(hr){
realrm1->Release();
return hr;
}
INTERNAL_CREATE(_dxj_Direct3dRM3,realrm3,ret);
realrm1->Release();
if (*ret==NULL) {
realrm3->Release();
return E_FAIL;
}
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::directDraw4Create(BSTR strGuid, I_dxj_DirectDraw4 **ret){
HRESULT hr;
LPDIRECTDRAW realdraw1=NULL;
LPDIRECTDRAW4 realdraw4=NULL;
GUID guid;
LPGUID pguid=&guid;
DWORD i=0;
LoadDDRAW();
hr=BSTRtoPPGUID(&pguid,strGuid);
if FAILED(hr) return E_FAIL;
if (!m_pDirectDrawCreate) return E_FAIL;
hr=(m_pDirectDrawCreate)((GUID*)pguid,&realdraw1,NULL);
if FAILED(hr) return hr;
hr=realdraw1->QueryInterface(IID_IDirectDraw4,(void**) &realdraw4);
if FAILED(hr){
i=realdraw1->Release();
return hr;
}
i=realdraw1->Release();
INTERNAL_CREATE(_dxj_DirectDraw4,realdraw4,ret);
if (*ret==NULL) {
i=realdraw4->Release();
return E_FAIL;
}
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::directDrawCreate(BSTR strGuid, I_dxj_DirectDraw7 **ret){
HRESULT hr;
LPDIRECTDRAW7 realdraw7=NULL;
GUID guid;
LPGUID pguid=&guid;
DWORD i=0;
LoadDDRAW();
ZeroMemory(&guid,sizeof(GUID)); //andrewke bugfix post dx7
hr=BSTRtoPPGUID(&pguid,strGuid);
if FAILED(hr) return E_FAIL;
if (!m_pDirectDrawCreateEx) return E_FAIL;
hr=(m_pDirectDrawCreateEx)(pguid,(void**)&realdraw7,IID_IDirectDraw7,NULL);
if FAILED(hr) return hr;
INTERNAL_CREATE(_dxj_DirectDraw7,realdraw7,ret);
if (*ret==NULL) {
i=realdraw7->Release();
return E_FAIL;
}
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::directSoundCreate(BSTR strGuid, I_dxj_DirectSound **ret){
HRESULT hr;
LPDIRECTSOUND realsound1=NULL;
GUID guid;
LPGUID pguid=&guid;
LoadDSOUND();
hr=BSTRtoPPGUID(&pguid,strGuid);
if FAILED(hr) return hr;
if (!m_pDirectSoundCreate) return E_FAIL;
hr=(m_pDirectSoundCreate)((GUID*)pguid,&realsound1,NULL);
if FAILED(hr) return hr;
INTERNAL_CREATE(_dxj_DirectSound,realsound1,ret);
if (*ret==NULL) {
realsound1->Release();
return E_FAIL;
}
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::directSoundCaptureCreate(BSTR strGuid, I_dxj_DirectSoundCapture **ret){
HRESULT hr;
LPDIRECTSOUNDCAPTURE realsound1=NULL;
GUID guid;
LPGUID pguid=&guid;
hr=BSTRtoPPGUID(&pguid,strGuid);
if FAILED(hr) return hr;
LoadDSOUND();
if (!m_pDirectSoundCaptureCreate) return E_FAIL;
hr=(m_pDirectSoundCaptureCreate)(pguid,&realsound1,NULL);
if FAILED(hr) return hr;
INTERNAL_CREATE(_dxj_DirectSoundCapture,realsound1,ret);
if (*ret==NULL) {
realsound1->Release();
return E_FAIL;
}
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::getDSCaptureEnum( I_dxj_DSEnum **retVal)
{
HRESULT hr;
LoadDSOUND();
if (!m_pDirectSoundCaptureEnumerate) return E_FAIL;
hr=C_dxj_DSEnumObject::create(NULL,m_pDirectSoundCaptureEnumerate,retVal);
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::getDDEnum( I_dxj_DirectDrawEnum **retVal)
{
LoadDDRAW();
HRESULT hr;
if (!m_pDirectDrawEnumerateEx) return E_FAIL;
hr=C_dxj_DirectDrawEnumObject::create(m_pDirectDrawEnumerateEx,retVal);
return hr;
}
/////////////////////////////////////////////////////////////////////////////
STDMETHODIMP C_dxj_DirectX7Object::getDSEnum( I_dxj_DSEnum **retVal)
{
HRESULT hr;
LoadDSOUND();
if (!m_pDirectSoundEnumerate) return E_FAIL;
hr=C_dxj_DSEnumObject::create(m_pDirectSoundEnumerate,NULL,retVal);
return hr;
}
/////////////////////////////////////////////////////////////////////////////
STDMETHODIMP C_dxj_DirectX7Object::colorGetAlpha(long color, float *retval)
{
static float (WINAPI *ProcAdd)(long)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (float (WINAPI*)(long)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMColorGetAlpha");
if (ProcAdd==NULL) return E_FAIL;
}
*retval=(ProcAdd) ((D3DCOLOR) color);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::colorGetRed(long color, float *retval)
{
static float (WINAPI *ProcAdd)(long)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (float (WINAPI*)(long)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMColorGetRed");
if (ProcAdd==NULL) return E_FAIL;
}
*retval=(ProcAdd) ((D3DCOLOR) color);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::colorGetGreen(long color, float *retval)
{
static float (WINAPI *ProcAdd)(long)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (float (WINAPI*)(long)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMColorGetGreen");
if (ProcAdd==NULL) return E_FAIL;
}
*retval=(ProcAdd) ((D3DCOLOR) color);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::colorGetBlue(long color, float *retval)
{
static float (WINAPI *ProcAdd)(long)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (float (WINAPI*)(long)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMColorGetBlue");
if (ProcAdd==NULL) return E_FAIL;
}
*retval=(ProcAdd) ((D3DCOLOR) color);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createColorRGB(float r, float g, float b, long *color)
{
static long (WINAPI *ProcAdd)(float,float,float)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (long (WINAPI*)(float,float,float)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMCreateColorRGB");
if (ProcAdd==NULL) return E_FAIL;
}
*color=(ProcAdd) (r,g,b);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createColorRGBA(float r, float g, float b, float a, long *color)
{
static long (WINAPI *ProcAdd)(float,float,float,float)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (long (WINAPI*)(float,float,float,float)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMCreateColorRGBA");
if (ProcAdd==NULL) return E_FAIL;
}
*color=(ProcAdd) (r,g,b,a);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::matrixFromQuaternion(D3dMatrix *matrix, D3dRMQuaternion *quat)
{
if ((!matrix) || (!quat)) return E_INVALIDARG;
D3DMATRIX *mat=(D3DMATRIX*)matrix;
FLOAT w=quat->s;
FLOAT x=quat->v.x;
FLOAT y=quat->v.y;
FLOAT z=quat->v.z;
FLOAT xx = x*x, xy = x*y, xz = x*z, xw = x*w;
FLOAT yy = y*y, yz = y*z, yw = y*w;
FLOAT zz = z*z, zw = z*w;
FLOAT ww = w*w;
mat->_11 = xx-yy-zz+ww;
mat->_12 = 2.0f*(xy-zw);
mat->_13 = 2.0f*(xz+yw);
mat->_21 = 2.0f*(xy+zw);
mat->_22 = -xx+yy-zz+ww;
mat->_23 = 2.0f*(yz-xw);
mat->_31 = 2.0f*(xz-yw);
mat->_32 = 2.0f*(yz+xw);
mat->_33 = -xx-yy+zz+ww;
mat->_14 = mat->_41 = 0.0f;
mat->_24 = mat->_42 = 0.0f;
mat->_34 = mat->_43 = 0.0f;
mat->_44 = xx+yy+zz+ww;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::quaternionRotation(D3dRMQuaternion *quat, D3dVector *axis, float theta)
{
static D3dRMQuaternion* (WINAPI *ProcAdd)(D3dRMQuaternion*,D3dVector*,float)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dRMQuaternion* (WINAPI*)(D3dRMQuaternion*,D3dVector*,float)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMQuaternionRotation");
if (ProcAdd==NULL) return E_FAIL;
}
if (NULL== (ProcAdd) (quat,axis,theta)) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::quaternionMultiply(D3dRMQuaternion *quatr, D3dRMQuaternion *quat1, D3dRMQuaternion *quat2)
{
static D3dRMQuaternion* (WINAPI *ProcAdd)(D3dRMQuaternion*,D3dRMQuaternion*,D3dRMQuaternion*)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dRMQuaternion* (WINAPI*)(D3dRMQuaternion*,D3dRMQuaternion*,D3dRMQuaternion*)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMQuaternionMultiply");
if (ProcAdd==NULL) return E_FAIL;
}
if (NULL== (ProcAdd) (quatr,quat1,quat2)) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::quaternionSlerp(D3dRMQuaternion *quatr, D3dRMQuaternion *quat1, D3dRMQuaternion *quat2, float t)
{
static D3dRMQuaternion* (WINAPI *ProcAdd)(D3dRMQuaternion*,D3dRMQuaternion*,D3dRMQuaternion*,float)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dRMQuaternion* (WINAPI*)(D3dRMQuaternion*,D3dRMQuaternion*,D3dRMQuaternion*,float)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMQuaternionSlerp");
if (ProcAdd==NULL) return E_FAIL;
}
if (NULL== (ProcAdd) (quatr,quat1,quat2,t)) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorAdd(D3dVector *v, D3dVector *a, D3dVector *b)
{
v->x = a->x + b->x;
v->y = a->y + b->y;
v->z = a->z + b->z;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorCopy(D3dVector *dest, D3dVector *src)
{
memcpy(dest,src,sizeof(D3DVECTOR));
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorCrossProduct(D3dVector *v, D3dVector *a, D3dVector *b)
{
v->x = ((a->y) * (b->z)) - ((a->z) * (b->y));
v->y = ((a->z) * (b->x)) - ((a->x) * (b->z));
v->z = ((a->x) * (b->y)) - ((a->y) * (b->x));
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorDotProduct(D3dVector *a, D3dVector *b, float *ret)
{
//TODO consider overflow?
*ret = (a->x * b->x) + (a->y *b->y) + (a->z * b->z);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorModulus(D3dVector *a, float *ret)
{
//NOTE: would do it myself but dont want to link in c runtime for Sqrt
static float (WINAPI *ProcAdd)(D3dVector*)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (float (WINAPI*)(D3dVector*)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMVectorModulus");
if (ProcAdd==NULL) return E_FAIL;
}
*ret=(ProcAdd) (a);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorNormalize(D3dVector *dest)
{
double l;
l = dest->x * dest->x + dest->y * dest->y + dest->z * dest->z;
l = sqrt(l);
dest->x = dest->x / (float)l;
dest->y = dest->y / (float)l;
dest->z = dest->z / (float)l;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorRandom(D3dVector *a)
{
static D3dVector* (WINAPI *ProcAdd)(D3dVector*)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dVector* (WINAPI*)(D3dVector*)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMVectorRandom");
if (ProcAdd==NULL) return E_FAIL;
}
if ((ProcAdd)(a)==NULL) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorReflect(D3dVector *dest,D3dVector *ray,D3dVector *norm)
{
static D3dVector* (WINAPI *ProcAdd)(D3dVector*,D3dVector*,D3dVector*)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dVector* (WINAPI*)(D3dVector*,D3dVector*,D3dVector*)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMVectorReflect");
if (ProcAdd==NULL) return E_FAIL;
}
if ((ProcAdd)(dest,ray,norm)==NULL) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorRotate(D3dVector *dest,D3dVector *vA,D3dVector *vAxis, float theta)
{
static D3dVector* (WINAPI *ProcAdd)(D3dVector*,D3dVector*,D3dVector*,float)=NULL;
LoadD3DRMDLL();
if (ProcAdd==NULL){
if (g_hInstD3DRMDLL==NULL) return E_NOINTERFACE;
ProcAdd = (D3dVector* (WINAPI*)(D3dVector*,D3dVector*,D3dVector*,float)) GetProcAddress(g_hInstD3DRMDLL, "D3DRMVectorRotate");
if (ProcAdd==NULL) return E_FAIL;
}
if ((ProcAdd)(dest,vA,vAxis,theta)==NULL) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorScale(D3dVector *v, D3dVector *a, float factor)
{
v->x=a->x *factor;
v->y=a->y *factor;
v->z=a->z *factor;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::vectorSubtract(D3dVector *v, D3dVector *a, D3dVector *b)
{
v->x = a->x - b->x;
v->y = a->y - b->y;
v->z = a->z - b->z;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::directPlayCreate(BSTR strGuid, I_dxj_DirectPlay4 **directPlay)
{
LPDIRECTPLAY4 dp;
HRESULT hr=E_FAIL;
LPDIRECTPLAY lpDP=NULL;
GUID guid;
LoadDPLAY();
ZeroMemory(&guid,sizeof(GUID));
hr=DPLBSTRtoGUID(&guid,strGuid);
if FAILED(hr) return E_INVALIDARG;
if (m_pDirectPlayCreate == NULL ) return E_FAIL;
// create a DirectPlay1 interface
hr = (m_pDirectPlayCreate)(&guid, &lpDP, NULL);
if FAILED(hr) return hr;
if (!lpDP) return E_FAIL;
// now get Dplay3 interface
hr = lpDP->QueryInterface(IID_IDirectPlay4,(LPVOID *)&dp);
lpDP->Release();
if FAILED(hr) return hr;
if (!dp) return E_FAIL;
*directPlay=NULL;
INTERNAL_CREATE(_dxj_DirectPlay4, dp, directPlay);
if (*directPlay==NULL) return E_FAIL;
return hr;
}
STDMETHODIMP C_dxj_DirectX7Object::directPlayLobbyCreate(I_dxj_DirectPlayLobby3 **ret)
{
LPDIRECTPLAYLOBBY3 dp=NULL;
HRESULT hr=E_FAIL;
LPDIRECTPLAYLOBBY lpDP=NULL;
LoadDPLAY();
if (m_pDirectPlayLobbyCreate == NULL ) return E_FAIL;
// create a DirectPlay1 interface
hr = m_pDirectPlayLobbyCreate(NULL, &lpDP, NULL,NULL,0);
if FAILED(hr) return hr;
if (!lpDP) return E_FAIL;
// no get Dplay3 interface
hr = lpDP->QueryInterface(IID_IDirectPlayLobby3,(LPVOID *)&dp);
lpDP->Release();
if FAILED(hr) return hr;
*ret=NULL;
if (!dp) return E_FAIL;
INTERNAL_CREATE(_dxj_DirectPlayLobby3, dp, ret);
if (*ret==NULL) return E_FAIL;
return hr;
}
/////////////////////////////////////////////////////////////////////////////
STDMETHODIMP C_dxj_DirectX7Object::getDPEnum( I_dxj_DPEnumServiceProviders **retVal)
{
HRESULT hr;
LoadDPLAY();
if (!m_pDirectPlayEnumerate) return E_FAIL;
hr=C_dxj_DPEnumObject::create(m_pDirectPlayEnumerate,retVal);
return hr;
}
////////////////////////////////////////////////////////////////
STDMETHODIMP C_dxj_DirectX7Object::directInputCreate(I_dxj_DirectInput **ret){
LoadDINPUTDLL();
HRESULT hr;
static HRESULT (WINAPI *ProcAdd)(HINSTANCE,DWORD,LPDIRECTINPUT*,LPUNKNOWN)=NULL;
if (ProcAdd==NULL){
if (g_hInstDINPUTDLL==NULL) return E_NOINTERFACE;
ProcAdd = (HRESULT (WINAPI*)(HINSTANCE,DWORD,LPDIRECTINPUT*,LPUNKNOWN)) GetProcAddress(g_hInstDINPUTDLL, "DirectInputCreateA");
if (ProcAdd==NULL) return E_FAIL;
}
LPDIRECTINPUT lpInput=NULL;
hr= (ProcAdd)(g_hInst,(DWORD)DIRECTINPUT_VERSION,&lpInput,NULL);
if FAILED(hr) return hr;
INTERNAL_CREATE(_dxj_DirectInput,lpInput,ret);
return hr;
}
/////////////////////////////////////////////////////////////////////////////
//
STDMETHODIMP C_dxj_DirectX7Object::tickCount( long *retval)
{
*retval = GetTickCount();
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::systemBpp(long *retval)
{
HDC hdc;
hdc = ::GetDC(NULL);
if (!hdc) return E_OUTOFMEMORY;
*retval = GetDeviceCaps(hdc, BITSPIXEL);
::ReleaseDC(NULL, hdc);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::rotateXMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest,
/* [in] */ float radians){
//id matrix
float fcos;
float fsin;
ZeroMemory(mDest,sizeof(D3dMatrix));
mDest->rc11 = 1;
mDest->rc44 = 1;
fsin=(float)sin(radians);
fcos=(float)cos(radians);
mDest->rc22 = fcos;
mDest->rc33 = fcos;
mDest->rc23 = -fsin;
mDest->rc32 = fsin;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::rotateYMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest,
/* [in] */ float radians){
//id matrix
float fcos;
float fsin;
ZeroMemory(mDest,sizeof(D3dMatrix));
mDest->rc22 = 1;
mDest->rc44 = 1;
fsin=(float)sin(radians);
fcos=(float)cos(radians);
mDest->rc11 = fcos;
mDest->rc33 = fcos;
mDest->rc13 = fsin;
mDest->rc31 = -fsin;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::rotateZMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest,
/* [in] */ float radians){
//id matrix
float fcos;
float fsin;
ZeroMemory(mDest,sizeof(D3dMatrix));
mDest->rc33 = 1;
mDest->rc44 = 1;
fsin=(float)sin(radians);
fcos=(float)cos(radians);
mDest->rc11 = fcos;
mDest->rc22 = fcos;
mDest->rc12 = -fsin;
mDest->rc21 = fsin;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::viewMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *view,
/* [in] */ D3dVector __RPC_FAR *from,
/* [in] */ D3dVector __RPC_FAR *at,
/* [in] */ D3dVector __RPC_FAR *world_up,
/* [in] */ float roll)
{
D3dVector up;
D3dVector right;
D3dVector view_Dir;
ZeroMemory(view,sizeof(D3dMatrix));
view->rc11 = 1;
view->rc22 = 1;
view->rc33 = 1;
view->rc44 = 1;
view_Dir.x=at->x-from->x;
view_Dir.y=at->y-from->y;
view_Dir.z=at->z-from->z;
vectorNormalize(&view_Dir);
//think lefthanded coords
vectorCrossProduct(&right, world_up, &view_Dir);
vectorCrossProduct(&up, &view_Dir, &right);
vectorNormalize(&right);
vectorNormalize(&up);
view->rc11 = right.x;
view->rc21 = right.y;
view->rc31 = right.z;
view->rc12 = up.x ; //AK? should this be negative?
view->rc22 = up.y;
view->rc32 = up.z;
view->rc13 = view_Dir.x;
view->rc23 = view_Dir.y;
view->rc33 = view_Dir.z;
view->rc41 = -((right.x * from->x) + (right.y * from->y) + (right.z * from->z));
view->rc42 = -((up.x * from->x) + (up.y * from->y) + (up.z * from->z));
view->rc43 = -((view_Dir.x * from->x) + (view_Dir.y * from->y) + (view_Dir.z * from->z));
if (roll!=0){
D3dMatrix rotZMat;
rotateZMatrix(&rotZMat,-roll);
matrixMultiply(view,&rotZMat,view);
}
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::matrixMultiply(
/* [out][in] */ D3dMatrix __RPC_FAR *result,
/* [in] */ D3dMatrix __RPC_FAR *a,
/* [in] */ D3dMatrix __RPC_FAR *b)
{
D3dMatrix ret;
ZeroMemory(&ret,sizeof(D3dMatrix));
ret.rc11 = b->rc11 * a->rc11 + b->rc21 * a->rc12 + b->rc31 * a->rc13 + b->rc41 * a->rc14;
ret.rc12 = b->rc12 * a->rc11 + b->rc22 * a->rc12 + b->rc32 * a->rc13 + b->rc42 * a->rc14;
ret.rc13 = b->rc13 * a->rc11 + b->rc23 * a->rc12 + b->rc33 * a->rc13 + b->rc43 * a->rc14;
ret.rc14 = b->rc14 * a->rc11 + b->rc24 * a->rc12 + b->rc34 * a->rc13 + b->rc44 * a->rc14;
ret.rc21 = b->rc11 * a->rc21 + b->rc21 * a->rc22 + b->rc31 * a->rc23 + b->rc41 * a->rc24;
ret.rc22 = b->rc12 * a->rc21 + b->rc22 * a->rc22 + b->rc32 * a->rc23 + b->rc42 * a->rc24;
ret.rc23 = b->rc13 * a->rc21 + b->rc23 * a->rc22 + b->rc33 * a->rc23 + b->rc43 * a->rc24;
ret.rc24 = b->rc14 * a->rc21 + b->rc24 * a->rc22 + b->rc34 * a->rc23 + b->rc44 * a->rc24;
ret.rc31 = b->rc11 * a->rc31 + b->rc21 * a->rc32 + b->rc31 * a->rc33 + b->rc41 * a->rc34;
ret.rc32 = b->rc12 * a->rc31 + b->rc22 * a->rc32 + b->rc32 * a->rc33 + b->rc42 * a->rc34;
ret.rc33 = b->rc13 * a->rc31 + b->rc23 * a->rc32 + b->rc33 * a->rc33 + b->rc43 * a->rc34;
ret.rc34 = b->rc14 * a->rc31 + b->rc24 * a->rc32 + b->rc34 * a->rc33 + b->rc44 * a->rc34;
ret.rc41 = b->rc11 * a->rc41 + b->rc21 * a->rc42 + b->rc31 * a->rc43 + b->rc41 * a->rc44;
ret.rc42 = b->rc12 * a->rc41 + b->rc22 * a->rc42 + b->rc32 * a->rc43 + b->rc42 * a->rc44;
ret.rc43 = b->rc13 * a->rc41 + b->rc23 * a->rc42 + b->rc33 * a->rc43 + b->rc43 * a->rc44;
ret.rc44 = b->rc14 * a->rc41 + b->rc24 * a->rc42 + b->rc34 * a->rc43 + b->rc44 * a->rc44;
memcpy(result,&ret,sizeof(D3dMatrix));
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::projectionMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest,
/* [in] */ float near_plane,
/* [in] */ float far_plane,
/* [in] */ float fov)
{
float c,s,Q;
c =(float) cos(fov * 0.5);
s = (float)sin(fov * 0.5);
if (far_plane==0) return E_INVALIDARG;
Q = s / (1 - near_plane / far_plane);
ZeroMemory(mDest,sizeof(D3dMatrix));
mDest->rc11 = c;
mDest->rc22 = c;
mDest->rc33 = Q;
mDest->rc43 = -Q * near_plane;
mDest->rc34 = s;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::copyMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest,
/* [in] */ D3dMatrix __RPC_FAR *mSrc)
{
memcpy(mDest,mSrc,sizeof(D3dMatrix));
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::identityMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest)
{
ZeroMemory (mDest,sizeof(D3dMatrix));
mDest->rc11=1;
mDest->rc22=1;
mDest->rc33=1;
mDest->rc44=1;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::zeroMatrix(
/* [out][in] */ D3dMatrix __RPC_FAR *mDest)
{
ZeroMemory (mDest,sizeof(D3dMatrix));
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::directMusicLoaderCreate (
/* [retval][out] */ I_dxj_DirectMusicLoader __RPC_FAR *__RPC_FAR *ret)
{
HRESULT hr;
IDirectMusicLoader *pLoader=NULL;
hr =CoCreateInstance(
CLSID_DirectMusicLoader,
NULL,
CLSCTX_INPROC,
IID_IDirectMusicLoader,
(void**)&pLoader);
if (FAILED(hr)) return E_NOINTERFACE;
if (!pLoader) return E_FAIL;
INTERNAL_CREATE_NOADDREF(_dxj_DirectMusicLoader,pLoader,ret);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::directMusicComposerCreate (
/* [retval][out] */ I_dxj_DirectMusicComposer __RPC_FAR *__RPC_FAR *ret)
{
IDirectMusicComposer *pComp=NULL;
if (FAILED(CoCreateInstance(
CLSID_DirectMusicComposer,
NULL,
CLSCTX_INPROC,
IID_IDirectMusicComposer,
(void**)&pComp ))) {
return E_NOINTERFACE;
}
if (!pComp) return E_FAIL;
INTERNAL_CREATE_NOADDREF(_dxj_DirectMusicComposer,pComp,ret);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::directMusicPerformanceCreate (
/* [retval][out] */ I_dxj_DirectMusicPerformance __RPC_FAR *__RPC_FAR *ret)
{
IDirectMusicPerformance *pPerf=NULL;
if (FAILED(CoCreateInstance(
CLSID_DirectMusicPerformance,
NULL,
CLSCTX_INPROC,
IID_IDirectMusicPerformance,
(void**)&pPerf ))) {
return E_NOINTERFACE;
}
if (!pPerf) return E_FAIL;
INTERNAL_CREATE_NOADDREF(_dxj_DirectMusicPerformance,pPerf,ret);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::getWindowRect(
/* [in] */ long hwnd,
/* [out][in] */ Rect __RPC_FAR *r)
{
BOOL b=GetWindowRect((HWND)hwnd,(LPRECT)r);
if (!b) return E_FAIL;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createEvent(
/* [in] */ I_dxj_DirectXEvent __RPC_FAR *event,
/* [retval][out] */ long __RPC_FAR *h)
{
HRESULT hr;
LPSTREAM pStm=NULL;
IUnknown *pUnk=NULL;
HANDLE hEvent=NULL;
EVENTTHREADINFO *pNewEvent=NULL;
EVENTTHREADINFO *pTemp=NULL;
if (!event) return E_INVALIDARG;
if (!h) return E_INVALIDARG;
pNewEvent=(EVENTTHREADINFO*)malloc(sizeof(EVENTTHREADINFO));
if (!pNewEvent) return E_OUTOFMEMORY;
pNewEvent->pNext=NULL;
pNewEvent->fEnd=FALSE;
pNewEvent->pCallback=event;
pNewEvent->pStream=NULL;
pNewEvent->threadID=0;
event->AddRef();
pNewEvent->hEvent = CreateEvent(NULL,FALSE,FALSE,NULL);
if (!pNewEvent->hEvent){
free(pNewEvent);
event->Release();
return E_FAIL;
}
//pNewEvent->hEndEvent = CreateEvent(NULL,FALSE,FALSE,NULL);
hr=event->QueryInterface(IID_IUnknown,(void**)&pUnk);
if FAILED(hr) {
free(pNewEvent);
event->Release();
return E_FAIL;
}
hr=CoMarshalInterThreadInterfaceInStream(IID_IUnknown,pUnk,&pStm);
if (pUnk) pUnk->Release();
if FAILED(hr) {
free(pNewEvent);
event->Release();
return E_FAIL;
}
pNewEvent->pStream=pStm;
pNewEvent->hThread=CreateThread(NULL,0,ThreadFunc,(unsigned long*)pNewEvent,CREATE_SUSPENDED ,&pNewEvent->threadID);
if (!pNewEvent->threadID) {
CloseHandle(pNewEvent->hEvent);
free(pNewEvent);
event->Release();
return E_FAIL;
}
if (!m_pEventList){
m_pEventList=pNewEvent;
}
else{
pTemp=m_pEventList;
m_pEventList=pNewEvent;
pNewEvent->pNext=pTemp;
}
ResumeThread(pNewEvent->hThread);
*h=(long)PtrToLong(pNewEvent->hEvent); //bugbug SUNDOWN -
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::setEvent(
/* [in] */ long eventId)
{
SetEvent((HANDLE)eventId);
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::destroyEvent(
/* [in] */ long eventId)
{
//find the info on the stack
if (!m_pEventList) return E_INVALIDARG;
EVENTTHREADINFO *pTemp=NULL;
EVENTTHREADINFO *pLast=NULL;
//rely on lazy evaluation
for (pTemp=m_pEventList; ((pTemp)&&(pTemp->hEvent!=(HANDLE)eventId));pLast=pTemp,pTemp=pTemp->pNext);
if (!pTemp) return E_INVALIDARG;
//remove it from our Link List
if (!pLast) {
m_pEventList=pTemp->pNext;
}
else {
pLast->pNext=pTemp->pNext;
}
//indicate that we want to kill the thread
pTemp->fEnd=TRUE;
//Fire the event in case we are waiting
if (pTemp->hEvent) SetEvent(pTemp->hEvent);
//Wait for it to finish out
if (pTemp->hThread) WaitForSingleObject(pTemp->hThread,1000);
//wait for the end event to signal
//if (pTemp->hEndEvent) WaitForSingleObject(pTemp->hEndEvent,1000);
//desctroy the event
if (pTemp->hEvent) CloseHandle(pTemp->hEvent);
//if (pTemp->hEndEvent) CloseHandle (pTemp->hEndEvent);
if (pTemp->pCallback) pTemp->pCallback->Release();
//thread is gone..
//free the memory
free(pTemp);
return S_OK;
}
DWORD WINAPI ThreadFunc(LPVOID param){
HRESULT hr;
IUnknown *pUnk=NULL;
EVENTTHREADINFO *pCntrl=(EVENTTHREADINFO *)param;
I_dxj_DirectXEvent *pVBCallback=NULL;
OleInitialize(NULL);
LCID LOCAL_SYSTEM_DEFAULT=GetSystemDefaultLCID();
//note pstrm is released even on failure
hr=CoGetInterfaceAndReleaseStream(pCntrl->pStream,IID_IUnknown,(void**)&pUnk);
pCntrl->pCallback=NULL; //since released to 0
if FAILED(hr) return -1;
if (!pUnk) return -1;
hr=pUnk->QueryInterface(IID_I_dxj_DirectXEvent,(void**)&pVBCallback);
pUnk->Release();
if FAILED(hr) return -1;
while (pCntrl->fEnd==FALSE)
{
WaitForSingleObject(pCntrl->hEvent,INFINITE);
if ((pVBCallback )&&(pCntrl->fEnd==FALSE))
{
pVBCallback->AddRef();
pVBCallback->DXCallback((long)PtrToLong(pCntrl->hEvent)); //bugbug SUNDOWN
pVBCallback->Release();
}
}
if (pVBCallback) pVBCallback->Release();
OleUninitialize();
//we need to syncronize the ending of the thread..
//if (pCntrl->hEndEvent) SetEvent(pCntrl->hEndEvent);
return 0;
}
STDMETHODIMP C_dxj_DirectX7Object::createD3DVertex(float x, float y, float z, float nx, float ny, float nz, float tu, float tv, D3dVertex *v)
{
if (!v) return E_INVALIDARG;
((D3DVERTEX*)v)->x=x;
((D3DVERTEX*)v)->y=y;
((D3DVERTEX*)v)->z=z;
((D3DVERTEX*)v)->nx=nx;
((D3DVERTEX*)v)->ny=ny;
((D3DVERTEX*)v)->nz=nz;
((D3DVERTEX*)v)->tu=tu;
((D3DVERTEX*)v)->tv=tv;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createD3DLVertex(float x, float y, float z, long color, long specular, float tu, float tv, D3dLVertex *v)
{
if (!v) return E_INVALIDARG;
((D3DLVERTEX*)v)->x=x;
((D3DLVERTEX*)v)->y=y;
((D3DLVERTEX*)v)->z=z;
((D3DLVERTEX*)v)->color=color;
((D3DLVERTEX*)v)->specular=specular;
((D3DLVERTEX*)v)->dwReserved=0;
((D3DLVERTEX*)v)->tu=tu;
((D3DLVERTEX*)v)->tv=tv;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createD3DTLVertex(float sx, float sy, float sz, float rhw, long color, long specular, float tu, float tv, D3dTLVertex *v)
{
if (!v) return E_INVALIDARG;
((D3DTLVERTEX*)v)->sx=sx;
((D3DTLVERTEX*)v)->sy=sy;
((D3DTLVERTEX*)v)->sz=sz;
((D3DTLVERTEX*)v)->rhw=rhw;
((D3DTLVERTEX*)v)->color=color;
((D3DTLVERTEX*)v)->specular=specular;
((D3DTLVERTEX*)v)->tu=tu;
((D3DTLVERTEX*)v)->tv=tv;
return S_OK;
}
STDMETHODIMP C_dxj_DirectX7Object::createNewGuid(BSTR *ret)
{
HRESULT hr;
GUID g=GUID_NULL;
if (!ret) return E_INVALIDARG;
hr=::CoCreateGuid(&g);
if FAILED(hr) return hr;
*ret=GUIDtoBSTR(&g);
return S_OK;
}