mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-01 14:36:00 +00:00
472 lines
15 KiB
C++
472 lines
15 KiB
C++
|
#include "GracenoteApi.h"
|
|||
|
#include "api.h"
|
|||
|
#include "../winamp/api_decodefile.h"
|
|||
|
#include <bfc/error.h>
|
|||
|
#include <limits.h>
|
|||
|
#include <shlwapi.h>
|
|||
|
#include <strsafe.h>
|
|||
|
|
|||
|
GracenoteApi::GracenoteApi()
|
|||
|
{
|
|||
|
cddbInitialized = false;
|
|||
|
playlistInitialized = false;
|
|||
|
pCDDBControl=0;
|
|||
|
}
|
|||
|
|
|||
|
GracenoteApi::~GracenoteApi()
|
|||
|
{
|
|||
|
}
|
|||
|
|
|||
|
void GracenoteApi::Close()
|
|||
|
{
|
|||
|
if (pCDDBControl)
|
|||
|
{
|
|||
|
pCDDBControl->Shutdown();
|
|||
|
pCDDBControl->Release();
|
|||
|
pCDDBControl=0;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
static void SetProxy(const wchar_t *proxy, ICddbOptions *options)
|
|||
|
{
|
|||
|
wchar_t user[1024]=L"";
|
|||
|
wchar_t pass[1024]=L"";
|
|||
|
wchar_t server[1024]=L"";
|
|||
|
int port=80;
|
|||
|
if (!_wcsnicmp(proxy,L"https:",6))
|
|||
|
port = 443;
|
|||
|
|
|||
|
const wchar_t *skip = wcsstr(proxy, L"://");
|
|||
|
if (skip)
|
|||
|
proxy = skip+3;
|
|||
|
|
|||
|
skip = wcsstr(proxy, L"@");
|
|||
|
if (skip)
|
|||
|
{
|
|||
|
const wchar_t *delimiter = wcsstr(proxy, L":");
|
|||
|
if (delimiter < skip) // make sure there's really a password (and we didn't end up finding the port number)
|
|||
|
{
|
|||
|
StringCchCopyNW(user, 1024, proxy, delimiter-proxy);
|
|||
|
StringCchCopyNW(pass, 1024, delimiter+1, skip-(delimiter+1));
|
|||
|
proxy=skip+1;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
StringCchCopyNW(user, 1024, proxy, skip-proxy);
|
|||
|
proxy=skip+1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
skip = wcsstr(proxy, L":"); // look for port
|
|||
|
if (skip)
|
|||
|
{
|
|||
|
StringCchCopyNW(server, 1024, proxy, skip-proxy);
|
|||
|
port = _wtoi(skip+1);
|
|||
|
}
|
|||
|
else
|
|||
|
StringCchCopyW(server, 1024, proxy);
|
|||
|
|
|||
|
if (server[0])
|
|||
|
options->put_ProxyServer(server);
|
|||
|
if (port)
|
|||
|
options->put_ProxyServerPort(port);
|
|||
|
if (user[0])
|
|||
|
options->put_ProxyUserName(user);
|
|||
|
if (pass[0])
|
|||
|
options->put_ProxyPassword(pass);
|
|||
|
}
|
|||
|
|
|||
|
// {C0A565DC-0CFE-405a-A27C-468B0C8A3A5C}
|
|||
|
static const GUID internetConfigGroupGUID =
|
|||
|
{ 0xc0a565dc, 0xcfe, 0x405a, { 0xa2, 0x7c, 0x46, 0x8b, 0xc, 0x8a, 0x3a, 0x5c } };
|
|||
|
|
|||
|
ICDDBControl2 *GracenoteApi::GetCDDB()
|
|||
|
{
|
|||
|
Nullsoft::Utility::AutoLock lock(cddbGuard);
|
|||
|
if (!cddbInitialized)
|
|||
|
{
|
|||
|
CoCreateInstance(__uuidof(CDDBNSWinampControl), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&pCDDBControl);
|
|||
|
if (pCDDBControl == NULL)
|
|||
|
return 0;
|
|||
|
|
|||
|
#if 1 // TODO: benski> put back in once we can match winamp lang pack to a gracenote language ID
|
|||
|
// translate if necessary
|
|||
|
if (WASABI_API_LNG)
|
|||
|
{
|
|||
|
const wchar_t *langFolder = WASABI_API_LNG->GetLanguageFolder();
|
|||
|
if (langFolder)
|
|||
|
{
|
|||
|
WIN32_FIND_DATAW find;
|
|||
|
wchar_t mask[MAX_PATH] = {0}, maskLoc[16] = {0};
|
|||
|
|
|||
|
// attempt to get the language code to help better guess the CDDB dll needed
|
|||
|
if(WASABI_API_LNG->GetLanguageIdentifier(LANG_IDENT_STR))
|
|||
|
StringCchPrintfW(maskLoc, 16, L"CddbLang%s.dll", WASABI_API_LNG->GetLanguageIdentifier(LANG_LANG_CODE));
|
|||
|
|
|||
|
PathCombineW(mask, langFolder, maskLoc);
|
|||
|
HANDLE hFind = FindFirstFileW(mask, &find);
|
|||
|
|
|||
|
// if the the guess provides nothing then scan for any valid dll (not ideal)
|
|||
|
if (hFind == INVALID_HANDLE_VALUE)
|
|||
|
{
|
|||
|
PathCombineW(mask, langFolder, L"CddbLang*.dll");
|
|||
|
hFind = FindFirstFileW(mask, &find);
|
|||
|
}
|
|||
|
|
|||
|
if (hFind != INVALID_HANDLE_VALUE)
|
|||
|
{
|
|||
|
ICddbUIPtr ui;
|
|||
|
ui.CreateInstance(__uuidof(CddbNSWinampUI));
|
|||
|
if (ui)
|
|||
|
{
|
|||
|
long val = 0;
|
|||
|
wchar_t cddb_lang_fn[MAX_PATH] = {0};
|
|||
|
PathCombineW(cddb_lang_fn, langFolder, find.cFileName);
|
|||
|
// TODO: benski> gracenote wants a "language ID" but we don't have a good way of knowing it :(
|
|||
|
ui->SetUILanguage(0, cddb_lang_fn, &val);
|
|||
|
// TODO: benski> also need to set ICddbOptions language ID
|
|||
|
}
|
|||
|
}
|
|||
|
FindClose(hFind);
|
|||
|
}
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
// winamp browser id
|
|||
|
//HRESULT hr = pCDDBControl->SetClientInfo(L"7944448", L"F8DE207FBA826F136FF2C7EFE0AAB181", L"1", L"regstring");
|
|||
|
//wa5's id
|
|||
|
|
|||
|
const wchar_t *appVersion = WASABI_API_APP->main_getVersionNumString();
|
|||
|
|
|||
|
/* development client ID */
|
|||
|
//HRESULT hr = pCDDBControl->SetClientInfo(L"3714048", L"B49286AE14F73CCD73C23B371A56DB00", const_cast<BSTR>(appVersion), L"regstring");
|
|||
|
|
|||
|
/* Beta Client ID */
|
|||
|
//HRESULT hr = pCDDBControl->SetClientInfo(L"8337664", L"A222F2FA8B3E047291DFDBF465FD3C95", const_cast<BSTR>(appVersion), L"regstring");
|
|||
|
|
|||
|
/* Production Client ID */
|
|||
|
BSTR appVersionBSTR = SysAllocString(appVersion);
|
|||
|
HRESULT hr = pCDDBControl->SetClientInfo(L"4896768", L"C1519CAE91489E405BCA93531837F2BE", appVersionBSTR, L"regstring");
|
|||
|
SysFreeString(appVersionBSTR);
|
|||
|
|
|||
|
if (FAILED(hr))
|
|||
|
return 0;
|
|||
|
|
|||
|
long flags = CACHE_UPDATE_FUZZY | CACHE_DONT_WRITE_ANY | CACHE_NO_LOOKUP_MEDIA; //CACHE_SUBMIT_ALL | CACHE_SUBMIT_OFFLINE | CACHE_SUBMIT_NEW | CACHE_NO_LOOKUP_MEDIA;
|
|||
|
|
|||
|
// set cache path for cddb control
|
|||
|
ICddbOptionsPtr pOptions;
|
|||
|
hr = pCDDBControl->GetOptions(&pOptions);
|
|||
|
if (SUCCEEDED(hr))
|
|||
|
{
|
|||
|
wchar_t dataPath[MAX_PATH] = {0};
|
|||
|
PathCombineW(dataPath, WASABI_API_APP->path_getUserSettingsPath(), L"Plugins");
|
|||
|
CreateDirectoryW(dataPath, 0);
|
|||
|
PathAppendW(dataPath, L"Gracenote");
|
|||
|
CreateDirectoryW(dataPath, 0);
|
|||
|
|
|||
|
hr = pOptions->put_LocalCachePath(dataPath);
|
|||
|
|
|||
|
// initial cache flags
|
|||
|
|
|||
|
//BOOL bOnline = SendMessage(line.hMainWindow, WM_USER, 0, 242);
|
|||
|
//if (!bOnline)
|
|||
|
//flags |= CACHE_DONT_CONNECT;
|
|||
|
|
|||
|
// other needed settings
|
|||
|
hr = pOptions->put_ProgressEvents(FALSE);
|
|||
|
//hr = pOptions->put_LocalCacheFlags(CACHE_DONT_CREATE | CACHE_UPDATE_FUZZY | CACHE_SUBMIT_ALL);
|
|||
|
hr = pOptions->put_LocalCacheFlags(flags);
|
|||
|
hr = pOptions->put_LocalCacheSize(128 * 1024); // 128 megabyte limit on local cache size
|
|||
|
hr = pOptions->put_LocalCacheTimeout(5 * 365); // 5 years (e.g. when Gracenote contract runs out)
|
|||
|
hr = pOptions->put_TestSubmitMode(FALSE);
|
|||
|
//hr = pOptions->put_TestSubmitMode(TRUE); //CT> for BETA cycle...
|
|||
|
|
|||
|
// this is supposed to turn off the spinning logo in the upper-left-hand corner
|
|||
|
hr = pOptions->put_ResourceModule(-1);
|
|||
|
|
|||
|
// get n set proxy settings
|
|||
|
const wchar_t *proxy = AGAVE_API_CONFIG->GetString(internetConfigGroupGUID, L"proxy", L"");
|
|||
|
if (proxy && proxy[0])
|
|||
|
SetProxy(proxy, pOptions);
|
|||
|
|
|||
|
// save settings
|
|||
|
hr = pCDDBControl->SetOptions(pOptions);
|
|||
|
}
|
|||
|
|
|||
|
hr = pCDDBControl->Initialize(0/*(long)line.hMainWindow*/, (CDDBCacheFlags)flags);
|
|||
|
|
|||
|
// checks for user registration
|
|||
|
long pVal=0;
|
|||
|
|
|||
|
// this must be called when control is first initialized
|
|||
|
// this will load existing registration into control
|
|||
|
hr = pCDDBControl->IsRegistered(FALSE, &pVal);
|
|||
|
|
|||
|
// if not reg'd, bring up reg UI (param1 = TRUE)
|
|||
|
if (!pVal)
|
|||
|
{
|
|||
|
// do headless registration
|
|||
|
ICddbUserInfoPtr pUser;
|
|||
|
|
|||
|
hr = pCDDBControl->GetUserInfo(&pUser);
|
|||
|
if (pUser != NULL)
|
|||
|
{
|
|||
|
do
|
|||
|
{
|
|||
|
wchar_t strdata[129] = {0};
|
|||
|
size_t size = sizeof(strdata)/sizeof(*strdata);
|
|||
|
wchar_t *str = strdata;
|
|||
|
|
|||
|
GUID uid = GUID_NULL;
|
|||
|
int x;
|
|||
|
unsigned char *p;
|
|||
|
CoCreateGuid(&uid);
|
|||
|
p = (unsigned char *) & uid;
|
|||
|
//lstrcpynW(str, L"WA2_", 129);
|
|||
|
StringCchCopyExW(str, size, L"WA5_", &str, &size, 0);
|
|||
|
for (x = 0; x < sizeof(uid); x ++)
|
|||
|
{
|
|||
|
StringCchPrintfExW(str, size, &str, &size, 0, L"%02X", p[x]);
|
|||
|
//wsprintfW(str + wcslen(str), L"%02X", p[x]);
|
|||
|
}
|
|||
|
|
|||
|
// user name will have to be unique per install
|
|||
|
hr = pUser->put_UserHandle(strdata);
|
|||
|
hr = pUser->put_Password(strdata);
|
|||
|
|
|||
|
hr = pCDDBControl->SetUserInfo(pUser);
|
|||
|
}
|
|||
|
while (hr == CDDBSVCHandleUsed);
|
|||
|
}
|
|||
|
|
|||
|
// this is just to check again that the user is now registered
|
|||
|
hr = pCDDBControl->IsRegistered(FALSE, &pVal);
|
|||
|
}
|
|||
|
|
|||
|
cddbInitialized = true;
|
|||
|
}
|
|||
|
|
|||
|
if (pCDDBControl)
|
|||
|
pCDDBControl->AddRef();
|
|||
|
|
|||
|
return pCDDBControl;
|
|||
|
}
|
|||
|
|
|||
|
#if 0
|
|||
|
/// This is the deprecated version of GetPlaylistManager that is no longer used without the MLDB manager
|
|||
|
ICddbPlaylist25Mgr *GracenoteApi::GetPlaylistManager()
|
|||
|
{
|
|||
|
ICddbPlaylist25Mgr *playlistMgr;
|
|||
|
|
|||
|
ICDDBControl2 *cddb = GetCDDB();
|
|||
|
if (!cddb)
|
|||
|
return 0;
|
|||
|
|
|||
|
CoCreateInstance(__uuidof(CddbNSWinampPlaylist2Mgr), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&playlistMgr);
|
|||
|
if (playlistMgr)
|
|||
|
{
|
|||
|
playlistMgr->AddRef();
|
|||
|
wchar_t dataPath[MAX_PATH] = {0};
|
|||
|
PathCombineW(dataPath, WASABI_API_APP->path_getUserSettingsPath(), L"Plugins");
|
|||
|
CreateDirectoryW(dataPath, 0);
|
|||
|
PathAppendW(dataPath, L"Gracenote");
|
|||
|
CreateDirectoryW(dataPath, 0);
|
|||
|
|
|||
|
if (SUCCEEDED(playlistMgr->Initialize(cddb, dataPath)))
|
|||
|
{
|
|||
|
playlistMgr->DownloadCorrelates(0);
|
|||
|
playlistInitialized = true;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
playlistMgr->Release();
|
|||
|
playlistMgr=0;
|
|||
|
}
|
|||
|
}
|
|||
|
cddb->Release();
|
|||
|
return playlistMgr;
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
/// Caller is responsible for freeing the returned BSTR !!!
|
|||
|
BSTR SetAndCreatePath(const wchar_t *node)
|
|||
|
{
|
|||
|
wchar_t path_to_create[MAX_PATH] = {0};
|
|||
|
|
|||
|
BSTR bPath = 0;
|
|||
|
|
|||
|
PathCombineW(path_to_create, WASABI_API_APP->path_getUserSettingsPath(), L"Plugins");
|
|||
|
CreateDirectoryW(path_to_create, 0);
|
|||
|
PathAppendW(path_to_create, node);
|
|||
|
CreateDirectoryW(path_to_create, 0);
|
|||
|
|
|||
|
bPath = SysAllocString(path_to_create);
|
|||
|
// modified path as return value
|
|||
|
return bPath;
|
|||
|
}
|
|||
|
|
|||
|
/// This has superceded the old GetPlaylistManager
|
|||
|
/// Returns both a playlist manager in 'playlistMgr' and an mldb manager in 'mldbMgr'
|
|||
|
//int GracenoteApi::GetPlaylistManagerWithMLDBManager(ICddbPlaylist25Mgr **playlistMgr, ICddbMLDBManager **mldbMgr)
|
|||
|
int GracenoteApi::GetPlaylistManager(ICddbPlaylist25Mgr **playlistMgr, ICddbMLDBManager **mldbMgr)
|
|||
|
{
|
|||
|
ICddbPlaylist25Mgr *playlistMgrCreated;
|
|||
|
ICddbMLDBManager *mldbMgrCreated;
|
|||
|
|
|||
|
ICDDBControl2 *cddb = GetCDDB();
|
|||
|
if (!cddb)
|
|||
|
return 0;
|
|||
|
|
|||
|
// Create the mldb manager
|
|||
|
CoCreateInstance(__uuidof(CddbMLDBManager), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&mldbMgrCreated);
|
|||
|
if (mldbMgrCreated)
|
|||
|
{
|
|||
|
//PL_MLDB_FLAGS_ZERO 0x00000000 Used for initialization.
|
|||
|
//PL_MLDB_CHECK_BASE 0x00000001 Causes CheckDB to verify the consistency of the MLDB data file.
|
|||
|
//PL_MLDB_CHECK_INDEX 0x00000002 Causes CheckDB to verify the consistency of the MLDB index file.
|
|||
|
//PL_MLDB_CHECK_DEEP 0x00000004 Causes CheckDB to perform a thorough check of the MLDB file(s).
|
|||
|
//PL_MLDB_CHECK_DEFAULT 0x00000007 Default operation of CheckDB.
|
|||
|
//PL_MLDB_CLEAR_INIT_FLAG 0x00000010 Causes ModifyInitFlag to remove the file indicating Playlist is initialized.
|
|||
|
//PL_MLDB_SET_INIT_FLAG 0x00000020 Causes ModifyInitFlag to create the file indicating Playlist is initialized.
|
|||
|
//PL_MLDB_BACKUP_BASE 0x00000100 Causes BackupDB to create a backup copy of the MLDB data file.
|
|||
|
//PL_MLDB_BACKUP_INDEX 0x00000200 Causes BackupDB to create a backup copy of the MLDB index file.
|
|||
|
//PL_MLDB_RESTORE_BASE 0x00000400 Causes RestoreDB to restore the MLDB data file from the backup copy.
|
|||
|
//PL_MLDB_RESTORE_INDEX 0x00000800 Causes RestoreDB to restore the MLDB index file from the backup copy.
|
|||
|
//PL_MLDB_DELETE_INDEX 0x00001000 Causes DeleteDBFiles to remove the MLDB index file.
|
|||
|
//PL_MLDB_DELETE_BASE 0x00002000 Causes DeleteDBFiles to remove the MLDB data file.
|
|||
|
//PL_MLDB_DELETE_BACKUPS 0x00004000 Causes DeleteDBFiles to remove the MLDBbackup files.
|
|||
|
//PL_MLDB_DELETE_OTHER 0x00008000 Causes DeleteDBFiles to remove the other (non-MLDB) files used by Playlist.
|
|||
|
//PL_MLDB_AUTO_REINDEX 0x00010000 When specified in SetOptions, will cause theindex file to be automatically rebuilt atinitialization if it is corrupt.
|
|||
|
//PL_MLDB_AUTO_BACKUP 0x00020000 When specified in SetOptions, will cause the MLDB files to be automatically backed up at shutdown (if they have been modified). PL_MLDB_AUTO_MANAGE_INIT_FLAG 0x00040000 When specified in SetOptions, will cause the <20>init file<6C> to be managed automatically (created at initialization, deleted at shut down).
|
|||
|
//PL_MLDB_AUTO_CHECK_IF_INIT_SET 0x00080000 When specified in SetOptions, will cause the MLDB files to be check at initialization if the <20>init file<6C> exists (meaning shut down wasn<73>t called).
|
|||
|
//PL_MLDB_AUTO_CHECK_AT_INIT 0x00100000 When specified in SetOptions, will cause the MLDB files to be checked always at initialization.
|
|||
|
//PL_MLDB_AUTO_DEFAULT 0x000C0000 The default automatic behavior if no flags are specified with SetOptions.
|
|||
|
//PL_MLDB_DEVICE_MLDB_42 0x01000000 Enable Gracenote Device SDK 4.2 compatibility for MLDB, list, and correlates files
|
|||
|
|
|||
|
//long autoFlags = PL_MLDB_AUTO_DEFAULT;
|
|||
|
//long autoFlags = PL_MLDB_AUTO_REINDEX | PL_MLDB_AUTO_BACKUP | PL_MLDB_AUTO_MANAGE_INIT_FLAG | PL_MLDB_AUTO_CHECK_IF_INIT_SET | PL_MLDB_AUTO_CHECK_AT_INIT;
|
|||
|
long autoFlags = PL_MLDB_AUTO_REINDEX | PL_MLDB_AUTO_BACKUP | PL_MLDB_AUTO_MANAGE_INIT_FLAG | PL_MLDB_AUTO_CHECK_IF_INIT_SET;
|
|||
|
BSTR bDataPath = SetAndCreatePath(L"Gracenote");
|
|||
|
BSTR bBackupPath = SetAndCreatePath(L"Gracenote/Backup");
|
|||
|
|
|||
|
mldbMgrCreated->AddRef();
|
|||
|
mldbMgrCreated->SetOptions(autoFlags, bBackupPath);
|
|||
|
|
|||
|
|
|||
|
CoCreateInstance(__uuidof(CddbNSWinampPlaylist2Mgr), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&playlistMgrCreated);
|
|||
|
if (playlistMgrCreated)
|
|||
|
{
|
|||
|
playlistMgrCreated->AddRef();
|
|||
|
|
|||
|
|
|||
|
|
|||
|
mldbMgrCreated->Attach(playlistMgrCreated); // Attach the MLDB manager to the playlistMgr
|
|||
|
|
|||
|
if (SUCCEEDED(playlistMgrCreated->Initialize(cddb, bDataPath)))
|
|||
|
{
|
|||
|
playlistMgrCreated->DownloadCorrelates(0);
|
|||
|
playlistInitialized = true;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
playlistMgrCreated->Release();
|
|||
|
playlistMgrCreated=0;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
SysFreeString(bDataPath);
|
|||
|
SysFreeString(bBackupPath);
|
|||
|
}
|
|||
|
cddb->Release();
|
|||
|
|
|||
|
*mldbMgr = mldbMgrCreated;
|
|||
|
*playlistMgr = playlistMgrCreated;
|
|||
|
|
|||
|
if (mldbMgr && playlistMgr)
|
|||
|
return NErr_Success;
|
|||
|
else
|
|||
|
return NErr_FailedCreate;
|
|||
|
}
|
|||
|
|
|||
|
/// Dont really have to use this, get the MLDB manager when creating the playlist manager
|
|||
|
ICddbMLDBManager *GracenoteApi::GetMLDBManager()
|
|||
|
{
|
|||
|
ICddbMLDBManager *mldbMgr;
|
|||
|
|
|||
|
ICDDBControl2 *cddb = GetCDDB();
|
|||
|
if (!cddb)
|
|||
|
return 0;
|
|||
|
|
|||
|
CoCreateInstance(__uuidof(CddbMLDBManager), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&mldbMgr);
|
|||
|
if (mldbMgr)
|
|||
|
{
|
|||
|
mldbMgr->AddRef();
|
|||
|
}
|
|||
|
cddb->Release();
|
|||
|
return mldbMgr;
|
|||
|
}
|
|||
|
|
|||
|
HRESULT GracenoteApi::CreateFingerprint(ICDDBMusicIDManager *musicID, api_decodefile *decodeApi, ICddbFileInfo *info, const wchar_t *filename, long *killswitch)
|
|||
|
{
|
|||
|
if (!musicID || !decodeApi)
|
|||
|
return E_FAIL;
|
|||
|
|
|||
|
ICddbMusicIDFingerprinterPtr fingerprinter;
|
|||
|
musicID->CreateFingerprinter(NULL, &fingerprinter);
|
|||
|
|
|||
|
AudioParameters parameters;
|
|||
|
parameters.bitsPerSample = 16;
|
|||
|
parameters.channels = 2;
|
|||
|
parameters.sampleRate = 44100;
|
|||
|
ifc_audiostream *decoder = decodeApi->OpenAudioBackground(filename, ¶meters);
|
|||
|
if (decoder)
|
|||
|
{
|
|||
|
HRESULT hr = fingerprinter->BeginAudioStream((long)parameters.sampleRate, (long)parameters.bitsPerSample, (long)parameters.channels);
|
|||
|
char data[65536] = {0};
|
|||
|
size_t decodeSize;
|
|||
|
int decode_killswitch=0, decode_error;
|
|||
|
while (decodeSize = decoder->ReadAudio((void *)data, sizeof(data), &decode_killswitch, &decode_error))
|
|||
|
{
|
|||
|
if (decodeSize > LONG_MAX) // I _really_ doubt this is going to happen, but just in case, since we cast down to a long
|
|||
|
break;
|
|||
|
|
|||
|
if (*killswitch)
|
|||
|
break;
|
|||
|
hr = fingerprinter->WriteAudioData(data, (long)decodeSize);
|
|||
|
if (hr == CDDBMusicID_FPAcquired)
|
|||
|
break;
|
|||
|
}
|
|||
|
ICddbMusicIDFingerprintPtr fingerprint;
|
|||
|
fingerprinter->EndAudioStream(&fingerprint);
|
|||
|
decodeApi->CloseAudio(decoder);
|
|||
|
|
|||
|
hr=info->put_Fingerprint(fingerprint);
|
|||
|
return S_OK;
|
|||
|
}
|
|||
|
return E_FAIL;
|
|||
|
}
|
|||
|
|
|||
|
ICDDBMusicIDManager3 *GracenoteApi::GetMusicID()
|
|||
|
{
|
|||
|
ICDDBControl2 *cddb = GetCDDB();
|
|||
|
if (!cddb)
|
|||
|
return 0;
|
|||
|
ICDDBMusicIDManager3 *musicID;
|
|||
|
CoCreateInstance(__uuidof(CDDBNSWinampMusicIDManager), 0, CLSCTX_INPROC_SERVER, IID_IUnknown, (void **)&musicID);
|
|||
|
if (musicID)
|
|||
|
musicID->Initialize(cddb);
|
|||
|
cddb->Release();
|
|||
|
return musicID;
|
|||
|
}
|
|||
|
|
|||
|
#define CBCLASS GracenoteApi
|
|||
|
START_DISPATCH;
|
|||
|
CB(API_GRACENOTE_GETCDDB, GetCDDB)
|
|||
|
CB(API_GRACENOTE_GETMUSICID, GetMusicID)
|
|||
|
CB(API_GRACENOTE_GETPLAYLISTMGR, GetPlaylistManager)
|
|||
|
//CB(API_GRACENOTE_GETPLAYLISTMGRWITHMLDBMGR, GetPlaylistManagerWithMLDBManager)
|
|||
|
CB(API_GRACENOTE_GETMLDBMGR, GetMLDBManager)
|
|||
|
CB(API_GRACENOTE_CREATEFINGERPRINT, CreateFingerprint)
|
|||
|
END_DISPATCH;
|
|||
|
#undef CBCLASS
|