You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
353 lines
8.5 KiB
353 lines
8.5 KiB
/////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (c) 1998 Active Voice Corporation. All Rights Reserved.
|
|
//
|
|
// Active Agent(r) and Unified Communications(tm) are trademarks of Active Voice Corporation.
|
|
//
|
|
// Other brand and product names used herein are trademarks of their respective owners.
|
|
//
|
|
// The entire program and user interface including the structure, sequence, selection,
|
|
// and arrangement of the dialog, the exclusively "yes" and "no" choices represented
|
|
// by "1" and "2," and each dialog message are protected by copyrights registered in
|
|
// the United States and by international treaties.
|
|
//
|
|
// Protected by one or more of the following United States patents: 5,070,526, 5,488,650,
|
|
// 5,434,906, 5,581,604, 5,533,102, 5,568,540, 5,625,676, 5,651,054.
|
|
//
|
|
// Active Voice Corporation
|
|
// Seattle, Washington
|
|
// USA
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
////
|
|
// tsmthunk.c - tsm thunk functions
|
|
////
|
|
|
|
// This is a thunk layer to the telephone functions in avtsm.dll.
|
|
// It's purpose is to allow an application to use avtsm.dll functions
|
|
// only if they are available.
|
|
//
|
|
// To use this module, link TSMTHUNK.OBJ with your application
|
|
// rather than with AVTSM.LIB. Before calling any Tsm
|
|
// functions, call TsmThunkInit. Before exiting your application,
|
|
// call TsmThunkTerm.
|
|
//
|
|
|
|
#include "winlocal.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "avtsm.h"
|
|
#include "tsmthunk.h"
|
|
#include "loadlib.h"
|
|
#include "mem.h"
|
|
#include "trace.h"
|
|
|
|
extern HINSTANCE g_hInstLib;
|
|
|
|
////
|
|
// private definitions
|
|
////
|
|
|
|
#define TSMTHUNK_LIBNAME "avtsm.dll"
|
|
|
|
#ifdef TSMTHUNK
|
|
#undef TSMTHUNK
|
|
#endif
|
|
|
|
extern HINSTANCE g_hInstLib;
|
|
|
|
// tsmthunk control struct
|
|
//
|
|
typedef struct TSMTHUNK
|
|
{
|
|
DWORD dwVersion;
|
|
HINSTANCE hInst;
|
|
HTASK hTask;
|
|
HINSTANCE hInstLib;
|
|
} TSMTHUNK, FAR *LPTSMTHUNK;
|
|
|
|
// tsmthunk function struct
|
|
//
|
|
typedef struct TSMTHUNKFN
|
|
{
|
|
int index;
|
|
LPSTR lpszName;
|
|
FARPROC lpfn;
|
|
} TSMTHUNKFN, FAR *LPTSMTHUNKFN;
|
|
|
|
enum
|
|
{
|
|
iTsmInit = 0,
|
|
iTsmTerm,
|
|
iTsmSetSpeed,
|
|
iTsmConvert,
|
|
iTsmSupportsSpeed,
|
|
|
|
TSMTHUNK_MAXFUNCTIONS
|
|
};
|
|
|
|
static TSMTHUNKFN aTsmThunkFn[] =
|
|
{
|
|
iTsmInit, "TsmInit", NULL,
|
|
iTsmTerm, "TsmTerm", NULL,
|
|
iTsmSetSpeed, "TsmSetSpeed", NULL,
|
|
iTsmConvert, "TsmConvert", NULL,
|
|
iTsmSupportsSpeed, "TsmSupportsSpeed", NULL
|
|
};
|
|
|
|
// helper functions
|
|
//
|
|
static LPTSMTHUNK TsmThunkGetPtr(HTSMTHUNK hTsmThunk);
|
|
static HTSMTHUNK TsmThunkGetHandle(LPTSMTHUNK lpTsmThunk);
|
|
|
|
////
|
|
// public functions
|
|
////
|
|
|
|
// TsmThunkInit - initialize tsmthunk engine
|
|
// <dwVersion> (i) must be TSMTHUNK_VERSION
|
|
// <hInst> (i) instance handle of calling module
|
|
// return handle (NULL if error)
|
|
//
|
|
HTSMTHUNK DLLEXPORT WINAPI TsmThunkInit(DWORD dwVersion, HINSTANCE hInst)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
LPTSMTHUNK lpTsmThunk = NULL;
|
|
|
|
if (dwVersion != TSMTHUNK_VERSION)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (hInst == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if ((lpTsmThunk = (LPTSMTHUNK) MemAlloc(NULL, sizeof(TSMTHUNK), 0)) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
int i;
|
|
|
|
lpTsmThunk->dwVersion = dwVersion;
|
|
lpTsmThunk->hInst = hInst;
|
|
lpTsmThunk->hTask = GetCurrentTask();
|
|
lpTsmThunk->hInstLib = NULL;
|
|
|
|
// load the library if possible
|
|
//
|
|
if ((lpTsmThunk->hInstLib = LoadLibraryPath(TSMTHUNK_LIBNAME, g_hInstLib, 0)) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
// get the address of each function in library
|
|
//
|
|
else for (i = 0; i < SIZEOFARRAY(aTsmThunkFn); ++i)
|
|
{
|
|
if (aTsmThunkFn[i].index != i)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if ((aTsmThunkFn[i].lpfn = GetProcAddress(lpTsmThunk->hInstLib,
|
|
aTsmThunkFn[i].lpszName)) == NULL)
|
|
{
|
|
TracePrintf_1(NULL, 6,
|
|
TEXT("GetProcAddress failed\n fn=%s\n"),
|
|
(LPTSTR) aTsmThunkFn[i].lpszName);
|
|
fSuccess = TraceFALSE(NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!fSuccess)
|
|
{
|
|
TsmThunkTerm(TsmThunkGetHandle(lpTsmThunk));
|
|
lpTsmThunk = NULL;
|
|
}
|
|
|
|
return fSuccess ? TsmThunkGetHandle(lpTsmThunk) : NULL;
|
|
}
|
|
|
|
|
|
// TsmThunkTerm - shut down tsmthunk engine
|
|
// <hTsmThunk> (i) handle returned from TsmThunkInit
|
|
// return 0 if success
|
|
//
|
|
int DLLEXPORT WINAPI TsmThunkTerm(HTSMTHUNK hTsmThunk)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
LPTSMTHUNK lpTsmThunk;
|
|
|
|
if ((lpTsmThunk = TsmThunkGetPtr(hTsmThunk)) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
// library no longer needed
|
|
//
|
|
FreeLibrary(lpTsmThunk->hInstLib);
|
|
|
|
if ((lpTsmThunk = MemFree(NULL, lpTsmThunk)) != NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
}
|
|
|
|
return fSuccess ? 0 : -1;
|
|
}
|
|
|
|
// Tsm thunk functions
|
|
//
|
|
|
|
HTSM DLLEXPORT WINAPI TsmInit(DWORD dwVersion, HINSTANCE hInst,
|
|
LPWAVEFORMATEX lpwfx, int nScaleEfficiency, long sizBufSrcMax,
|
|
DWORD dwFlags)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
HTSM (WINAPI *lpfnTsmInit)(DWORD dwVersion, HINSTANCE hInst,
|
|
LPWAVEFORMATEX lpwfx, int nScaleEfficiency, long sizBufSrcMax,
|
|
DWORD dwFlags);
|
|
HTSM hTsm;
|
|
|
|
if (aTsmThunkFn[iTsmInit].index != iTsmInit)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (((FARPROC) lpfnTsmInit = aTsmThunkFn[iTsmInit].lpfn) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
hTsm = (*lpfnTsmInit)(dwVersion, hInst,
|
|
lpwfx, nScaleEfficiency, sizBufSrcMax, dwFlags);
|
|
}
|
|
|
|
return fSuccess ? hTsm : NULL;
|
|
}
|
|
|
|
int DLLEXPORT WINAPI TsmTerm(HTSM hTsm)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
int (WINAPI *lpfnTsmTerm)(HTSM hTsm);
|
|
int iRet;
|
|
|
|
if (aTsmThunkFn[iTsmTerm].index != iTsmTerm)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (((FARPROC) lpfnTsmTerm = aTsmThunkFn[iTsmTerm].lpfn) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
iRet = (*lpfnTsmTerm)(hTsm);
|
|
}
|
|
|
|
return fSuccess ? iRet : -1;
|
|
}
|
|
|
|
int DLLEXPORT WINAPI TsmSetSpeed(HTSM hTsm, int nLevel, DWORD dwFlags)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
int (WINAPI *lpfnTsmSetSpeed)(HTSM hTsm, int nLevel, DWORD dwFlags);
|
|
int iRet;
|
|
|
|
if (aTsmThunkFn[iTsmSetSpeed].index != iTsmSetSpeed)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (((FARPROC) lpfnTsmSetSpeed = aTsmThunkFn[iTsmSetSpeed].lpfn) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
iRet = (*lpfnTsmSetSpeed)(hTsm, nLevel, dwFlags);
|
|
}
|
|
|
|
return fSuccess ? iRet : -1;
|
|
}
|
|
|
|
long DLLEXPORT WINAPI TsmConvert(HTSM hTsm,
|
|
void _huge *hpBufSrc, long sizBufSrc,
|
|
void _huge *hpBufDst, long sizBufDst,
|
|
DWORD dwFlags)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
long (WINAPI *lpfnTsmConvert)(HTSM hTsm,
|
|
void _huge *hpBufSrc, long sizBufSrc,
|
|
void _huge *hpBufDst, long sizBufDst,
|
|
DWORD dwFlags);
|
|
long lRet;
|
|
|
|
if (aTsmThunkFn[iTsmConvert].index != iTsmConvert)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (((FARPROC) lpfnTsmConvert = aTsmThunkFn[iTsmConvert].lpfn) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
lRet = (*lpfnTsmConvert)(hTsm,
|
|
hpBufSrc, sizBufSrc, hpBufDst, sizBufDst, dwFlags);
|
|
}
|
|
|
|
return fSuccess ? lRet : -1;
|
|
}
|
|
|
|
BOOL DLLEXPORT WINAPI TsmSupportsSpeed(int nLevel, LPWAVEFORMATEX lpwfx, DWORD dwFlags)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
BOOL (WINAPI *lpfnTsmSupportsSpeed)(int nLevel, LPWAVEFORMATEX lpwfx, DWORD dwFlags);
|
|
BOOL fRet;
|
|
|
|
if (aTsmThunkFn[iTsmSupportsSpeed].index != iTsmSupportsSpeed)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (((FARPROC) lpfnTsmSupportsSpeed = aTsmThunkFn[iTsmSupportsSpeed].lpfn) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else
|
|
{
|
|
fRet = (*lpfnTsmSupportsSpeed)(nLevel, lpwfx, dwFlags);
|
|
}
|
|
|
|
return fSuccess ? fRet : FALSE;
|
|
}
|
|
|
|
////
|
|
// helper functions
|
|
////
|
|
|
|
// TsmThunkGetPtr - verify that tsmthunk handle is valid,
|
|
// <hTsmThunk> (i) handle returned from TsmThunkInit
|
|
// return corresponding tsmthunk pointer (NULL if error)
|
|
//
|
|
static LPTSMTHUNK TsmThunkGetPtr(HTSMTHUNK hTsmThunk)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
LPTSMTHUNK lpTsmThunk;
|
|
|
|
if ((lpTsmThunk = (LPTSMTHUNK) hTsmThunk) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
else if (IsBadWritePtr(lpTsmThunk, sizeof(TSMTHUNK)))
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
#ifdef CHECKTASK
|
|
// make sure current task owns the tsmthunk handle
|
|
//
|
|
else if (lpTsmThunk->hTask != GetCurrentTask())
|
|
fSuccess = TraceFALSE(NULL);
|
|
#endif
|
|
|
|
return fSuccess ? lpTsmThunk : NULL;
|
|
}
|
|
|
|
// TsmThunkGetHandle - verify that tsmthunk pointer is valid,
|
|
// <lpTsmThunk> (i) pointer to TSMTHUNK struct
|
|
// return corresponding tsmthunk handle (NULL if error)
|
|
//
|
|
static HTSMTHUNK TsmThunkGetHandle(LPTSMTHUNK lpTsmThunk)
|
|
{
|
|
BOOL fSuccess = TRUE;
|
|
HTSMTHUNK hTsmThunk;
|
|
|
|
if ((hTsmThunk = (HTSMTHUNK) lpTsmThunk) == NULL)
|
|
fSuccess = TraceFALSE(NULL);
|
|
|
|
return fSuccess ? hTsmThunk : NULL;
|
|
}
|
|
|