blob: a4863362cf5c1373b77cd2f2e5653e62d3cba245 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
|
// This may look like C, but it's really -*- C++ -*-
// -*- C++ -*-
// ===================================================================
/**
* @file GIOP_Message_Handler.h
*
* $Id$
*
* @author Balachandran Natarajan <bala@cs.wustl.edu>
**/
// ===================================================================
#ifndef TAO_GIOP_MESSAGE_HANDLER_H
#define TAO_GIOP_MESSAGE_HANDLER_H
#include "ace/pre.h"
#include "ace/Message_Block.h"
#if !defined (ACE_LACKS_PRAGMA_ONCE)
# pragma once
#endif /* ACE_LACKS_PRAGMA_ONCE */
#include "tao/GIOP_Message_State.h"
class TAO_Transport;
class TAO_ORB_Core;
class TAO_GIOP_Message_Base;
enum TAO_GIOP_Message_Status
{
/// The buffer is waiting for the header of the message yet
TAO_GIOP_WAITING_FOR_HEADER = 0,
/// The buffer is waiting for the payload to appear on the socket
TAO_GIOP_WAITING_FOR_PAYLOAD,
/// The buffer has got multiple messages
TAO_GIOP_MULTIPLE_MESSAGES
};
/**
* @class TAO_GIOP_Message_Handler
*
* @brief GIOP specific message handler class
*
* This class does some of the message handling for GIOP. This class
* reads the message from the socket, splits the messages to create a
* CDR stream out of it and passes that to the higher layers of the ORB.
* The read from the socket is done using a single 'read' instead of
* reading the header and the payload seperately.
*/
class TAO_GIOP_Message_Handler
{
public:
/// Ctor
TAO_GIOP_Message_Handler (TAO_ORB_Core *orb_core,
TAO_GIOP_Message_Base *base,
size_t input_cdr_size = ACE_CDR::DEFAULT_BUFSIZE);
/// Reads the message from the <transport> and sets the <wr_ptr> of
/// the buffer appropriately.
int read_messages (TAO_Transport *transport);
/// Parse the GIOP message header if we have read bytes suffcient
/// bytes. There are four possibilities
/// - We did not read sufficient bytes, then make the reactor to
/// call us back. (return -2)
/// - We read a piece of message that was left out in the
/// socket. In such cases we just go ahead with more processing
/// (return 0).
/// - We have sufficient info for processing the header and we
/// processed it succesfully. (return 1);
/// - Any errors in processing will return a -1.
int parse_message_header (void);
/// Check whether we have atleast one complete message ready for
/// processing.
int is_message_ready (void);
/// Return the underlying data block of the <current_buffer_>. At
/// the sametime making a new data_block for itself. The read and
/// write pointer positions would be reset.
ACE_Data_Block *steal_data_block (void);
/// Reset the contents of the <current_buffer_> if no more requests
/// need to be processed. We reset the contents of the
/// <message_state_> to parse and process the next request.
void reset (int reset_flag = 0);
/// Return the underlying message state
TAO_GIOP_Message_State &message_state (void);
/// Return the pointer to the data block within the message block
ACE_Data_Block *data_block (void) const;
/// Return the position of the read pointer in the <current_buffer_>
size_t rd_pos (void) const;
/// Return the position of the write pointer in the <current_buffer_>
size_t wr_pos (void) const;
private:
/// Actually parses the header information from the
/// <current_buffer_>.
int parse_message_header_i (void);
/// Validates the first 4 bytes that contain the magic word
/// "GIOP". Also calls the validate_version () on the incoming
/// stream.
int parse_magic_bytes (void);
/// Gets the size of the payload from the <current_buffer_>. If the
/// size of the current buffer is less than the payload size, the
/// size of the buffer is increased.
CORBA::ULong get_payload_size (void);
/// Extract a CORBA::ULong from the <current_buffer_>
CORBA::ULong read_ulong (const char *buf);
/// Get the next message from the <supp_buffer_> in to the
/// <current_buffer_>
int get_message (void);
private:
/// The pointer to the object that holds us
TAO_GIOP_Message_Base *mesg_base_;
/// The state of the message in the buffer
TAO_GIOP_Message_Status message_status_;
/// The size of the message that is being read of the socket. This
/// value is originally set to 1024 bytes. It is reset if we start
/// receiving messages with payloads greater than that. The current
/// value of <message_size_> would be the size of the last message
/// received (ie. payload+headers).
size_t message_size_;
/// The buffer. rd_ptr() points to the beginning of the current
/// message, properly aligned wr_ptr() points to where the next
/// read() should put the data.
ACE_Message_Block current_buffer_;
/// The supplementary buffer that holds just one message if the
/// <current_buffer_> has more than one message. One message from
/// the <current_buffer_> is taken and filled in this buffer, which
/// is then sent to the higher layers of the ORB.
ACE_Message_Block supp_buffer_;
/// The message state. It represents the status of the messages that
/// have been read from the current_buffer_
TAO_GIOP_Message_State message_state_;
/// Our copy the ORB_Core
TAO_ORB_Core *orb_core_;
};
const size_t TAO_GIOP_MESSAGE_HEADER_LEN = 12;
const size_t TAO_GIOP_MESSAGE_SIZE_OFFSET = 8;
const size_t TAO_GIOP_MESSAGE_FLAGS_OFFSET = 6;
const size_t TAO_GIOP_MESSAGE_TYPE_OFFSET = 7;
const size_t TAO_GIOP_VERSION_MINOR_OFFSET = 5;
const size_t TAO_GIOP_VERSION_MAJOR_OFFSET = 4;
const size_t TAO_GIOP_MESSAGE_FRAGMENT_HEADER = 4;
#if defined (__ACE_INLINE__)
# include "tao/GIOP_Message_Handler.inl"
#endif /* __ACE_INLINE__ */
#include "ace/post.h"
#endif /*TAO_GIOP_MESSAGE_HANDLER_H*/
|