summaryrefslogtreecommitdiffstats
path: root/net/base/io_buffer.h
blob: eaf3a39baebf94a565a8bc4b5ff9373d775435c2 (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
// Copyright (c) 2009 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 NET_BASE_IO_BUFFER_H_
#define NET_BASE_IO_BUFFER_H_

#include <string>

#include "base/ref_counted.h"
#include "base/scoped_ptr.h"

namespace net {

// This is a simple wrapper around a buffer that provides ref counting for
// easier asynchronous IO handling.
class IOBuffer : public base::RefCountedThreadSafe<IOBuffer> {
 public:
  IOBuffer() : data_(NULL) {}
  explicit IOBuffer(int buffer_size);
  virtual ~IOBuffer() {
    delete[] data_;
  }

  char* data() { return data_; }

 protected:
  // Only allow derived classes to specify data_.
  // In all other cases, we own data_, and must delete it at destruction time.
  explicit IOBuffer(char* data) : data_(data) {}
  char* data_;
};

// This version stores the size of the buffer so that the creator of the object
// doesn't have to keep track of that value.
// NOTE: This doesn't mean that we want to stop sending the size as an explicit
// argument to IO functions. Please keep using IOBuffer* for API declarations.
class IOBufferWithSize : public IOBuffer {
 public:
  explicit IOBufferWithSize(int size) : IOBuffer(size), size_(size) {}
  ~IOBufferWithSize() {}

  int size() const { return size_; }

 private:
  int size_;
};

// This is a read only IOBuffer.  The data is stored in a string and
// the IOBuffer interface does not provide a proper way to modify it.
class StringIOBuffer : public IOBuffer {
 public:
  explicit StringIOBuffer(const std::string& s)
      : IOBuffer(static_cast<char*>(NULL)),
        string_data_(s) {
    data_ = const_cast<char*>(string_data_.data());
  }
  ~StringIOBuffer() {
    // We haven't allocated the buffer, so remove it before the base class
    // destructor tries to delete[] it.
    data_ = NULL;
  }

  int size() const { return string_data_.size(); }

 private:
  std::string string_data_;
};

// This version wraps an existing IOBuffer and provides convenient functions
// to progressively read all the data.
class DrainableIOBuffer : public IOBuffer {
 public:
  DrainableIOBuffer(IOBuffer* base, int size)
      : IOBuffer(base->data()), base_(base), size_(size), used_(0) {}
  ~DrainableIOBuffer() {
    // The buffer is owned by the |base_| instance.
    data_ = NULL;
  }

  // DidConsume() changes the |data_| pointer so that |data_| always points
  // to the first unconsumed byte.
  void DidConsume(int bytes) { SetOffset(used_ + bytes); }

  // Returns the number of unconsumed bytes.
  int BytesRemaining() const { return size_ - used_; }

  // Returns the number of consumed bytes.
  int BytesConsumed() const { return used_; }

  // Seeks to an arbitrary point in the buffer. The notion of bytes consumed
  // and remaining are updated appropriately.
  void SetOffset(int bytes);

  int size() const { return size_; }

 private:
  scoped_refptr<IOBuffer> base_;
  int size_;
  int used_;
};

// This version provides a resizable buffer and a changeable offset.
class GrowableIOBuffer : public IOBuffer {
 public:
  GrowableIOBuffer() : IOBuffer(), capacity_(0), offset_(0) {}
  ~GrowableIOBuffer() { data_ = NULL; }

  int capacity() { return capacity_; }
  void set_capacity(int capacity);

  // |offset| moves the |data_| pointer, allowing "seeking" in the data.
  int offset() { return offset_; }
  void set_offset(int offset);

  int RemainingCapacity() { return capacity_ - offset_; }
  char* StartOfBuffer() { return real_data_.get(); }

 private:
  scoped_ptr_malloc<char> real_data_;
  int capacity_;
  int offset_;
};

// This class allows the creation of a temporary IOBuffer that doesn't really
// own the underlying buffer. Please use this class only as a last resort.
// A good example is the buffer for a synchronous operation, where we can be
// sure that nobody is keeping an extra reference to this object so the lifetime
// of the buffer can be completely managed by its intended owner.
class WrappedIOBuffer : public IOBuffer {
 public:
  explicit WrappedIOBuffer(const char* data)
      : IOBuffer(const_cast<char*>(data)) {}
  ~WrappedIOBuffer() {
    data_ = NULL;
  }
};

}  // namespace net

#endif  // NET_BASE_IO_BUFFER_H_