3 //==========================================================================
5 * @file Message_Block.h
7 * @author Douglas C. Schmidt <d.schmidt@vanderbilt.edu>
9 //==========================================================================
11 #ifndef ACE_MESSAGE_BLOCK_H
12 #define ACE_MESSAGE_BLOCK_H
14 #include /**/ "ace/pre.h"
16 #include /**/ "ace/config-lite.h"
17 #include /**/ "ace/ACE_export.h"
19 #if !defined (ACE_LACKS_PRAGMA_ONCE)
21 #endif /* ACE_LACKS_PRAGMA_ONCE */
23 #include "ace/Default_Constants.h"
24 #include "ace/Global_Macros.h"
25 #include "ace/Time_Value.h"
27 ACE_BEGIN_VERSIONED_NAMESPACE_DECL
29 // Forward declaration.
36 * @class ACE_Message_Block
38 * @brief Stores messages for use throughout ACE (particularly
39 * in an ACE_Message_Queue).
41 * An ACE_Message_Block is modeled after the message data
42 * structures used in System V STREAMS. Its purpose is to
43 * enable efficient manipulation of arbitrarily large messages
44 * without incurring much memory copying overhead. Here are the
45 * main characteristics of an ACE_Message_Block:
46 * - Contains a pointer to a reference-counted
47 * ACE_Data_Block, which in turn points to the actual data
48 * buffer. This allows very flexible and efficient sharing of
49 * data by multiple ACE_Message_Block objects.
50 * - One or more ACE_Message_Blocks can be linked to form a
52 * - ACE_Message_Blocks can be linked together in a doubly linked fashion
53 * to form a queue of messages (this is how ACE_Message_Queue works).
55 * @see C++NPv1, section 4.2; APG, section 12.3.2.
57 class ACE_Export ACE_Message_Block
60 friend class ACE_Data_Block
;
65 /// Undifferentiated data message
67 /// Undifferentiated protocol control
71 /// Line break (regular and priority)
75 /// Post an event to an event queue
77 /// Generate process signal
79 /// ioctl; set/get params
81 /// Set various stream head options
85 /// Acknowledge ioctl (high priority; go to head of queue)
87 /// Negative ioctl acknowledge
89 /// Priority proto message
91 /// Generate process signal
93 /// Generate read notification
97 /// Stop transmission immediately
99 /// Restart transmission after stop
103 /// Fatal error used to set u.u_error
105 /// Post an event to an event queue
108 // = Message class masks
109 /// Normal priority message mask
111 /// High priority control message mask
113 /// User-defined message mask
117 typedef int ACE_Message_Type
;
118 typedef unsigned long Message_Flags
;
122 /// Don't delete the data on exit since we don't own it.
124 /// user defined flags start here
128 /// Create an empty message.
129 ACE_Message_Block (ACE_Allocator
*message_block_allocator
= 0);
132 * Create an ACE_Message_Block that owns the specified ACE_Data_Block
133 * without copying it. If the @a flags is set to @c DONT_DELETE we
134 * don't delete the ACE_Data_Block. It is left to the client's
135 * responsibility to take care of the memory allocated for the
138 ACE_Message_Block (ACE_Data_Block
*data_block
,
139 Message_Flags flags
= 0,
140 ACE_Allocator
*message_block_allocator
= 0);
143 * Create an ACE_Message_Block that refers to @a data without
144 * copying it. The @a data memory will not be freed when this block is
145 * destroyed; memory management of @a data is left to the caller.
146 * Note that the @c size of the new ACE_Message_Block will be @a size, but
147 * the @c length will be 0 until the write pointer is changed.
149 ACE_Message_Block (const char *data
,
151 unsigned long priority
= ACE_DEFAULT_MESSAGE_BLOCK_PRIORITY
);
154 * Create an initialized message of type @a type containing @a size
155 * bytes. The @a cont argument initializes the continuation field in
156 * the ACE_Message_Block. If @a data == 0 then this block allocates and
157 * owns the block's memory, using @a allocator to get the data if it's
158 * non-0. If @a data != 0 then this block refers to that memory until
159 * this this block ceases to exist; this object will not free @a data on
160 * destruction. If @a locking_strategy is non-0 then this is used
161 * to protect regions of code that access shared state (e.g.,
162 * reference counting) from race conditions. Note that the @c size
163 * of the ACE_Message_Block will be @a size, but the @c length will be 0
164 * until the write pointer is set. The @a data_block_allocator is used to
165 * allocate the data blocks while the @a allocator_strategy is used
166 * to allocate the buffers contained by those. The
167 * @a message_block_allocator is used to allocate new ACE_Message_Block
168 * objects when the duplicate() method is called. If a
169 * @a message_block_allocator is given, this ACE_Message_Block and
170 * future ACE_Message_Block objects created by duplicate() will be
171 * freed using this allocator when they are released.
172 * @note If you use this allocator, the ACE_Message_Block you created
173 * should have been created using this allocator because it will be
174 * released to the same allocator.
176 ACE_Message_Block (size_t size
,
177 ACE_Message_Type type
= MB_DATA
,
178 ACE_Message_Block
*cont
= nullptr,
179 const char *data
= nullptr,
180 ACE_Allocator
*allocator_strategy
= nullptr,
181 ACE_Lock
*locking_strategy
= nullptr,
182 unsigned long priority
= ACE_DEFAULT_MESSAGE_BLOCK_PRIORITY
,
183 const ACE_Time_Value
&execution_time
= ACE_Time_Value::zero
,
184 const ACE_Time_Value
&deadline_time
= ACE_Time_Value::max_time
,
185 ACE_Allocator
*data_block_allocator
= nullptr,
186 ACE_Allocator
*message_block_allocator
= nullptr);
189 * A copy constructor. This constructor is a bit different. If the
190 * incoming Message Block has a data block from the stack this
191 * constructor does a deep copy ie. allocates a new data block on
192 * the heap and does a copy of the data from the incoming message
193 * block. As a final note, the alignment information is used to
194 * align the data block if it is created afresh. If the incoming
195 * @a mb has a data block has a data block allocated from the heap,
196 * then this constructor just duplicates (ie. a shallow copy) the
197 * data block of the incoming @a mb.
199 ACE_Message_Block (const ACE_Message_Block
&mb
,
203 * Create a Message Block that assumes it has ownership of @a data,
204 * but in reality it doesn't (i.e., cannot delete it since it didn't
205 * malloc it!). Note that the @c size of the Message_Block will
206 * be @a size, but the @a length will be 0 until <wr_ptr> is set.
208 int init (const char *data
,
212 * Create an initialized message of type @a type containing @a size
213 * bytes. The @a cont argument initializes the continuation field in
214 * the Message_Block. If @a data == 0 then we create and own the
215 * @a data, using @a allocator_strategy to get the data if it's non-0. If
216 * @a data != 0 we assume that we have ownership of the @a data till
217 * this object ceases to exist (and don't delete it during
218 * destruction). If @a locking_strategy is non-0 then this is used
219 * to protect regions of code that access shared state (e.g.,
220 * reference counting) from race conditions. Note that the @a size
221 * of the Message_Block will be @a size, but the @a length will be 0
222 * until <wr_ptr> is set. The @a data_block_allocator is use to
223 * allocate the data blocks while the @a allocator_strategy is used
224 * to allocate the buffers contained by those.
226 int init (size_t size
,
227 ACE_Message_Type type
= MB_DATA
,
228 ACE_Message_Block
*cont
= 0,
229 const char *data
= 0,
230 ACE_Allocator
*allocator_strategy
= 0,
231 ACE_Lock
*locking_strategy
= 0,
232 unsigned long priority
= ACE_DEFAULT_MESSAGE_BLOCK_PRIORITY
,
233 const ACE_Time_Value
&execution_time
= ACE_Time_Value::zero
,
234 const ACE_Time_Value
&deadline_time
= ACE_Time_Value::max_time
,
235 ACE_Allocator
*data_block_allocator
= 0,
236 ACE_Allocator
*message_block_allocator
= 0);
239 * Delete all the resources held in the message.
241 * @note Note that release() is designed to release the continuation
242 * chain; the destructor is not. See release() for details.
244 virtual ~ACE_Message_Block ();
246 // = Message Type accessors and mutators.
248 /// Get type of the message.
249 ACE_Message_Type
msg_type () const;
251 /// Set type of the message.
252 void msg_type (ACE_Message_Type type
);
254 /// Find out what type of message this is.
255 int is_data_msg () const;
257 /// Find out what class of message this is (there are two classes,
258 /// @c normal messages and @c high-priority messages).
259 ACE_Message_Type
msg_class () const;
261 // = Message flag accessors and mutators.
262 /// Bitwise-or the @a more_flags into the existing message flags and
263 /// return the new value.
264 Message_Flags
set_flags (Message_Flags more_flags
);
266 /// Clear the message flag bits specified in @a less_flags and return
268 Message_Flags
clr_flags (Message_Flags less_flags
);
270 /// Get the current message flags.
271 Message_Flags
flags () const;
273 // = Data Block flag accessors and mutators.
274 /// Bitwise-or the @a more_flags into the existing message flags and
275 /// return the new value.
276 /** @todo I think the following set of methods could not be used at
277 * all. May be they are useless. Let us have it so that we don't
278 * mess up memory management of the Message_Block. Somebody correct
279 * me if I am totally totally wrong..
281 Message_Flags
set_self_flags (ACE_Message_Block::Message_Flags more_flags
);
283 /// Clear the message flag bits specified in @a less_flags and return
285 Message_Flags
clr_self_flags (ACE_Message_Block::Message_Flags less_flags
);
287 /// Get the current message flags.
288 Message_Flags
self_flags () const;
290 /// Get priority of the message.
291 unsigned long msg_priority () const;
293 /// Set priority of the message.
294 void msg_priority (unsigned long priority
);
296 /// Get execution time associated with the message.
297 const ACE_Time_Value
&msg_execution_time () const;
299 /// Set execution time associated with the message.
300 void msg_execution_time (const ACE_Time_Value
&et
);
302 /// Get absolute time of deadline associated with the message.
303 const ACE_Time_Value
&msg_deadline_time () const;
305 /// Set absolute time of deadline associated with the message.
306 void msg_deadline_time (const ACE_Time_Value
&dt
);
308 // = Deep copy and shallow copy methods.
310 /// Return an exact "deep copy" of the message, i.e., create fresh
311 /// new copies of all the Data_Blocks and continuations.
312 virtual ACE_Message_Block
*clone (Message_Flags mask
= 0) const;
314 /// Return a "shallow" copy that increments our reference count by 1.
315 virtual ACE_Message_Block
*duplicate () const;
318 * Return a "shallow" copy that increments our reference count by 1.
319 * This is similar to CORBA's _duplicate() method, which is useful
320 * if you want to eliminate lots of checks for NULL @a mb pointers
321 * before calling _duplicate() on them.
323 static ACE_Message_Block
*duplicate (const ACE_Message_Block
*mb
);
326 * Decrease the shared ACE_Data_Block's reference count by 1. If the
327 * ACE_Data_Block's reference count goes to 0, it is deleted.
328 * In all cases, this ACE_Message_Block is deleted - it must have come
329 * from the heap, or there will be trouble.
331 * release() is designed to release the continuation chain; the
332 * destructor is not. If we make the destructor release the
333 * continuation chain by calling release() or delete on the message
334 * blocks in the continuation chain, the following code will not
335 * work since the message block in the continuation chain is not off
338 * ACE_Message_Block mb1 (1024);
339 * ACE_Message_Block mb2 (1024);
343 * And hence, call release() on a dynamically allocated message
344 * block. This will release all the message blocks in the
345 * continuation chain. If you call delete or let the message block
346 * fall off the stack, cleanup of the message blocks in the
347 * continuation chain becomes the responsibility of the user.
349 * @retval 0, always, and the object this method was invoked on is no
352 virtual ACE_Message_Block
*release ();
355 * This behaves like the non-static method release(), except that it
356 * checks if @a mb is 0. This is similar to CORBA::release(), which
357 * is useful if you want to eliminate lots of checks for NULL
358 * pointers before calling release() on them. Returns @a mb.
360 static ACE_Message_Block
*release (ACE_Message_Block
*mb
);
362 // = Operations on Message data
365 * Copies data into this ACE_Message_Block. Data is copied into the
366 * block starting at the current write pointer.
368 * @param buf Pointer to the buffer to copy from.
369 * @param n The number of bytes to copy.
371 * @retval 0 on success; the write pointer is advanced by @arg n.
372 * @retval -1 if the amount of free space following the write pointer
373 * in the block is less than @arg n. Free space can be checked
374 * by calling space().
376 int copy (const char *buf
, size_t n
);
379 * Copies a 0-terminated character string into this ACE_Message_Block.
380 * The string is copied into the block starting at the current write
381 * pointer. The 0-terminator is included in the copied data.
383 * @param buf Pointer to the character string to copy from.
385 * @retval 0 on success; the write pointer is advanced by the string's
386 * length, including the 0 terminator.
387 * @retval -1 if the amount of free space following the write pointer
388 * in the block is less than required to hold the entire string.
389 * Free space can be checked by calling space().
391 int copy (const char *buf
);
393 /// Normalizes data in the top-level Message_Block to align with the base,
394 /// i.e., it "shifts" the data pointed to by <rd_ptr> down to the <base> and
395 /// then readjusts <rd_ptr> to point to <base> and <wr_ptr> to point
396 /// to <base> + the length of the moved data. Returns -1 and does
397 /// nothing if the <rd_ptr> is > <wr_ptr>, else 0 on success.
400 /// Resets the Message Block data to contain nothing, i.e., sets the
401 /// read and write pointers to align with the base.
404 /// Access all the allocators in the message block.
405 /// @todo Not sure whether we would need finer control while
406 /// trying to access allocators ie. a method for every allocator.
408 * This method returns the allocators only from the first message
409 * block in the chain.
411 * @param allocator_strategy Strategy used to allocate the
414 * @param data_block_allocator Strategy used to allocate the
415 * underlying data block
417 * @param message_block_allocator Strategy used to allocate the
420 void access_allocators (ACE_Allocator
*&allocator_strategy
,
421 ACE_Allocator
*&data_block_allocator
,
422 ACE_Allocator
*&message_block_allocator
);
424 /// Reset all the allocators in the message block.
425 /// @todo Not sure whether we would need finer control while
426 /// trying to reset allocators ie. a method for every allocator.
428 * This method resets the allocators in all the message blocks in
431 void reset_allocators (ACE_Allocator
*allocator_strategy
= 0,
432 ACE_Allocator
*data_block_allocator
= 0,
433 ACE_Allocator
*message_block_allocator
= 0);
435 /// Get message data.
438 /// Set message data (doesn't reallocate).
439 void base (char *data
,
441 Message_Flags
= DONT_DELETE
);
443 /// Return a pointer to 1 past the end of the allocated data in a message.
447 * Return a pointer to 1 past the end of the allotted data in a message.
448 * Allotted data may be less than allocated data if a value smaller than
449 * capacity() to is passed to size().
453 /// Get the read pointer.
454 char *rd_ptr () const;
456 /// Set the read pointer to @a ptr.
457 void rd_ptr (char *ptr
);
459 /// Set the read pointer ahead @a n bytes.
460 void rd_ptr (size_t n
);
462 /// Get the write pointer.
463 char *wr_ptr () const;
465 /// Set the write pointer to @a ptr.
466 void wr_ptr (char *ptr
);
468 /// Set the write pointer ahead @a n bytes. This is used to compute
469 /// the <length> of a message.
470 void wr_ptr (size_t n
);
472 /** @name Message length and size operations
474 * Message length is (wr_ptr - rd_ptr).
476 * Message size is capacity of the message, including data outside
477 * the [rd_ptr,wr_ptr] range.
480 /// Get the length of the message
481 size_t length () const;
483 /// Set the length of the message
484 void length (size_t n
);
486 /// Get the length of the Message_Blocks, including chained
488 size_t total_length () const;
490 /// Get the total number of bytes in all Message_Blocks, including
491 /// chained Message_Blocks.
492 size_t total_size () const;
494 /// Get the total number of bytes and total length in all
495 /// Message_Blocks, including chained Message_Blocks.
496 void total_size_and_length (size_t &mb_size
,
497 size_t &mb_length
) const;
499 /// Get the number of bytes in the top-level Message_Block (i.e.,
500 /// does not consider the bytes in chained Message_Blocks).
501 size_t size () const;
504 * Set the number of bytes in the top-level Message_Block,
505 * reallocating space if necessary. However, the @c rd_ptr_ and
506 * @c wr_ptr_ remain at the original offsets into the buffer, even if
507 * it is reallocated. Returns 0 if successful, else -1.
509 int size (size_t length
);
511 /// Get the number of allocated bytes in all Message_Block, including
512 /// chained Message_Blocks.
513 size_t total_capacity () const;
515 /// Get the number of allocated bytes in the top-level Message_Block.
516 size_t capacity () const;
518 /// Get the number of bytes available after the <wr_ptr_> in the
519 /// top-level Message_Block.
520 size_t space () const;
523 // = ACE_Data_Block methods.
526 * Get a pointer to the data block. Note that the ACE_Message_Block
527 * still references the block; this call does not change the reference
530 ACE_Data_Block
*data_block () const;
533 * Set a new data block pointer. The original ACE_Data_Block is released
534 * as a result of this call. If you need to keep the original block, call
535 * <replace_data_block> instead. Upon return, this ACE_Message_Block
536 * holds a pointer to the new ACE_Data_Block, taking over the reference
537 * you held on it prior to the call.
539 void data_block (ACE_Data_Block
*);
541 /// Set a new data block pointer. A pointer to the original ACE_Data_Block
542 /// is returned, and not released (as it is with <data_block>).
543 ACE_Data_Block
*replace_data_block (ACE_Data_Block
*);
545 // = The continuation field chains together composite messages.
546 /// Get the continuation field.
547 ACE_Message_Block
*cont () const;
549 /// Set the continuation field.
550 void cont (ACE_Message_Block
*);
552 // = Pointer to the Message_Block directly ahead in the ACE_Message_Queue.
553 /// Get link to next message.
554 ACE_Message_Block
*next () const;
556 /// Set link to next message.
557 void next (ACE_Message_Block
*);
559 // = Pointer to the Message_Block directly behind in the ACE_Message_Queue.
560 /// Get link to prev message.
561 ACE_Message_Block
*prev () const;
563 /// Set link to prev message.
564 void prev (ACE_Message_Block
*);
566 // = The locking strategy prevents race conditions.
567 /// Get the locking strategy.
568 ACE_Lock
*locking_strategy ();
570 /// Set a new locking strategy and return the hold one.
571 ACE_Lock
*locking_strategy (ACE_Lock
*);
573 /// Get the current reference count.
574 int reference_count () const;
576 /// Dump the state of an object.
579 /// Declare the dynamic allocation hooks.
580 ACE_ALLOC_HOOK_DECLARE
;
583 // = Internal initialization methods.
584 /// Perform the actual initialization.
585 ACE_Message_Block (size_t size
,
586 ACE_Message_Type type
,
587 ACE_Message_Block
*cont
,
589 ACE_Allocator
*allocator_strategy
,
590 ACE_Lock
*locking_strategy
,
592 unsigned long priority
,
593 const ACE_Time_Value
&execution_time
,
594 const ACE_Time_Value
&deadline_time
,
596 ACE_Allocator
*data_block_allocator
,
597 ACE_Allocator
*message_block_allocator
);
599 /// Internal release implementation
600 /// Returns 1 if the data block has to be destroyed.
601 int release_i (ACE_Lock
*lock
);
603 /// Perform the actual initialization.
604 int init_i (size_t size
,
605 ACE_Message_Type type
,
606 ACE_Message_Block
*cont
,
608 ACE_Allocator
*allocator_strategy
,
609 ACE_Lock
*locking_strategy
,
611 unsigned long priority
,
612 const ACE_Time_Value
&execution_time
,
613 const ACE_Time_Value
&deadline_time
,
615 ACE_Allocator
*data_block_allocator
,
616 ACE_Allocator
*message_block_allocator
);
618 /// Pointer to beginning of next read.
621 /// Pointer to beginning of next write.
624 /// Priority of message.
625 unsigned long priority_
;
627 #if defined (ACE_HAS_TIMED_MESSAGE_BLOCKS)
628 /// Execution time associated with the message.
629 ACE_Time_Value execution_time_
;
631 /// Absolute deadline time for message.
632 ACE_Time_Value deadline_time_
;
633 #endif /* ACE_HAS_TIMED_MESSAGE_BLOCKS */
635 // = Links to other ACE_Message_Block *s.
636 /// Pointer to next message block in the chain.
637 ACE_Message_Block
*cont_
;
639 /// Pointer to next message in the list.
640 ACE_Message_Block
*next_
;
642 /// Pointer to previous message in the list.
643 ACE_Message_Block
*prev_
;
645 /// Misc flags (e.g., DONT_DELETE and USER_FLAGS).
646 ACE_Message_Block::Message_Flags flags_
;
648 /// Pointer to the reference counted data structure that contains the
649 /// actual memory buffer.
650 ACE_Data_Block
*data_block_
;
652 /// The allocator used to destroy ourselves when release is called
653 /// and create new message blocks on duplicate.
654 ACE_Allocator
*message_block_allocator_
;
657 // = Disallow these operations for now (use <clone> instead).
658 ACE_Message_Block
&operator= (const ACE_Message_Block
&);
659 ACE_Message_Block (const ACE_Message_Block
&);
663 * @class ACE_Data_Block
665 * @brief Stores the data payload that is accessed via one or more
666 * ACE_Message_Block's.
668 * This data structure is reference counted to maximize
669 * sharing. It also contains the <locking_strategy_> (which
670 * protects the reference count from race conditions in
671 * concurrent programs) and the <allocation_strategy_> (which
672 * determines what memory pool is used to allocate the memory).
674 class ACE_Export ACE_Data_Block
677 /// Default "do-nothing" constructor.
681 ACE_Data_Block (size_t size
,
682 ACE_Message_Block::ACE_Message_Type msg_type
,
683 const char *msg_data
,
684 ACE_Allocator
*allocator_strategy
,
685 ACE_Lock
*locking_strategy
,
686 ACE_Message_Block::Message_Flags flags
,
687 ACE_Allocator
*data_block_allocator
);
689 /// Delete all the resources held in the message.
690 virtual ~ACE_Data_Block ();
692 /// Get type of the message.
693 ACE_Message_Block::ACE_Message_Type
msg_type () const;
695 /// Set type of the message.
696 void msg_type (ACE_Message_Block::ACE_Message_Type type
);
698 /// Get message data pointer
701 /// Set message data pointer (doesn't reallocate).
702 void base (char *data
,
704 ACE_Message_Block::Message_Flags mflags
= ACE_Message_Block::DONT_DELETE
);
706 /// Return a pointer to 1 past the end of the allocated data in a message.
710 * Return a pointer to 1 past the end of the allotted data in a message.
711 * The allotted data may be less than allocated data if <size()> is passed
712 * an argument less than <capacity()>.
716 // = Message size is the total amount of space allocated.
718 /// Get the total amount of allotted space in the message. The amount of
719 /// allotted space may be less than allocated space.
720 size_t size () const;
722 /// Set the total amount of space in the message. Returns 0 if
723 /// successful, else -1.
724 int size (size_t length
);
726 /// Get the total amount of allocated space.
727 size_t capacity () const;
730 * Return an exact "deep copy" of the message, i.e., create fresh
731 * new copies of all the Data_Blocks and continuations.
732 * Notice that Data_Blocks can act as "Prototypes", i.e. derived
733 * classes can override this method and create instances of
736 virtual ACE_Data_Block
*clone (ACE_Message_Block::Message_Flags mask
= 0) const;
739 * As clone above, but it does not copy the contents of the buffer,
740 * i.e., create a new Data_Block of the same dynamic type, with the
741 * same allocator, locking_strategy, and with the same amount of
742 * storage available (if @a max_size is zero) but the buffer is unitialized.
743 * If @a max_size is specified other than zero, it will be used when
744 * creating the new data block.
746 virtual ACE_Data_Block
*clone_nocopy (ACE_Message_Block::Message_Flags mask
= 0,
747 size_t max_size
= 0) const;
749 /// Return a "shallow" copy that increments our reference count by 1.
750 ACE_Data_Block
*duplicate ();
753 * Decrease the shared reference count by 1. If the reference count
754 * is > 0 then return this; else if reference count == 0 then delete
755 * @c this and @a mb and return 0. Behavior is undefined if reference
758 ACE_Data_Block
*release (ACE_Lock
*lock
= 0);
760 // = Message flag accessors and mutators.
761 /// Bitwise-or the @a more_flags into the existing message flags and
762 /// return the new value.
763 ACE_Message_Block::Message_Flags
set_flags (ACE_Message_Block::Message_Flags more_flags
);
765 /// Clear the message flag bits specified in @a less_flags and return
767 ACE_Message_Block::Message_Flags
clr_flags (ACE_Message_Block::Message_Flags less_flags
);
769 /// Get the current message flags.
770 ACE_Message_Block::Message_Flags
flags () const;
772 /// Obtain the allocator strategy.
773 ACE_Allocator
*allocator_strategy () const;
775 // = The locking strategy prevents race conditions.
776 /// Get the locking strategy.
777 ACE_Lock
*locking_strategy ();
779 /// Set a new locking strategy and return the hold one.
780 ACE_Lock
*locking_strategy (ACE_Lock
*);
782 /// Dump the state of an object.
785 /// Get the current reference count.
786 int reference_count () const;
788 /// Get the allocator used to create this object
789 ACE_Allocator
*data_block_allocator () const;
792 /// Internal release implementation
793 virtual ACE_Data_Block
*release_i ();
795 /// Internal get the current reference count.
796 int reference_count_i () const;
799 * Decrease the reference count, but don't delete the object.
800 * Returns 0 if the object should be removed.
801 * If @a lock is equal to the locking strategy then we assume that
802 * the lock is being held by the current thread; this is used to
803 * release all the data blocks in a chain while holding a single
806 friend class ACE_Message_Block
;
807 ACE_Data_Block
*release_no_delete (ACE_Lock
*lock
);
810 ACE_Message_Block::ACE_Message_Type type_
;
812 /// Current size of message block.
815 /// Total size of buffer.
818 /// Misc flags (e.g., DONT_DELETE and USER_FLAGS).
819 ACE_Message_Block::Message_Flags flags_
;
821 /// Pointer To beginning of message payload.
826 * Pointer to the allocator defined for this ACE_Data_Block. Note
827 * that this pointer is shared by all owners of this
830 ACE_Allocator
*allocator_strategy_
;
833 * Pointer to the locking strategy defined for this
834 * ACE_Data_Block. This is used to protect regions of code that
835 * access shared ACE_Data_Block state. Note that this lock is
836 * shared by all owners of the ACE_Data_Block's data.
838 ACE_Lock
*locking_strategy_
;
841 * Reference count for this ACE_Data_Block, which is used to avoid
842 * deep copies (i.e., clone()). Note that this pointer value is
843 * shared by all owners of the <Data_Block>'s data, i.e., all the
844 * ACE_Message_Blocks.
846 int reference_count_
;
848 /// The allocator use to destroy ourselves.
849 ACE_Allocator
*data_block_allocator_
;
852 // = Disallow these operations.
853 ACE_Data_Block
&operator= (const ACE_Data_Block
&);
854 ACE_Data_Block (const ACE_Data_Block
&);
857 ACE_END_VERSIONED_NAMESPACE_DECL
859 #if defined (__ACE_INLINE__)
860 #include "ace/Message_Block.inl"
861 #endif /* __ACE_INLINE__ */
863 #include "ace/Message_Block_T.h"
865 #include /**/ "ace/post.h"
867 #endif /* ACE_MESSAGE_BLOCK_H */