2005-09-18 00:17:51 -07:00
/*
* net / dccp / ackvec . c
*
* An implementation of the DCCP protocol
* Copyright ( c ) 2005 Arnaldo Carvalho de Melo < acme @ ghostprotocols . net >
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; version 2 of the License ;
*/
# include "ackvec.h"
# include "dccp.h"
# include <linux/dccp.h>
2006-03-20 17:16:17 -08:00
# include <linux/init.h>
# include <linux/errno.h>
# include <linux/kernel.h>
2005-09-18 00:17:51 -07:00
# include <linux/skbuff.h>
2006-03-20 17:16:17 -08:00
# include <linux/slab.h>
2005-09-18 00:17:51 -07:00
# include <net/sock.h>
2006-12-06 20:33:20 -08:00
static struct kmem_cache * dccp_ackvec_slab ;
static struct kmem_cache * dccp_ackvec_record_slab ;
2006-03-20 17:19:55 -08:00
static struct dccp_ackvec_record * dccp_ackvec_record_new ( void )
{
struct dccp_ackvec_record * avr =
kmem_cache_alloc ( dccp_ackvec_record_slab , GFP_ATOMIC ) ;
if ( avr ! = NULL )
INIT_LIST_HEAD ( & avr - > dccpavr_node ) ;
return avr ;
}
static void dccp_ackvec_record_delete ( struct dccp_ackvec_record * avr )
{
if ( unlikely ( avr = = NULL ) )
return ;
/* Check if deleting a linked record */
WARN_ON ( ! list_empty ( & avr - > dccpavr_node ) ) ;
kmem_cache_free ( dccp_ackvec_record_slab , avr ) ;
}
static void dccp_ackvec_insert_avr ( struct dccp_ackvec * av ,
struct dccp_ackvec_record * avr )
{
/*
* AVRs are sorted by seqno . Since we are sending them in order , we
* just add the AVR at the head of the list .
* - sorbo .
*/
if ( ! list_empty ( & av - > dccpav_records ) ) {
const struct dccp_ackvec_record * head =
list_entry ( av - > dccpav_records . next ,
struct dccp_ackvec_record ,
dccpavr_node ) ;
BUG_ON ( before48 ( avr - > dccpavr_ack_seqno ,
head - > dccpavr_ack_seqno ) ) ;
}
list_add ( & avr - > dccpavr_node , & av - > dccpav_records ) ;
}
2006-03-20 17:16:17 -08:00
2005-09-18 00:17:51 -07:00
int dccp_insert_option_ackvec ( struct sock * sk , struct sk_buff * skb )
{
struct dccp_sock * dp = dccp_sk ( sk ) ;
struct dccp_ackvec * av = dp - > dccps_hc_rx_ackvec ;
2006-11-26 01:04:40 -02:00
/* Figure out how many options do we need to represent the ackvec */
const u16 nr_opts = ( av - > dccpav_vec_len +
DCCP_MAX_ACKVEC_OPT_LEN - 1 ) /
DCCP_MAX_ACKVEC_OPT_LEN ;
u16 len = av - > dccpav_vec_len + 2 * nr_opts , i ;
2005-09-18 00:17:51 -07:00
struct timeval now ;
u32 elapsed_time ;
2006-11-26 01:04:40 -02:00
const unsigned char * tail , * from ;
unsigned char * to ;
2006-03-20 17:19:55 -08:00
struct dccp_ackvec_record * avr ;
if ( DCCP_SKB_CB ( skb ) - > dccpd_opt_len + len > DCCP_MAX_OPT_LEN )
return - 1 ;
2005-09-18 00:17:51 -07:00
dccp_timestamp ( sk , & now ) ;
elapsed_time = timeval_delta ( & now , & av - > dccpav_time ) / 10 ;
2006-03-20 22:32:06 -08:00
if ( elapsed_time ! = 0 & &
dccp_insert_option_elapsed_time ( sk , skb , elapsed_time ) )
return - 1 ;
avr = dccp_ackvec_record_new ( ) ;
if ( avr = = NULL )
return - 1 ;
2005-09-18 00:17:51 -07:00
DCCP_SKB_CB ( skb ) - > dccpd_opt_len + = len ;
2006-11-26 01:04:40 -02:00
to = skb_push ( skb , len ) ;
2005-09-18 00:17:51 -07:00
len = av - > dccpav_vec_len ;
from = av - > dccpav_buf + av - > dccpav_buf_head ;
2006-11-26 01:04:40 -02:00
tail = av - > dccpav_buf + DCCP_MAX_ACKVEC_LEN ;
for ( i = 0 ; i < nr_opts ; + + i ) {
int copylen = len ;
if ( len > DCCP_MAX_ACKVEC_OPT_LEN )
copylen = DCCP_MAX_ACKVEC_OPT_LEN ;
* to + + = DCCPO_ACK_VECTOR_0 ;
* to + + = copylen + 2 ;
/* Check if buf_head wraps */
if ( from + copylen > tail ) {
const u16 tailsize = tail - from ;
2005-09-18 00:17:51 -07:00
2006-11-26 01:04:40 -02:00
memcpy ( to , from , tailsize ) ;
to + = tailsize ;
len - = tailsize ;
copylen - = tailsize ;
from = av - > dccpav_buf ;
}
2005-09-18 00:17:51 -07:00
2006-11-26 01:04:40 -02:00
memcpy ( to , from , copylen ) ;
from + = copylen ;
to + = copylen ;
len - = copylen ;
2005-09-18 00:17:51 -07:00
}
/*
2006-10-24 16:17:51 -07:00
* From RFC 4340 , A .2 :
2005-09-18 00:17:51 -07:00
*
* For each acknowledgement it sends , the HC - Receiver will add an
* acknowledgement record . ack_seqno will equal the HC - Receiver
* sequence number it used for the ack packet ; ack_ptr will equal
* buf_head ; ack_ackno will equal buf_ackno ; and ack_nonce will
* equal buf_nonce .
*/
2006-03-20 17:19:55 -08:00
avr - > dccpavr_ack_seqno = DCCP_SKB_CB ( skb ) - > dccpd_seq ;
avr - > dccpavr_ack_ptr = av - > dccpav_buf_head ;
avr - > dccpavr_ack_ackno = av - > dccpav_buf_ackno ;
avr - > dccpavr_ack_nonce = av - > dccpav_buf_nonce ;
avr - > dccpavr_sent_len = av - > dccpav_vec_len ;
dccp_ackvec_insert_avr ( av , avr ) ;
2005-09-18 00:17:51 -07:00
2006-11-14 12:57:34 -02:00
dccp_pr_debug ( " %s ACK Vector 0, len=%d, ack_seqno=%llu, "
2005-09-18 00:17:51 -07:00
" ack_ackno=%llu \n " ,
2006-11-14 12:57:34 -02:00
dccp_role ( sk ) , avr - > dccpavr_sent_len ,
2006-03-20 17:19:55 -08:00
( unsigned long long ) avr - > dccpavr_ack_seqno ,
( unsigned long long ) avr - > dccpavr_ack_ackno ) ;
return 0 ;
2005-09-18 00:17:51 -07:00
}
2006-03-20 17:15:42 -08:00
struct dccp_ackvec * dccp_ackvec_alloc ( const gfp_t priority )
2005-09-18 00:17:51 -07:00
{
2006-03-20 17:16:17 -08:00
struct dccp_ackvec * av = kmem_cache_alloc ( dccp_ackvec_slab , priority ) ;
2006-01-04 01:46:34 -02:00
2005-09-18 00:17:51 -07:00
if ( av ! = NULL ) {
2006-09-19 13:06:16 -07:00
av - > dccpav_buf_head = DCCP_MAX_ACKVEC_LEN - 1 ;
2007-03-20 12:26:51 -03:00
av - > dccpav_buf_ackno = UINT48_MAX + 1 ;
2005-09-18 00:17:51 -07:00
av - > dccpav_buf_nonce = av - > dccpav_buf_nonce = 0 ;
av - > dccpav_time . tv_sec = 0 ;
av - > dccpav_time . tv_usec = 0 ;
2006-09-19 13:06:16 -07:00
av - > dccpav_vec_len = 0 ;
2006-03-20 17:19:55 -08:00
INIT_LIST_HEAD ( & av - > dccpav_records ) ;
2005-09-18 00:17:51 -07:00
}
return av ;
}
void dccp_ackvec_free ( struct dccp_ackvec * av )
{
2006-03-20 17:19:55 -08:00
if ( unlikely ( av = = NULL ) )
return ;
2006-03-20 17:20:46 -08:00
if ( ! list_empty ( & av - > dccpav_records ) ) {
struct dccp_ackvec_record * avr , * next ;
list_for_each_entry_safe ( avr , next , & av - > dccpav_records ,
dccpavr_node ) {
list_del_init ( & avr - > dccpavr_node ) ;
dccp_ackvec_record_delete ( avr ) ;
}
}
2006-03-20 17:16:17 -08:00
kmem_cache_free ( dccp_ackvec_slab , av ) ;
2005-09-18 00:17:51 -07:00
}
static inline u8 dccp_ackvec_state ( const struct dccp_ackvec * av ,
2006-11-24 13:02:42 -02:00
const u32 index )
2005-09-18 00:17:51 -07:00
{
return av - > dccpav_buf [ index ] & DCCP_ACKVEC_STATE_MASK ;
}
static inline u8 dccp_ackvec_len ( const struct dccp_ackvec * av ,
2006-11-24 13:02:42 -02:00
const u32 index )
2005-09-18 00:17:51 -07:00
{
return av - > dccpav_buf [ index ] & DCCP_ACKVEC_LEN_MASK ;
}
/*
* If several packets are missing , the HC - Receiver may prefer to enter multiple
* bytes with run length 0 , rather than a single byte with a larger run length ;
* this simplifies table updates if one of the missing packets arrives .
*/
static inline int dccp_ackvec_set_buf_head_state ( struct dccp_ackvec * av ,
const unsigned int packets ,
2006-01-04 01:46:34 -02:00
const unsigned char state )
2005-09-18 00:17:51 -07:00
{
unsigned int gap ;
2006-01-17 13:03:54 -08:00
long new_head ;
2005-09-18 00:17:51 -07:00
2006-03-20 17:15:42 -08:00
if ( av - > dccpav_vec_len + packets > DCCP_MAX_ACKVEC_LEN )
2005-09-18 00:17:51 -07:00
return - ENOBUFS ;
gap = packets - 1 ;
new_head = av - > dccpav_buf_head - packets ;
if ( new_head < 0 ) {
if ( gap > 0 ) {
memset ( av - > dccpav_buf , DCCP_ACKVEC_STATE_NOT_RECEIVED ,
gap + new_head + 1 ) ;
gap = - new_head ;
}
2006-03-20 17:15:42 -08:00
new_head + = DCCP_MAX_ACKVEC_LEN ;
2006-12-10 16:01:18 -02:00
}
2005-09-18 00:17:51 -07:00
av - > dccpav_buf_head = new_head ;
if ( gap > 0 )
memset ( av - > dccpav_buf + av - > dccpav_buf_head + 1 ,
DCCP_ACKVEC_STATE_NOT_RECEIVED , gap ) ;
av - > dccpav_buf [ av - > dccpav_buf_head ] = state ;
av - > dccpav_vec_len + = packets ;
return 0 ;
}
/*
2006-10-24 16:17:51 -07:00
* Implements the RFC 4340 , Appendix A
2005-09-18 00:17:51 -07:00
*/
int dccp_ackvec_add ( struct dccp_ackvec * av , const struct sock * sk ,
const u64 ackno , const u8 state )
{
/*
* Check at the right places if the buffer is full , if it is , tell the
* caller to start dropping packets till the HC - Sender acks our ACK
* vectors , when we will free up space in dccpav_buf .
*
* We may well decide to do buffer compression , etc , but for now lets
* just drop .
*
2006-10-24 16:17:51 -07:00
* From Appendix A .1 .1 ( ` New Packets ' ) :
2005-09-18 00:17:51 -07:00
*
* Of course , the circular buffer may overflow , either when the
* HC - Sender is sending data at a very high rate , when the
* HC - Receiver ' s acknowledgements are not reaching the HC - Sender ,
* or when the HC - Sender is forgetting to acknowledge those acks
* ( so the HC - Receiver is unable to clean up old state ) . In this
* case , the HC - Receiver should either compress the buffer ( by
* increasing run lengths when possible ) , transfer its state to
* a larger buffer , or , as a last resort , drop all received
* packets , without processing them whatsoever , until its buffer
* shrinks again .
*/
/* See if this is the first ackno being inserted */
if ( av - > dccpav_vec_len = = 0 ) {
av - > dccpav_buf [ av - > dccpav_buf_head ] = state ;
av - > dccpav_vec_len = 1 ;
} else if ( after48 ( ackno , av - > dccpav_buf_ackno ) ) {
const u64 delta = dccp_delta_seqno ( av - > dccpav_buf_ackno ,
ackno ) ;
/*
* Look if the state of this packet is the same as the
* previous ackno and if so if we can bump the head len .
*/
if ( delta = = 1 & &
dccp_ackvec_state ( av , av - > dccpav_buf_head ) = = state & &
( dccp_ackvec_len ( av , av - > dccpav_buf_head ) <
DCCP_ACKVEC_LEN_MASK ) )
av - > dccpav_buf [ av - > dccpav_buf_head ] + + ;
else if ( dccp_ackvec_set_buf_head_state ( av , delta , state ) )
return - ENOBUFS ;
} else {
/*
* A .1 .2 . Old Packets
*
2006-10-24 16:17:51 -07:00
* When a packet with Sequence Number S < = buf_ackno
* arrives , the HC - Receiver will scan the table for
* the byte corresponding to S . ( Indexing structures
2005-09-18 00:17:51 -07:00
* could reduce the complexity of this scan . )
*/
u64 delta = dccp_delta_seqno ( ackno , av - > dccpav_buf_ackno ) ;
2006-11-24 13:02:42 -02:00
u32 index = av - > dccpav_buf_head ;
2005-09-18 00:17:51 -07:00
while ( 1 ) {
const u8 len = dccp_ackvec_len ( av , index ) ;
const u8 state = dccp_ackvec_state ( av , index ) ;
/*
* valid packets not yet in dccpav_buf have a reserved
* entry , with a len equal to 0.
*/
if ( state = = DCCP_ACKVEC_STATE_NOT_RECEIVED & &
len = = 0 & & delta = = 0 ) { /* Found our
reserved seat ! */
dccp_pr_debug ( " Found %llu reserved seat! \n " ,
( unsigned long long ) ackno ) ;
av - > dccpav_buf [ index ] = state ;
goto out ;
}
/* len == 0 means one packet */
if ( delta < len + 1 )
goto out_duplicate ;
delta - = len + 1 ;
2006-03-20 17:15:42 -08:00
if ( + + index = = DCCP_MAX_ACKVEC_LEN )
2005-09-18 00:17:51 -07:00
index = 0 ;
}
}
av - > dccpav_buf_ackno = ackno ;
dccp_timestamp ( sk , & av - > dccpav_time ) ;
out :
return 0 ;
out_duplicate :
/* Duplicate packet */
dccp_pr_debug ( " Received a dup or already considered lost "
" packet: %llu \n " , ( unsigned long long ) ackno ) ;
return - EILSEQ ;
}
# ifdef CONFIG_IP_DCCP_DEBUG
void dccp_ackvector_print ( const u64 ackno , const unsigned char * vector , int len )
{
2006-11-20 18:26:03 -02:00
dccp_pr_debug_cat ( " ACK vector len=%d, ackno=%llu | " , len ,
2006-12-10 16:01:18 -02:00
( unsigned long long ) ackno ) ;
2005-09-18 00:17:51 -07:00
while ( len - - ) {
const u8 state = ( * vector & DCCP_ACKVEC_STATE_MASK ) > > 6 ;
const u8 rl = * vector & DCCP_ACKVEC_LEN_MASK ;
2006-11-20 18:26:03 -02:00
dccp_pr_debug_cat ( " %d,%d| " , state , rl ) ;
2005-09-18 00:17:51 -07:00
+ + vector ;
}
2006-11-20 18:26:03 -02:00
dccp_pr_debug_cat ( " \n " ) ;
2005-09-18 00:17:51 -07:00
}
void dccp_ackvec_print ( const struct dccp_ackvec * av )
{
dccp_ackvector_print ( av - > dccpav_buf_ackno ,
av - > dccpav_buf + av - > dccpav_buf_head ,
av - > dccpav_vec_len ) ;
}
# endif
2006-03-20 17:19:55 -08:00
static void dccp_ackvec_throw_record ( struct dccp_ackvec * av ,
struct dccp_ackvec_record * avr )
2005-09-18 00:17:51 -07:00
{
2006-03-20 17:19:55 -08:00
struct dccp_ackvec_record * next ;
2006-09-19 13:04:54 -07:00
/* sort out vector length */
if ( av - > dccpav_buf_head < = avr - > dccpavr_ack_ptr )
av - > dccpav_vec_len = avr - > dccpavr_ack_ptr - av - > dccpav_buf_head ;
else
av - > dccpav_vec_len = DCCP_MAX_ACKVEC_LEN - 1
- av - > dccpav_buf_head
+ avr - > dccpavr_ack_ptr ;
2006-03-20 17:19:55 -08:00
/* free records */
list_for_each_entry_safe_from ( avr , next , & av - > dccpav_records ,
dccpavr_node ) {
list_del_init ( & avr - > dccpavr_node ) ;
dccp_ackvec_record_delete ( avr ) ;
}
2005-09-18 00:17:51 -07:00
}
void dccp_ackvec_check_rcv_ackno ( struct dccp_ackvec * av , struct sock * sk ,
const u64 ackno )
{
2006-03-20 17:19:55 -08:00
struct dccp_ackvec_record * avr ;
2005-09-18 00:17:51 -07:00
2006-03-20 17:19:55 -08:00
/*
* If we traverse backwards , it should be faster when we have large
* windows . We will be receiving ACKs for stuff we sent a while back
* - sorbo .
*/
list_for_each_entry_reverse ( avr , & av - > dccpav_records , dccpavr_node ) {
if ( ackno = = avr - > dccpavr_ack_seqno ) {
2006-11-14 12:57:34 -02:00
dccp_pr_debug ( " %s ACK packet 0, len=%d, ack_seqno=%llu, "
2006-03-20 17:19:55 -08:00
" ack_ackno=%llu, ACKED! \n " ,
2006-11-14 12:57:34 -02:00
dccp_role ( sk ) , 1 ,
2006-03-20 17:19:55 -08:00
( unsigned long long ) avr - > dccpavr_ack_seqno ,
( unsigned long long ) avr - > dccpavr_ack_ackno ) ;
dccp_ackvec_throw_record ( av , avr ) ;
break ;
2006-11-14 13:19:45 -02:00
} else if ( avr - > dccpavr_ack_seqno > ackno )
break ; /* old news */
2005-09-18 00:17:51 -07:00
}
}
static void dccp_ackvec_check_rcv_ackvector ( struct dccp_ackvec * av ,
2006-11-24 13:02:42 -02:00
struct sock * sk , u64 * ackno ,
2005-09-18 00:17:51 -07:00
const unsigned char len ,
const unsigned char * vector )
{
unsigned char i ;
2006-03-20 17:19:55 -08:00
struct dccp_ackvec_record * avr ;
2005-09-18 00:17:51 -07:00
/* Check if we actually sent an ACK vector */
2006-03-20 17:19:55 -08:00
if ( list_empty ( & av - > dccpav_records ) )
2005-09-18 00:17:51 -07:00
return ;
i = len ;
2006-03-20 17:19:55 -08:00
/*
* XXX
* I think it might be more efficient to work backwards . See comment on
* rcv_ackno . - sorbo .
*/
avr = list_entry ( av - > dccpav_records . next , struct dccp_ackvec_record ,
dccpavr_node ) ;
2005-09-18 00:17:51 -07:00
while ( i - - ) {
const u8 rl = * vector & DCCP_ACKVEC_LEN_MASK ;
u64 ackno_end_rl ;
2006-11-24 13:02:42 -02:00
dccp_set_seqno ( & ackno_end_rl , * ackno - rl ) ;
2005-09-18 00:17:51 -07:00
/*
2006-03-20 17:19:55 -08:00
* If our AVR sequence number is greater than the ack , go
* forward in the AVR list until it is not so .
2005-09-18 00:17:51 -07:00
*/
2006-03-20 17:19:55 -08:00
list_for_each_entry_from ( avr , & av - > dccpav_records ,
dccpavr_node ) {
2006-11-24 13:02:42 -02:00
if ( ! after48 ( avr - > dccpavr_ack_seqno , * ackno ) )
2006-03-20 17:19:55 -08:00
goto found ;
}
/* End of the dccpav_records list, not found, exit */
break ;
found :
2006-11-24 13:02:42 -02:00
if ( between48 ( avr - > dccpavr_ack_seqno , ackno_end_rl , * ackno ) ) {
2006-09-19 13:05:35 -07:00
const u8 state = * vector & DCCP_ACKVEC_STATE_MASK ;
2005-09-18 00:17:51 -07:00
if ( state ! = DCCP_ACKVEC_STATE_NOT_RECEIVED ) {
2006-11-14 12:57:34 -02:00
dccp_pr_debug ( " %s ACK vector 0, len=%d, "
2005-09-18 00:17:51 -07:00
" ack_seqno=%llu, ack_ackno=%llu, "
" ACKED! \n " ,
2006-11-14 12:57:34 -02:00
dccp_role ( sk ) , len ,
2005-09-18 00:17:51 -07:00
( unsigned long long )
2006-03-20 17:19:55 -08:00
avr - > dccpavr_ack_seqno ,
2005-09-18 00:17:51 -07:00
( unsigned long long )
2006-03-20 17:19:55 -08:00
avr - > dccpavr_ack_ackno ) ;
dccp_ackvec_throw_record ( av , avr ) ;
2006-06-11 20:58:33 -07:00
break ;
2005-09-18 00:17:51 -07:00
}
/*
2006-03-20 17:19:55 -08:00
* If it wasn ' t received , continue scanning . . . we might
* find another one .
2005-09-18 00:17:51 -07:00
*/
}
2006-11-24 13:02:42 -02:00
dccp_set_seqno ( ackno , ackno_end_rl - 1 ) ;
2005-09-18 00:17:51 -07:00
+ + vector ;
}
}
int dccp_ackvec_parse ( struct sock * sk , const struct sk_buff * skb ,
2006-11-24 13:02:42 -02:00
u64 * ackno , const u8 opt , const u8 * value , const u8 len )
2005-09-18 00:17:51 -07:00
{
2006-11-24 13:02:42 -02:00
if ( len > DCCP_MAX_ACKVEC_OPT_LEN )
2005-09-18 00:17:51 -07:00
return - 1 ;
/* dccp_ackvector_print(DCCP_SKB_CB(skb)->dccpd_ack_seq, value, len); */
dccp_ackvec_check_rcv_ackvector ( dccp_sk ( sk ) - > dccps_hc_rx_ackvec , sk ,
2006-11-24 13:02:42 -02:00
ackno , len , value ) ;
2005-09-18 00:17:51 -07:00
return 0 ;
}
2006-03-20 17:16:17 -08:00
int __init dccp_ackvec_init ( void )
{
dccp_ackvec_slab = kmem_cache_create ( " dccp_ackvec " ,
sizeof ( struct dccp_ackvec ) , 0 ,
SLAB_HWCACHE_ALIGN , NULL , NULL ) ;
2006-03-20 17:19:55 -08:00
if ( dccp_ackvec_slab = = NULL )
goto out_err ;
dccp_ackvec_record_slab =
kmem_cache_create ( " dccp_ackvec_record " ,
sizeof ( struct dccp_ackvec_record ) ,
0 , SLAB_HWCACHE_ALIGN , NULL , NULL ) ;
if ( dccp_ackvec_record_slab = = NULL )
goto out_destroy_slab ;
2006-03-20 17:16:17 -08:00
return 0 ;
2006-03-20 17:19:55 -08:00
out_destroy_slab :
kmem_cache_destroy ( dccp_ackvec_slab ) ;
dccp_ackvec_slab = NULL ;
out_err :
2006-11-20 18:39:23 -02:00
DCCP_CRIT ( " Unable to create Ack Vector slab cache " ) ;
2006-03-20 17:19:55 -08:00
return - ENOBUFS ;
2006-03-20 17:16:17 -08:00
}
void dccp_ackvec_exit ( void )
{
if ( dccp_ackvec_slab ! = NULL ) {
kmem_cache_destroy ( dccp_ackvec_slab ) ;
dccp_ackvec_slab = NULL ;
}
2006-03-20 17:19:55 -08:00
if ( dccp_ackvec_record_slab ! = NULL ) {
kmem_cache_destroy ( dccp_ackvec_record_slab ) ;
dccp_ackvec_record_slab = NULL ;
}
2006-03-20 17:16:17 -08:00
}