2009-07-28 15:32:10 -06:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
2008-12-07 22:30:24 -07:00
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
#include "stdafx.h"
|
|
|
|
|
|
|
|
#include "BannerLoader.h"
|
|
|
|
#include "BannerLoaderWii.h"
|
|
|
|
#include "BannerLoaderGC.h"
|
|
|
|
|
|
|
|
#include "VolumeCreator.h"
|
|
|
|
#include "FileUtil.h"
|
|
|
|
|
2009-01-21 09:31:44 -07:00
|
|
|
// HyperIris: dunno if this suitable, may be need move.
|
|
|
|
#ifdef WIN32
|
|
|
|
#include <Windows.h>
|
2010-07-08 11:59:56 -06:00
|
|
|
#else
|
2010-07-23 17:51:34 -06:00
|
|
|
#include <sys/param.h>
|
2010-07-08 11:46:32 -06:00
|
|
|
#include <iconv.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
|
|
|
|
2010-07-22 01:55:35 -06:00
|
|
|
#ifndef ICONV_CONST
|
2010-07-23 17:51:34 -06:00
|
|
|
#if defined __FreeBSD__ || __NetBSD__
|
|
|
|
#define ICONV_CONST const
|
|
|
|
#else
|
2010-07-22 01:55:35 -06:00
|
|
|
#define ICONV_CONST
|
|
|
|
#endif
|
2010-07-23 17:51:34 -06:00
|
|
|
#endif
|
2010-07-22 01:55:35 -06:00
|
|
|
|
2008-12-07 22:30:24 -07:00
|
|
|
namespace DiscIO
|
|
|
|
{
|
2009-07-28 17:38:49 -06:00
|
|
|
void IBannerLoader::CopyToStringAndCheck(std::string& _rDestination, const char* _src)
|
2008-12-07 22:30:24 -07:00
|
|
|
{
|
|
|
|
static bool bValidChars[256];
|
|
|
|
static bool bInitialized = false;
|
|
|
|
|
|
|
|
if (!bInitialized)
|
|
|
|
{
|
2009-07-28 17:38:49 -06:00
|
|
|
for (int i = 0; i < 0x20; i++)
|
2008-12-07 22:30:24 -07:00
|
|
|
{
|
|
|
|
bValidChars[i] = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate valid chars
|
2009-02-09 08:41:28 -07:00
|
|
|
for (int i = 0x20; i < 256; i++)
|
2008-12-07 22:30:24 -07:00
|
|
|
{
|
2009-02-09 08:41:28 -07:00
|
|
|
bValidChars[i] = true;
|
2008-12-07 22:30:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bValidChars[0x0a] = true;
|
2009-02-09 08:41:28 -07:00
|
|
|
//bValidChars[0xa9] = true;
|
|
|
|
//bValidChars[0xe9] = true;
|
2008-12-07 22:30:24 -07:00
|
|
|
|
|
|
|
bInitialized = true;
|
|
|
|
}
|
|
|
|
|
2009-02-09 08:41:28 -07:00
|
|
|
char destBuffer[2048] = {0};
|
2008-12-07 22:30:24 -07:00
|
|
|
char* dest = destBuffer;
|
|
|
|
const char* src = _src;
|
|
|
|
|
|
|
|
// copy the string and check for "unknown" characters
|
|
|
|
while (*src != 0x00)
|
|
|
|
{
|
|
|
|
u8 c = *src;
|
|
|
|
|
|
|
|
if (c == 0x0a){c = 0x20;}
|
|
|
|
|
|
|
|
if (bValidChars[c] == false)
|
|
|
|
{
|
2009-07-28 17:38:49 -06:00
|
|
|
src++;
|
|
|
|
continue;
|
2008-12-07 22:30:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
*dest = c;
|
|
|
|
dest++;
|
|
|
|
src++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// finalize the string
|
2009-07-28 17:38:49 -06:00
|
|
|
*dest = 0x00;
|
2008-12-07 22:30:24 -07:00
|
|
|
|
|
|
|
_rDestination = destBuffer;
|
|
|
|
}
|
|
|
|
|
2009-02-06 09:53:22 -07:00
|
|
|
bool IBannerLoader::CopyBeUnicodeToString( std::string& _rDestination, const u16* _src, int length )
|
2009-01-21 09:31:44 -07:00
|
|
|
{
|
|
|
|
bool returnCode = false;
|
|
|
|
#ifdef WIN32
|
|
|
|
if (_src)
|
|
|
|
{
|
2009-02-06 09:53:22 -07:00
|
|
|
u16* buffer = new u16[length];
|
|
|
|
if (buffer)
|
2009-01-21 09:31:44 -07:00
|
|
|
{
|
2009-02-06 09:53:22 -07:00
|
|
|
memcpy(buffer, _src, sizeof(u16)*length);
|
|
|
|
for (int i = 0; i < length; i++)
|
|
|
|
{
|
|
|
|
buffer[i] = swap16(buffer[i]);
|
|
|
|
}
|
|
|
|
|
2009-02-28 11:44:39 -07:00
|
|
|
u32 ansiNameSize = WideCharToMultiByte(932, 0,
|
|
|
|
(LPCWSTR)buffer, (int)wcslen((LPCWSTR)buffer),
|
2009-02-06 09:53:22 -07:00
|
|
|
NULL, NULL, NULL, NULL);
|
|
|
|
if (ansiNameSize > 0)
|
2009-01-21 09:31:44 -07:00
|
|
|
{
|
2009-02-06 09:53:22 -07:00
|
|
|
char* pAnsiStrBuffer = new char[ansiNameSize + 1];
|
|
|
|
if (pAnsiStrBuffer)
|
2009-01-21 09:31:44 -07:00
|
|
|
{
|
2009-02-06 09:53:22 -07:00
|
|
|
memset(pAnsiStrBuffer, 0, (ansiNameSize + 1) * sizeof(char));
|
2009-02-28 11:44:39 -07:00
|
|
|
if (WideCharToMultiByte(932, 0,
|
|
|
|
(LPCWSTR)buffer, (int)wcslen((LPCWSTR)buffer),
|
2009-02-06 09:53:22 -07:00
|
|
|
pAnsiStrBuffer, ansiNameSize, NULL, NULL))
|
|
|
|
{
|
|
|
|
_rDestination = pAnsiStrBuffer;
|
|
|
|
returnCode = true;
|
|
|
|
}
|
2009-08-03 16:51:13 -06:00
|
|
|
delete[] pAnsiStrBuffer;
|
2009-01-21 09:31:44 -07:00
|
|
|
}
|
2009-02-06 09:53:22 -07:00
|
|
|
}
|
2009-08-03 16:51:13 -06:00
|
|
|
delete[] buffer;
|
2009-02-06 09:53:22 -07:00
|
|
|
}
|
2009-01-21 09:31:44 -07:00
|
|
|
}
|
2010-07-08 11:59:56 -06:00
|
|
|
#else
|
2010-07-08 11:46:32 -06:00
|
|
|
if (_src)
|
|
|
|
{
|
|
|
|
iconv_t conv_desc = iconv_open("UTF-8", "CP932");
|
|
|
|
if (conv_desc == (iconv_t) -1)
|
|
|
|
{
|
|
|
|
// Initialization failure.
|
|
|
|
if (errno == EINVAL)
|
|
|
|
{
|
|
|
|
ERROR_LOG(DISCIO, "Conversion from CP932 to UTF-8 is not supported.");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG(DISCIO, "Iconv initialization failure: %s\n", strerror (errno));
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
char* src_buffer = new char[length];
|
|
|
|
for (int i = 0; i < length; i++)
|
|
|
|
src_buffer[i] = swap16(_src[i]);
|
|
|
|
|
|
|
|
size_t inbytes = sizeof(char) * length;
|
|
|
|
size_t outbytes = 2 * inbytes;
|
|
|
|
char* utf8_buffer = new char[outbytes + 1];
|
|
|
|
memset(utf8_buffer, 0, (outbytes + 1) * sizeof(char));
|
|
|
|
|
|
|
|
// Save the buffer locations because iconv increments them
|
|
|
|
char* utf8_buffer_start = utf8_buffer;
|
|
|
|
char* src_buffer_start = src_buffer;
|
|
|
|
|
2010-07-22 01:55:35 -06:00
|
|
|
size_t iconv_size = iconv(conv_desc,
|
|
|
|
(ICONV_CONST char**)&src_buffer, &inbytes,
|
|
|
|
&utf8_buffer, &outbytes);
|
2010-07-08 11:46:32 -06:00
|
|
|
|
|
|
|
// Handle failures
|
|
|
|
if (iconv_size == (size_t) -1)
|
|
|
|
{
|
|
|
|
ERROR_LOG(DISCIO, "iconv failed.");
|
|
|
|
switch (errno) {
|
|
|
|
case EILSEQ:
|
|
|
|
ERROR_LOG(DISCIO, "Invalid multibyte sequence.");
|
|
|
|
break;
|
|
|
|
case EINVAL:
|
|
|
|
ERROR_LOG(DISCIO, "Incomplete multibyte sequence.");
|
|
|
|
break;
|
|
|
|
case E2BIG:
|
|
|
|
ERROR_LOG(DISCIO, "Insufficient space allocated for output buffer.");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ERROR_LOG(DISCIO, "Error: %s.", strerror(errno));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_rDestination = utf8_buffer_start;
|
|
|
|
returnCode = true;
|
|
|
|
}
|
|
|
|
delete[] utf8_buffer_start;
|
|
|
|
delete[] src_buffer_start;
|
|
|
|
iconv_close(conv_desc);
|
|
|
|
}
|
2009-01-21 09:31:44 -07:00
|
|
|
#endif
|
|
|
|
return returnCode;
|
|
|
|
}
|
|
|
|
|
2009-06-06 20:54:07 -06:00
|
|
|
IBannerLoader* CreateBannerLoader(DiscIO::IFileSystem& _rFileSystem, DiscIO::IVolume *pVolume)
|
2008-12-07 22:30:24 -07:00
|
|
|
{
|
2009-06-06 20:54:07 -06:00
|
|
|
if (IsVolumeWiiDisc(pVolume) || IsVolumeWadFile(pVolume))
|
2008-12-07 22:30:24 -07:00
|
|
|
{
|
2010-06-03 14:37:32 -06:00
|
|
|
return new CBannerLoaderWii(pVolume);
|
|
|
|
}
|
|
|
|
if (_rFileSystem.IsValid())
|
|
|
|
{
|
|
|
|
return new CBannerLoaderGC(_rFileSystem);
|
2008-12-07 22:30:24 -07:00
|
|
|
}
|
|
|
|
|
2010-06-03 14:37:32 -06:00
|
|
|
return NULL;
|
2008-12-07 22:30:24 -07:00
|
|
|
}
|
|
|
|
|
2010-06-03 14:37:32 -06:00
|
|
|
} // namespace
|