error.c
上传用户:lyxiangda
上传日期:2007-01-12
资源大小:3042k
文件大小:7k
源码类别:

CA认证

开发平台:

WINDOWS

  1. /* 
  2.  * The contents of this file are subject to the Mozilla Public
  3.  * License Version 1.1 (the "License"); you may not use this file
  4.  * except in compliance with the License. You may obtain a copy of
  5.  * the License at http://www.mozilla.org/MPL/
  6.  * 
  7.  * Software distributed under the License is distributed on an "AS
  8.  * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
  9.  * implied. See the License for the specific language governing
  10.  * rights and limitations under the License.
  11.  * 
  12.  * The Original Code is the Netscape security libraries.
  13.  * 
  14.  * The Initial Developer of the Original Code is Netscape
  15.  * Communications Corporation.  Portions created by Netscape are 
  16.  * Copyright (C) 1994-2000 Netscape Communications Corporation.  All
  17.  * Rights Reserved.
  18.  * 
  19.  * Contributor(s):
  20.  * 
  21.  * Alternatively, the contents of this file may be used under the
  22.  * terms of the GNU General Public License Version 2 or later (the
  23.  * "GPL"), in which case the provisions of the GPL are applicable 
  24.  * instead of those above.  If you wish to allow use of your 
  25.  * version of this file only under the terms of the GPL and not to
  26.  * allow others to use your version of this file under the MPL,
  27.  * indicate your decision by deleting the provisions above and
  28.  * replace them with the notice and other provisions required by
  29.  * the GPL.  If you do not delete the provisions above, a recipient
  30.  * may use your version of this file under either the MPL or the
  31.  * GPL.
  32.  */
  33. #ifdef DEBUG
  34. static const char CVS_ID[] = "@(#) $RCSfile: error.c,v $ $Revision: 1.2 $ $Date: 2000/05/17 20:19:23 $ $Name: NSS_3_1_1_RTM $";
  35. #endif /* DEBUG */
  36. /*
  37.  * error.c
  38.  *
  39.  * This file contains the code implementing the per-thread error 
  40.  * stacks upon which most NSS routines report their errors.
  41.  */
  42. #ifndef BASE_H
  43. #include "base.h"
  44. #endif /* BASE_H */
  45. /*
  46.  * The stack itself has a header, and a sequence of integers.
  47.  * The header records the amount of space (as measured in stack
  48.  * slots) already allocated for the stack, and the count of the
  49.  * number of records currently being used.
  50.  */
  51. struct stack_header_str {
  52.   PRUint16 space;
  53.   PRUint16 count;
  54. };
  55. struct error_stack_str {
  56.   struct stack_header_str header;
  57.   PRInt32 stack[1];
  58. };
  59. typedef struct error_stack_str error_stack;
  60. /*
  61.  * error_stack_index
  62.  *
  63.  * Thread-private data must be indexed.  This is that index.
  64.  * See PR_NewThreadPrivateIndex for more information.
  65.  */
  66. static PRUintn error_stack_index;
  67. /*
  68.  * call_once
  69.  *
  70.  * The thread-private index must be obtained (once!) at runtime.
  71.  * This block is used for that one-time call.
  72.  */
  73. static PRCallOnceType error_call_once;
  74. /*
  75.  * error_once_function
  76.  *
  77.  * This is the once-called callback.
  78.  */
  79. static PRStatus
  80. error_once_function
  81. (
  82.   void
  83. )
  84. {
  85.   return nss_NewThreadPrivateIndex(&error_stack_index);
  86.   /* return PR_NewThreadPrivateIndex(&error_stack_index, PR_Free); */
  87. }
  88. /*
  89.  * error_get_my_stack
  90.  *
  91.  * This routine returns the calling thread's error stack, creating
  92.  * it if necessary.  It may return NULL upon error, which implicitly
  93.  * means that it ran out of memory.
  94.  */
  95. static error_stack *
  96. error_get_my_stack
  97. (
  98.   void
  99. )
  100. {
  101.   PRStatus st;
  102.   error_stack *rv;
  103.   PRUintn new_size;
  104.   PRUint32 new_bytes;
  105.   error_stack *new_stack;
  106.   if( 0 == error_stack_index ) {
  107.     st = PR_CallOnce(&error_call_once, error_once_function);
  108.     if( PR_SUCCESS != st ) {
  109.       return (error_stack *)NULL;
  110.     }
  111.   }
  112.   rv = (error_stack *)nss_GetThreadPrivate(error_stack_index);
  113.   if( (error_stack *)NULL == rv ) {
  114.     /* Doesn't exist; create one */
  115.     new_size = 16;
  116.   } else {
  117.     if( rv->header.count == rv->header.space ) {
  118.       /* Too small, expand it */
  119.       new_size = rv->header.space + 16;
  120.     } else {
  121.       /* Okay, return it */
  122.       return rv;
  123.     }
  124.   }
  125.   new_bytes = (new_size * sizeof(PRInt32)) + 
  126.     sizeof(struct stack_header_str);
  127.   /* Use NSPR's calloc/realloc, not NSS's, to avoid loops! */
  128.   if( (error_stack *)NULL == rv ) {
  129.     new_stack = PR_Calloc(1, new_bytes);
  130.   } else {
  131.     new_stack = PR_Realloc(rv, new_bytes);
  132.   }
  133.   
  134.   if( (error_stack *)NULL != new_stack ) {
  135.     new_stack->header.space = new_size;
  136.   }
  137.   /* Set the value, whether or not the allocation worked */
  138.   nss_SetThreadPrivate(error_stack_index, new_stack);
  139.   return new_stack;
  140. }
  141. /*
  142.  * The error stack
  143.  *
  144.  * The public methods relating to the error stack are:
  145.  *
  146.  *  NSS_GetError
  147.  *  NSS_GetErrorStack
  148.  *
  149.  * The nonpublic methods relating to the error stack are:
  150.  *
  151.  *  nss_SetError
  152.  *  nss_ClearErrorStack
  153.  *
  154.  */
  155. /*
  156.  * NSS_GetError
  157.  *
  158.  * This routine returns the highest-level (most general) error set
  159.  * by the most recent NSS library routine called by the same thread
  160.  * calling this routine.
  161.  *
  162.  * This routine cannot fail.  However, it may return zero, which
  163.  * indicates that the previous NSS library call did not set an error.
  164.  *
  165.  * Return value:
  166.  *  0 if no error has been set
  167.  *  A nonzero error number
  168.  */
  169. NSS_IMPLEMENT PRInt32
  170. NSS_GetError
  171. (
  172.   void
  173. )
  174. {
  175.   error_stack *es = error_get_my_stack();
  176.   if( (error_stack *)NULL == es ) {
  177.     return NSS_ERROR_NO_MEMORY; /* Good guess! */
  178.   }
  179.   if( 0 == es->header.count ) {
  180.     return 0;
  181.   }
  182.   return es->stack[ es->header.count-1 ];
  183. }
  184. /*
  185.  * NSS_GetErrorStack
  186.  *
  187.  * This routine returns a pointer to an array of integers, containing
  188.  * the entire sequence or "stack" of errors set by the most recent NSS
  189.  * library routine called by the same thread calling this routine.
  190.  * NOTE: the caller DOES NOT OWN the memory pointed to by the return
  191.  * value.  The pointer will remain valid until the calling thread
  192.  * calls another NSS routine.  The lowest-level (most specific) error 
  193.  * is first in the array, and the highest-level is last.  The array is
  194.  * zero-terminated.  This routine may return NULL upon error; this
  195.  * indicates a low-memory situation.
  196.  *
  197.  * Return value:
  198.  *  NULL upon error, which is an implied NSS_ERROR_NO_MEMORY
  199.  *  A NON-caller-owned pointer to an array of integers
  200.  */
  201. NSS_IMPLEMENT PRInt32 *
  202. NSS_GetErrorStack
  203. (
  204.   void
  205. )
  206. {
  207.   error_stack *es = error_get_my_stack();
  208.   if( (error_stack *)NULL == es ) {
  209.     return (PRInt32 *)NULL;
  210.   }
  211.   /* Make sure it's terminated */
  212.   es->stack[ es->header.count ] = 0;
  213.   return es->stack;
  214. }
  215. /*
  216.  * nss_SetError
  217.  *
  218.  * This routine places a new error code on the top of the calling 
  219.  * thread's error stack.  Calling this routine wiht an error code
  220.  * of zero will clear the error stack.
  221.  */
  222. NSS_IMPLEMENT void
  223. nss_SetError
  224. (
  225.   PRUint32 error
  226. )
  227. {
  228.   error_stack *es;
  229.   if( 0 == error ) {
  230.     nss_ClearErrorStack();
  231.     return;
  232.   }
  233.   es = error_get_my_stack();
  234.   if( (error_stack *)NULL == es ) {
  235.     /* Oh, well. */
  236.     return;
  237.   }
  238.   es->stack[ es->header.count ] = error;
  239.   es->header.count++;
  240.   return;
  241. }
  242. /*
  243.  * nss_ClearErrorStack
  244.  *
  245.  * This routine clears the calling thread's error stack.
  246.  */
  247. NSS_IMPLEMENT void
  248. nss_ClearErrorStack
  249. (
  250.   void
  251. )
  252. {
  253.   error_stack *es = error_get_my_stack();
  254.   if( (error_stack *)NULL == es ) {
  255.     /* Oh, well. */
  256.     return;
  257.   }
  258.   es->header.count = 0;
  259.   es->stack[0] = 0;
  260.   return;
  261. }