/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 | | //////////////////////////////////////////////////////////////////////////////// |