Coverage Report

Created: 2018-09-25 14:53

/src/mozilla-central/xpcom/io/nsSegmentedBuffer.cpp
Line
Count
Source (jump to first uncovered line)
1
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
3
/* This Source Code Form is subject to the terms of the Mozilla Public
4
 * License, v. 2.0. If a copy of the MPL was not distributed with this
5
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
6
7
#include "nsSegmentedBuffer.h"
8
#include "nsMemory.h"
9
10
nsresult
11
nsSegmentedBuffer::Init(uint32_t aSegmentSize, uint32_t aMaxSize)
12
0
{
13
0
  if (mSegmentArrayCount != 0) {
14
0
    return NS_ERROR_FAILURE;  // initialized more than once
15
0
  }
16
0
  mSegmentSize = aSegmentSize;
17
0
  mMaxSize = aMaxSize;
18
#if 0 // testing...
19
  mSegmentArrayCount = 2;
20
#else
21
0
  mSegmentArrayCount = NS_SEGMENTARRAY_INITIAL_COUNT;
22
0
#endif
23
0
  return NS_OK;
24
0
}
25
26
char*
27
nsSegmentedBuffer::AppendNewSegment()
28
0
{
29
0
  if (GetSize() >= mMaxSize) {
30
0
    return nullptr;
31
0
  }
32
0
33
0
  if (!mSegmentArray) {
34
0
    uint32_t bytes = mSegmentArrayCount * sizeof(char*);
35
0
    mSegmentArray = (char**)moz_xmalloc(bytes);
36
0
    memset(mSegmentArray, 0, bytes);
37
0
  }
38
0
39
0
  if (IsFull()) {
40
0
    uint32_t newArraySize = mSegmentArrayCount * 2;
41
0
    uint32_t bytes = newArraySize * sizeof(char*);
42
0
    mSegmentArray = (char**)moz_xrealloc(mSegmentArray, bytes);
43
0
    // copy wrapped content to new extension
44
0
    if (mFirstSegmentIndex > mLastSegmentIndex) {
45
0
      // deal with wrap around case
46
0
      memcpy(&mSegmentArray[mSegmentArrayCount],
47
0
             mSegmentArray,
48
0
             mLastSegmentIndex * sizeof(char*));
49
0
      memset(mSegmentArray, 0, mLastSegmentIndex * sizeof(char*));
50
0
      mLastSegmentIndex += mSegmentArrayCount;
51
0
      memset(&mSegmentArray[mLastSegmentIndex], 0,
52
0
             (newArraySize - mLastSegmentIndex) * sizeof(char*));
53
0
    } else {
54
0
      memset(&mSegmentArray[mLastSegmentIndex], 0,
55
0
             (newArraySize - mLastSegmentIndex) * sizeof(char*));
56
0
    }
57
0
    mSegmentArrayCount = newArraySize;
58
0
  }
59
0
60
0
  char* seg = (char*)malloc(mSegmentSize);
61
0
  if (!seg) {
62
0
    return nullptr;
63
0
  }
64
0
  mSegmentArray[mLastSegmentIndex] = seg;
65
0
  mLastSegmentIndex = ModSegArraySize(mLastSegmentIndex + 1);
66
0
  return seg;
67
0
}
68
69
bool
70
nsSegmentedBuffer::DeleteFirstSegment()
71
0
{
72
0
  NS_ASSERTION(mSegmentArray[mFirstSegmentIndex] != nullptr, "deleting bad segment");
73
0
  free(mSegmentArray[mFirstSegmentIndex]);
74
0
  mSegmentArray[mFirstSegmentIndex] = nullptr;
75
0
  int32_t last = ModSegArraySize(mLastSegmentIndex - 1);
76
0
  if (mFirstSegmentIndex == last) {
77
0
    mLastSegmentIndex = last;
78
0
    return true;
79
0
  } else {
80
0
    mFirstSegmentIndex = ModSegArraySize(mFirstSegmentIndex + 1);
81
0
    return false;
82
0
  }
83
0
}
84
85
bool
86
nsSegmentedBuffer::DeleteLastSegment()
87
0
{
88
0
  int32_t last = ModSegArraySize(mLastSegmentIndex - 1);
89
0
  NS_ASSERTION(mSegmentArray[last] != nullptr, "deleting bad segment");
90
0
  free(mSegmentArray[last]);
91
0
  mSegmentArray[last] = nullptr;
92
0
  mLastSegmentIndex = last;
93
0
  return (bool)(mLastSegmentIndex == mFirstSegmentIndex);
94
0
}
95
96
bool
97
nsSegmentedBuffer::ReallocLastSegment(size_t aNewSize)
98
0
{
99
0
  int32_t last = ModSegArraySize(mLastSegmentIndex - 1);
100
0
  NS_ASSERTION(mSegmentArray[last] != nullptr, "realloc'ing bad segment");
101
0
  char* newSegment = (char*)realloc(mSegmentArray[last], aNewSize);
102
0
  if (newSegment) {
103
0
    mSegmentArray[last] = newSegment;
104
0
    return true;
105
0
  }
106
0
  return false;
107
0
}
108
109
void
110
nsSegmentedBuffer::Empty()
111
0
{
112
0
  if (mSegmentArray) {
113
0
    for (uint32_t i = 0; i < mSegmentArrayCount; i++) {
114
0
      if (mSegmentArray[i]) {
115
0
        free(mSegmentArray[i]);
116
0
      }
117
0
    }
118
0
    free(mSegmentArray);
119
0
    mSegmentArray = nullptr;
120
0
  }
121
0
  mSegmentArrayCount = NS_SEGMENTARRAY_INITIAL_COUNT;
122
0
  mFirstSegmentIndex = mLastSegmentIndex = 0;
123
0
}
124
125
#if 0
126
void
127
TestSegmentedBuffer()
128
{
129
  nsSegmentedBuffer* buf = new nsSegmentedBuffer();
130
  NS_ASSERTION(buf, "out of memory");
131
  buf->Init(4, 16);
132
  char* seg;
133
  bool empty;
134
  seg = buf->AppendNewSegment();
135
  NS_ASSERTION(seg, "AppendNewSegment failed");
136
  seg = buf->AppendNewSegment();
137
  NS_ASSERTION(seg, "AppendNewSegment failed");
138
  seg = buf->AppendNewSegment();
139
  NS_ASSERTION(seg, "AppendNewSegment failed");
140
  empty = buf->DeleteFirstSegment();
141
  NS_ASSERTION(!empty, "DeleteFirstSegment failed");
142
  empty = buf->DeleteFirstSegment();
143
  NS_ASSERTION(!empty, "DeleteFirstSegment failed");
144
  seg = buf->AppendNewSegment();
145
  NS_ASSERTION(seg, "AppendNewSegment failed");
146
  seg = buf->AppendNewSegment();
147
  NS_ASSERTION(seg, "AppendNewSegment failed");
148
  seg = buf->AppendNewSegment();
149
  NS_ASSERTION(seg, "AppendNewSegment failed");
150
  empty = buf->DeleteFirstSegment();
151
  NS_ASSERTION(!empty, "DeleteFirstSegment failed");
152
  empty = buf->DeleteFirstSegment();
153
  NS_ASSERTION(!empty, "DeleteFirstSegment failed");
154
  empty = buf->DeleteFirstSegment();
155
  NS_ASSERTION(!empty, "DeleteFirstSegment failed");
156
  empty = buf->DeleteFirstSegment();
157
  NS_ASSERTION(empty, "DeleteFirstSegment failed");
158
  delete buf;
159
}
160
#endif
161
162
////////////////////////////////////////////////////////////////////////////////