| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597 | 
							- /***************************************************************************
 
-  *                                  _   _ ____  _
 
-  *  Project                     ___| | | |  _ \| |
 
-  *                             / __| | | | |_) | |
 
-  *                            | (__| |_| |  _ <| |___
 
-  *                             \___|\___/|_| \_\_____|
 
-  *
 
-  * Copyright (C) 1998 - 2020, Daniel Stenberg, <[email protected]>, et al.
 
-  *
 
-  * This software is licensed as described in the file COPYING, which
 
-  * you should have received as part of this distribution. The terms
 
-  * are also available at https://curl.haxx.se/docs/copyright.html.
 
-  *
 
-  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
 
-  * copies of the Software, and permit persons to whom the Software is
 
-  * furnished to do so, under the terms of the COPYING file.
 
-  *
 
-  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 
-  * KIND, either express or implied.
 
-  *
 
-  ***************************************************************************/
 
- #include "curl_setup.h"
 
- #include <curl/curl.h>
 
- #include "urldata.h"
 
- #include "getinfo.h"
 
- #include "vtls/vtls.h"
 
- #include "connect.h" /* Curl_getconnectinfo() */
 
- #include "progress.h"
 
- /* The last #include files should be: */
 
- #include "curl_memory.h"
 
- #include "memdebug.h"
 
- /*
 
-  * Initialize statistical and informational data.
 
-  *
 
-  * This function is called in curl_easy_reset, curl_easy_duphandle and at the
 
-  * beginning of a perform session. It must reset the session-info variables,
 
-  * in particular all variables in struct PureInfo.
 
-  */
 
- CURLcode Curl_initinfo(struct Curl_easy *data)
 
- {
 
-   struct Progress *pro = &data->progress;
 
-   struct PureInfo *info = &data->info;
 
-   pro->t_nslookup = 0;
 
-   pro->t_connect = 0;
 
-   pro->t_appconnect = 0;
 
-   pro->t_pretransfer = 0;
 
-   pro->t_starttransfer = 0;
 
-   pro->timespent = 0;
 
-   pro->t_redirect = 0;
 
-   pro->is_t_startransfer_set = false;
 
-   info->httpcode = 0;
 
-   info->httpproxycode = 0;
 
-   info->httpversion = 0;
 
-   info->filetime = -1; /* -1 is an illegal time and thus means unknown */
 
-   info->timecond = FALSE;
 
-   info->header_size = 0;
 
-   info->request_size = 0;
 
-   info->proxyauthavail = 0;
 
-   info->httpauthavail = 0;
 
-   info->numconnects = 0;
 
-   free(info->contenttype);
 
-   info->contenttype = NULL;
 
-   free(info->wouldredirect);
 
-   info->wouldredirect = NULL;
 
-   info->conn_primary_ip[0] = '\0';
 
-   info->conn_local_ip[0] = '\0';
 
-   info->conn_primary_port = 0;
 
-   info->conn_local_port = 0;
 
-   info->retry_after = 0;
 
-   info->conn_scheme = 0;
 
-   info->conn_protocol = 0;
 
- #ifdef USE_SSL
 
-   Curl_ssl_free_certinfo(data);
 
- #endif
 
-   return CURLE_OK;
 
- }
 
- static CURLcode getinfo_char(struct Curl_easy *data, CURLINFO info,
 
-                              const char **param_charp)
 
