/*
* Copyright 2013 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "SkTLS.h"
#include "SkTypes.h"
#include "SkError.h"
#include "SkErrorInternals.h"
#include <stdio.h>
#include <stdarg.h>
namespace {
void *CreateThreadError() { return new SkError(kNoError_SkError); }
void DeleteThreadError(void *v) { delete reinterpret_cast<SkError *>(v); }
#define THREAD_ERROR \
(*reinterpret_cast<SkError*>(SkTLS::Get(CreateThreadError, DeleteThreadError)))
void *CreateThreadErrorCallback() {
return new SkErrorCallbackFunction(SkErrorInternals::DefaultErrorCallback);
}
void DeleteThreadErrorCallback(void* v) {
delete reinterpret_cast<SkErrorCallbackFunction *>(v);
}
#define THREAD_ERROR_CALLBACK \
*(reinterpret_cast<SkErrorCallbackFunction *>(SkTLS::Get(CreateThreadErrorCallback, \
DeleteThreadErrorCallback)))
void *CreateThreadErrorContext() { return new void **(nullptr); }
void DeleteThreadErrorContext(void *v) { delete reinterpret_cast<void **>(v); }
#define THREAD_ERROR_CONTEXT \
(*reinterpret_cast<void **>(SkTLS::Get(CreateThreadErrorContext, DeleteThreadErrorContext)))
#define ERROR_STRING_LENGTH 2048
void *CreateThreadErrorString() { return new char[(ERROR_STRING_LENGTH)]; }
void DeleteThreadErrorString(void *v) { delete[] reinterpret_cast<char *>(v); }
#define THREAD_ERROR_STRING \
(reinterpret_cast<char *>(SkTLS::Get(CreateThreadErrorString, DeleteThreadErrorString)))
}
SkError SkGetLastError() {
return SkErrorInternals::GetLastError();
}
void SkClearLastError() {
SkErrorInternals::ClearError();
}
void SkSetErrorCallback(SkErrorCallbackFunction cb, void *context) {
SkErrorInternals::SetErrorCallback(cb, context);
}
const char *SkGetLastErrorString() {
return SkErrorInternals::GetLastErrorString();
}
// ------------ Private Error functions ---------
void SkErrorInternals::SetErrorCallback(SkErrorCallbackFunction cb, void *context) {
if (cb == nullptr) {
THREAD_ERROR_CALLBACK = SkErrorInternals::DefaultErrorCallback;
} else {
THREAD_ERROR_CALLBACK = cb;
}
THREAD_ERROR_CONTEXT = context;
}
void SkErrorInternals::DefaultErrorCallback(SkError code, void *context) {
SkDebugf("Skia Error: %s\n", SkGetLastErrorString());
}
void SkErrorInternals::ClearError() {
SkErrorInternals::SetError( kNoError_SkError, "All is well" );
}
SkError SkErrorInternals::GetLastError() {
return THREAD_ERROR;
}
const char *SkErrorInternals::GetLastErrorString() {
return THREAD_ERROR_STRING;
}
void SkErrorInternals::SetError(SkError code, const char *fmt, ...) {
THREAD_ERROR = code;
va_list args;
char *str = THREAD_ERROR_STRING;
const char *error_name = nullptr;
switch( code ) {
case kNoError_SkError:
error_name = "No Error";
break;
case kInvalidArgument_SkError:
error_name = "Invalid Argument";
break;
case kInvalidOperation_SkError:
error_name = "Invalid Operation";
break;
case kInvalidHandle_SkError:
error_name = "Invalid Handle";
break;
case kInvalidPaint_SkError:
error_name = "Invalid Paint";
break;
case kOutOfMemory_SkError:
error_name = "Out Of Memory";
break;
case kParseError_SkError:
error_name = "Parse Error";
break;
default:
error_name = "Unknown error";
break;
}
sprintf( str, "%s: ", error_name );
int string_left = SkToInt(ERROR_STRING_LENGTH - strlen(str));
str += strlen(str);
va_start( args, fmt );
vsnprintf( str, string_left, fmt, args );
va_end( args );
SkErrorCallbackFunction fn = THREAD_ERROR_CALLBACK;
if (fn && code != kNoError_SkError) {
fn(code, THREAD_ERROR_CONTEXT);
}
}