mq5.1-source.src.share.cclient.error.ErrorTrace.cpp Maven / Gradle / Ivy
The newest version!
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2000-2010 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
/*
* @(#)ErrorTrace.cpp 1.11 06/26/07
*/
#include
#include
#include
#include "ErrorTrace.h"
#define ASSERT assert
#ifdef __cplusplus
extern "C" {
#endif /* __cplusplus */
static PRUintn _errorTraceKey;
static PRCallOnceType once;
static void PR_CALLBACK deleteErrorTrace(void * data)
{
PRUint32 i;
ErrorTrace * trace = (ErrorTrace *)data;
if (trace == NULL) return;
if (trace->trace != NULL) {
for (i = 0; i < trace->num_elements; i++) {
ASSERT( trace->trace[i] != NULL );
free(trace->trace[i]);
}
trace->num_elements = 0;
free(trace->trace);
}
trace->num_allocated = 0;
free(trace);
}
static PRStatus once_func(void)
{
return PR_NewThreadPrivateIndex(&_errorTraceKey, &deleteErrorTrace);
}
static PRStatus
getErrorTraceAlloc(ErrorTrace **trace, PRBool alloc)
{
if (once.initialized == 0) {
if(PR_CallOnce(&once, once_func) != PR_SUCCESS) return PR_FAILURE;
}
*trace = (ErrorTrace *)PR_GetThreadPrivate(_errorTraceKey);
if (*trace != (ErrorTrace *)NULL) {
if ((*trace)->usable == PR_FALSE) {
*trace = (ErrorTrace *)NULL;
return PR_FAILURE;
}
return PR_SUCCESS;
}
if (alloc == PR_FALSE) return PR_SUCCESS;
*trace = (ErrorTrace *)calloc(1, sizeof(ErrorTrace));
if (*trace == (ErrorTrace *)NULL) {
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
(*trace)->trace = (char **)calloc(16, sizeof(char *));
if ((*trace)->trace == NULL) {
free(*trace);
*trace = NULL;
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
(*trace)->usable = PR_TRUE;
(*trace)->num_allocated = 16;
(*trace)->num_elements = 0;
if (PR_SetThreadPrivate(_errorTraceKey, *trace) != PR_SUCCESS) {
PR_SetThreadPrivate(_errorTraceKey, NULL);
free((*trace)->trace);
free(*trace);
*trace = NULL;
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
return PR_SUCCESS;
}
#ifdef __cplusplus
}
#endif /* __cplusplus */
PRStatus
getErrorTrace(ErrorTrace **trace)
{
return getErrorTraceAlloc(trace, PR_FALSE);
}
PRStatus
setErrorTraceElement(const char * method ,
const char * file, PRInt32 lineNumber,
const char * errorType, PRUint32 errorCode)
{
return setVErrorTraceElement(method, file, lineNumber, errorType, errorCode, "");
}
PRStatus
setVErrorTraceElement(const char * method ,
const char * file, PRInt32 lineNumber,
const char * errorType, PRUint32 errorCode,
const char * const format, ...)
{
va_list argptr;
ErrorTrace * trace = NULL;
char * element = NULL;
/* :::::\0 line error */
size_t size = 7 + 12 + 12;
char ** newtrace = NULL;
char errorStr[10000];
size_t errorStrLen = 0;
va_start(argptr, format);
PR_vsnprintf(errorStr, sizeof(errorStr), format, argptr);
va_end(argptr);
errorStr[sizeof(errorStr)-1] = '\0';
errorStrLen = strlen(errorStr);
PRStatus status = getErrorTraceAlloc(&trace, PR_TRUE);
if (status != PR_SUCCESS) return status;
if (trace == (ErrorTrace *)NULL) {
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
if (method != NULL) size += strlen(method);
if (file != NULL) size += strlen(file);
if (errorType != NULL) size += strlen(errorType);
size += errorStrLen;
element = (char *)malloc(size);
if (element == NULL) {
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
if (trace->num_elements == trace->num_allocated -1) {
newtrace = (char **)realloc(trace->trace, (trace->num_allocated+16) * sizeof(char *));
if (newtrace == NULL) {
free(element);
PR_SetError(PR_OUT_OF_MEMORY_ERROR, 0);
return PR_FAILURE;
}
trace->trace = newtrace;
trace->num_allocated += 16;
}
if (errorStrLen == 0) {
sprintf(element, "%s:%s:%d:%s:%d", method, file, lineNumber, errorType, errorCode);
} else {
sprintf(element, "%s:%s:%d:%s:%d:%s", method, file, lineNumber, errorType, errorCode, errorStr);
}
ASSERT( trace->num_elements < (trace->num_allocated-1) );
trace->trace[trace->num_elements] = element;
trace->trace[++(trace->num_elements)] = NULL;
return PR_SUCCESS;
}
PRStatus
clearErrorTrace(PRBool all)
{
ErrorTrace * trace = NULL;
PRUint32 i;
PRStatus status = getErrorTraceAlloc(&trace, PR_FALSE);
if (status != PR_SUCCESS) return status;
if (trace == (ErrorTrace *)NULL) return PR_SUCCESS;
if (trace->usable == PR_FALSE) return PR_FAILURE;
if (all == PR_TRUE) {
if (PR_SetThreadPrivate(_errorTraceKey, NULL) != PR_SUCCESS) {
trace->usable = PR_FALSE;
return PR_FAILURE;
}
return PR_SUCCESS;
}
ASSERT( trace->trace != NULL );
for (i = 0; i < trace->num_elements; i++) {
ASSERT( trace->trace[i] != NULL );
free(trace->trace[i]);
}
trace->num_elements = 0;
return PR_SUCCESS;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy