src/proto2/NetworkTCP.cc
author terom
Mon, 10 Nov 2008 18:21:23 +0000
branchno-netsession
changeset 32 2ff929186c90
parent 31 d0d7489d4e8b
child 33 e53f09b378f4
permissions -rw-r--r--
write some mode code...
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     1
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     2
#include "NetworkTCP.hh"
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     3
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     4
#include <cstdlib>
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     5
#include <cassert>
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     6
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     7
NetworkBuffer::NetworkBuffer (NetworkSocket &socket, size_t size_hint) :
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     8
    socket(socket), buf(0), size(0), offset(0) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
     9
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    10
    // allocate initial buffer
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    11
    if ((buf = (char *) malloc(size_hint)) == NULL)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    12
       throw CL_Error("malloc failed");
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    13
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    14
    // remember size
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    15
    size = size_hint;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    16
}
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    17
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    18
~NetworkBuffer::NetworkBuffer (void) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    19
    free(buf);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    20
}
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    21
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    22
void NetworkBuffer::resize (size_t suffix_size) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    23
    size_t new_size = size;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    24
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    25
    // grow new_size until item_size fits
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    26
    while (offset + item_size > new_size)
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    27
        new_size *= 2;
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    28
    
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    29
    // grow if needed
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    30
    if (new_size != size) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    31
        // realloc buffer
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    32
        if ((buf = (char *) realloc((void *) buf, new_size)) == NULL)
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    33
            throw CL_Error("realloc failed");
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    34
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    35
        // update size
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    36
        size = new_size;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    37
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    38
    } else if (new_size > (offset + item_size) * 4) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    39
        // XXX: shrink?
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    40
    }
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    41
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
    42
        
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    43
void NetworkBuffer::trim (size_t prefix_size) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    44
    // update offset
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    45
    offset -= prefix;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    46
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    47
    // shift the buffer forwards from (buf + prefix) -> (buf), copying (old_offset - prefix) bytes
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    48
    memmove(buf, buf + ret, offset);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    49
}
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    50
     
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    51
bool try_read (size_t item_size) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    52
    int ret;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    53
    size_t to_read = item_size;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    54
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    55
    // keept reads at at least NETWORK_CHUNK_SIZE bytes
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    56
    if (to_read < NETWORK_TCP_CHUNK_SIZE)
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    57
        to_read = NETWORK_TCP_CHUNK_SIZE;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    58
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    59
    // resize buffer if needed
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    60
    resize(to_read);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    61
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    62
    // read once
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    63
    try {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    64
        ret = socket.recv(buf + offset, to_read);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    65
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    66
    } catch (CL_Error &e) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    67
        if (errno == EAGAIN)
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    68
            return false;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    69
        else
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    70
            throw;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    71
    }
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    72
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    73
    // update offset
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    74
    offset += ret;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    75
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    76
    // did we get enough?
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    77
    if (ret < item_size)
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    78
        return false;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    79
    else
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    80
        return true;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    81
} 
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    82
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    83
bool peek_prefix (uint16_t &ref) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    84
    if (offset < sizeof(uint16_t))
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    85
        return false;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    86
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    87
    ret = ntohs(*((uint16_t *) (buf)));
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    88
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    89
    return true;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    90
}
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    91
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    92
bool peek_prefix (uint32_t &ref) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    93
    if (offset < sizeof(uint32_t))
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    94
        return false;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    95
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    96
    ret = ntohl(*((uint32_t *) (buf)));
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    97
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    98
    return true;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
    99
}
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   100
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   101
template <typename PrefixType> PrefixType read_prefix (char *buf_ptr, size_t buf_max) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   102
    PrefixType prefix = 0;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   103
    size_t missing = 0;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   104
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   105
    do {    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   106
        // do we have the prefix?
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   107
        if (peek_prefix(prefix)) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   108
            // do we already have the payload?
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   109
            if (offset >= sizeof(PrefixType) + prefix) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   110
                break;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   111
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   112
            } else {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   113
                missing = (sizeof(PrefixType) + prefix) - offset;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   114
            }
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   115
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   116
        } else {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   117
            missing = sizeof(PrefixType);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   118
        }
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   119
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   120
        // sanity-check
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   121
        assert(missing);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   122
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   123
        // try and read the missing data
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   124
        if (try_read(missing) == false) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   125
            // if unable to read what we need, return zero.
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   126
            return 0;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   127
        }
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   128
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   129
        // assess the situation again
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   130
    } while (true);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   131
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   132
    // copy the data over, unless it's too large
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   133
    if (prefix <= buf_max) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   134
        memcpy(buf_ptr, buf, prefix);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   135
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   136
        // trim the bytes out
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   137
        trim(prefix);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   138
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   139
        // return
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   140
        return prefix;
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   141
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   142
    } else {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   143
        // trim the bytes out
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   144
        trim(prefix);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   145
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   146
        throw CL_Error("recv prefix overflow");   
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   147
    }
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   148
}
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   149
   
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   150
void NetworkBuffer::push_write (char *buf_ptr, size_t buf_size) {
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   151
    int ret;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   152
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   153
    // try and short-circuit writes unless we have already buffered data
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   154
    if (offset == 0) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   155
        try {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   156
            // attempt to send something
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   157
            ret = socket.send(buf_ptr, buf_size);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   158
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   159
        } catch (CL_Error &e) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   160
            // ignore EAGAIN, detect this by setting ret to -1
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   161
            if (errno != EAGAIN)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   162
                throw;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   163
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   164
            ret = -1;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   165
        }
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   166
        
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   167
        // if we managed to send something, adjust buf/size and buffer
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   168
        if (ret > 0) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   169
            buf_ptr += ret;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   170
            buf_size -= ret;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   171
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   172
            // sanity-check
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   173
            assert(buf_size >= 0);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   174
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   175
            // if that was all, we're done
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   176
            if (buf_size == 0)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   177
                return;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   178
        }
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   179
    }
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   180
    
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   181
    // resize to fit buf_size more bytes
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   182
    resize(buf_size);
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   183
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   184
    // copy into our internal buffer
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   185
    memcpy(buf + offset, buf_ptr, buf_size);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   186
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   187
        
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   188
void NetworkBuffer::flush_write (void) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   189
    int ret;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   190
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   191
    // ignore if we don't have any data buffered
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   192
    if (offset == 0)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   193
        return;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   194
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   195
    // attempt to write as much as possible
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   196
    try {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   197
        ret = socket.send(buf, offset)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   198
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   199
    } catch (CL_Error &e) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   200
        // ignore EAGAIN and just return
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   201
        if (errno == EAGAIN)
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   202
            return;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   203
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   204
        else
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   205
            throw;
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   206
    }
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   207
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   208
    // trim the buffer
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   209
    trim(ret);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   210
}
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   211
        
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   212
template <typename PrefixType> void NetworkBuffer::write_prefix (char *buf, PrefixType prefix) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   213
    push_write(&prefix, sizeof(PrefixType)); 
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   214
    push_write(buf, prefix);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   215
}
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   216
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   217
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   218
NetworkTCPTransport::NetworkTCPTransport (CL_Socket socket) :
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   219
    socket(socket), in(NETWORK_TCP_INITIAL_IN_BUF), out(NETWORK_TCP_INITIAL_OUT_BUF) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   220
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   221
    // use nonblocking sockets
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   222
    socket.set_nonblocking(true);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   223
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   224
    // connect signals
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   225
    slots.connect(socket.sig_read_triggered(), this, &NetworkTCPTransport::on_read);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   226
    slots.connect(socket.sig_write_triggered(), this, &NetworkTCPTransport::on_write);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   227
    slots.connect(socket.sig_disconnected_triggered(), this, &NetworkTCPTransport::on_disconnected);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   228
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   229
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   230
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   231
void NetworkTCPTransport::on_read (void) {
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   232
    uint16_t prefix;
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   233
    NetworkPacket packet;
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   234
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   235
    // let the in stream read length-prefixed packets and pass them on to handle_packet
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   236
    while ((prefix = in.read_prefix(packet.get_buf(), packet.get_buf_size)) > 0) {
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   237
        packet.set_data_size(prefix);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   238
        sig_packet(packet);
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   239
    }
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   240
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   241
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   242
void NetworkTCPTransport::on_write (void) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   243
    // just flush the output buffer
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   244
    out.flush_write();
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   245
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   246
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   247
void NetworkTCPTransport::on_disconnected (void) {
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   248
    // pass right through
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   249
    sig_disconnect();
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   250
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   251
        
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   252
void NetworkTCPTransport::write_packet (const NetworkPacket &packet) {
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   253
    uint16_t prefix = packet.get_data_size();
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   254
    
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   255
    if (prefix != packet.get_data_size())
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   256
        throw CL_Error("send prefix overflow");
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   257
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   258
    // just write to the output buffer
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   259
    out.write_prefix(packet.get_buf(), prefix);
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   260
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   261
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   262
NetworkTCPServer::NetworkTCPServer (const NetworkAddress &listen_addr) :
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   263
    socket(tcp, ipv4) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   264
    
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   265
    // bind
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   266
    socket.bind(listen_addr);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   267
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   268
    // assign slots
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   269
    slots.connect(socket.sig_read_triggered(), this, &NetworkTCPServer::on_accept);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   270
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   271
    // listen
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   272
    socket.listen(NETWORK_LISTEN_BACKLOG);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   273
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   274
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   275
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   276
void NetworkTCPServer::on_accept (void) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   277
    // accept a new socket
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   278
    NetworkSocket client_sock = socket.accept();
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   279
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   280
    // create a new NetworkTCPTransport
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   281
    NetworkTCPTransport *client = new NetworkTCPTransport(client_sock);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   282
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   283
    // let our user handle it
32
2ff929186c90 write some mode code...
terom
parents: 31
diff changeset
   284
    sig_client(client);
31
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   285
}
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   286
        
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   287
NetworkTCPClient::NetworkTCPClient (const NetworkAddress &connect_addr) :
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   288
    NetworkTCPTransport(NetworkSocket(tcp, ipv4)) {
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   289
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   290
    // connect
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   291
    socket.connect(connect_addr);
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   292
d0d7489d4e8b add initial code written so far
terom
parents:
diff changeset
   293
}