Yoba Perl
array.cpp
1 #include "yobaperl/array.hpp"
2 #include "yobaperl/perl.hpp"
3 #include "yobaperl/scalar.hpp"
4 
5 namespace yoba {
6 
7 
8 
9 Array::Array(Perl & perl, AV * av, bool increase_refcount)
10  : Variable(perl, reinterpret_cast<SV *>(av), increase_refcount)
11 {
12 }
13 
14 
15 
16 Scalar Array::get(const SSize_t index) const
17 {
18  if(_checkIndex(index))
19  {
20  SV ** elements = av_fetch(getAV(), index, /* lval */FALSE); //TODO lval?
21  YOBAPERL_ASSERT(elements);
22 
23  SV * result = *elements;
24  YOBAPERL_ASSERT(result);
25 
26  return Scalar(_perl, result, /* ref++ */true);
27  }
28  else
29  {
30  std::string error = "Array element not exists (index: " + std::to_string(index) + ")";
31 
32  if(_perl.isExceptionsEnabled())
33  throw PerlException(_perl.getId(), error);
34 
35  if(_perl.isWarningsEnabled())
36  warn("%s", error.c_str());
37 
38  return _perl.newScalar();
39  }
40 }
41 
43 {
44  return get(0);
45 }
46 
48 {
49  return get(getSize() - 1);
50 }
51 
52 Array & Array::replace(const SSize_t index, const Scalar & scalar)
53 {
54  if(_checkIndex(index))
55  {
56  _store(index, scalar);
57  }
58  else
59  {
60  //TODO Complete
61  std::string error = "Array element not exists (index: " + std::to_string(index) + ")";
62 
63 // if(_perl.isExceptionsEnabled())
64  throw PerlException(_perl.getId(), error);
65 
66 // if(_perl.isWarningsEnabled())
67 // warn("%s", error.c_str());
68 
69 // _store(index, scalar.getSV());
70  }
71 
72  return *this;
73 }
74 
75 Array & Array::push(const Scalar & scalar)
76 {
77  _push(scalar);
78  return *this;
79 }
80 
81 Array & Array::push(const Array & array)
82 {
83  reserve(getSize() + array.getSize());
84  for(int i = 0; i < array.getSize(); i++)
85  push(array[i]);
86  return *this;
87 }
88 
89 Array & Array::push(const std::vector<Scalar> & scalars)
90 {
91  reserve(getSize() + scalars.size());
92  for(auto it = scalars.cbegin(); it != scalars.cend(); it++)
93  push(*it);
94  return *this;
95 }
96 
97 Array & Array::push(const std::list<Scalar> & scalars)
98 {
99  reserve(getSize() + scalars.size());
100  for(auto it = scalars.cbegin(); it != scalars.cend(); it++)
101  push(*it);
102  return *this;
103 }
104 
105 Array & Array::unshift(const Scalar & scalar)
106 {
107  av_unshift(getAV(), 1);
108  _store(0, scalar);
109 
110  return *this;
111 }
112 
113 Array & Array::unshift(const Array & array)
114 {
115  av_unshift(getAV(), array.getSize());
116 
117  const int size = array.getSize();
118  for(int i = 0; i < size; i++)
119  _store(i, array[i]);
120 
121  return *this;
122 }
123 
124 Array & Array::unshift(const std::vector<Scalar> & scalars)
125 {
126  av_unshift(getAV(), scalars.size());
127 
128  for(std::size_t i = 0; i < scalars.size(); i++)
129  _store(i, scalars[i]);
130 
131  return *this;
132 }
133 
134 Array & Array::unshift(const std::list<Scalar> & scalars)
135 {
136  av_unshift(getAV(), scalars.size());
137 
138  SSize_t i = 0;
139  auto end_ = scalars.cend();
140  for(auto it = scalars.cbegin(); it != end_; ++it)
141  _store(i++, *it);
142 
143  return *this;
144 }
145 
147 {
148  if(!isEmpty())
149  {
150  SV * result = av_pop(getAV());
151  YOBAPERL_ASSERT(result);
152  return Scalar(_perl, result, false);
153  }
154  else
155  {
156  static const std::string error = "Pop empty array";
157 
158  if(_perl.isExceptionsEnabled())
159  throw PerlException(_perl.getId(), error);
160 
161  if(_perl.isWarningsEnabled())
162  warn("%s", error.c_str());
163 
164  return _perl.newScalar();
165  }
166 }
167 
169 {
170  if(!isEmpty())
171  {
172  SV * result = av_shift(getAV());
173  YOBAPERL_ASSERT(result);
174  return Scalar(_perl, result, false);
175  }
176  else
177  {
178  static const std::string error = "Pop empty array";
179 
180  if(_perl.isExceptionsEnabled())
181  throw PerlException(_perl.getId(), error);
182 
183  if(_perl.isWarningsEnabled())
184  warn("%s", error.c_str());
185 
186  return _perl.newScalar();
187  }
188 }
189 
191 {
192  av_clear(getAV());
193  return *this;
194 }
195 
196 
197 
198 int Array::getSize() const
199 {
200  return av_len(getAV()) + 1; // Perl_av_len returns -1 if array is empty
201 }
202 
203 bool Array::isEmpty() const
204 {
205  return getSize() == 0;
206 }
207 
208 bool Array::isExists(SSize_t index) const
209 {
210  return av_exists(getAV(), index);
211 }
212 
213 Array & Array::reserve(SSize_t size)
214 {
215  av_extend(getAV(), size - 1);
216  return *this;
217 }
218 
220 {
221  Array result = _perl.newArray();
222  result.reserve(getSize());
223 
224  for(Scalar elem : *this)
225  result.push(elem.makeCopy());
226 
227  return result;
228 }
229 
231 {
232  return Scalar(_perl, newRV_inc(getSV()), /*ref++*/false);
233 }
234 
235 std::string Array::toString() const
236 {
237  std::stringstream ss;
238 
239  ss << "(";
240 
241  const int array_size = getSize();
242  for(SSize_t i = 0; i < array_size; i++)
243  {
244  ss << (*this).get(i);
245  if(i < array_size - 1)
246  ss << ", ";
247  }
248 
249  ss << ")";
250 
251  return ss.str();
252 }
253 
254 std::vector<Scalar> Array::toVector() const
255 {
256  std::vector<Scalar> result;
257  result.reserve(getSize());
258 
259  auto end_ = end();
260  for(auto it = begin(); it != end_; ++it)
261  result.push_back(*it);
262 
263  return result;
264 }
265 
266 std::list<Scalar> Array::toList() const
267 {
268  std::list<Scalar> result;
269 
270  auto end_ = end();
271  for(auto it = begin(); it != end_; ++it)
272  result.push_back(*it);
273 
274  return result;
275 }
276 
278 {
279  return Iterator(getPerl(), _getFirstSV());
280 }
281 
283 {
284  return Iterator(getPerl(), _getLastSV() + 1);
285 }
286 
287 
288 
289 AV * Array::getAV() const
290 {
291  return reinterpret_cast<AV *>(getSV());
292 }
293 
294 
295 
296 Scalar Array::operator[] (const SSize_t index) const noexcept
297 {
298  return Scalar(_perl, *av_fetch(getAV(), index, /* lval */FALSE), true);
299 }
300 
301 Array & Array::operator+= (const Scalar & scalar)
302 {
303  push(scalar);
304  return *this;
305 }
306 
307 Array & Array::operator<< (const Scalar & scalar)
308 {
309  push(scalar);
310  return *this;
311 }
312 
313 Array Array::operator+ (const Array & other) const
314 {
315  Array result = _perl.newArray();
316  result.reserve(getSize() + other.getSize());
317 
318  result.push(*this);
319  result.push(other);
320 
321  return result;
322 }
323 
324 Array::operator bool() const
325 {
326  return !isEmpty();
327 }
328 
329 
330 
331 void Array::_store(const int index, Scalar scalar)
332 {
333  av_store(getAV(), index, scalar.detachSV());
334 }
335 
336 void Array::_push(Scalar scalar)
337 {
338  av_push(getAV(), scalar.detachSV());
339 }
340 
341 bool Array::_checkIndex(const SSize_t index) const
342 {
343  const SSize_t array_size = getSize();
344  return array_size > 0 && index >= 0 && index < array_size && isExists(index);
345 }
346 
347 SV ** Array::_getFirstSV() const
348 {
349  SV ** result = av_fetch(getAV(), 0, /* lval */FALSE);
350  YOBAPERL_ASSERT(result);
351  YOBAPERL_ASSERT(*result);
352  return result;
353 }
354 
355 SV ** Array::_getLastSV() const
356 {
357  SV ** result = _getFirstSV() + (getSize() - 1);
358  YOBAPERL_ASSERT(result);
359  YOBAPERL_ASSERT(*result);
360  return result;
361 }
362 
363 
364 
365 } // namespace yoba
Array reference
Definition: array.hpp:26
std::list< Scalar > toList() const
Convert to C++ list.
Definition: array.cpp:266
Scalar get(const SSize_t index) const
Fetch element by index.
Definition: array.cpp:16
std::string toString() const
Convert to C++ string.
Definition: array.cpp:235
std::vector< Scalar > toVector() const
Convert to C++ vector.
Definition: array.cpp:254
Iterator begin() const
Iterator to first element.
Definition: array.cpp:277
SV * getSV() const
Get raw scalar.
Definition: variable.cpp:63
Array & reserve(SSize_t size)
Reserve space.
Definition: array.cpp:213
SV * detachSV()
Nullify object and return SV.
Definition: variable.cpp:70
Array & push(const Scalar &scalar)
Add element to end.
Definition: array.cpp:75
Definition: array.cpp:5
Perl & getPerl() const
Get Perl instance.
Definition: variable.cpp:46
Array & operator+=(const Scalar &scalar)
Alias to push()
Definition: array.cpp:301
bool isEmpty() const
Check if array is empty.
Definition: array.cpp:203
Array operator+(const Array &other) const
Merge two arrays.
Definition: array.cpp:313
Array(Perl &perl, AV *av, bool increase_refcount)
Constructor.
Definition: array.cpp:9
Scalar operator[](const SSize_t index) const noexcept
Unsafe version of get()
Definition: array.cpp:296
bool isExists(SSize_t index) const
Check element exists.
Definition: array.cpp:208
Scalar makeCopy() const
Create copy.
Definition: scalar.cpp:187
Array & operator<<(const Scalar &scalar)
Alias to push()
Definition: array.cpp:307
Array makeCopy() const
Copy each element to new array.
Definition: array.cpp:219
Array newArray()
Create anonymous empty array.
Definition: perl.cpp:61
Base class for perl variables.
Definition: variable.hpp:20
Scalar newScalar()
Create anonymous empty scalar.
Definition: perl.cpp:51
Main class.
Definition: perl.hpp:32
Scalar pop()
Remove last element and return.
Definition: array.cpp:146
Scalar shift()
Remove first element and return.
Definition: array.cpp:168
int getSize() const
Elements count.
Definition: array.cpp:198
int getId() const
Perl object id.
Definition: perl.cpp:298
Scalar getLast() const
Get last element.
Definition: array.cpp:47
class ArrayIterator Iterator
Type alias.
Definition: array.hpp:29
Perl exception.
AV * getAV() const
Get raw array.
Definition: array.cpp:289
Array & replace(const SSize_t index, const Scalar &scalar)
Replace element.
Definition: array.cpp:52
Array & unshift(const Scalar &scalar)
Add element to begin.
Definition: array.cpp:105
Scalar makeRef() const
Take reference.
Definition: array.cpp:230
Scalar reference
Definition: scalar.hpp:24
Array & clear()
Remove all elements.
Definition: array.cpp:190
Scalar getFirst() const
Get first element.
Definition: array.cpp:42
Iterator end() const
Iterator to last element + 1.
Definition: array.cpp:282