2007-05-30 21:56:52 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
2003-10-02 22:52:57 +00: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; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
* 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 for more details.
|
|
|
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2003-10-02 22:52:57 +00:00
|
|
|
*
|
2006-02-11 10:05:31 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2003-10-02 22:52:57 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2004-12-02 00:33:42 +00:00
|
|
|
#include "common/config-manager.h"
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/error.h"
|
2004-12-02 00:33:42 +00:00
|
|
|
#include "common/str.h"
|
2006-10-21 12:44:10 +00:00
|
|
|
#include "common/system.h"
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/textconsole.h"
|
2005-01-10 22:35:43 +00:00
|
|
|
#include "common/util.h"
|
2011-02-09 01:09:01 +00:00
|
|
|
#include "audio/mididrv.h"
|
|
|
|
#include "audio/musicplugin.h"
|
2003-10-02 22:52:57 +00:00
|
|
|
|
2005-04-03 22:01:38 +00:00
|
|
|
const byte MidiDriver::_mt32ToGm[128] = {
|
|
|
|
// 0 1 2 3 4 5 6 7 8 9 A B C D E F
|
|
|
|
0, 1, 0, 2, 4, 4, 5, 3, 16, 17, 18, 16, 16, 19, 20, 21, // 0x
|
|
|
|
6, 6, 6, 7, 7, 7, 8, 112, 62, 62, 63, 63, 38, 38, 39, 39, // 1x
|
|
|
|
88, 95, 52, 98, 97, 99, 14, 54, 102, 96, 53, 102, 81, 100, 14, 80, // 2x
|
|
|
|
48, 48, 49, 45, 41, 40, 42, 42, 43, 46, 45, 24, 25, 28, 27, 104, // 3x
|
|
|
|
32, 32, 34, 33, 36, 37, 35, 35, 79, 73, 72, 72, 74, 75, 64, 65, // 4x
|
|
|
|
66, 67, 71, 71, 68, 69, 70, 22, 56, 59, 57, 57, 60, 60, 58, 61, // 5x
|
|
|
|
61, 11, 11, 98, 14, 9, 14, 13, 12, 107, 107, 77, 78, 78, 76, 76, // 6x
|
|
|
|
47, 117, 127, 118, 118, 116, 115, 119, 115, 112, 55, 124, 123, 0, 14, 117 // 7x
|
|
|
|
};
|
|
|
|
|
|
|
|
const byte MidiDriver::_gmToMt32[128] = {
|
|
|
|
// 0 1 2 3 4 5 6 7 8 9 A B C D E F
|
|
|
|
5, 1, 2, 7, 3, 5, 16, 21, 22, 101, 101, 97, 104, 103, 102, 20, // 0x
|
|
|
|
8, 9, 11, 12, 14, 15, 87, 15, 59, 60, 61, 62, 67, 44, 79, 23, // 1x
|
|
|
|
64, 67, 66, 70, 68, 69, 28, 31, 52, 54, 55, 56, 49, 51, 57, 112, // 2x
|
|
|
|
48, 50, 45, 26, 34, 35, 45, 122, 89, 90, 94, 81, 92, 95, 24, 25, // 3x
|
|
|
|
80, 78, 79, 78, 84, 85, 86, 82, 74, 72, 76, 77, 110, 107, 108, 76, // 4x
|
|
|
|
47, 44, 111, 45, 44, 34, 44, 30, 32, 33, 88, 34, 35, 35, 38, 33, // 5x
|
|
|
|
41, 36, 100, 37, 40, 34, 43, 40, 63, 21, 99, 105, 103, 86, 55, 84, // 6x
|
|
|
|
101, 103, 100, 120, 117, 113, 99, 128, 128, 128, 128, 124, 123, 128, 128, 128, // 7x
|
|
|
|
};
|
2003-10-02 22:52:57 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
static const uint32 GUIOMapping[] = {
|
|
|
|
MT_PCSPK, Common::GUIO_MIDIPCSPK,
|
2010-09-17 20:02:29 +00:00
|
|
|
MT_CMS, Common::GUIO_MIDICMS,
|
2010-06-21 21:36:36 +00:00
|
|
|
MT_PCJR, Common::GUIO_MIDIPCJR,
|
|
|
|
MT_ADLIB, Common::GUIO_MIDIADLIB,
|
2010-08-11 19:32:07 +00:00
|
|
|
MT_C64, Common::GUIO_MIDIC64,
|
|
|
|
MT_AMIGA, Common::GUIO_MIDIAMIGA,
|
2010-08-11 18:54:56 +00:00
|
|
|
MT_APPLEIIGS, Common::GUIO_MIDIAPPLEIIGS,
|
2010-08-11 19:32:07 +00:00
|
|
|
MT_TOWNS, Common::GUIO_MIDITOWNS,
|
|
|
|
MT_PC98, Common::GUIO_MIDIPC98,
|
|
|
|
MT_GM, Common::GUIO_MIDIGM,
|
2010-06-22 16:42:49 +00:00
|
|
|
MT_MT32, Common::GUIO_MIDIMT32,
|
2010-08-11 19:32:07 +00:00
|
|
|
0, 0
|
2010-06-21 21:36:36 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
uint32 MidiDriver::musicType2GUIO(uint32 musicType) {
|
|
|
|
uint32 res = 0;
|
2003-10-18 00:22:46 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
for (int i = 0; GUIOMapping[i] || GUIOMapping[i + 1]; i += 2) {
|
|
|
|
if (musicType == GUIOMapping[i] || musicType == (uint32)-1)
|
|
|
|
res |= GUIOMapping[i + 1];
|
|
|
|
}
|
2003-10-18 00:22:46 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
return res;
|
|
|
|
}
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
bool MidiDriver::_forceTypeMT32 = false;
|
2003-10-18 00:22:46 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
MusicType MidiDriver::getMusicType(MidiDriver::DeviceHandle handle) {
|
|
|
|
if (_forceTypeMT32)
|
|
|
|
return MT_MT32;
|
|
|
|
|
|
|
|
if (handle) {
|
|
|
|
const MusicPlugin::List p = MusicMan.getPlugins();
|
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); m++) {
|
|
|
|
MusicDevices i = (**m)->getDevices();
|
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); d++) {
|
|
|
|
if (handle == d->getHandle())
|
|
|
|
return d->getMusicType();
|
|
|
|
}
|
2003-10-18 00:22:46 +00:00
|
|
|
}
|
|
|
|
}
|
2010-07-21 18:17:51 +00:00
|
|
|
|
2010-06-29 00:29:35 +00:00
|
|
|
return MT_INVALID;
|
2003-10-18 00:22:46 +00:00
|
|
|
}
|
2004-12-02 00:33:42 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
Common::String MidiDriver::getDeviceString(DeviceHandle handle, DeviceStringType type) {
|
|
|
|
if (handle) {
|
|
|
|
const MusicPlugin::List p = MusicMan.getPlugins();
|
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); m++) {
|
|
|
|
MusicDevices i = (**m)->getDevices();
|
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); d++) {
|
|
|
|
if (handle == d->getHandle()) {
|
|
|
|
if (type == kDriverName)
|
|
|
|
return d->getMusicDriverName();
|
|
|
|
else if (type == kDriverId)
|
|
|
|
return d->getMusicDriverId();
|
|
|
|
else if (type == kDeviceId)
|
|
|
|
return d->getCompleteId();
|
|
|
|
else
|
|
|
|
return Common::String("auto");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-12-30 14:18:21 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
return Common::String("auto");
|
2009-03-26 13:52:43 +00:00
|
|
|
}
|
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
MidiDriver::DeviceHandle MidiDriver::detectDevice(int flags) {
|
|
|
|
// Query the selected music device (defaults to MT_AUTO device).
|
|
|
|
DeviceHandle hdl = getDeviceHandle(ConfMan.get("music_driver"));
|
2009-03-26 13:52:43 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
_forceTypeMT32 = false;
|
2005-12-30 14:18:21 +00:00
|
|
|
|
|
|
|
// Check whether the selected music driver is compatible with the
|
|
|
|
// given flags.
|
2010-06-21 21:36:36 +00:00
|
|
|
switch (getMusicType(hdl)) {
|
|
|
|
case MT_PCSPK:
|
|
|
|
if (flags & MDT_PCSPK)
|
|
|
|
return hdl;
|
|
|
|
break;
|
|
|
|
|
2010-06-22 16:42:49 +00:00
|
|
|
case MT_PCJR:
|
2010-06-22 15:30:41 +00:00
|
|
|
if (flags & MDT_PCJR)
|
|
|
|
return hdl;
|
|
|
|
break;
|
2010-07-21 18:17:51 +00:00
|
|
|
|
2010-06-25 18:47:52 +00:00
|
|
|
case MT_CMS:
|
|
|
|
if (flags & MDT_CMS)
|
|
|
|
return hdl;
|
|
|
|
break;
|
2010-06-22 15:30:41 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
case MT_ADLIB:
|
|
|
|
if (flags & MDT_ADLIB)
|
|
|
|
return hdl;
|
|
|
|
break;
|
2010-07-21 18:17:51 +00:00
|
|
|
|
2010-08-11 19:32:07 +00:00
|
|
|
case MT_C64:
|
2010-08-11 18:54:56 +00:00
|
|
|
if (flags & MDT_C64)
|
|
|
|
return hdl;
|
|
|
|
break;
|
|
|
|
|
2010-08-11 19:32:07 +00:00
|
|
|
case MT_AMIGA:
|
2010-08-11 18:54:56 +00:00
|
|
|
if (flags & MDT_AMIGA)
|
|
|
|
return hdl;
|
2010-10-12 02:18:11 +00:00
|
|
|
break;
|
2010-08-11 18:54:56 +00:00
|
|
|
|
2010-08-11 19:32:07 +00:00
|
|
|
case MT_APPLEIIGS:
|
2010-08-11 18:54:56 +00:00
|
|
|
if (flags & MDT_APPLEIIGS)
|
|
|
|
return hdl;
|
|
|
|
break;
|
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
case MT_TOWNS:
|
|
|
|
if (flags & MDT_TOWNS)
|
|
|
|
return hdl;
|
|
|
|
break;
|
|
|
|
|
2010-06-25 18:47:52 +00:00
|
|
|
case MT_PC98:
|
|
|
|
if (flags & MDT_PC98)
|
|
|
|
return hdl;
|
|
|
|
break;
|
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
case MT_GM:
|
|
|
|
case MT_GS:
|
|
|
|
case MT_MT32:
|
|
|
|
if (flags & MDT_MIDI)
|
|
|
|
return hdl;
|
2010-06-25 18:47:52 +00:00
|
|
|
break;
|
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
case MT_NULL:
|
2010-06-25 18:47:52 +00:00
|
|
|
return hdl;
|
2005-12-30 14:18:21 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2005-12-30 14:18:21 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
// If the selected driver did not match the flags setting,
|
|
|
|
// we try to determine a suitable and "optimal" music driver.
|
|
|
|
const MusicPlugin::List p = MusicMan.getPlugins();
|
2010-06-25 20:51:57 +00:00
|
|
|
// If only MDT_MIDI but not MDT_PREFER_MT32 or MDT_PREFER_GM is set we prefer the other devices (which will always be
|
2010-06-21 21:36:36 +00:00
|
|
|
// detected since they are hard coded and cannot be disabled.
|
2010-06-29 00:29:57 +00:00
|
|
|
for (int l = (flags & (MDT_PREFER_GM | MDT_PREFER_MT32)) ? 1 : 0; l < 2; ++l) {
|
2010-06-21 21:36:36 +00:00
|
|
|
if ((flags & MDT_MIDI) && (l == 1)) {
|
|
|
|
// If a preferred MT32 or GM device has been selected that device gets returned
|
2010-06-29 00:30:16 +00:00
|
|
|
if (flags & MDT_PREFER_MT32)
|
|
|
|
hdl = getDeviceHandle(ConfMan.get("mt32_device"));
|
|
|
|
else if (flags & MDT_PREFER_GM)
|
|
|
|
hdl = getDeviceHandle(ConfMan.get("gm_device"));
|
|
|
|
else
|
|
|
|
hdl = getDeviceHandle("auto");
|
2010-06-29 00:29:57 +00:00
|
|
|
|
|
|
|
const MusicType type = getMusicType(hdl);
|
2010-06-21 21:36:36 +00:00
|
|
|
|
2010-10-24 19:15:17 +00:00
|
|
|
// If have a "Don't use GM/MT-32" setting we skip this part and jump
|
|
|
|
// to AdLib, PC Speaker etc. detection right away.
|
|
|
|
if (type != MT_NULL) {
|
|
|
|
if (type != MT_AUTO && type != MT_INVALID) {
|
|
|
|
if (flags & MDT_PREFER_MT32)
|
|
|
|
// If we have a preferred MT32 device we disable the gm/mt32 mapping (more about this in mididrv.h)
|
|
|
|
_forceTypeMT32 = true;
|
|
|
|
|
|
|
|
return hdl;
|
|
|
|
}
|
2010-06-22 16:42:49 +00:00
|
|
|
|
2010-10-24 19:15:17 +00:00
|
|
|
// If we have no specific device selected (neither in the scummvm nor in the game domain)
|
|
|
|
// and no preferred MT32 or GM device selected we arrive here.
|
|
|
|
// If MT32 is preferred we try for the first available device with music type 'MT_MT32' (usually the mt32 emulator)
|
|
|
|
if (flags & MDT_PREFER_MT32) {
|
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
|
|
|
MusicDevices i = (**m)->getDevices();
|
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
|
|
|
if (d->getMusicType() == MT_MT32)
|
|
|
|
return d->getHandle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now we default to the first available device with music type 'MT_GM'
|
2010-06-29 00:29:57 +00:00
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
2010-06-21 21:36:36 +00:00
|
|
|
MusicDevices i = (**m)->getDevices();
|
2010-06-29 00:29:57 +00:00
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
2010-10-24 19:15:17 +00:00
|
|
|
if (d->getMusicType() == MT_GM || d->getMusicType() == MT_GS)
|
2010-06-29 00:29:57 +00:00
|
|
|
return d->getHandle();
|
2010-06-21 21:36:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-29 00:29:57 +00:00
|
|
|
}
|
2010-06-22 16:42:49 +00:00
|
|
|
|
2010-06-25 18:47:52 +00:00
|
|
|
MusicType tp = MT_AUTO;
|
2010-06-21 21:36:36 +00:00
|
|
|
if (flags & MDT_TOWNS)
|
|
|
|
tp = MT_TOWNS;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (flags & MDT_PC98)
|
2010-08-11 18:54:56 +00:00
|
|
|
tp = MT_PC98;
|
2010-06-21 21:36:36 +00:00
|
|
|
else if (flags & MDT_ADLIB)
|
|
|
|
tp = MT_ADLIB;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (flags & MDT_PCJR)
|
2010-08-11 18:54:56 +00:00
|
|
|
tp = MT_PCJR;
|
2010-10-10 07:43:12 +00:00
|
|
|
else if (flags & MDT_PCSPK)
|
|
|
|
tp = MT_PCSPK;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (flags & MDT_C64)
|
2010-08-11 18:54:56 +00:00
|
|
|
tp = MT_C64;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (flags & MDT_AMIGA)
|
2010-08-11 18:54:56 +00:00
|
|
|
tp = MT_AMIGA;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (flags & MDT_APPLEIIGS)
|
2010-08-11 18:54:56 +00:00
|
|
|
tp = MT_APPLEIIGS;
|
2010-08-11 19:32:07 +00:00
|
|
|
else if (l == 0)
|
2010-06-29 21:56:04 +00:00
|
|
|
// if we haven't tried to find a MIDI device yet we do this now.
|
|
|
|
continue;
|
2010-06-21 21:36:36 +00:00
|
|
|
else
|
2010-06-25 18:47:52 +00:00
|
|
|
tp = MT_AUTO;
|
2010-06-21 21:36:36 +00:00
|
|
|
|
2010-06-29 00:29:57 +00:00
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
2010-06-21 21:36:36 +00:00
|
|
|
MusicDevices i = (**m)->getDevices();
|
2010-06-29 00:29:57 +00:00
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
2010-06-21 21:36:36 +00:00
|
|
|
if (d->getMusicType() == tp)
|
|
|
|
return d->getHandle();
|
|
|
|
}
|
2005-12-30 14:18:21 +00:00
|
|
|
}
|
2004-12-02 00:33:42 +00:00
|
|
|
}
|
|
|
|
|
2010-06-29 00:38:06 +00:00
|
|
|
return 0;
|
2004-12-02 00:33:42 +00:00
|
|
|
}
|
2004-12-25 18:34:44 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
MidiDriver *MidiDriver::createMidi(MidiDriver::DeviceHandle handle) {
|
|
|
|
MidiDriver *driver = 0;
|
|
|
|
const MusicPlugin::List p = MusicMan.getPlugins();
|
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); m++) {
|
|
|
|
if (getDeviceString(handle, MidiDriver::kDriverId).equals((**m)->getId()))
|
|
|
|
(**m)->createInstance(&driver, handle);
|
2004-12-25 18:34:44 +00:00
|
|
|
}
|
2010-06-22 16:42:49 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
return driver;
|
|
|
|
}
|
|
|
|
|
|
|
|
MidiDriver::DeviceHandle MidiDriver::getDeviceHandle(const Common::String &identifier) {
|
|
|
|
const MusicPlugin::List p = MusicMan.getPlugins();
|
2010-06-22 18:27:00 +00:00
|
|
|
|
|
|
|
if (p.begin() == p.end())
|
2010-07-26 06:10:47 +00:00
|
|
|
error("Music plugins must be loaded prior to calling this method");
|
2010-06-22 18:27:00 +00:00
|
|
|
|
2010-06-21 21:36:36 +00:00
|
|
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); m++) {
|
|
|
|
MusicDevices i = (**m)->getDevices();
|
|
|
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); d++) {
|
2010-07-26 06:19:57 +00:00
|
|
|
// The music driver id isn't unique, but it will match
|
|
|
|
// driver's first device. This is useful when selecting
|
|
|
|
// the driver from the command line.
|
|
|
|
if (identifier.equals(d->getMusicDriverId()) || identifier.equals(d->getCompleteId()) || identifier.equals(d->getCompleteName())) {
|
2010-06-21 21:36:36 +00:00
|
|
|
return d->getHandle();
|
2010-06-29 00:29:35 +00:00
|
|
|
}
|
2010-06-21 21:36:36 +00:00
|
|
|
}
|
2010-06-22 16:42:49 +00:00
|
|
|
}
|
2004-12-25 18:34:44 +00:00
|
|
|
|
2010-06-29 00:29:35 +00:00
|
|
|
return 0;
|
2004-12-25 18:34:44 +00:00
|
|
|
}
|
MIDI: Send a reset MIDI device signal on startup.
This is currently done in the engine code. I adapted AGI, AGOS, DRACI,
GROOVIE, LURE, MADE, QUEEN, SAGA, SKY, TINSEL and TOUCHE to send a reset
device on startup. The sound output still works fine (started up a game
from every engine), so this should hopefully not introduce any regressions.
As far as I can tell it seems that SCUMM does send a proper device reset, so
I did not touch it. KYRA only sends a proper reset for MT-32 currently. I am
not sure about SCI though.
This fixes bug #3066826 "SIMON: MIDI notes off when using RTL after SCI".
svn-id: r52736
2010-09-15 22:00:20 +00:00
|
|
|
|
|
|
|
void MidiDriver::sendMT32Reset() {
|
|
|
|
static const byte resetSysEx[] = { 0x41, 0x10, 0x16, 0x12, 0x7F, 0x00, 0x00, 0x01, 0x00 };
|
|
|
|
sysEx(resetSysEx, sizeof(resetSysEx));
|
|
|
|
g_system->delayMillis(100);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MidiDriver::sendGMReset() {
|
|
|
|
static const byte resetSysEx[] = { 0x7E, 0x7F, 0x09, 0x01 };
|
|
|
|
sysEx(resetSysEx, sizeof(resetSysEx));
|
|
|
|
g_system->delayMillis(100);
|
|
|
|
}
|
|
|
|
|