0ad/source/sound/JSI_Sound.cpp
janwas c0ed950657 had to remove uint and ulong from lib/types.h due to conflict with other library.
this snowballed into a massive search+destroy of the hodgepodge of
mostly equivalent types we had in use (int, uint, unsigned, unsigned
int, i32, u32, ulong, uintN).

it is more efficient to use 64-bit types in 64-bit mode, so the
preferred default is size_t (for anything remotely resembling a size or
index). tile coordinates are ssize_t to allow more efficient conversion
to/from floating point. flags are int because we almost never need more
than 15 distinct bits, bit test/set is not slower and int is fastest to
type. finally, some data that is pretty much directly passed to OpenGL
is now typed accordingly.

after several hours, the code now requires fewer casts and less
guesswork.

other changes:
- unit and player IDs now have an "invalid id" constant in the
respective class to avoid casting and -1
- fix some endian/64-bit bugs in the map (un)packing. added a
convenience function to write/read a size_t.
- ia32: change CPUID interface to allow passing in ecx (required for
cache topology detection, which I need at work). remove some unneeded
functions from asm, replace with intrinsics where possible.

This was SVN commit r5942.
2008-05-11 18:48:32 +00:00

206 lines
5.2 KiB
C++

#include "precompiled.h"
#include "JSI_Sound.h"
#include "maths/Vector3D.h"
#include "lib/res/sound/snd_mgr.h"
#include "lib/res/h_mgr.h" // h_filename
JSI_Sound::JSI_Sound(const CStr& Filename)
{
m_Handle = snd_open(Filename);
// special-case to avoid throwing exceptions if quickstart has
// disabled sound: set a flag queried by Construct; the object will
// then immediately be freed.
if (m_Handle == ERR::AGAIN)
{
m_SoundDisabled = true;
return;
}
m_SoundDisabled = false;
// if open failed, raise an exception - it's the only way to
// report errors, since we're in the ctor and don't want to move
// the open call elsewhere (by requiring an explicit open() call).
if (m_Handle < 0)
throw std::exception(); // caught by JSI_Sound::Construct.
snd_set_pos(m_Handle, 0,0,0, true);
}
JSI_Sound::~JSI_Sound()
{
this->Free(0, 0, 0);
}
bool JSI_Sound::SetGain(JSContext* cx, uintN argc, jsval* argv)
{
if (! m_Handle)
return false;
debug_assert(argc >= 1); // FIXME
float gain;
if (! ToPrimitive<float>(cx, argv[0], gain))
return false;
snd_set_gain(m_Handle, gain);
return true;
}
bool JSI_Sound::SetPitch(JSContext* cx, uintN argc, jsval* argv)
{
if (! m_Handle)
return false;
debug_assert(argc >= 1); // FIXME
float pitch;
if (! ToPrimitive<float>(cx, argv[0], pitch))
return false;
snd_set_pitch(m_Handle, pitch);
return true;
}
bool JSI_Sound::SetPosition(JSContext* cx, uintN argc, jsval* argv)
{
if (! m_Handle)
return false;
debug_assert(argc >= 1); // FIXME
CVector3D pos;
// absolute world coords
if (ToPrimitive<CVector3D>(cx, argv[0], pos))
snd_set_pos(m_Handle, pos[0], pos[1], pos[2]);
// relative, 0 offset - right on top of the listener
// (we don't need displacement from the listener, e.g. always behind)
else
snd_set_pos(m_Handle, 0,0,0, true);
return true;
}
bool JSI_Sound::Fade(JSContext* cx, uintN argc, jsval* argv)
{
if (! m_Handle)
return false;
debug_assert(argc >= 3); // FIXME
float initial_gain, final_gain;
float length;
if (! (ToPrimitive<float>(cx, argv[0], initial_gain)
&& ToPrimitive<float>(cx, argv[1], final_gain)
&& ToPrimitive<float>(cx, argv[2], length)))
return false;
snd_fade(m_Handle, initial_gain, final_gain, length, FT_S_CURVE);
// HACK: snd_fade causes <m_Handle> to be automatically freed when a
// fade to 0 has completed. however, we're still holding on to a
// reference, which will cause a double-free warning when Free() is
// called from the dtor. solution is to neuter our Handle by
// setting it to 0 (ok since it'll be freed). this does mean that
// no further operations can be carried out during that final fade.
m_Handle = 0;
return true;
}
// start playing the sound (one-shot).
// it will automatically be freed when done.
bool JSI_Sound::Play(JSContext* UNUSED(cx), uintN UNUSED(argc), jsval* UNUSED(argv))
{
if (! m_Handle)
return false;
snd_play(m_Handle);
// We can't do anything else with this sound now, since it's impossible to
// know whether or not it's still valid (since it might have finished playing
// already). So set it to 0, so we don't try doing anything (like freeing it)
// in the future.
m_Handle = 0;
return true;
}
// request the sound be played until free() is called. returns immediately.
bool JSI_Sound::Loop(JSContext* UNUSED(cx), uintN UNUSED(argc), jsval* UNUSED(argv))
{
if (! m_Handle)
return false;
snd_set_loop(m_Handle, true);
snd_play(m_Handle);
return true;
}
// stop sound if currently playing and free resources.
// doesn't need to be called unless played via loop() -
// sounds are freed automatically when done playing.
bool JSI_Sound::Free(JSContext* UNUSED(cx), uintN UNUSED(argc), jsval* UNUSED(argv))
{
if (! m_Handle)
return false;
snd_free(m_Handle); // resets it to 0
return true;
}
// Script-bound functions
void JSI_Sound::ScriptingInit()
{
AddMethod<CStr, &JSI_Sound::ToString>("toString", 0);
AddMethod<bool, &JSI_Sound::Play>("play", 0);
AddMethod<bool, &JSI_Sound::Loop>("loop", 0);
AddMethod<bool, &JSI_Sound::Free>("free", 0);
AddMethod<bool, &JSI_Sound::SetGain>("setGain", 0);
AddMethod<bool, &JSI_Sound::SetPitch>("setPitch", 0);
AddMethod<bool, &JSI_Sound::SetPosition>("setPosition", 0);
AddMethod<bool, &JSI_Sound::Fade>("fade", 0);
CJSObject<JSI_Sound>::ScriptingInit("Sound", &JSI_Sound::Construct, 1);
}
CStr JSI_Sound::ToString(JSContext* UNUSED(cx), uintN UNUSED(argc), jsval* UNUSED(argv))
{
return "[object Sound: " + h_filename(m_Handle).string() + "]";
}
JSBool JSI_Sound::Construct(JSContext* cx, JSObject* UNUSED(obj), uintN argc, jsval* argv, jsval* rval)
{
debug_assert(argc >= 1); // FIXME
CStrW filename;
if (! ToPrimitive<CStrW>(cx, argv[0], filename))
return JS_FALSE;
try
{
JSI_Sound* newObject = new JSI_Sound(filename);
// somewhat of a hack to avoid throwing exceptions if
// sound was disabled due to quickstart. see JSI_Sound().
if (newObject->m_SoundDisabled)
{
delete newObject;
*rval = JSVAL_NULL;
return JS_TRUE;
}
newObject->m_EngineOwned = false;
*rval = OBJECT_TO_JSVAL(newObject->GetScript());
}
catch (std::exception)
{
// failed, but this can happen easily enough that we don't want to
// return JS_FALSE (since that stops the script).
*rval = JSVAL_NULL;
}
return JS_TRUE;
}