/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef _LOG4CXX_HELPERS_DATAGRAM_PACKET
#define _LOG4CXX_HELPERS_DATAGRAM_PACKET

#include <log4cxx/helpers/objectimpl.h>
#include <log4cxx/helpers/objectptr.h>
#include <log4cxx/helpers/inetaddress.h>

namespace log4cxx
{
namespace helpers
{

/** This class represents a datagram packet.
<p>Datagram packets are used to implement a connectionless packet
delivery service. Each message is routed from one machine to another
based solely on information contained within that packet. Multiple
packets sent from one machine to another might be routed differently,
and might arrive in any order.
*/
class LOG4CXX_EXPORT DatagramPacket : public helpers::ObjectImpl
{
	protected:
		/** the data for this packet. */
		void* buf;

		/** The offset of the data for this packet. */
		int offset;

		/** The length of the data for this packet. */
		int length;

		/** The IP address for this packet. */
		InetAddressPtr address;

		/** The UDP port number of the remote host. */
		int port;

	public:
		DECLARE_ABSTRACT_LOG4CXX_OBJECT(DatagramPacket)
		BEGIN_LOG4CXX_CAST_MAP()
		LOG4CXX_CAST_ENTRY(DatagramPacket)
		END_LOG4CXX_CAST_MAP()

		/** Constructs a DatagramPacket for receiving packets of length
		<code>length</code>. */
		DatagramPacket(void* buf, int length);

		/** Constructs a datagram packet for sending packets of length
		<code>length</code> to the specified port number on the specified
		host. */
		DatagramPacket(void* buf, int length, InetAddressPtr address, int port);

		/** Constructs a DatagramPacket for receiving packets of length
		<code>length</code>, specifying an offset into the buffer. */
		DatagramPacket(void* buf, int offset, int length);

		/** Constructs a datagram packet for sending packets of length
		<code>length</code> with offset <code>offset</code> to the
		specified port number on the specified host. */
		DatagramPacket(void* buf, int offset, int length, InetAddressPtr address,
			int port);

		~DatagramPacket();

		/** Returns the IP address of the machine to which this datagram
		is being sent or from which the datagram was received. */
		inline InetAddressPtr getAddress() const
		{
			return address;
		}

		/** Returns the data received or the data to be sent. */
		inline void* getData() const
		{
			return buf;
		}

		/** Returns the length of the data to be sent or the length of the
		data received. */
		inline int getLength() const
		{
			return length;
		}

		/** Returns the offset of the data to be sent or the offset of the
		data received. */
		inline int getOffset() const
		{
			return offset;
		}

		/** Returns the port number on the remote host to which this
		 datagram is being sent or from which the datagram was received. */
		inline int getPort() const
		{
			return port;
		}

		inline void setAddress(InetAddressPtr address1)
		{
			this->address = address1;
		}

		/** Set the data buffer for this packet. */
		inline void setData(void* buf1)
		{
			this->buf = buf1;
		}

		/** Set the data buffer for this packet. */
		inline void setData(void* buf1, int offset1, int length1)
		{
			this->buf = buf1;
			this->offset = offset1;
			this->length = length1;
		}

		/** Set the length for this packet. */
		inline void setLength(int length1)
		{
			this->length = length1;
		}

		inline void setPort(int port1)
		{
			this->port = port1;
		}

	private:
		//
		//  prevent copy and assignment statements
		DatagramPacket(const DatagramPacket&);
		DatagramPacket& operator=(const DatagramPacket&);

}; // class DatagramPacket
LOG4CXX_PTR_DEF(DatagramPacket);
}  // namespace helpers
} // namespace log4cxx

#endif // _LOG4CXX_HELPERS_DATAGRAM_PACKET