- {
 
-   switch(info) {
 
-   case CURLINFO_EFFECTIVE_URL:
 
-     *param_charp = data->change.url?data->change.url:(char *)"";
 
-     break;
 
-   case CURLINFO_EFFECTIVE_METHOD: {
 
-     const char *m = data->set.str[STRING_CUSTOMREQUEST];
 
-     if(!m) {
 
-       if(data->set.opt_no_body)
 
-         m = "HEAD";
 
-       else {
 
-         switch(data->state.httpreq) {
 
-         case HTTPREQ_POST:
 
-         case HTTPREQ_POST_FORM:
 
-         case HTTPREQ_POST_MIME:
 
-           m = "POST";
 
-           break;
 
-         case HTTPREQ_PUT:
 
-           m = "PUT";
 
-           break;
 
-         default: /* this should never happen */
 
-         case HTTPREQ_GET:
 
-           m = "GET";
 
-           break;
 
-         case HTTPREQ_HEAD:
 
-           m = "HEAD";
 
-           break;
 
-         }
 
-       }
 
-     }
 
-     *param_charp = m;
 
-   }
 
-     break;
 
-   case CURLINFO_CONTENT_TYPE:
 
-     *param_charp = data->info.contenttype;
 
-     break;
 
-   case CURLINFO_PRIVATE:
 
-     *param_charp = (char *) data->set.private_data;
 
-     break;
 
-   case CURLINFO_FTP_ENTRY_PATH:
 
-     /* Return the entrypath string from the most recent connection.
 
-        This pointer was copied from the connectdata structure by FTP.
 
-        The actual string may be free()ed by subsequent libcurl calls so
 
-        it must be copied to a safer area before the next libcurl call.
 
-        Callers must never free it themselves. */
 
-     *param_charp = data->state.most_recent_ftp_entrypath;
 
-     break;
 
-   case CURLINFO_REDIRECT_URL:
 
-     /* Return the URL this request would have been redirected to if that
 
-        option had been enabled! */
 
-     *param_charp = data->info.wouldredirect;
 
-     break;
 
-   case CURLINFO_PRIMARY_IP:
 
-     /* Return the ip address of the most recent (primary) connection */
 
-     *param_charp = data->info.conn_primary_ip;
 
-     break;
 
-   case CURLINFO_LOCAL_IP:
 
-     /* Return the source/local ip address of the most recent (primary)
 
-        connection */
 
-     *param_charp = data->info.conn_local_ip;
 
-     break;
 
-   case CURLINFO_RTSP_SESSION_ID:
 
-     *param_charp = data->set.str[STRING_RTSP_SESSION_ID];
 
-     break;
 
-   case CURLINFO_SCHEME:
 
-     *param_charp = data->info.conn_scheme;
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- static CURLcode getinfo_long(struct Curl_easy *data, CURLINFO info,
 
-                              long *param_longp)
 
- {
 
-   curl_socket_t sockfd;
 
-   union {
 
-     unsigned long *to_ulong;
 
-     long          *to_long;
 
-   } lptr;
 
- #ifdef DEBUGBUILD
 
-   char *timestr = getenv("CURL_TIME");
 
-   if(timestr) {
 
-     unsigned long val = strtol(timestr, NULL, 10);
 
-     switch(info) {
 
-     case CURLINFO_LOCAL_PORT:
 
-       *param_longp = (long)val;
 
-       return CURLE_OK;
 
-     default:
 
-       break;
 
-     }
 
-   }
 
-   /* use another variable for this to allow different values */
 
-   timestr = getenv("CURL_DEBUG_SIZE");
 
-   if(timestr) {
 
-     unsigned long val = strtol(timestr, NULL, 10);
 
-     switch(info) {
 
-     case CURLINFO_HEADER_SIZE:
 
-     case CURLINFO_REQUEST_SIZE:
 
-       *param_longp = (long)val;
 
-       return CURLE_OK;
 
-     default:
 
-       break;
 
-     }
 
-   }
 
- #endif
 
-   switch(info) {
 
-   case CURLINFO_RESPONSE_CODE:
 
-     *param_longp = data->info.httpcode;
 
-     break;
 
-   case CURLINFO_HTTP_CONNECTCODE:
 
-     *param_longp = data->info.httpproxycode;
 
-     break;
 
-   case CURLINFO_FILETIME:
 
-     if(data->info.filetime > LONG_MAX)
 
-       *param_longp = LONG_MAX;
 
-     else if(data->info.filetime < LONG_MIN)
 
-       *param_longp = LONG_MIN;
 
-     else
 
-       *param_longp = (long)data->info.filetime;
 
-     break;
 
-   case CURLINFO_HEADER_SIZE:
 
-     *param_longp = (long)data->info.header_size;
 
-     break;
 
-   case CURLINFO_REQUEST_SIZE:
 
-     *param_longp = (long)data->info.request_size;
 
-     break;
 
-   case CURLINFO_SSL_VERIFYRESULT:
 
-     *param_longp = data->set.ssl.certverifyresult;
 
-     break;
 
- #ifndef CURL_DISABLE_PROXY
 
-   case CURLINFO_PROXY_SSL_VERIFYRESULT:
 
-     *param_longp = data->set.proxy_ssl.certverifyresult;
 
-     break;
 
- #endif
 
-   case CURLINFO_REDIRECT_COUNT:
 
-     *param_longp = data->set.followlocation;
 
-     break;
 
-   case CURLINFO_HTTPAUTH_AVAIL:
 
-     lptr.to_long = param_longp;
 
-     *lptr.to_ulong = data->info.httpauthavail;
 
-     break;
 
-   case CURLINFO_PROXYAUTH_AVAIL:
 
-     lptr.to_long = param_longp;
 
-     *lptr.to_ulong = data->info.proxyauthavail;
 
-     break;
 
-   case CURLINFO_OS_ERRNO:
 
-     *param_longp = data->state.os_errno;
 
-     break;
 
-   case CURLINFO_NUM_CONNECTS:
 
-     *param_longp = data->info.numconnects;
 
-     break;
 
-   case CURLINFO_LASTSOCKET:
 
-     sockfd = Curl_getconnectinfo(data, NULL);
 
-     /* note: this is not a good conversion for systems with 64 bit sockets and
 
-        32 bit longs */
 
-     if(sockfd != CURL_SOCKET_BAD)
 
-       *param_longp = (long)sockfd;
 
-     else
 
-       /* this interface is documented to return -1 in case of badness, which
 
-          may not be the same as the CURL_SOCKET_BAD value */
 
-       *param_longp = -1;
 
-     break;
 
-   case CURLINFO_PRIMARY_PORT:
 
-     /* Return the (remote) port of the most recent (primary) connection */
 
-     *param_longp = data->info.conn_primary_port;
 
-     break;
 
-   case CURLINFO_LOCAL_PORT:
 
-     /* Return the local port of the most recent (primary) connection */
 
-     *param_longp = data->info.conn_local_port;
 
-     break;
 
-   case CURLINFO_CONDITION_UNMET:
 
-     if(data->info.httpcode == 304)
 
-       *param_longp = 1L;
 
-     else
 
-       /* return if the condition prevented the document to get transferred */
 
-       *param_longp = data->info.timecond ? 1L : 0L;
 
-     break;
 
-   case CURLINFO_RTSP_CLIENT_CSEQ:
 
-     *param_longp = data->state.rtsp_next_client_CSeq;
 
-     break;
 
-   case CURLINFO_RTSP_SERVER_CSEQ:
 
-     *param_longp = data->state.rtsp_next_server_CSeq;
 
-     break;
 
-   case CURLINFO_RTSP_CSEQ_RECV:
 
-     *param_longp = data->state.rtsp_CSeq_recv;
 
-     break;
 
-   case CURLINFO_HTTP_VERSION:
 
-     switch(data->info.httpversion) {
 
-     case 10:
 
-       *param_longp = CURL_HTTP_VERSION_1_0;
 
-       break;
 
-     case 11:
 
-       *param_longp = CURL_HTTP_VERSION_1_1;
 
-       break;
 
-     case 20:
 
-       *param_longp = CURL_HTTP_VERSION_2_0;
 
-       break;
 
-     case 30:
 
-       *param_longp = CURL_HTTP_VERSION_3;
 
-       break;
 
-     default:
 
-       *param_longp = CURL_HTTP_VERSION_NONE;
 
-       break;
 
-     }
 
-     break;
 
-   case CURLINFO_PROTOCOL:
 
-     *param_longp = data->info.conn_protocol;
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- #define DOUBLE_SECS(x) (double)(x)/1000000
 
- static CURLcode getinfo_offt(struct Curl_easy *data, CURLINFO info,
 
-                              curl_off_t *param_offt)
 
- {
 
- #ifdef DEBUGBUILD
 
-   char *timestr = getenv("CURL_TIME");
 
-   if(timestr) {
 
-     unsigned long val = strtol(timestr, NULL, 10);
 
-     switch(info) {
 
-     case CURLINFO_TOTAL_TIME_T:
 
-     case CURLINFO_NAMELOOKUP_TIME_T:
 
-     case CURLINFO_CONNECT_TIME_T:
 
-     case CURLINFO_APPCONNECT_TIME_T:
 
-     case CURLINFO_PRETRANSFER_TIME_T:
 
-     case CURLINFO_STARTTRANSFER_TIME_T:
 
-     case CURLINFO_REDIRECT_TIME_T:
 
-     case CURLINFO_SPEED_DOWNLOAD_T:
 
-     case CURLINFO_SPEED_UPLOAD_T:
 
-       *param_offt = (curl_off_t)val;
 
-       return CURLE_OK;
 
-     default:
 
-       break;
 
-     }
 
-   }
 
- #endif
 
-   switch(info) {
 
-   case CURLINFO_FILETIME_T:
 
-     *param_offt = (curl_off_t)data->info.filetime;
 
-     break;
 
-   case CURLINFO_SIZE_UPLOAD_T:
 
-     *param_offt = data->progress.uploaded;
 
-     break;
 
-   case CURLINFO_SIZE_DOWNLOAD_T:
 
-     *param_offt = data->progress.downloaded;
 
-     break;
 
-   case CURLINFO_SPEED_DOWNLOAD_T:
 
-     *param_offt = data->progress.dlspeed;
 
-     break;
 
-   case CURLINFO_SPEED_UPLOAD_T:
 
-     *param_offt = data->progress.ulspeed;
 
-     break;
 
-   case CURLINFO_CONTENT_LENGTH_DOWNLOAD_T:
 
-     *param_offt = (data->progress.flags & PGRS_DL_SIZE_KNOWN)?
 
-       data->progress.size_dl:-1;
 
-     break;
 
-   case CURLINFO_CONTENT_LENGTH_UPLOAD_T:
 
-     *param_offt = (data->progress.flags & PGRS_UL_SIZE_KNOWN)?
 
-       data->progress.size_ul:-1;
 
-     break;
 
-    case CURLINFO_TOTAL_TIME_T:
 
-     *param_offt = data->progress.timespent;
 
-     break;
 
-   case CURLINFO_NAMELOOKUP_TIME_T:
 
-     *param_offt = data->progress.t_nslookup;
 
-     break;
 
-   case CURLINFO_CONNECT_TIME_T:
 
-     *param_offt = data->progress.t_connect;
 
-     break;
 
-   case CURLINFO_APPCONNECT_TIME_T:
 
-     *param_offt = data->progress.t_appconnect;
 
-     break;
 
-   case CURLINFO_PRETRANSFER_TIME_T:
 
-     *param_offt = data->progress.t_pretransfer;
 
-     break;
 
-   case CURLINFO_STARTTRANSFER_TIME_T:
 
-     *param_offt = data->progress.t_starttransfer;
 
-     break;
 
-   case CURLINFO_REDIRECT_TIME_T:
 
-     *param_offt = data->progress.t_redirect;
 
-     break;
 
-   case CURLINFO_RETRY_AFTER:
 
-     *param_offt = data->info.retry_after;
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- static CURLcode getinfo_double(struct Curl_easy *data, CURLINFO info,
 
-                                double *param_doublep)
 
- {
 
- #ifdef DEBUGBUILD
 
-   char *timestr = getenv("CURL_TIME");
 
-   if(timestr) {
 
-     unsigned long val = strtol(timestr, NULL, 10);
 
-     switch(info) {
 
-     case CURLINFO_TOTAL_TIME:
 
-     case CURLINFO_NAMELOOKUP_TIME:
 
-     case CURLINFO_CONNECT_TIME:
 
-     case CURLINFO_APPCONNECT_TIME:
 
-     case CURLINFO_PRETRANSFER_TIME:
 
-     case CURLINFO_STARTTRANSFER_TIME:
 
-     case CURLINFO_REDIRECT_TIME:
 
-     case CURLINFO_SPEED_DOWNLOAD:
 
-     case CURLINFO_SPEED_UPLOAD:
 
-       *param_doublep = (double)val;
 
-       return CURLE_OK;
 
-     default:
 
-       break;
 
-     }
 
-   }
 
- #endif
 
-   switch(info) {
 
-   case CURLINFO_TOTAL_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.timespent);
 
-     break;
 
-   case CURLINFO_NAMELOOKUP_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_nslookup);
 
-     break;
 
-   case CURLINFO_CONNECT_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_connect);
 
-     break;
 
-   case CURLINFO_APPCONNECT_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_appconnect);
 
-     break;
 
-   case CURLINFO_PRETRANSFER_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_pretransfer);
 
-     break;
 
-   case CURLINFO_STARTTRANSFER_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_starttransfer);
 
-     break;
 
-   case CURLINFO_SIZE_UPLOAD:
 
-     *param_doublep = (double)data->progress.uploaded;
 
-     break;
 
-   case CURLINFO_SIZE_DOWNLOAD:
 
-     *param_doublep = (double)data->progress.downloaded;
 
-     break;
 
-   case CURLINFO_SPEED_DOWNLOAD:
 
-     *param_doublep = (double)data->progress.dlspeed;
 
-     break;
 
-   case CURLINFO_SPEED_UPLOAD:
 
-     *param_doublep = (double)data->progress.ulspeed;
 
-     break;
 
-   case CURLINFO_CONTENT_LENGTH_DOWNLOAD:
 
-     *param_doublep = (data->progress.flags & PGRS_DL_SIZE_KNOWN)?
 
-       (double)data->progress.size_dl:-1;
 
-     break;
 
-   case CURLINFO_CONTENT_LENGTH_UPLOAD:
 
-     *param_doublep = (data->progress.flags & PGRS_UL_SIZE_KNOWN)?
 
-       (double)data->progress.size_ul:-1;
 
-     break;
 
-   case CURLINFO_REDIRECT_TIME:
 
-     *param_doublep = DOUBLE_SECS(data->progress.t_redirect);
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- static CURLcode getinfo_slist(struct Curl_easy *data, CURLINFO info,
 
-                               struct curl_slist **param_slistp)
 
- {
 
-   union {
 
-     struct curl_certinfo *to_certinfo;
 
-     struct curl_slist    *to_slist;
 
-   } ptr;
 
-   switch(info) {
 
-   case CURLINFO_SSL_ENGINES:
 
-     *param_slistp = Curl_ssl_engines_list(data);
 
-     break;
 
-   case CURLINFO_COOKIELIST:
 
-     *param_slistp = Curl_cookie_list(data);
 
-     break;
 
-   case CURLINFO_CERTINFO:
 
-     /* Return the a pointer to the certinfo struct. Not really an slist
 
-        pointer but we can pretend it is here */
 
-     ptr.to_certinfo = &data->info.certs;
 
-     *param_slistp = ptr.to_slist;
 
-     break;
 
-   case CURLINFO_TLS_SESSION:
 
-   case CURLINFO_TLS_SSL_PTR:
 
-     {
 
-       struct curl_tlssessioninfo **tsip = (struct curl_tlssessioninfo **)
 
-                                           param_slistp;
 
-       struct curl_tlssessioninfo *tsi = &data->tsi;
 
- #ifdef USE_SSL
 
-       struct connectdata *conn = data->conn;
 
- #endif
 
-       *tsip = tsi;
 
-       tsi->backend = Curl_ssl_backend();
 
-       tsi->internals = NULL;
 
- #ifdef USE_SSL
 
-       if(conn && tsi->backend != CURLSSLBACKEND_NONE) {
 
-         unsigned int i;
 
-         for(i = 0; i < (sizeof(conn->ssl) / sizeof(conn->ssl[0])); ++i) {
 
-           if(conn->ssl[i].use) {
 
-             tsi->internals = Curl_ssl->get_internals(&conn->ssl[i], info);
 
-             break;
 
-           }
 
-         }
 
-       }
 
- #endif
 
-     }
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- static CURLcode getinfo_socket(struct Curl_easy *data, CURLINFO info,
 
-                                curl_socket_t *param_socketp)
 
- {
 
-   switch(info) {
 
-   case CURLINFO_ACTIVESOCKET:
 
-     *param_socketp = Curl_getconnectinfo(data, NULL);
 
-     break;
 
-   default:
 
-     return CURLE_UNKNOWN_OPTION;
 
-   }
 
-   return CURLE_OK;
 
- }
 
- CURLcode Curl_getinfo(struct Curl_easy *data, CURLINFO info, ...)
 
- {
 
-   va_list arg;
 
-   long *param_longp = NULL;
 
-   double *param_doublep = NULL;
 
-   curl_off_t *param_offt = NULL;
 
-   const char **param_charp = NULL;
 
-   struct curl_slist **param_slistp = NULL;
 
-   curl_socket_t *param_socketp = NULL;
 
-   int type;
 
-   CURLcode result = CURLE_UNKNOWN_OPTION;
 
-   if(!data)
 
-     return result;
 
-   va_start(arg, info);
 
-   type = CURLINFO_TYPEMASK & (int)info;
 
-   switch(type) {
 
-   case CURLINFO_STRING:
 
-     param_charp = va_arg(arg, const char **);
 
-     if(param_charp)
 
-       result = getinfo_char(data, info, param_charp);
 
-     break;
 
-   case CURLINFO_LONG:
 
-     param_longp = va_arg(arg, long *);
 
-     if(param_longp)
 
-       result = getinfo_long(data, info, param_longp);
 
-     break;
 
-   case CURLINFO_DOUBLE:
 
-     param_doublep = va_arg(arg, double *);
 
-     if(param_doublep)
 
-       result = getinfo_double(data, info, param_doublep);
 
-     break;
 
-   case CURLINFO_OFF_T:
 
-     param_offt = va_arg(arg, curl_off_t *);
 
-     if(param_offt)
 
-       result = getinfo_offt(data, info, param_offt);
 
-     break;
 
-   case CURLINFO_SLIST:
 
-     param_slistp = va_arg(arg, struct curl_slist **);
 
-     if(param_slistp)
 
-       result = getinfo_slist(data, info, param_slistp);
 
-     break;
 
-   case CURLINFO_SOCKET:
 
-     param_socketp = va_arg(arg, curl_socket_t *);
 
-     if(param_socketp)
 
-       result = getinfo_socket(data, info, param_socketp);
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
-   va_end(arg);
 
-   return result;
 
- }
 
 
  |