summaryrefslogtreecommitdiffstats
path: root/src/3rdparty/angle/src/libANGLE/Error.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/3rdparty/angle/src/libANGLE/Error.h')
-rw-r--r--src/3rdparty/angle/src/libANGLE/Error.h205
1 files changed, 185 insertions, 20 deletions
diff --git a/src/3rdparty/angle/src/libANGLE/Error.h b/src/3rdparty/angle/src/libANGLE/Error.h
index a5f760956a..1d57bb8707 100644
--- a/src/3rdparty/angle/src/libANGLE/Error.h
+++ b/src/3rdparty/angle/src/libANGLE/Error.h
@@ -9,23 +9,89 @@
#ifndef LIBANGLE_ERROR_H_
#define LIBANGLE_ERROR_H_
-#include "angle_gl.h"
#include <EGL/egl.h>
+#include <EGL/eglext.h>
+#include "angle_gl.h"
+#include "common/angleutils.h"
+#include "common/debug.h"
-#include <string>
#include <memory>
+#include <ostream>
+#include <string>
+
+namespace angle
+{
+template <typename ErrorT, typename ResultT, typename ErrorBaseT, ErrorBaseT NoErrorVal>
+class ANGLE_NO_DISCARD ErrorOrResultBase
+{
+ public:
+ ErrorOrResultBase(const ErrorT &error) : mError(error) {}
+ ErrorOrResultBase(ErrorT &&error) : mError(std::move(error)) {}
+
+ ErrorOrResultBase(ResultT &&result) : mError(NoErrorVal), mResult(std::forward<ResultT>(result))
+ {
+ }
+
+ ErrorOrResultBase(const ResultT &result) : mError(NoErrorVal), mResult(result) {}
+
+ bool isError() const { return mError.isError(); }
+ const ErrorT &getError() const { return mError; }
+ ResultT &&getResult() { return std::move(mResult); }
+
+ private:
+ ErrorT mError;
+ ResultT mResult;
+};
+
+template <typename ErrorT, typename ErrorBaseT, ErrorBaseT NoErrorVal, typename CodeT, CodeT EnumT>
+class ErrorStreamBase : angle::NonCopyable
+{
+ public:
+ ErrorStreamBase() : mID(EnumT) {}
+ ErrorStreamBase(GLuint id) : mID(id) {}
+
+ template <typename T>
+ ErrorStreamBase &operator<<(T value)
+ {
+ mErrorStream << value;
+ return *this;
+ }
+
+ operator ErrorT() { return ErrorT(EnumT, mID, mErrorStream.str()); }
+
+ template <typename ResultT>
+ operator ErrorOrResultBase<ErrorT, ResultT, ErrorBaseT, NoErrorVal>()
+ {
+ return static_cast<ErrorT>(*this);
+ }
+
+ private:
+ GLuint mID;
+ std::ostringstream mErrorStream;
+};
+} // namespace angle
+
+namespace egl
+{
+class Error;
+} // namespace egl
namespace gl
{
-class Error final
+class ANGLE_NO_DISCARD Error final
{
public:
explicit inline Error(GLenum errorCode);
- Error(GLenum errorCode, const char *msg, ...);
- Error(GLenum errorCode, GLuint id, const char *msg, ...);
+ Error(GLenum errorCode, std::string &&message);
+ Error(GLenum errorCode, GLuint id, std::string &&message);
inline Error(const Error &other);
inline Error(Error &&other);
+ inline ~Error() = default;
+
+ // automatic error type conversion
+ inline Error(egl::Error &&eglErr);
+ inline Error(egl::Error eglErr);
inline Error &operator=(const Error &other);
inline Error &operator=(Error &&other);
@@ -43,40 +109,60 @@ class Error final
private:
void createMessageString() const;
+ friend std::ostream &operator<<(std::ostream &os, const Error &err);
+ friend class egl::Error;
+
GLenum mCode;
GLuint mID;
mutable std::unique_ptr<std::string> mMessage;
};
-template <typename T>
-class ErrorOrResult
+template <typename ResultT>
+using ErrorOrResult = angle::ErrorOrResultBase<Error, ResultT, GLenum, GL_NO_ERROR>;
+
+namespace priv
{
- public:
- ErrorOrResult(const gl::Error &error) : mError(error) {}
- ErrorOrResult(T &&result) : mError(GL_NO_ERROR), mResult(std::move(result)) {}
- bool isError() const { return mError.isError(); }
- const gl::Error &getError() const { return mError; }
- T &&getResult() { return std::move(mResult); }
+template <GLenum EnumT>
+using ErrorStream = angle::ErrorStreamBase<Error, GLenum, GL_NO_ERROR, GLenum, EnumT>;
- private:
- Error mError;
- T mResult;
-};
+} // namespace priv
+
+using InternalError = priv::ErrorStream<GL_INVALID_OPERATION>;
+
+using InvalidEnum = priv::ErrorStream<GL_INVALID_ENUM>;
+using InvalidValue = priv::ErrorStream<GL_INVALID_VALUE>;
+using InvalidOperation = priv::ErrorStream<GL_INVALID_OPERATION>;
+using StackOverflow = priv::ErrorStream<GL_STACK_OVERFLOW>;
+using StackUnderflow = priv::ErrorStream<GL_STACK_UNDERFLOW>;
+using OutOfMemory = priv::ErrorStream<GL_OUT_OF_MEMORY>;
+using InvalidFramebufferOperation = priv::ErrorStream<GL_INVALID_FRAMEBUFFER_OPERATION>;
+
+inline Error NoError()
+{
+ return Error(GL_NO_ERROR);
+}
+
+using LinkResult = ErrorOrResult<bool>;
} // namespace gl
namespace egl
{
-class Error final
+class ANGLE_NO_DISCARD Error final
{
public:
explicit inline Error(EGLint errorCode);
- Error(EGLint errorCode, const char *msg, ...);
- Error(EGLint errorCode, EGLint id, const char *msg, ...);
+ Error(EGLint errorCode, std::string &&message);
+ Error(EGLint errorCode, EGLint id, std::string &&message);
inline Error(const Error &other);
inline Error(Error &&other);
+ inline ~Error() = default;
+
+ // automatic error type conversion
+ inline Error(gl::Error &&glErr);
+ inline Error(gl::Error glErr);
inline Error &operator=(const Error &other);
inline Error &operator=(Error &&other);
@@ -90,13 +176,92 @@ class Error final
private:
void createMessageString() const;
+ friend std::ostream &operator<<(std::ostream &os, const Error &err);
+ friend class gl::Error;
+
EGLint mCode;
EGLint mID;
mutable std::unique_ptr<std::string> mMessage;
};
+template <typename ResultT>
+using ErrorOrResult = angle::ErrorOrResultBase<Error, ResultT, EGLint, EGL_SUCCESS>;
+
+namespace priv
+{
+
+template <EGLint EnumT>
+using ErrorStream = angle::ErrorStreamBase<Error, EGLint, EGL_SUCCESS, EGLint, EnumT>;
+
+} // namespace priv
+
+using EglNotInitialized = priv::ErrorStream<EGL_NOT_INITIALIZED>;
+using EglBadAccess = priv::ErrorStream<EGL_BAD_ACCESS>;
+using EglBadAlloc = priv::ErrorStream<EGL_BAD_ALLOC>;
+using EglBadAttribute = priv::ErrorStream<EGL_BAD_ATTRIBUTE>;
+using EglBadConfig = priv::ErrorStream<EGL_BAD_CONFIG>;
+using EglBadContext = priv::ErrorStream<EGL_BAD_CONTEXT>;
+using EglBadCurrentSurface = priv::ErrorStream<EGL_BAD_CURRENT_SURFACE>;
+using EglBadDisplay = priv::ErrorStream<EGL_BAD_DISPLAY>;
+using EglBadMatch = priv::ErrorStream<EGL_BAD_MATCH>;
+using EglBadNativeWindow = priv::ErrorStream<EGL_BAD_NATIVE_WINDOW>;
+using EglBadParameter = priv::ErrorStream<EGL_BAD_PARAMETER>;
+using EglBadSurface = priv::ErrorStream<EGL_BAD_SURFACE>;
+using EglContextLost = priv::ErrorStream<EGL_CONTEXT_LOST>;
+using EglBadStream = priv::ErrorStream<EGL_BAD_STREAM_KHR>;
+using EglBadState = priv::ErrorStream<EGL_BAD_STATE_KHR>;
+using EglBadDevice = priv::ErrorStream<EGL_BAD_DEVICE_EXT>;
+
+inline Error NoError()
+{
+ return Error(EGL_SUCCESS);
+}
+
} // namespace egl
+#define ANGLE_CONCAT1(x, y) x##y
+#define ANGLE_CONCAT2(x, y) ANGLE_CONCAT1(x, y)
+#define ANGLE_LOCAL_VAR ANGLE_CONCAT2(_localVar, __LINE__)
+
+#define ANGLE_TRY_TEMPLATE(EXPR, FUNC) \
+ { \
+ auto ANGLE_LOCAL_VAR = EXPR; \
+ if (ANGLE_LOCAL_VAR.isError()) \
+ { \
+ FUNC(ANGLE_LOCAL_VAR); \
+ } \
+ } \
+ ANGLE_EMPTY_STATEMENT
+
+#define ANGLE_RETURN(X) return X;
+#define ANGLE_TRY(EXPR) ANGLE_TRY_TEMPLATE(EXPR, ANGLE_RETURN);
+
+#define ANGLE_TRY_RESULT(EXPR, RESULT) \
+ { \
+ auto ANGLE_LOCAL_VAR = EXPR; \
+ if (ANGLE_LOCAL_VAR.isError()) \
+ { \
+ return ANGLE_LOCAL_VAR.getError(); \
+ } \
+ RESULT = ANGLE_LOCAL_VAR.getResult(); \
+ } \
+ ANGLE_EMPTY_STATEMENT
+
+// TODO(jmadill): Introduce way to store errors to a const Context.
+#define ANGLE_SWALLOW_ERR(EXPR) \
+ { \
+ auto ANGLE_LOCAL_VAR = EXPR; \
+ if (ANGLE_LOCAL_VAR.isError()) \
+ { \
+ ERR() << "Unhandled internal error: " << ANGLE_LOCAL_VAR; \
+ } \
+ } \
+ ANGLE_EMPTY_STATEMENT
+
+#undef ANGLE_LOCAL_VAR
+#undef ANGLE_CONCAT2
+#undef ANGLE_CONCAT1
+
#include "Error.inl"
#endif // LIBANGLE_ERROR_H_