winamp/Src/Plugins/Input/in_wv/in_wv.cpp
2024-09-24 14:54:57 +02:00

2062 lines
55 KiB
C++

/*
** .WV input plug-in for WavPack
** Copyright (c) 2000 - 2006, Conifer Software, All Rights Reserved
*/
#include <windows.h>
#include <fcntl.h>
#include <stdio.h>
#include <mmreg.h>
#include <msacm.h>
#include <math.h>
#include <sys/stat.h>
#include <io.h>
#include <strsafe.h>
#include "in2.h"
#include "wavpack.h"
#include "resource.h"
#include "wasabi/winamp/wa_ipc.h"
#include "wasabi/wasabi.h"
#include "wasabi/nu/autochar.h"
#include "wasabi/nu/autowide.h"
#define fileno _fileno
static float calculate_gain(WavpackContext *wpc, bool allowDefault=true);
#define PLUGIN_VERSION "2.8.1"
//#define DEBUG_CONSOLE
//#define ANSI_METADATA
#define UNICODE_METADATA
//#define OLD_INFO_DIALOG
// post this to the main window at end of file (after playback as stopped)
#define WM_WA_MPEG_EOF WM_USER+2
#define MAX_NCH 8
static struct wpcnxt {
WavpackContext *wpc; // WavPack context provided by library
float play_gain; // playback gain (for replaygain support)
//int soft_clipping; // soft clipping active for playback
int output_bits; // 16, 24, or 32 bits / sample
long sample_buffer[576*MAX_NCH*2]; // sample buffer
float error [MAX_NCH]; // error term for noise shaping
char lastfn[MAX_PATH]; // filename stored for comparisons only
wchar_t w_lastfn[MAX_PATH]; // w_filename stored for comparisons only
FILE *wv_id, *wvc_id; // file pointer when we use reader callbacks
} curr, edit, info;
int decode_pos_ms; // current decoding position, in milliseconds
int paused; // are we paused?
int seek_needed; // if != -1, it is the point that the decode thread should seek to, in ms.
#define ALLOW_WVC 0x1
#define REPLAYGAIN_TRACK 0x2
#define REPLAYGAIN_ALBUM 0x4
#define SOFTEN_CLIPPING 0x8
#define PREVENT_CLIPPING 0x10
#define ALWAYS_16BIT 0x20 // new flags added for version 2.5
#define ALLOW_MULTICHANNEL 0x40
#define REPLAYGAIN_24BIT 0x80
int config_bits = ALLOW_WVC | ALLOW_MULTICHANNEL; // all configuration goes here
int killDecodeThread=0; // the kill switch for the decode thread
HANDLE thread_handle=INVALID_HANDLE_VALUE; // the handle to the decode thread
DWORD WINAPI __stdcall DecodeThread(void *b); // the decode thread procedure
static api_service* WASABI_API_SVC;
static api_language* WASABI_API_LNG;
static api_config *AGAVE_API_CONFIG;
static api_memmgr *WASABI_API_MEMMGR;
HINSTANCE WASABI_API_LNG_HINST = 0, WASABI_API_ORIG_HINST = 0;
static char file_extensions[128] = {"WV\0WavPack Files (*.WV)\0"};
// function definitions for the In_Module stucture
void about (HWND hwndParent);
void init();
void quit();
void getfileinfo(const char *filename, char *title, int *length_in_ms);
int infoDlg(const char *fn, HWND hwnd);
int isourfile(const char *fn);
int play(const char *fn);
void pause();
void unpause();
int ispaused();
void stop();
int getlength();
int getoutputtime();
void setoutputtime(int time_in_ms);
void setvolume(int volume);
void setpan(int pan);
void eq_set(int on, char data [10], int preamp);
In_Module mod = // the output module
{
IN_VER,
"WavPack Decoder v"PLUGIN_VERSION,
0, // hMainWindow
0, // hDllInstance
file_extensions,
1, // is_seekable
1, // uses output
about,
about,
init,
quit,
getfileinfo,
infoDlg,
isourfile,
play,
pause,
unpause,
ispaused,
stop,
getlength,
getoutputtime,
setoutputtime,
setvolume,
setpan,
0,0,0,0,0,0,0,0,0, // vis stuff
0,0, // dsp
eq_set,
NULL, // setinfo
0 // out_mod
};
static BOOL CALLBACK WavPackDlgProc(HWND, UINT, WPARAM, LPARAM);
extern long dump_alloc (void);
int DoAboutMessageBox(HWND parent, wchar_t* title, wchar_t* message)
{
MSGBOXPARAMSW msgbx = {sizeof(MSGBOXPARAMSW),0};
msgbx.lpszText = message;
msgbx.lpszCaption = title;
msgbx.lpszIcon = MAKEINTRESOURCEW(102);
msgbx.hInstance = GetModuleHandle(0);
msgbx.dwStyle = MB_USERICON;
msgbx.hwndOwner = parent;
return MessageBoxIndirectW(&msgbx);
}
void about(HWND hwndParent)
{
wchar_t about_string[512];
#ifdef DEBUG_ALLOC
sprintf (about_string, "alloc_count = %d", dump_alloc ());
#else
StringCchPrintfW(about_string, 512, WASABI_API_LNGSTRINGW(IDS_ABOUT_MESSAGE), PLUGIN_VERSION, "1998-2010", __DATE__);
#endif
DoAboutMessageBox(hwndParent, WASABI_API_LNGSTRINGW(IDS_ABOUT), about_string);
}
void init() /* any one-time initialization goes here (configuration reading, etc) */
{
if (mod.hMainWindow)
{
// load all of the required wasabi services from the winamp client
WASABI_API_SVC = (api_service *)SendMessage(mod.hMainWindow, WM_WA_IPC, 0, IPC_GET_API_SERVICE);
if (WASABI_API_SVC == (api_service *)1)
WASABI_API_SVC=0;
WASABI_API_SVC->service_register(&albumArtFactory);
}
ServiceBuild(AGAVE_API_CONFIG, AgaveConfigGUID);
ServiceBuild(WASABI_API_LNG, languageApiGUID);
ServiceBuild(WASABI_API_MEMMGR, memMgrApiServiceGuid);
// need to have this initialised before we try to do anything with localisation features
WASABI_API_START_LANG(mod.hDllInstance,InWvLangGuid);
static char szDescription[256];
StringCchPrintfA(szDescription,256,WASABI_API_LNGSTRING(IDS_DESCRIPTION),PLUGIN_VERSION);
mod.description = szDescription;
// set the file extension to the localised version
char tmp [64], *tmp_ptr = tmp, *fex_ptr = file_extensions;
WASABI_API_LNGSTRING_BUF(IDS_FILETYPE, tmp, sizeof (tmp));
*fex_ptr++ = 'W';
*fex_ptr++ = 'V';
*fex_ptr++ = 0;
while (*tmp_ptr)
*fex_ptr++ = *tmp_ptr++;
*fex_ptr++ = 0;
*fex_ptr++ = 0;
}
#ifdef DEBUG_CONSOLE
HANDLE debug_console=INVALID_HANDLE_VALUE; // debug console
void debug_write (char *str)
{
static int cant_debug;
if (cant_debug)
return;
if (debug_console == INVALID_HANDLE_VALUE) {
AllocConsole ();
#if 1
debug_console = GetStdHandle (STD_OUTPUT_HANDLE);
#else
debug_console = CreateConsoleScreenBuffer (GENERIC_WRITE, FILE_SHARE_WRITE,
NULL, CONSOLE_TEXTMODE_BUFFER, NULL);
#endif
if (debug_console == INVALID_HANDLE_VALUE) {
MessageBox(NULL, "Can't get a console handle", "WavPack",MB_OK);
cant_debug = 1;
return;
}
else if (!SetConsoleActiveScreenBuffer (debug_console)) {
MessageBox(NULL, "Can't activate console buffer", "WavPack",MB_OK);
cant_debug = 1;
return;
}
}
WriteConsole (debug_console, str, strlen (str), NULL, NULL);
}
#endif
void quit() /* one-time deinit, such as memory freeing */
{
#ifdef DEBUG_CONSOLE
if (debug_console != INVALID_HANDLE_VALUE) {
FreeConsole ();
if (debug_console != GetStdHandle (STD_OUTPUT_HANDLE))
CloseHandle (debug_console);
debug_console = INVALID_HANDLE_VALUE;
}
#endif
ServiceRelease(AGAVE_API_CONFIG, AgaveConfigGUID);
ServiceRelease(WASABI_API_LNG, languageApiGUID);
ServiceRelease(WASABI_API_MEMMGR, memMgrApiServiceGuid);
WASABI_API_SVC->service_deregister(&albumArtFactory);
}
// used for detecting URL streams.. unused here. strncmp(fn,"http://",7) to detect HTTP streams, etc
int isourfile(const char *fn)
{
return 0;
}
int play(const char *fn)
{
int num_chans, sample_rate;
char error[128];
int maxlatency;
int thread_id;
int open_flags;
#ifdef DEBUG_CONSOLE
sprintf (error, "play (%s)\n", fn);
debug_write (error);
#endif
open_flags = OPEN_TAGS | OPEN_NORMALIZE;
if (config_bits & ALLOW_WVC)
open_flags |= OPEN_WVC;
if (!(config_bits & ALLOW_MULTICHANNEL))
open_flags |= OPEN_2CH_MAX;
curr.wpc = WavpackOpenFileInput(fn, error, open_flags, 0);
if (!curr.wpc) // error opening file, just return error
return -1;
num_chans = WavpackGetReducedChannels(curr.wpc);
sample_rate = WavpackGetSampleRate(curr.wpc);
curr.output_bits = WavpackGetBitsPerSample(curr.wpc) > 16 ? 24 : 16;
if (config_bits & ALWAYS_16BIT)
curr.output_bits = 16;
else if ((config_bits & (REPLAYGAIN_TRACK | REPLAYGAIN_ALBUM)) &&
(config_bits & REPLAYGAIN_24BIT))
curr.output_bits = 24;
if (num_chans > MAX_NCH) // don't allow too many channels!
{
WavpackCloseFile(curr.wpc);
return -1;
}
curr.play_gain = calculate_gain(curr.wpc);
lstrcpyn(curr.lastfn, fn, MAX_PATH);
paused = 0;
decode_pos_ms = 0;
seek_needed = -1;
maxlatency = mod.outMod->Open(sample_rate, num_chans, curr.output_bits, -1, -1);
if (maxlatency < 0) // error opening device
{
curr.wpc = WavpackCloseFile(curr.wpc);
return -1;
}
// dividing by 1000 for the first parameter of setinfo makes it
// display 'H'... for hundred.. i.e. 14H Kbps.
mod.SetInfo(0, (sample_rate + 500) / 1000, num_chans, 1);
// initialize vis stuff
mod.SAVSAInit(maxlatency, sample_rate);
mod.VSASetInfo(sample_rate, num_chans);
mod.outMod->SetVolume(-666); // set the output plug-ins default volume
killDecodeThread=0;
thread_handle = (HANDLE)CreateThread (NULL, 0, (LPTHREAD_START_ROUTINE) DecodeThread, (void *) &killDecodeThread, 0, (LPDWORD)&thread_id);
if (SetThreadPriority(thread_handle, THREAD_PRIORITY_HIGHEST) == 0) {
curr.wpc = WavpackCloseFile(curr.wpc);
return -1;
}
return 0;
}
void pause()
{
#ifdef DEBUG_CONSOLE
debug_write ("pause ()\n");
#endif
paused = 1;
mod.outMod->Pause(1);
}
void unpause()
{
#ifdef DEBUG_CONSOLE
debug_write ("unpause ()\n");
#endif
paused = 0;
mod.outMod->Pause(0);
}
int ispaused()
{
return paused;
}
void stop()
{
#ifdef DEBUG_CONSOLE
debug_write ("stop ()\n");
#endif
if (thread_handle != INVALID_HANDLE_VALUE)
{
killDecodeThread = 1;
if (WaitForSingleObject(thread_handle, INFINITE) == WAIT_TIMEOUT)
{
MessageBox(mod.hMainWindow,"error asking thread to die!\n", "error killing decode thread", 0);
TerminateThread(thread_handle,0);
}
CloseHandle(thread_handle);
thread_handle = INVALID_HANDLE_VALUE;
}
if (curr.wpc)
curr.wpc = WavpackCloseFile(curr.wpc);
mod.outMod->Close();
mod.SAVSADeInit();
}
int getlength()
{
return (int)(WavpackGetNumSamples (curr.wpc) * 1000.0 / WavpackGetSampleRate (curr.wpc));
}
int getoutputtime()
{
if (seek_needed == -1)
return decode_pos_ms + (mod.outMod->GetOutputTime () - mod.outMod->GetWrittenTime ());
else
return seek_needed;
}
void setoutputtime (int time_in_ms)
{
#ifdef DEBUG_CONSOLE
char str [40];
sprintf (str, "setoutputtime (%d)\n", time_in_ms);
debug_write (str);
#endif
seek_needed = time_in_ms;
}
void setvolume (int volume)
{
mod.outMod->SetVolume(volume);
}
void setpan (int pan)
{
mod.outMod->SetPan(pan);
}
static void generate_format_string(WavpackContext *wpc, wchar_t *string, int maxlen, int wide);
static void AnsiToUTF8(char *string, int len);
static int UTF8ToWideChar(const char *pUTF8, wchar_t *pWide);
static void UTF8ToAnsi(char *string, int len);
int infoDlg(const char *fn, HWND hwnd)
{
#ifdef OLD_INFO_DIALOG
char string[2048];
wchar_t w_string[2048];
WavpackContext *wpc;
int open_flags;
open_flags = OPEN_TAGS | OPEN_NORMALIZE;
if (config_bits & ALLOW_WVC)
open_flags |= OPEN_WVC;
if (!(config_bits & ALLOW_MULTICHANNEL))
open_flags |= OPEN_2CH_MAX;
wpc = WavpackOpenFileInput(fn, string, open_flags, 0);
if (wpc)
{
int mode = WavpackGetMode(wpc);
//generate_format_string(wpc, string, sizeof (string), 1);
wchar_t *temp = (wchar_t *)malloc(sizeof(string) * sizeof(wchar_t));
generate_format_string(wpc, temp, sizeof(string), 0);
lstrcpyn(string, AutoChar(temp, CP_UTF8), sizeof(string));
free(temp);
if (WavpackGetMode(wpc) & MODE_VALID_TAG)
{
char value [128];
if (config_bits & (REPLAYGAIN_TRACK | REPLAYGAIN_ALBUM))
{
int local_clipping = 0;
float local_gain;
local_gain = calculate_gain(wpc);
if (local_gain != 1.0)
StringCchPrintf(string + strlen (string), 2048, "Gain: %+.2f dB %s\n",
log10 (local_gain) * 20.0, local_clipping ? "(w/soft clipping)" : "");
}
if (WavpackGetTagItem(wpc, "title", value, sizeof (value)))
{
if (!(mode & MODE_APETAG))
AnsiToUTF8(value, sizeof (value));
StringCchPrintf(string + strlen (string), 2048, "\nTitle: %s", value);
}
if (WavpackGetTagItem(wpc, "artist", value, sizeof (value)))
{
if (!(mode & MODE_APETAG))
AnsiToUTF8(value, sizeof (value));
StringCchPrintf(string + strlen (string), 2048, "\nArtist: %s", value);
}
if (WavpackGetTagItem (wpc, "album", value, sizeof (value)))
{
if (!(mode & MODE_APETAG))
AnsiToUTF8(value, sizeof (value));
StringCchPrintf (string + strlen (string), 2048, "\nAlbum: %s", value);
}
if (WavpackGetTagItem (wpc, "genre", value, sizeof (value)))
{
if (!(mode & MODE_APETAG))
AnsiToUTF8(value, sizeof (value));
StringCchPrintf(string + strlen (string), 2048, "\nGenre: %s", value);
}
if (WavpackGetTagItem (wpc, "comment", value, sizeof (value)))
{
if (!(mode & MODE_APETAG))
AnsiToUTF8(value, sizeof (value));
StringCchPrintf(string + strlen (string), 2048, "\nComment: %s", value);
}
if (WavpackGetTagItem(wpc, "year", value, sizeof (value)))
StringCchPrintf(string + strlen (string), 2048, "\nYear: %s", value);
if (WavpackGetTagItem(wpc, "track", value, sizeof (value)))
StringCchPrintf(string + strlen (string), 2048, "\nTrack: %s", value);
StringCchCat(string, 2048, "\n");
}
UTF8ToWideChar(string, w_string);
MessageBoxW(hwnd, w_string, L"WavPack File Info Box", MB_OK);
wpc = WavpackCloseFile(wpc);
}
else
MessageBox(hwnd, string, "WavPack Decoder", MB_OK);
return 0;
#else
return 1;
#endif
}
void getfileinfo(const char *filename, char *title, int *length_in_ms)
{
if (!filename || !*filename) // currently playing file
{
if (length_in_ms)
*length_in_ms = getlength ();
if (title)
{
if (WavpackGetTagItem(curr.wpc, "title", NULL, 0))
{
char art [128], ttl [128];
WavpackGetTagItem(curr.wpc, "title", ttl, sizeof (ttl));
if (WavpackGetMode(curr.wpc) & MODE_APETAG)
UTF8ToAnsi(ttl, sizeof (ttl));
if (WavpackGetTagItem(curr.wpc, "artist", art, sizeof (art)))
{
if (WavpackGetMode(curr.wpc) & MODE_APETAG)
UTF8ToAnsi(art, sizeof (art));
StringCchPrintf(title, GETFILEINFO_TITLE_LENGTH, "%s - %s", art, ttl);
}
else
lstrcpyn(title, ttl, GETFILEINFO_TITLE_LENGTH);
}
else
{
char *p = curr.lastfn + strlen (curr.lastfn);
while (*p != '\\' && p >= curr.lastfn)
p--;
lstrcpyn(title, ++p, GETFILEINFO_TITLE_LENGTH);
}
}
}
else // some other file
{
WavpackContext *wpc;
char error [128];
int open_flags;
if (length_in_ms)
*length_in_ms = -1000;
if (title)
*title = 0;
open_flags = OPEN_TAGS | OPEN_NORMALIZE;
if (config_bits & ALLOW_WVC)
open_flags |= OPEN_WVC;
if (!(config_bits & ALLOW_MULTICHANNEL))
open_flags |= OPEN_2CH_MAX;
wpc = WavpackOpenFileInput(filename, error, open_flags, 0);
if (wpc)
{
if (length_in_ms)
*length_in_ms = (int)(WavpackGetNumSamples(wpc) * 1000.0 / WavpackGetSampleRate(wpc));
if (title && WavpackGetTagItem(wpc, "title", NULL, 0))
{
char art [128], ttl [128];
WavpackGetTagItem(wpc, "title", ttl, sizeof (ttl));
if (WavpackGetMode(wpc) & MODE_APETAG)
UTF8ToAnsi(ttl, sizeof (ttl));
if (WavpackGetTagItem(wpc, "artist", art, sizeof (art)))
{
if (WavpackGetMode(wpc) & MODE_APETAG)
UTF8ToAnsi(art, sizeof (art));
StringCchPrintf(title, GETFILEINFO_TITLE_LENGTH, "%s - %s", art, ttl);
}
else
lstrcpyn(title, ttl, GETFILEINFO_TITLE_LENGTH);
}
wpc = WavpackCloseFile(wpc);
}
if (title && !*title)
{
char *p = (char*)filename + strlen (filename);
while (*p != '\\' && p >= filename) p--;
lstrcpyn(title, ++p, GETFILEINFO_TITLE_LENGTH);
}
}
}
void eq_set(int on, char data [10], int preamp)
{
// most plug-ins can't even do an EQ anyhow.. I'm working on writing
// a generic PCM EQ, but it looks like it'll be a little too CPU
// consuming to be useful :)
}
static int read_samples(struct wpcnxt *cnxt, int num_samples);
DWORD WINAPI __stdcall DecodeThread(void *b)
{
int num_chans, sample_rate;
int done = 0;
memset(curr.error, 0, sizeof (curr.error));
num_chans = WavpackGetReducedChannels(curr.wpc);
sample_rate = WavpackGetSampleRate(curr.wpc);
while (!*((int *)b) )
{
if (seek_needed != -1)
{
int seek_position = seek_needed;
int bc = 0;
seek_needed = -1;
if (seek_position > getlength() - 1000 && getlength() > 1000)
seek_position = getlength() - 1000; // don't seek to last second
mod.outMod->Flush(decode_pos_ms = seek_position);
if (WavpackSeekSample(curr.wpc, (int)(sample_rate / 1000.0 * seek_position))) {
decode_pos_ms = (int)(WavpackGetSampleIndex(curr.wpc) * 1000.0 / sample_rate);
mod.outMod->Flush(decode_pos_ms);
continue;
}
else
done = 1;
}
if (done) {
mod.outMod->CanWrite();
if (!mod.outMod->IsPlaying()) {
PostMessage(mod.hMainWindow, WM_WA_MPEG_EOF, 0, 0);
return 0;
}
Sleep(10);
}
else if (mod.outMod->CanWrite() >= ((576 * num_chans * (curr.output_bits / 8)) << (mod.dsp_isactive () ? 1 : 0)))
{
int tsamples = read_samples (&curr, 576) * num_chans;
int tbytes = tsamples * (curr.output_bits/8);
if (tsamples)
{
mod.SAAddPCMData((char *) curr.sample_buffer, num_chans, curr.output_bits, decode_pos_ms);
mod.VSAAddPCMData((char *) curr.sample_buffer, num_chans, curr.output_bits, decode_pos_ms);
decode_pos_ms = (int)(WavpackGetSampleIndex(curr.wpc) * 1000.0 / sample_rate);
if (mod.dsp_isactive())
tbytes = mod.dsp_dosamples ((short *) curr.sample_buffer,
tsamples / num_chans, curr.output_bits, num_chans, sample_rate) * (num_chans * (curr.output_bits/8));
mod.outMod->Write ((char *) curr.sample_buffer, tbytes);
}
else
done = 1;
}
else
{
mod.SetInfo((int) ((WavpackGetInstantBitrate (curr.wpc) + 500.0) / 1000.0), -1, -1, 1);
Sleep(20);
}
}
return 0;
}
/********* These functions provide the "transcoding" mode of winamp. *********/
extern "C" __declspec (dllexport) intptr_t winampGetExtendedRead_open (const char *fn, int *size, int *bps, int *nch, int *srate)
{
struct wpcnxt *cnxt = (struct wpcnxt *)malloc(sizeof (struct wpcnxt));
int num_chans, sample_rate, open_flags;
char error[128];
#ifdef DEBUG_CONSOLE
sprintf (error, "Read_open (%s)\n", fn);
debug_write (error);
#endif
if (!cnxt)
return 0;
memset(cnxt, 0, sizeof (struct wpcnxt));
open_flags = OPEN_NORMALIZE | OPEN_WVC;
if (!(config_bits & ALLOW_MULTICHANNEL) || *nch == 2)
open_flags |= OPEN_2CH_MAX;
if (config_bits & (REPLAYGAIN_TRACK | REPLAYGAIN_ALBUM))
open_flags |= OPEN_TAGS;
cnxt->wpc = WavpackOpenFileInput(fn, error, open_flags, 0);
if (!cnxt->wpc) // error opening file, just return error
{
free (cnxt);
return 0;
}
num_chans = WavpackGetReducedChannels(cnxt->wpc);
sample_rate = WavpackGetSampleRate(cnxt->wpc);
if (num_chans > MAX_NCH)
{
WavpackCloseFile(cnxt->wpc);
free (cnxt);
return 0;
}
if (*bps != 16 && *bps != 24 && *bps != 32)
{
cnxt->output_bits = WavpackGetBitsPerSample(cnxt->wpc) > 16 ? 24 : 16;
if (config_bits & ALWAYS_16BIT)
cnxt->output_bits = 16;
else if ((config_bits & (REPLAYGAIN_TRACK | REPLAYGAIN_ALBUM)) &&
(config_bits & REPLAYGAIN_24BIT))
cnxt->output_bits = 24;
}
else
cnxt->output_bits = *bps;
if (num_chans > MAX_NCH) // don't allow too many channels!
{
WavpackCloseFile(cnxt->wpc);
free (cnxt);
return 0;
}
*nch = num_chans;
*srate = sample_rate;
*bps = cnxt->output_bits;
*size = WavpackGetNumSamples(cnxt->wpc) * (*bps / 8) * (*nch);
cnxt->play_gain = calculate_gain(cnxt->wpc);
#ifdef DEBUG_CONSOLE
sprintf (error, "Read_open success! nch=%d, srate=%d, bps=%d, size=%d\n",
*nch, *srate, *bps, *size);
debug_write (error);
#endif
return (intptr_t) cnxt;
}
extern "C" __declspec (dllexport) intptr_t winampGetExtendedRead_getData (intptr_t handle, char *dest, int len, int *killswitch)
{
struct wpcnxt *cnxt = (struct wpcnxt *)handle;
int num_chans = WavpackGetReducedChannels(cnxt->wpc);
int bytes_per_sample = num_chans * cnxt->output_bits / 8;
int used = 0;
#ifdef DEBUG_CONSOLE
char error [128];
#endif
while (used < len && !*killswitch)
{
int nsamples = (len - used) / bytes_per_sample, tsamples;
if (!nsamples)
break;
else if (nsamples > 576)
nsamples = 576;
tsamples = read_samples(cnxt, nsamples) * num_chans;
if (tsamples)
{
int tbytes = tsamples * (cnxt->output_bits/8);
memcpy (dest + used, cnxt->sample_buffer, tbytes);
used += tbytes;
}
else
break;
}
#ifdef DEBUG_CONSOLE
sprintf (error, "Read_getData (%d), actualy read %d\n", len, used);
debug_write (error);
#endif
return used;
}
extern "C" __declspec (dllexport) int winampGetExtendedRead_setTime (intptr_t handle, int millisecs)
{
struct wpcnxt *cnxt = (struct wpcnxt *) handle;
int sample_rate = WavpackGetSampleRate(cnxt->wpc);
return WavpackSeekSample(cnxt->wpc, (int)(sample_rate / 1000.0 * millisecs));
}
extern "C" __declspec (dllexport) void winampGetExtendedRead_close (intptr_t handle)
{
struct wpcnxt *cnxt = (struct wpcnxt *) handle;
#ifdef DEBUG_CONSOLE
char error [128];
sprintf (error, "Read_close ()\n");
debug_write (error);
#endif
WavpackCloseFile(cnxt->wpc);
free (cnxt);
}
/* This is a generic function to read WavPack samples and convert them to a
* form usable by winamp. It includes conversion of any WavPack format
* (including ieee float) to 16, 24, or 32-bit integers (with noise shaping
* for the 16-bit case) and replay gain implementation (with optional soft
* clipping). It is used by both the regular "play" code and the newer
* transcoding functions.
*
* The num_samples parameter is the number of "composite" samples to
* convert and is limited currently to 576 samples for legacy reasons. The
* return value is the number of samples actually converted and will be
* equal to the number requested unless an error occurs or the end-of-file
* is encountered. The converted samples are stored (interleaved) at
* cnxt->sample_buffer[].
*/
static int read_samples (struct wpcnxt *cnxt, int num_samples)
{
int num_chans = WavpackGetReducedChannels(cnxt->wpc), samples, tsamples;
samples = WavpackUnpackSamples(cnxt->wpc, (int32_t*) cnxt->sample_buffer, num_samples);
tsamples = samples * num_chans;
if (tsamples)
{
if (!(WavpackGetMode(cnxt->wpc) & MODE_FLOAT))
{
float scaler = (float) (1.0 / ((unsigned long) 1 << (WavpackGetBytesPerSample(cnxt->wpc) * 8 - 1)));
float *fptr = (float *) cnxt->sample_buffer;
long *lptr = cnxt->sample_buffer;
int cnt = tsamples;
while (cnt--)
*fptr++ = *lptr++ * scaler;
}
if (cnxt->play_gain != 1.0)
{
float *fptr = (float *) cnxt->sample_buffer;
int cnt = tsamples;
double outval;
while (cnt--)
{
outval = *fptr * cnxt->play_gain;
/*if (cnxt->soft_clipping)
{
if (outval > 0.75)
outval = 1.0 - (0.0625 / (outval - 0.5));
else if (outval < -0.75)
outval = -1.0 - (0.0625 / (outval + 0.5));
}*/
*fptr++ = (float) outval;
}
}
if (cnxt->output_bits == 16)
{
float *fptr = (float *) cnxt->sample_buffer;
short *sptr = (short *) cnxt->sample_buffer;
int cnt = samples, ch;
while (cnt--)
for (ch = 0; ch < num_chans; ++ch)
{
int dst;
*fptr -= cnxt->error [ch];
if (*fptr >= 1.0)
dst = 32767;
else if (*fptr <= -1.0)
dst = -32768;
else
dst = (int) floor (*fptr * 32768.0);
cnxt->error [ch] = (float)(dst / 32768.0 - *fptr++);
*sptr++ = dst;
}
}
else if (cnxt->output_bits == 24)
{
unsigned char *cptr = (unsigned char *) cnxt->sample_buffer;
float *fptr = (float *) cnxt->sample_buffer;
int cnt = tsamples;
long outval;
while (cnt--) {
if (*fptr >= 1.0)
outval = 8388607;
else if (*fptr <= -1.0)
outval = -8388608;
else
outval = (int) floor (*fptr * 8388608.0);
*cptr++ = (unsigned char) outval;
*cptr++ = (unsigned char) (outval >> 8);
*cptr++ = (unsigned char) (outval >> 16);
fptr++;
}
}
else if (cnxt->output_bits == 32)
{
float *fptr = (float *) cnxt->sample_buffer;
long *sptr = (long *) cnxt->sample_buffer;
int cnt = tsamples;
while (cnt--)
{
if (*fptr >= 1.0)
*sptr++ = 8388607 << 8;
else if (*fptr <= -1.0)
*sptr++ = -8388608 << 8;
else
*sptr++ = ((int) floor (*fptr * 8388608.0)) << 8;
fptr++;
}
}
}
return samples;
}
extern "C" __declspec (dllexport) In_Module * winampGetInModule2()
{
return &mod;
}
// This code provides an interface between the reader callback mechanism that
// WavPack uses internally and the standard fstream C library.
static int32_t read_bytes(void *id, void *data, int32_t bcount)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return (int32_t) fread(data, 1, bcount, file);
else
return 0;
}
static uint32_t get_pos(void *id)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return ftell(file);
else
return -1;
}
static int set_pos_abs(void *id, uint32_t pos)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return fseek(file, pos, SEEK_SET);
else
return 0;
}
static int set_pos_rel(void *id, int32_t delta, int mode)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return fseek(file, delta, mode);
else
return -1;
}
static int push_back_byte(void *id, int c)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return ungetc(c, file);
else
return EOF;
}
static uint32_t get_length(void *id)
{
FILE *file = id ? *(FILE**)id : NULL;
struct stat statbuf;
if (!file || fstat (fileno (file), &statbuf) || !(statbuf.st_mode & S_IFREG))
return 0;
else
return statbuf.st_size;
}
static int can_seek(void *id)
{
FILE *file = id ? *(FILE**)id : NULL;
struct stat statbuf;
return file && !fstat (fileno (file), &statbuf) && (statbuf.st_mode & S_IFREG);
}
static int32_t write_bytes(void *id, void *data, int32_t bcount)
{
FILE *file = id ? *(FILE**)id : NULL;
if (file)
return (int32_t) fwrite (data, 1, bcount, file);
else
return 0;
}
static WavpackStreamReader freader = {
read_bytes, get_pos, set_pos_abs,
set_pos_rel, push_back_byte,
get_length, can_seek, write_bytes
};
/* These functions provide UNICODE support for the winamp media library */
static int metadata_we_can_write(const char *metadata);
static void close_context(struct wpcnxt *cxt)
{
if (cxt->wpc)
WavpackCloseFile(cxt->wpc);
if (cxt->wv_id)
fclose(cxt->wv_id);
if (cxt->wvc_id)
fclose(cxt->wvc_id);
memset(cxt, 0, sizeof (*cxt));
}
#ifdef ANSI_METADATA
extern "C" __declspec (dllexport) int winampGetExtendedFileInfo(char *filename, char *metadata, char *ret, int retlen)
{
int open_flags = OPEN_TAGS;
char error[128];
int retval = 0;
#ifdef DEBUG_CONSOLE
sprintf (error, "winampGetExtendedFileInfo (%s)\n", metadata);
debug_write (error);
#endif
if (!_stricmp(metadata, "type"))
{
ret[0] = '0';
ret[1] = 0;
return 1;
}
else if (!_stricmp(metadata, "family"))
{
int len;
const char *p;
if (!filename || !filename[0]) return 0;
len = lstrlen(filename);
if (len < 3 || '.' != filename[len - 3]) return 0;
p = &filename[len - 2];
if (!_stricmp(p, "wv")) { WASABI_API_LNGSTRING_BUF(IDS_FAMILY_STRING, ret, retlen); return 1; }
return 0;
}
if (!filename || !*filename)
return retval;
if (!_stricmp(metadata, "length")) { /* even if no file, return a 1 and write "0" */
StringCchPrintf(ret, retlen, "%d", 0);
retval = 1;
}
if (!info.wpc || strcmp(filename, info.lastfn) || !_stricmp(metadata, "formatinformation"))
{
close_context(&info);
if (!(info.wv_id = fopen(filename, "rb")))
return retval;
if (config_bits & ALLOW_WVC)
{
int length = strlen(filename) + 10;
char *wvc_name = (char *)malloc(length);
if (wvc_name)
{
lstrcpyn(wvc_name, filename, length);
StringCchCat(wvc_name, length, "c");
info.wvc_id = fopen(wvc_name, "rb");
free(wvc_name);
}
}
info.wpc = WavpackOpenFileInputEx(&freader, &info.wv_id, info.wvc_id ? &info.wvc_id : NULL, error, open_flags, 0);
if (!info.wpc)
{
close_context(&info);
return retval;
}
lstrcpyn(info.lastfn, filename, MAX_PATH);
info.w_lastfn [0] = 0;
}
if (!_stricmp(metadata, "formatinformation"))
{
wchar_t *temp = (wchar_t *)malloc(retlen * sizeof(wchar_t));
generate_format_string(info.wpc, temp, retlen, 0);
lstrcpyn(ret, AutoChar(temp), retlen);
free(temp);
retval = 1;
}
else if (!_stricmp(metadata, "length"))
{
StringCchPrintf(ret, retlen, "%d", (int)(WavpackGetNumSamples(info.wpc) * 1000.0 / WavpackGetSampleRate(info.wpc)));
retval = 1;
}
else if (!_stricmp(metadata, "lossless"))
{
StringCchPrintf (ret, retlen, "%d", (WavpackGetMode(info.wpc) & MODE_LOSSLESS) ? 1 : 0);
retval = 1;
}
else if (!_stricmp(metadata, "numsamples"))
{
StringCchPrintf(ret, retlen, "%d", WavpackGetNumSamples(info.wpc));
retval = 1;
}
else if (!_stricmp(metadata, "mime"))
{
lstrcpyn(ret, L"audio/x-wavpack", retlen);
retval = 1;
}
else if (!_stricmp(metadata, "gain"))
{
StringCchPrintf(ret, retlen, "%-+.2f dB", calculate_gain(info.wpc, false));
retval = 1;
}
else if (WavpackGetTagItem(info.wpc, metadata, ret, retlen))
{
if (!_stricmp(metadata, "rating"))
{
int rating = atoi(ret);
// appears to be generally 0-5 or 0-100
if (rating > 10) {
rating /= 20;
}
// or maybe we're dealing with a 1-10 range
else if (rating > 5) {
rating /= 2;
}
// otherwise it is hopefully in the 0-5 range
StringCchPrintf(ret, retlen, "%u", rating);
}
else
{
if (WavpackGetMode(info.wpc) & MODE_APETAG)
{
UTF8ToAnsi(ret, retlen);
}
}
retval = 1;
}
else if (metadata_we_can_write(metadata))
{
if (retlen)
*ret = 0;
retval = 1;
}
// This is a little ugly, but since the WavPack library has read the tags off the
// files, we can close the files (but not the WavPack context) now so that we don't
// leave handles open. We may access the file again for the "formatinformation"
// field, so we reopen the file if we get that one.
if (info.wv_id)
{
fclose(info.wv_id);
info.wv_id = NULL;
}
if (info.wvc_id)
{
fclose(info.wvc_id);
info.wvc_id = NULL;
}
return retval;
}
#endif
#ifdef UNICODE_METADATA
extern "C" __declspec (dllexport) int winampGetExtendedFileInfoW (wchar_t *filename, char *metadata, wchar_t *ret, int retlen)
{
char error[128], res[256];
int open_flags = OPEN_TAGS;
int retval = 0;
#ifdef DEBUG_CONSOLE
sprintf (error, "winampGetExtendedFileInfoW (%s)\n", metadata);
debug_write (error);
#endif
if (!_stricmp(metadata, "type"))
{
ret[0] = '0';
ret[1] = 0;
return 1;
}
else if (!_stricmp(metadata, "family"))
{
int len;
const wchar_t *p;
if (!filename || !filename[0]) return 0;
len = lstrlenW(filename);
if (len < 3 || L'.' != filename[len - 3]) return 0;
p = &filename[len - 2];
if (!_wcsicmp(p, L"wv")) { WASABI_API_LNGSTRINGW_BUF(IDS_FAMILY_STRING, ret, retlen); return 1; }
return 0;
}
if (!filename || !*filename)
return retval;
if (!_stricmp(metadata, "length")) /* even if no file, return a 1 and write "0" */
{
StringCchPrintfW(ret, retlen, L"%d", 0);
retval = 1;
}
if (!info.wpc || wcscmp(filename, info.w_lastfn) || !_stricmp(metadata, "formatinformation"))
{
close_context(&info);
if (!(info.wv_id = _wfopen(filename, L"rb")))
return retval;
if (config_bits & ALLOW_WVC) {
int length = wcslen(filename) + 10;
wchar_t *wvc_name = (wchar_t *)malloc(length * sizeof(wchar_t));
if (wvc_name) {
lstrcpynW(wvc_name, filename, length);
StringCchCatW(wvc_name, length, L"c");
info.wvc_id = _wfopen(wvc_name, L"rb");
free(wvc_name);
}
}
info.wpc = WavpackOpenFileInputEx(&freader, &info.wv_id, info.wvc_id ? &info.wvc_id : NULL, error, open_flags, 0);
if (!info.wpc)
{
close_context(&info);
return retval;
}
lstrcpynW(info.w_lastfn, filename, MAX_PATH);
info.lastfn[0] = 0;
}
if (!_stricmp(metadata, "formatinformation"))
{
generate_format_string(info.wpc, ret, retlen, 0);
retval = 1;
}
else if (!_stricmp (metadata, "length"))
{
StringCchPrintfW(ret, retlen, L"%d", (int)(WavpackGetNumSamples(info.wpc) * 1000.0 / WavpackGetSampleRate(info.wpc)));
retval = 1;
}
else if (!_stricmp(metadata, "lossless"))
{
StringCchPrintfW(ret, retlen, L"%d", (WavpackGetMode(info.wpc) & MODE_LOSSLESS) ? 1 : 0);
retval = 1;
}
else if (!_stricmp(metadata, "gain"))
{
StringCchPrintfW(ret, retlen, L"%-+.2f dB", calculate_gain(info.wpc, false));
retval = 1;
}
else if (!_stricmp(metadata, "numsamples"))
{
StringCchPrintfW(ret, retlen, L"%d", WavpackGetNumSamples(info.wpc));
retval = 1;
}
else if (!_stricmp(metadata, "mime"))
{
lstrcpynW(ret, L"audio/x-wavpack", retlen);
retval = 1;
}
else if (WavpackGetTagItem(info.wpc, metadata, res, sizeof (res)))
{
if (!_stricmp(metadata, "rating"))
{
int rating = atoi(res);
// appears to be generally 0-5 or 0-100
if (rating > 10) {
rating /= 20;
}
// or maybe we're dealing with a 1-10 range
else if (rating > 5) {
rating /= 2;
}
// otherwise it is hopefully in the 0-5 range
StringCchPrintfW(ret, retlen, L"%u", rating);
}
else
{
if (!(WavpackGetMode(info.wpc) & MODE_APETAG))
lstrcpynW(ret, AutoWide(res), retlen);
else
lstrcpynW(ret, AutoWide(res, CP_UTF8), retlen);
}
retval = 1;
}
else if (metadata_we_can_write (metadata))
{
if (retlen)
*ret = 0;
retval = 1;
}
// This is a little ugly, but since the WavPack library has read the tags off the
// files, we can close the files (but not the WavPack context) now so that we don't
// leave handles open. We may access the file again for the "formatinformation"
// field, so we reopen the file if we get that one.
if (info.wv_id)
{
fclose (info.wv_id);
info.wv_id = NULL;
}
if (info.wvc_id)
{
fclose (info.wvc_id);
info.wvc_id = NULL;
}
return retval;
}
#endif
#ifdef ANSI_METADATA
extern "C" int __declspec (dllexport) winampSetExtendedFileInfo(const char *filename, const char *metadata, char *val)
{
char error[128];
#ifdef DEBUG_CONSOLE
sprintf (error, "winampSetExtendedFileInfo (%s=%s)\n", metadata, val);
debug_write (error);
#endif
if (!filename || !*filename || !metadata_we_can_write(metadata))
return 0;
if (!edit.wpc || strcmp(filename, edit.lastfn))
{
if (edit.wpc)
WavpackCloseFile(edit.wpc);
edit.wpc = WavpackOpenFileInput(filename, error, OPEN_TAGS | OPEN_EDIT_TAGS, 0);
if (!edit.wpc)
return 0;
lstrcpyn(edit.lastfn, filename, MAX_PATH);
edit.w_lastfn [0] = 0;
}
if (strlen(val))
return WavpackAppendTagItem(edit.wpc, metadata, val, strlen (val));
else
return WavpackDeleteTagItem(edit.wpc, metadata);
}
#endif
#ifdef UNICODE_METADATA
extern "C" int __declspec (dllexport) winampSetExtendedFileInfoW(const wchar_t *filename, const char *metadata, wchar_t *val)
{
char error[128], utf8_val[256];
lstrcpyn(utf8_val,AutoChar(val, CP_UTF8),sizeof(utf8_val) - 1);
#ifdef DEBUG_CONSOLE
sprintf (error, "winampSetExtendedFileInfoW (%s=%s)\n", metadata, utf8_val);
debug_write (error);
#endif
if (!filename || !*filename || !metadata_we_can_write(metadata))
return 0;
if (!edit.wpc || wcscmp(filename, edit.w_lastfn))
{
if (edit.wpc)
{
WavpackCloseFile(edit.wpc);
edit.wpc = NULL;
}
if (edit.wv_id)
fclose(edit.wv_id);
if (!(edit.wv_id = _wfopen(filename, L"r+b")))
return 0;
edit.wpc = WavpackOpenFileInputEx(&freader, &edit.wv_id, NULL, error, OPEN_TAGS | OPEN_EDIT_TAGS, 0);
if (!edit.wpc)
{
fclose(edit.wv_id);
return 0;
}
lstrcpynW(edit.w_lastfn, filename, MAX_PATH);
edit.lastfn [0] = 0;
}
if (strlen(utf8_val))
return WavpackAppendTagItem(edit.wpc, metadata, utf8_val, strlen (utf8_val));
else
return WavpackDeleteTagItem(edit.wpc, metadata);
}
#endif
extern "C" int __declspec (dllexport) winampWriteExtendedFileInfo(void)
{
#ifdef DEBUG_CONSOLE
debug_write ("winampWriteExtendedFileInfo ()\n");
#endif
if (edit.wpc)
{
WavpackWriteTag(edit.wpc);
WavpackCloseFile(edit.wpc);
edit.wpc = NULL;
}
if (edit.wv_id)
{
fclose(edit.wv_id);
edit.wv_id = NULL;
}
close_context(&info); // make sure we re-read info on any open files
return 1;
}
// return 1 if you want winamp to show it's own file info dialogue, 0 if you want to show your own (via In_Module.InfoBox)
// if returning 1, remember to implement winampGetExtendedFileInfo("formatinformation")!
extern "C" __declspec(dllexport) int winampUseUnifiedFileInfoDlg(const wchar_t * fn)
{
return 1;
}
static const char *writable_metadata [] = {
"track", "genre", "year", "comment", "artist",
"album", "title", "albumartist", "composer",
"publisher", "disc", "tool", "encoder", "bpm",
"category", "rating",
"replaygain_track_gain", "replaygain_track_peak",
"replaygain_album_gain", "replaygain_album_peak"
};
#define NUM_KNOWN_METADATA (sizeof (writable_metadata) / sizeof (writable_metadata [0]))
static int metadata_we_can_write(const char *metadata)
{
if (!metadata || !*metadata)
return 0;
for (int i = 0; i < NUM_KNOWN_METADATA; ++i)
if (!_stricmp(metadata, writable_metadata[i]))
return 1;
return 0;
}
static void generate_format_string(WavpackContext *wpc, wchar_t *string, int maxlen, int wide)
{
int mode = WavpackGetMode(wpc);
unsigned char md5_sum[16];
wchar_t modes[256];
wchar_t fmt[256];
WASABI_API_LNGSTRINGW_BUF(IDS_ENCODER_VERSION, fmt, sizeof(fmt));
StringCchPrintfW(string, maxlen, fmt, WavpackGetVersion(wpc));
while (*string && string++ && maxlen--);
WASABI_API_LNGSTRINGW_BUF(IDS_SOURCE, fmt, sizeof (fmt));
StringCchPrintfW(string, maxlen, fmt, WavpackGetBitsPerSample(wpc),
WASABI_API_LNGSTRINGW((WavpackGetMode(wpc) & MODE_FLOAT) ? IDS_FLOATS : IDS_INTS),
WavpackGetSampleRate(wpc));
while (*string && string++ && maxlen--);
if (WavpackGetNumChannels(wpc) > 2)
{
WASABI_API_LNGSTRINGW_BUF(IDS_MULTICHANNEL, fmt, sizeof (fmt));
StringCchPrintfW(string, maxlen, fmt, WavpackGetNumChannels(wpc));
while (*string && string++ && maxlen--);
}
else if (WavpackGetNumChannels(wpc) == 2)
{
WASABI_API_LNGSTRINGW_BUF(IDS_STEREO, fmt, sizeof (fmt));
StringCchPrintfW(string, maxlen, fmt);
while (*string && string++ && maxlen--);
}
else
{
WASABI_API_LNGSTRINGW_BUF(IDS_MONO, fmt, sizeof (fmt));
StringCchPrintfW(string, maxlen, fmt);
while (*string && string++ && maxlen--);
}
modes [0] = 0;
if (WavpackGetMode(wpc) & MODE_HYBRID)
{
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW(IDS_HYBRID));
StringCchCatW(modes, 256, L" ");
}
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW((WavpackGetMode(wpc) & MODE_LOSSLESS) ? IDS_LOSSLESS : IDS_LOSSY));
if (WavpackGetMode(wpc) & MODE_FAST)
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW(IDS_FAST));
else if (WavpackGetMode(wpc) & MODE_VERY_HIGH)
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW(IDS_VHIGH));
else if (WavpackGetMode(wpc) & MODE_HIGH)
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW(IDS_HIGH));
if (WavpackGetMode(wpc) & MODE_EXTRA)
StringCchCatW(modes, 256, WASABI_API_LNGSTRINGW(IDS_EXTRA));
StringCchPrintfW(string, maxlen, L"%s:%s %s\n",
WASABI_API_LNGSTRINGW(IDS_MODES),
(wide || lstrlenW(modes) < 24) ? L"" : L"\n", modes);
while (*string && string++ && maxlen--);
if (WavpackGetRatio(wpc) != 0.0)
{
wchar_t str_kbps[32];
StringCchPrintfW(string, maxlen, L"%s: %d %s \n",
WASABI_API_LNGSTRINGW(IDS_BITRATE),
(int)((WavpackGetAverageBitrate(wpc, TRUE) + 500.0) / 1000.0),
WASABI_API_LNGSTRINGW_BUF(IDS_KBPS, str_kbps, sizeof(str_kbps)));
while (*string && string++ && maxlen--);
StringCchPrintfW(string, maxlen, L"%s: %.2f : 1 \n",
WASABI_API_LNGSTRINGW(IDS_RATIO), 1.0 / WavpackGetRatio(wpc));
while (*string && string++ && maxlen--);
}
if (WavpackGetMD5Sum(wpc, md5_sum))
{
wchar_t md5s1 [17], md5s2 [17];
int i;
for (i = 0; i < 8; ++i)
{
StringCchPrintfW(md5s1 + i * 2, sizeof(md5s1), L"%02x", md5_sum [i]);
StringCchPrintfW(md5s2 + i * 2, sizeof(md5s2), L"%02x", md5_sum [i+8]);
}
StringCchPrintfW(string, maxlen, (wide ? L"%s: %s%s\n" : L"%s:\n %s\n %s\n"),
WASABI_API_LNGSTRINGW(IDS_MD5), md5s1, md5s2);
}
}
///////////////////// native "C" functions required for AlbumArt support ///////////////////////////
#ifdef DEBUG_CONSOLE
static char temp_buff [256];
static char *wide2char (const wchar_t *src)
{
char *dst = temp_buff;
while (*src)
*dst++ = (char) *src++;
*dst = 0;
return temp_buff;
}
#endif
int WavPack_HandlesExtension(const wchar_t *extension)
{
return !_wcsicmp(extension, L"wv");
}
int WavPack_GetAlbumArt(const wchar_t *filename, const wchar_t *type, void **bits, size_t *len, wchar_t **mime_type)
{
char *buffer;
char error[128];
int tag_size, i;
int retval = 1;
#ifdef DEBUG_CONSOLE
sprintf (error, "WavPack_GetAlbumArt (%s)\n", wide2char (type));
debug_write (error);
#endif
if (!filename || !*filename || _wcsicmp(type, L"cover"))
return retval;
if (!info.wpc || wcscmp(filename, info.w_lastfn))
{
close_context(&info);
if (!(info.wv_id = _wfopen(filename, L"rb")))
return retval;
if (config_bits & ALLOW_WVC)
{
int length = wcslen(filename) + 10;
wchar_t *wvc_name = (wchar_t *)malloc(length * sizeof(wchar_t));
if (wvc_name)
{
lstrcpynW(wvc_name, filename, length);
StringCchCatW(wvc_name, length, L"c");
info.wvc_id = _wfopen(wvc_name, L"rb");
free(wvc_name);
}
}
info.wpc = WavpackOpenFileInputEx(&freader, &info.wv_id, info.wvc_id ? &info.wvc_id : NULL, error, OPEN_TAGS, 0);
if (!info.wpc)
{
close_context(&info);
return retval;
}
lstrcpynW(info.w_lastfn, filename, MAX_PATH);
info.lastfn[0] = 0;
}
tag_size = WavpackGetBinaryTagItem(info.wpc, "Cover Art (Front)", NULL, 0);
if (!tag_size)
return retval;
buffer = (char*)WASABI_API_MEMMGR->sysMalloc(tag_size);
WavpackGetBinaryTagItem(info.wpc, "Cover Art (Front)", buffer, tag_size);
for (i = 0; i < tag_size - 1; ++i)
if (!buffer[i] && strrchr(buffer, '.')) {
char *ext = strrchr(buffer, '.') + 1;
wchar_t *wcptr;
wcptr = *mime_type = (wchar_t*)WASABI_API_MEMMGR->sysMalloc(strlen(ext) * 2 + 2);
while (*ext)
*wcptr++ = *ext++;
*wcptr = 0;
*bits = buffer;
*len = tag_size - i - 1;
memmove(buffer, buffer + i + 1, *len);
retval = 0;
#ifdef DEBUG_CONSOLE
sprintf (error, "WavPack_GetAlbumArt (\"%s\", %d) success!\n", wide2char (*mime_type), *len);
debug_write (error);
#endif
}
if (retval)
WASABI_API_MEMMGR->sysFree(buffer);
// This is a little ugly, but since the WavPack library has read the tags off the
// files, we can close the files (but not the WavPack context) now so that we don't
// leave handles open. We may access the file again for the "formatinformation"
// field, so we reopen the file if we get that one.
if (info.wv_id)
{
fclose(info.wv_id);
info.wv_id = NULL;
}
if (info.wvc_id)
{
fclose(info.wvc_id);
info.wvc_id = NULL;
}
return retval;
}
int WavPack_SetAlbumArt(const wchar_t *filename, const wchar_t *type, void *bits, size_t len, const wchar_t *mime_type)
{
#if 1
return 2; // return 2 to indicate "read-only" cover art for now
#else
char error [128], name [50], *cp;
int tag_size, retval = 0;
unsigned char *buffer;
#ifdef DEBUG_CONSOLE
sprintf (error, "WavPack_SetAlbumArt (%s)\n", wide2char (mime_type));
debug_write (error);
#endif
if (!filename || !*filename || _wcsicmp (type, L"cover") || wcslen (mime_type) > 16)
return 1;
strcpy (name, "Cover Art (Front)");
cp = name + strlen (name);
*cp++ = '.';
while (*mime_type)
*cp++ = (char) *mime_type++;
*cp = 0;
tag_size = strlen (name) + 1 + len;
buffer = malloc (tag_size);
strcpy (buffer, name);
memcpy (buffer + strlen (buffer) + 1, bits, len);
if (!edit.wpc || wcscmp (filename, edit.w_lastfn)) {
if (edit.wpc) {
WavpackCloseFile (edit.wpc);
edit.wpc = NULL;
}
if (edit.wv_id)
fclose (edit.wv_id);
if (!(edit.wv_id = _wfopen (filename, L"r+b"))) {
free (buffer);
return 1;
}
edit.wpc = WavpackOpenFileInputEx (&freader, &edit.wv_id, NULL, error, OPEN_TAGS | OPEN_EDIT_TAGS, 0);
if (!edit.wpc) {
fclose (edit.wv_id);
free (buffer);
return 1;
}
wcscpy (edit.w_lastfn, filename);
edit.lastfn [0] = 0;
}
retval = WavpackAppendTagItem (edit.wpc, "Cover Art (Front)", buffer, tag_size);
free (buffer);
if (retval) {
winampWriteExtendedFileInfo ();
return 0;
}
else {
close_context (&edit);
return 1;
}
#endif
}
int WavPack_DeleteAlbumArt(const wchar_t *filename, const wchar_t *type)
{
#if 1
return 2; // return 2 to indicate "read-only" cover art for now
#else
char error [128];
#ifdef DEBUG_CONSOLE
sprintf (error, "WavPack_DeleteAlbumArt ()\n");
debug_write (error);
#endif
if (!filename || !*filename || _wcsicmp (type, L"cover"))
return 0;
if (!edit.wpc || wcscmp (filename, edit.w_lastfn)) {
if (edit.wpc) {
WavpackCloseFile (edit.wpc);
edit.wpc = NULL;
}
if (edit.wv_id)
fclose (edit.wv_id);
if (!(edit.wv_id = _wfopen (filename, L"r+b")))
return 1;
edit.wpc = WavpackOpenFileInputEx (&freader, &edit.wv_id, NULL, error, OPEN_TAGS | OPEN_EDIT_TAGS, 0);
if (!edit.wpc) {
fclose (edit.wv_id);
return 1;
}
wcscpy (edit.w_lastfn, filename);
edit.lastfn [0] = 0;
}
if (WavpackDeleteTagItem (edit.wpc, "Cover Art (Front)")) {
winampWriteExtendedFileInfo ();
return 0;
}
else {
close_context (&edit);
return 1;
}
#endif
}
//////////////////////////////////////////////////////////////////////////////
// This function uses the ReplayGain mode selected by the user and the info //
// stored in the specified tag to determine the gain value used to play the //
// file and whether "soft clipping" is required. Note that the gain is in //
// voltage scaling (not dB), so a value of 1.0 (not 0.0) is unity gain. //
//////////////////////////////////////////////////////////////////////////////
static float calculate_gain(WavpackContext *wpc, bool allowDefault)
{
if (AGAVE_API_CONFIG->GetBool(playbackConfigGroupGUID, L"replaygain", false))
{
float dB = 0, peak = 1.0f;
char gain[128] = "", peakVal[128] = "";
_locale_t C_locale = WASABI_API_LNG->Get_C_NumericLocale();
switch (AGAVE_API_CONFIG->GetUnsigned(playbackConfigGroupGUID, L"replaygain_source", 0))
{
case 0: // track
if ((!WavpackGetTagItem(wpc, "replaygain_track_gain", gain, sizeof(gain)) || !gain[0])
&& !AGAVE_API_CONFIG->GetBool(playbackConfigGroupGUID, L"replaygain_preferred_only", false))
WavpackGetTagItem(wpc, "replaygain_album_gain", gain, sizeof(gain));
if ((!WavpackGetTagItem(wpc, "replaygain_track_peak", peakVal, sizeof(peakVal)) || !peakVal[0])
&& !AGAVE_API_CONFIG->GetBool(playbackConfigGroupGUID, L"replaygain_preferred_only", false))
WavpackGetTagItem(wpc, "replaygain_album_peak", peakVal, sizeof(peakVal));
break;
case 1:
if ((!WavpackGetTagItem(wpc, "replaygain_album_gain", gain, sizeof(gain)) || !gain[0])
&& !AGAVE_API_CONFIG->GetBool(playbackConfigGroupGUID, L"replaygain_preferred_only", false))
WavpackGetTagItem(wpc, "replaygain_track_gain", gain, sizeof(gain));
if ((!WavpackGetTagItem(wpc, "replaygain_album_peak", peakVal, sizeof(peakVal)) || !peakVal[0])
&& !AGAVE_API_CONFIG->GetBool(playbackConfigGroupGUID, L"replaygain_preferred_only", false))
WavpackGetTagItem(wpc, "replaygain_track_peak", peakVal, sizeof(peakVal));
break;
}
if (gain[0])
{
if (gain[0] == L'+')
dB = (float)_atof_l(&gain[1],C_locale);
else
dB = (float)_atof_l(gain,C_locale);
}
else if (allowDefault)
{
dB = AGAVE_API_CONFIG->GetFloat(playbackConfigGroupGUID, L"non_replaygain", -6.0);
return powf(10.0f, dB / 20.0f);
}
if (peakVal[0])
{
peak = (float)_atof_l(peakVal,C_locale);
}
switch (AGAVE_API_CONFIG->GetUnsigned(playbackConfigGroupGUID, L"replaygain_mode", 1))
{
case 0: // apply gain
return powf(10.0f, dB / 20.0f);
case 1: // apply gain, but don't clip
return min(powf(10.0f, dB / 20.0f), 1.0f / peak);
case 2: // normalize
return 1.0f / peak;
case 3: // prevent clipping
if (peak > 1.0f)
return 1.0f / peak;
else
return 1.0f;
}
}
return 1.0f; // no gain
}
// Convert a Ansi string into its Unicode UTF-8 format equivalent. The
// conversion is done in-place so the maximum length of the string buffer must
// be specified because the string may become longer or shorter. If the
// resulting string will not fit in the specified buffer size then it is
// truncated.
#ifdef OLD_INFO_DIALOG
static void AnsiToUTF8(char *string, int len)
{
int max_chars = (int) strlen(string);
wchar_t *temp = (wchar_t *) malloc((max_chars + 1) * 2);
MultiByteToWideChar(CP_ACP, 0, string, -1, temp, max_chars + 1);
lstrcpyn(string, AutoChar(temp, CP_UTF8), len);
free(temp);
}
#endif
// Convert Unicode UTF-8 string to wide format. UTF-8 string must be NULL
// terminated. Resulting wide string must be able to fit in provided space
// and will also be NULL terminated. The number of characters converted will
// be returned (not counting terminator).
static int UTF8ToWideChar(const char *pUTF8, wchar_t *pWide)
{
int trail_bytes = 0;
int chrcnt = 0;
while (*pUTF8)
{
if (*pUTF8 & 0x80)
{
if (*pUTF8 & 0x40)
{
if (trail_bytes)
{
trail_bytes = 0;
chrcnt++;
}
else
{
char temp = *pUTF8;
while (temp & 0x80)
{
trail_bytes++;
temp <<= 1;
}
pWide [chrcnt] = temp >> trail_bytes--;
}
}
else if (trail_bytes)
{
pWide [chrcnt] = (pWide [chrcnt] << 6) | (*pUTF8 & 0x3f);
if (!--trail_bytes)
chrcnt++;
}
}
else
pWide [chrcnt++] = *pUTF8;
pUTF8++;
}
pWide [chrcnt] = 0;
return chrcnt;
}
// Convert a Unicode UTF-8 format string into its Ansi equivalent. The
// conversion is done in-place so the maximum length of the string buffer must
// be specified because the string may become longer or shorter. If the
// resulting string will not fit in the specified buffer size then it is
// truncated.
static void UTF8ToAnsi(char *string, int len)
{
int max_chars = (int)strlen(string);
wchar_t *temp = (wchar_t *)malloc((max_chars + 1) * 2);
int act_chars = UTF8ToWideChar(string, temp);
while (act_chars)
{
memset (string, 0, len);
if (WideCharToMultiByte(CP_ACP, 0, temp, act_chars, string, len - 1, NULL, NULL))
break;
else
act_chars--;
}
if (!act_chars)
*string = 0;
free (temp);
}
extern "C" __declspec(dllexport) int winampUninstallPlugin(HINSTANCE hDllInst, HWND hwndDlg, int param)
{
// as we're not hooking anything and have no settings we can support an on-the-fly uninstall action
return IN_PLUGIN_UNINSTALL_NOW;
}