mirror of
https://github.com/adulau/ssldump.git
synced 2024-12-22 16:05:58 +00:00
Merge branch 'wllm-rbnt-leaks'
This commit is contained in:
commit
c18ba4cc37
15 changed files with 137 additions and 38 deletions
|
@ -86,17 +86,22 @@ int network_handler_create(mod,handlerp)
|
|||
_status=0;
|
||||
abort:
|
||||
if(_status){
|
||||
network_handler_destroy(&handler);
|
||||
network_handler_destroy(mod, &handler);
|
||||
}
|
||||
return(_status);
|
||||
}
|
||||
|
||||
int network_handler_destroy(handlerp)
|
||||
int network_handler_destroy(mod,handlerp)
|
||||
proto_mod *mod;
|
||||
n_handler **handlerp;
|
||||
{
|
||||
n_handler *handler=0;
|
||||
if(!handlerp || !*handlerp)
|
||||
return(0);
|
||||
|
||||
handler = *handlerp;
|
||||
|
||||
mod->vtbl->destroy_ctx(mod->handle,&handler->ctx);
|
||||
free(*handlerp);
|
||||
*handlerp=0;
|
||||
return(0);
|
||||
|
@ -121,6 +126,12 @@ int network_process_packet(handler,timestamp,data,length)
|
|||
p.len=length;
|
||||
p.ip=(struct ip *)data;
|
||||
|
||||
if(p.len < 20) {
|
||||
if(!(NET_print_flags & NET_PRINT_JSON))
|
||||
printf("Malformed packet, packet too small to contain IP header, skipping ...\n");
|
||||
return(0);
|
||||
}
|
||||
|
||||
/*Handle, or rather mishandle, fragmentation*/
|
||||
off=ntohs(p.ip->ip_off);
|
||||
|
||||
|
|
|
@ -75,7 +75,7 @@ typedef struct packet_ packet;
|
|||
|
||||
int network_handler_create PROTO_LIST((proto_mod *mod,
|
||||
n_handler **handlerp));
|
||||
int network_handler_destroy PROTO_LIST((n_handler **handlerp));
|
||||
int network_handler_destroy PROTO_LIST((proto_mod *mod,n_handler **handlerp));
|
||||
int network_process_packet PROTO_LIST((n_handler *handler,
|
||||
struct timeval *timestamp,UCHAR *data,int length));
|
||||
int packet_copy PROTO_LIST((packet *in,packet **out));
|
||||
|
|
|
@ -104,6 +104,7 @@ int err_exit(str,num)
|
|||
int num;
|
||||
{
|
||||
fprintf(stderr,"ERROR: %s\n",str);
|
||||
sig_handler(SIGQUIT);
|
||||
exit(num);
|
||||
}
|
||||
|
||||
|
@ -127,6 +128,11 @@ int print_version()
|
|||
}
|
||||
|
||||
pcap_t *p;
|
||||
proto_mod *mod=&ssl_mod;
|
||||
n_handler *n;
|
||||
char *interface_name=0;
|
||||
char *file=0;
|
||||
char *filter=0;
|
||||
void sig_handler(int sig)
|
||||
{
|
||||
int freed_conn = 0;
|
||||
|
@ -138,9 +144,18 @@ void sig_handler(int sig)
|
|||
if(freed_conn && !(NET_print_flags & NET_PRINT_JSON))
|
||||
printf("Cleaned %d remaining connection(s) from connection pool\n", freed_conn);
|
||||
|
||||
network_handler_destroy(mod, &n);
|
||||
|
||||
if(p)
|
||||
pcap_close(p);
|
||||
exit(0);
|
||||
if(interface_name)
|
||||
free(interface_name);
|
||||
if(filter)
|
||||
free(filter);
|
||||
if(file)
|
||||
free(file);
|
||||
|
||||
exit(sig);
|
||||
}
|
||||
|
||||
void pcap_cb(ptr,hdr,data)
|
||||
|
@ -168,6 +183,12 @@ void pcap_cb(ptr,hdr,data)
|
|||
len-=4;
|
||||
break;
|
||||
case DLT_EN10MB:
|
||||
if(len < sizeof(struct ether_header)) {
|
||||
if(!(NET_print_flags & NET_PRINT_JSON))
|
||||
printf("Frame size too small to contain Ethernet header, skipping ...\n");
|
||||
return;
|
||||
}
|
||||
|
||||
type=ntohs(e_hdr->ether_type);
|
||||
|
||||
data+=sizeof(struct ether_header);
|
||||
|
@ -287,7 +308,6 @@ int main(argc,argv)
|
|||
char **argv;
|
||||
{
|
||||
int r;
|
||||
n_handler *n;
|
||||
#ifdef _WIN32
|
||||
__declspec(dllimport) char *optarg;
|
||||
__declspec(dllimport) int optind;
|
||||
|
@ -296,10 +316,6 @@ int main(argc,argv)
|
|||
extern int optind;
|
||||
#endif
|
||||
pcap_if_t *interfaces;
|
||||
char *interface_name=0;
|
||||
char *file=0;
|
||||
char *filter=0;
|
||||
proto_mod *mod=&ssl_mod;
|
||||
bpf_u_int32 localnet,netmask;
|
||||
int c;
|
||||
module_def *m=0;
|
||||
|
@ -493,6 +509,7 @@ int main(argc,argv)
|
|||
if(freed_conn && !(NET_print_flags & NET_PRINT_JSON))
|
||||
printf("Cleaned %d remaining connection(s) from connection pool\n", freed_conn);
|
||||
|
||||
network_handler_destroy(mod, &n);
|
||||
pcap_close(p);
|
||||
|
||||
free(n);
|
||||
|
|
|
@ -62,6 +62,7 @@ struct proto_mod_vtbl_ {
|
|||
proto_obj **objp,
|
||||
struct in_addr *i_addr,u_short i_port,
|
||||
struct in_addr *r_addr,u_short r_port,struct timeval *time_base));
|
||||
int (*destroy_ctx) PROTO_LIST((void *handle,proto_ctx **ctxp));
|
||||
int (*destroy) PROTO_LIST((proto_obj **objp));
|
||||
int (*data) PROTO_LIST((proto_obj *obj,segment *data,int direction));
|
||||
int (*close) PROTO_LIST((proto_obj *obj,packet *p,int direction));
|
||||
|
|
|
@ -170,7 +170,8 @@ int clean_old_conn() {
|
|||
while(conn) {
|
||||
tcpconn = &conn->conn;
|
||||
conn=conn->next;
|
||||
timestamp_diff(&last_packet_seen_time, &tcpconn->last_seen_time, &dt);
|
||||
if(timestamp_diff(&last_packet_seen_time, &tcpconn->last_seen_time, &dt))
|
||||
continue;
|
||||
if(dt.tv_sec > conn_ttl) {
|
||||
i++;
|
||||
tcp_destroy_conn(tcpconn);
|
||||
|
|
|
@ -80,7 +80,10 @@ int process_tcp_packet(handler,ctx,p)
|
|||
int direction;
|
||||
stream_data *stream;
|
||||
tcp_conn *conn;
|
||||
|
||||
|
||||
if(p->len < 20)
|
||||
ABORT(1);
|
||||
|
||||
p->tcp=(struct tcphdr *)p->data;
|
||||
|
||||
print_tcp_packet(p);
|
||||
|
@ -241,8 +244,7 @@ static int process_data_segment(conn,handler,p,stream,direction)
|
|||
l=p->len - p->tcp->th_off * 4;
|
||||
|
||||
if(l < 0) {
|
||||
if(!(NET_print_flags & NET_PRINT_JSON))
|
||||
printf("Malformed packet, computed TCP segment size is negative, skipping ...\n");
|
||||
fprintf(stderr,"Malformed packet, computed TCP segment size is negative, skipping ...\n");
|
||||
return(0);
|
||||
}
|
||||
|
||||
|
@ -363,7 +365,8 @@ static int process_data_segment(conn,handler,p,stream,direction)
|
|||
else
|
||||
conn->state=TCP_STATE_CLOSED;
|
||||
}
|
||||
|
||||
|
||||
free_tcp_segment_queue(stream->oo_queue);
|
||||
stream->oo_queue=seg->next;
|
||||
seg->next=0;
|
||||
stream->seq=seg->s_seq + seg->len;
|
||||
|
|
|
@ -121,6 +121,9 @@ int r_assoc_destroy(assocp)
|
|||
for(i=0;i<assoc->size;i++)
|
||||
destroy_assoc_chain(assoc->chains[i]);
|
||||
|
||||
free(assoc->chains);
|
||||
free(assoc);
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
# Process this file with autoconf to produce a configure script.
|
||||
|
||||
AC_PREREQ([2.69])
|
||||
AC_INIT([ssldump], [1.1])
|
||||
AC_INIT([ssldump], [1.3])
|
||||
AM_INIT_AUTOMAKE([subdir-objects])
|
||||
AC_CONFIG_SRCDIR([base/pcap-snoop.c])
|
||||
AC_CONFIG_HEADERS([config.h])
|
||||
|
|
|
@ -238,8 +238,9 @@ static int decode_HandshakeType_ClientHello(ssl,dir,seg,data)
|
|||
}
|
||||
|
||||
for(;cslen;cslen-=2){
|
||||
ssl_decode_enum(ssl,0,2,cipher_suite_decoder,
|
||||
0,data,&cs);
|
||||
if(ssl_decode_enum(ssl,0,2,cipher_suite_decoder,
|
||||
0,data,&cs))
|
||||
return(1);
|
||||
ssl_print_cipher_suite(ssl,(vj<<8)|vn,P_HL,cs);
|
||||
LF;
|
||||
}
|
||||
|
@ -2969,6 +2970,7 @@ static int decode_server_name_type_host_name(ssl,dir,seg,data)
|
|||
if (server_name != NULL)
|
||||
{
|
||||
if (ssl->server_name) free(ssl->server_name);
|
||||
if (l > data->len) l = data->len;
|
||||
memcpy(server_name,data->data,l);
|
||||
ssl->server_name = server_name;
|
||||
}
|
||||
|
|
|
@ -61,6 +61,7 @@ static int create_ssl_analyzer PROTO_LIST((void *handle,
|
|||
proto_ctx *ctx,tcp_conn *conn,proto_obj **objp,
|
||||
struct in_addr *i_addr,u_short i_port,
|
||||
struct in_addr *r_addr,u_short r_port, struct timeval *base_time));
|
||||
static int destroy_ssl_ctx PROTO_LIST((void *handle,proto_ctx **ctxp));
|
||||
static int destroy_ssl_analyzer PROTO_LIST((proto_obj **objp));
|
||||
static int read_ssl_record PROTO_LIST((ssl_obj *obj,r_queue *q,segment *seg,
|
||||
int offset,segment **lastp,int *offsetp));
|
||||
|
@ -228,6 +229,16 @@ static int create_ssl_ctx(handle,ctxp)
|
|||
return(_status);
|
||||
}
|
||||
|
||||
static int destroy_ssl_ctx(handle,ctxp)
|
||||
void *handle;
|
||||
proto_ctx **ctxp;
|
||||
{
|
||||
ssl_decode_ctx *ctx=0;
|
||||
ctx=(ssl_decode_ctx *) *ctxp;
|
||||
ssl_decode_ctx_destroy(&ctx);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int create_ssl_analyzer(void *handle, proto_ctx *ctx, tcp_conn *conn,
|
||||
proto_obj **objp, struct in_addr *i_addr, u_short i_port, struct in_addr *r_addr,
|
||||
u_short r_port, struct timeval *base_time)
|
||||
|
@ -635,6 +646,7 @@ static struct proto_mod_vtbl_ ssl_vtbl ={
|
|||
parse_ssl_flag,
|
||||
create_ssl_ctx,
|
||||
create_ssl_analyzer,
|
||||
destroy_ssl_ctx,
|
||||
destroy_ssl_analyzer,
|
||||
data_ssl_analyzer,
|
||||
close_ssl_analyzer,
|
||||
|
|
|
@ -191,6 +191,25 @@ int ssl_decode_ctx_create(dp,keyfile,pass,keylogfile)
|
|||
#endif
|
||||
}
|
||||
|
||||
int ssl_decode_ctx_destroy(dp)
|
||||
ssl_decode_ctx **dp;
|
||||
{
|
||||
#ifdef OPENSSL
|
||||
ssl_decode_ctx *d = *dp;
|
||||
if(d->ssl_key_log_file) {
|
||||
fclose(d->ssl_key_log_file);
|
||||
}
|
||||
|
||||
r_assoc *x = d->session_cache;
|
||||
r_assoc_destroy(&d->session_cache);
|
||||
|
||||
SSL_CTX_free(d->ssl_ctx);
|
||||
SSL_free(d->ssl);
|
||||
free(d);
|
||||
#endif
|
||||
return(0);
|
||||
}
|
||||
|
||||
int ssl_decoder_create(dp,ctx)
|
||||
ssl_decoder **dp;
|
||||
ssl_decode_ctx *ctx;
|
||||
|
@ -249,6 +268,7 @@ int ssl_set_client_random(d,msg,len)
|
|||
#ifdef OPENSSL
|
||||
int r;
|
||||
|
||||
r_data_destroy(&d->client_random);
|
||||
if((r=r_data_create(&d->client_random,msg,len)))
|
||||
ERETURN(r);
|
||||
#endif
|
||||
|
@ -262,7 +282,8 @@ int ssl_set_server_random(d,msg,len)
|
|||
{
|
||||
#ifdef OPENSSL
|
||||
int r;
|
||||
|
||||
|
||||
r_data_destroy(&d->server_random);
|
||||
if((r=r_data_create(&d->server_random,msg,len)))
|
||||
ERETURN(r);
|
||||
#endif
|
||||
|
@ -277,9 +298,11 @@ int ssl_set_client_session_id(d,msg,len)
|
|||
#ifdef OPENSSL
|
||||
int r;
|
||||
|
||||
if(len>0)
|
||||
if(len>0) {
|
||||
r_data_destroy(&d->session_id);
|
||||
if((r=r_data_create(&d->session_id,msg,len)))
|
||||
ERETURN(r);
|
||||
}
|
||||
#endif
|
||||
return(0);
|
||||
}
|
||||
|
|
|
@ -52,6 +52,7 @@
|
|||
|
||||
int ssl_decode_ctx_create PROTO_LIST((ssl_decode_ctx **ctx,
|
||||
char *keyfile,char *password,char *keylogfile));
|
||||
int ssl_decode_ctx_destroy(ssl_decode_ctx **dp);
|
||||
int ssl_decoder_destroy PROTO_LIST((ssl_decoder **dp));
|
||||
int ssl_decoder_create PROTO_LIST((ssl_decoder **dp,ssl_decode_ctx *ctx));
|
||||
int ssl_set_client_random PROTO_LIST((ssl_decoder *dp,
|
||||
|
|
|
@ -95,6 +95,7 @@ int process_v2_hello(ssl,seg)
|
|||
{
|
||||
int r;
|
||||
int rec_len;
|
||||
int _status;
|
||||
UINT4 cs_len;
|
||||
UINT4 sid_len;
|
||||
UINT4 chall_len;
|
||||
|
@ -104,36 +105,39 @@ int process_v2_hello(ssl,seg)
|
|||
UCHAR random[32];
|
||||
|
||||
if(seg->len==0)
|
||||
return(SSL_NO_DATA);
|
||||
|
||||
ABORT(SSL_NO_DATA);
|
||||
|
||||
d.data=seg->data;
|
||||
d.len=seg->len;
|
||||
|
||||
/* First check the message length. */
|
||||
if(d.len<4)
|
||||
return(SSL_BAD_CONTENT_TYPE);
|
||||
ABORT(SSL_BAD_CONTENT_TYPE);
|
||||
rec_len=((d.data[0] & 0x7f)<<8) | (d.data[1]);
|
||||
d.data+=2; d.len-=2;
|
||||
|
||||
if(d.len!=rec_len) /* Whatever this is it isn't valid SSLv2*/
|
||||
return(SSL_BAD_CONTENT_TYPE);
|
||||
ABORT(SSL_BAD_CONTENT_TYPE);
|
||||
|
||||
/* If msg_type==1 then we've got a v2 message (or trash)*/
|
||||
if(*d.data++!=1)
|
||||
return(SSL_BAD_CONTENT_TYPE);
|
||||
ABORT(SSL_BAD_CONTENT_TYPE);
|
||||
d.len--;
|
||||
|
||||
SSL_DECODE_UINT16(ssl,"Version number",P_DC,&d,&ver);
|
||||
SSL_DECODE_UINT16_ABORT(ssl,"Version number",P_DC,&d,&ver);
|
||||
/* We can't handle real v2 clients*/
|
||||
if(ver<=2){
|
||||
explain(ssl,"Version 2 Client.\n");
|
||||
return(SSL_BAD_DATA);
|
||||
ABORT(SSL_BAD_DATA);
|
||||
}
|
||||
|
||||
ssl->cur_json_st = json_object_new_object();
|
||||
ssl_print_record_num(ssl);
|
||||
ssl_print_timestamp(ssl,&seg->p->ts);
|
||||
ssl_print_direction_indicator(ssl,DIR_I2R);
|
||||
explain(ssl," SSLv2 compatible client hello\n");
|
||||
json_object_object_add(ssl->cur_json_st, "msg_type", json_object_new_string("Handshake"));
|
||||
json_object_object_add(ssl->cur_json_st, "handshake_type", json_object_new_string("ClientHello_v2_compat"));
|
||||
|
||||
INDENT_INCR;
|
||||
|
||||
|
@ -142,13 +146,13 @@ int process_v2_hello(ssl,seg)
|
|||
ver&0xff);
|
||||
LF;
|
||||
}
|
||||
SSL_DECODE_UINT16(ssl,"cipher_spec_length",P_DC,&d,&cs_len);
|
||||
SSL_DECODE_UINT16(ssl,"session_id_length",P_DC,&d,&sid_len);
|
||||
SSL_DECODE_UINT16(ssl,"challenge_length",P_DC,&d,&chall_len);
|
||||
SSL_DECODE_UINT16_ABORT(ssl,"cipher_spec_length",P_DC,&d,&cs_len);
|
||||
SSL_DECODE_UINT16_ABORT(ssl,"session_id_length",P_DC,&d,&sid_len);
|
||||
SSL_DECODE_UINT16_ABORT(ssl,"challenge_length",P_DC,&d,&chall_len);
|
||||
|
||||
if(cs_len%3){
|
||||
fprintf(stderr,"Bad cipher spec length %d\n",cs_len);
|
||||
return(SSL_BAD_DATA);
|
||||
ABORT(SSL_BAD_DATA);
|
||||
}
|
||||
P_(P_HL){
|
||||
explain(ssl,"cipher suites\n");
|
||||
|
@ -157,7 +161,7 @@ int process_v2_hello(ssl,seg)
|
|||
for(;cs_len;cs_len-=3){
|
||||
UINT4 val;
|
||||
|
||||
SSL_DECODE_UINT24(ssl,0,0,&d,&val);
|
||||
SSL_DECODE_UINT24_ABORT(ssl,0,0,&d,&val);
|
||||
ssl_print_cipher_suite(ssl,ver,P_HL,val);
|
||||
P_(P_HL){
|
||||
explain(ssl,"\n");
|
||||
|
@ -166,15 +170,15 @@ int process_v2_hello(ssl,seg)
|
|||
|
||||
if(sid_len!=0){
|
||||
fprintf(stderr,"Session ID field should be zero length\n");
|
||||
return(SSL_BAD_DATA);
|
||||
ABORT(SSL_BAD_DATA);
|
||||
}
|
||||
|
||||
if(chall_len<16 || chall_len>32){
|
||||
fprintf(stderr,"Invalid challenge length %d\n",chall_len);
|
||||
return(SSL_BAD_DATA);
|
||||
ABORT(SSL_BAD_DATA);
|
||||
}
|
||||
|
||||
SSL_DECODE_OPAQUE_ARRAY(ssl,0,chall_len,
|
||||
SSL_DECODE_OPAQUE_ARRAY_ABORT(ssl,0,chall_len,
|
||||
0,&d,&chall);
|
||||
P_(P_DC){
|
||||
exdump(ssl,"Challenge",&chall);
|
||||
|
@ -195,7 +199,18 @@ int process_v2_hello(ssl,seg)
|
|||
}
|
||||
|
||||
INDENT_POP;
|
||||
return(0);
|
||||
|
||||
_status=0;
|
||||
|
||||
abort:
|
||||
if(ssl->cur_json_st) {
|
||||
if(SSL_print_flags & SSL_PRINT_JSON)
|
||||
printf("%s\n", json_object_to_json_string(ssl->cur_json_st));
|
||||
json_object_put(ssl->cur_json_st);
|
||||
ssl->cur_json_st = NULL;
|
||||
}
|
||||
|
||||
return(_status);
|
||||
}
|
||||
|
||||
int ssl_decode_switch(ssl,dtable,value,dir,seg,data)
|
||||
|
@ -276,7 +291,10 @@ int ssl_expand_record(ssl,q,direction,data,len)
|
|||
printf(" unknown record type: %d\n", ct);
|
||||
ERETURN(r);
|
||||
}
|
||||
ssl_get_enum_str(ssl,enumstr,ContentType_decoder,ct);
|
||||
if((r=ssl_get_enum_str(ssl,enumstr,ContentType_decoder,ct))) {
|
||||
strncpy(enumstr, "Unknown", 20);
|
||||
}
|
||||
|
||||
json_object_object_add(jobj, "msg_type", json_object_new_string(enumstr));
|
||||
|
||||
if(!(SSL_print_flags & SSL_PRINT_JSON))
|
||||
|
|
|
@ -88,6 +88,12 @@ int exstr PROTO_LIST((ssl_obj *ssl,char *name,Data *data));
|
|||
#define SSL_DECODE_UINT32(a,n,b,c,d) if((r=ssl_decode_uintX(a,n,4,b,c,d))) ERETURN(r)
|
||||
#define SSL_DECODE_OPAQUE_ARRAY(a,n,b,c,d,e) if((r=ssl_decode_opaque_array(a,n,b,c,d,e))) ERETURN(r)
|
||||
#define SSL_DECODE_ENUM(a,b,c,d,e,f,g) if((r=ssl_decode_enum(a,b,c,d,e,f,g))) ERETURN(r)
|
||||
#define SSL_DECODE_UINT8_ABORT(a,n,b,c,d) if((r=ssl_decode_uintX(a,n,1,b,c,d))) ABORT(r)
|
||||
#define SSL_DECODE_UINT16_ABORT(a,n,b,c,d) if((r=ssl_decode_uintX(a,n,2,b,c,d))) ABORT(r)
|
||||
#define SSL_DECODE_UINT24_ABORT(a,n,b,c,d) if((r=ssl_decode_uintX(a,n,3,b,c,d))) ABORT(r)
|
||||
#define SSL_DECODE_UINT32_ABORT(a,n,b,c,d) if((r=ssl_decode_uintX(a,n,4,b,c,d))) ABORT(r)
|
||||
#define SSL_DECODE_OPAQUE_ARRAY_ABORT(a,n,b,c,d,e) if((r=ssl_decode_opaque_array(a,n,b,c,d,e))) ABORT(r)
|
||||
#define SSL_DECODE_ENUM_ABORT(a,b,c,d,e,f,g) if((r=ssl_decode_enum(a,b,c,d,e,f,g))) ABORT(r)
|
||||
#define P_(p) if((p==SSL_PRINT_ALL) || (p & SSL_print_flags))
|
||||
|
||||
#define INDENT if(!(NET_print_flags & NET_PRINT_JSON)) do {int i; for(i=0;i<(ssl->indent_depth + ssl->indent_name_len);i++) printf("%s",SSL_print_flags & SSL_PRINT_NROFF?" ":" ");} while(0)
|
||||
|
|
|
@ -73,6 +73,7 @@ int sslx_print_certificate(ssl,data,pf)
|
|||
#endif
|
||||
UCHAR *d;
|
||||
int _status;
|
||||
struct json_object *cert_obj;
|
||||
|
||||
#ifdef OPENSSL
|
||||
P_(P_ASN){
|
||||
|
@ -86,7 +87,6 @@ int sslx_print_certificate(ssl,data,pf)
|
|||
struct json_object *jobj;
|
||||
jobj = ssl->cur_json_st;
|
||||
|
||||
struct json_object *cert_obj;
|
||||
cert_obj = json_object_new_object();
|
||||
|
||||
d=data->data;
|
||||
|
@ -183,7 +183,8 @@ int sslx_print_certificate(ssl,data,pf)
|
|||
abort:
|
||||
#ifdef OPENSSL
|
||||
if(x) X509_free(x);
|
||||
#endif
|
||||
#endif
|
||||
if(_status && cert_obj) json_object_put(cert_obj);
|
||||
return(_status);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue