C++程序  |  94行  |  3.03 KB

// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef MOJO_EDK_SYSTEM_PORTS_MESSAGE_H_
#define MOJO_EDK_SYSTEM_PORTS_MESSAGE_H_

#include <stddef.h>

#include <memory>

#include "mojo/edk/system/ports/name.h"

namespace mojo {
namespace edk {
namespace ports {

// A message consists of a header (array of bytes), payload (array of bytes)
// and an array of ports. The header is used by the Node implementation.
//
// This class is designed to be subclassed, and the subclass is responsible for
// providing the underlying storage. The header size will be aligned, and it
// should be followed in memory by the array of ports and finally the payload.
//
// NOTE: This class does not manage the lifetime of the ports it references.
class Message {
 public:
  virtual ~Message() {}

  // Inspect the message at |bytes| and return the size of each section. Returns
  // |false| if the message is malformed and |true| otherwise.
  static bool Parse(const void* bytes,
                    size_t num_bytes,
                    size_t* num_header_bytes,
                    size_t* num_payload_bytes,
                    size_t* num_ports_bytes);

  void* mutable_header_bytes() { return start_; }
  const void* header_bytes() const { return start_; }
  size_t num_header_bytes() const { return num_header_bytes_; }

  void* mutable_payload_bytes() {
    return start_ + num_header_bytes_ + num_ports_bytes_;
  }
  const void* payload_bytes() const {
    return const_cast<Message*>(this)->mutable_payload_bytes();
  }
  size_t num_payload_bytes() const { return num_payload_bytes_; }

  PortName* mutable_ports() {
    return reinterpret_cast<PortName*>(start_ + num_header_bytes_);
  }
  const PortName* ports() const {
    return const_cast<Message*>(this)->mutable_ports();
  }
  size_t num_ports_bytes() const { return num_ports_bytes_; }
  size_t num_ports() const { return num_ports_bytes_ / sizeof(PortName); }

 protected:
  // Constructs a new Message base for a user message.
  //
  // Note: You MUST call InitializeUserMessageHeader() before this Message is
  // ready for transmission.
  Message(size_t num_payload_bytes, size_t num_ports);

  // Constructs a new Message base for an internal message. Do NOT call
  // InitializeUserMessageHeader() when using this constructor.
  Message(size_t num_header_bytes,
          size_t num_payload_bytes,
          size_t num_ports_bytes);

  Message(const Message& other) = delete;
  void operator=(const Message& other) = delete;

  // Initializes the header in a newly allocated message buffer to carry a
  // user message.
  void InitializeUserMessageHeader(void* start);

  // Note: storage is [header][ports][payload].
  char* start_ = nullptr;
  size_t num_ports_ = 0;
  size_t num_header_bytes_ = 0;
  size_t num_ports_bytes_ = 0;
  size_t num_payload_bytes_ = 0;
};

using ScopedMessage = std::unique_ptr<Message>;

}  // namespace ports
}  // namespace edk
}  // namespace mojo

#endif  // MOJO_EDK_SYSTEM_PORTS_MESSAGE_H_