123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- /**
- * OpenAL cross platform audio library
- * Copyright (C) 1999-2000 by authors.
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library 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
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
- * Or go to http://www.gnu.org/copyleft/lgpl.html
- */
- #include "config.h"
- #include "error.h"
- #ifdef _WIN32
- #define WIN32_LEAN_AND_MEAN
- #include <windows.h>
- #endif
- #include <atomic>
- #include <csignal>
- #include <cstdarg>
- #include <cstdio>
- #include <cstdlib>
- #include <cstring>
- #include <limits>
- #include <optional>
- #include <string>
- #include <utility>
- #include <vector>
- #include "AL/al.h"
- #include "AL/alc.h"
- #include "al/debug.h"
- #include "alc/alconfig.h"
- #include "alc/context.h"
- #include "alc/inprogext.h"
- #include "core/logging.h"
- #include "opthelpers.h"
- #include "strutils.h"
- namespace al {
- context_error::context_error(ALenum code, const char *msg, ...) : mErrorCode{code}
- {
- /* NOLINTBEGIN(*-array-to-pointer-decay) */
- std::va_list args;
- va_start(args, msg);
- setMessage(msg, args);
- va_end(args);
- /* NOLINTEND(*-array-to-pointer-decay) */
- }
- context_error::~context_error() = default;
- } /* namespace al */
- void ALCcontext::setError(ALenum errorCode, const char *msg, ...)
- {
- auto message = std::vector<char>(256);
- /* NOLINTBEGIN(*-array-to-pointer-decay) */
- std::va_list args, args2;
- va_start(args, msg);
- va_copy(args2, args);
- int msglen{std::vsnprintf(message.data(), message.size(), msg, args)};
- if(msglen >= 0 && static_cast<size_t>(msglen) >= message.size())
- {
- message.resize(static_cast<size_t>(msglen) + 1u);
- msglen = std::vsnprintf(message.data(), message.size(), msg, args2);
- }
- va_end(args2);
- va_end(args);
- /* NOLINTEND(*-array-to-pointer-decay) */
- if(msglen >= 0)
- msg = message.data();
- else
- {
- msg = "<internal error constructing message>";
- msglen = static_cast<int>(strlen(msg));
- }
- WARN("Error generated on context %p, code 0x%04x, \"%s\"\n",
- decltype(std::declval<void*>()){this}, errorCode, msg);
- if(TrapALError)
- {
- #ifdef _WIN32
- /* DebugBreak will cause an exception if there is no debugger */
- if(IsDebuggerPresent())
- DebugBreak();
- #elif defined(SIGTRAP)
- raise(SIGTRAP);
- #endif
- }
- if(mLastThreadError.get() == AL_NO_ERROR)
- mLastThreadError.set(errorCode);
- debugMessage(DebugSource::API, DebugType::Error, 0, DebugSeverity::High,
- {msg, static_cast<uint>(msglen)});
- }
- /* Special-case alGetError since it (potentially) raises a debug signal and
- * returns a non-default value for a null context.
- */
- AL_API auto AL_APIENTRY alGetError() noexcept -> ALenum
- {
- if(auto context = GetContextRef()) LIKELY
- return alGetErrorDirect(context.get());
- auto get_value = [](const char *envname, const char *optname) -> ALenum
- {
- auto optstr = al::getenv(envname);
- if(!optstr)
- optstr = ConfigValueStr({}, "game_compat", optname);
- if(optstr)
- {
- char *end{};
- auto value = std::strtoul(optstr->c_str(), &end, 0);
- if(end && *end == '\0' && value <= std::numeric_limits<ALenum>::max())
- return static_cast<ALenum>(value);
- ERR("Invalid default error value: \"%s\"", optstr->c_str());
- }
- return AL_INVALID_OPERATION;
- };
- static const ALenum deferror{get_value("__ALSOFT_DEFAULT_ERROR", "default-error")};
- WARN("Querying error state on null context (implicitly 0x%04x)\n", deferror);
- if(TrapALError)
- {
- #ifdef _WIN32
- if(IsDebuggerPresent())
- DebugBreak();
- #elif defined(SIGTRAP)
- raise(SIGTRAP);
- #endif
- }
- return deferror;
- }
- FORCE_ALIGN ALenum AL_APIENTRY alGetErrorDirect(ALCcontext *context) noexcept
- {
- ALenum ret{context->mLastThreadError.get()};
- if(ret != AL_NO_ERROR) UNLIKELY
- context->mLastThreadError.set(AL_NO_ERROR);
- return ret;
- }
|