cpp

Coverage Report

Created: 2024-07-28 06:14

/home/uke/oil/mycpp/gc_mylib.cc
Line
Count
Source (jump to first uncovered line)
1
#include "mycpp/gc_mylib.h"
2
3
#include <errno.h>
4
#include <stdio.h>
5
#include <unistd.h>  // isatty
6
7
namespace mylib {
8
9
0
void InitCppOnly() {
10
  // We don't seem need this now that we have ctx_FlushStdout().
11
  // setvbuf(stdout, 0, _IONBF, 0);
12
13
  // Arbitrary threshold of 50K objects based on eyeballing
14
  // benchmarks/osh-runtime 10K or 100K aren't too bad either.
15
0
  gHeap.Init(50000);
16
0
}
17
18
0
void print_stderr(BigStr* s) {
19
0
  fputs(s->data_, stderr);  // prints until first NUL
20
0
  fputc('\n', stderr);
21
0
}
22
23
#if 0
24
void writeln(BigStr* s, int fd) {
25
  // TODO: handle errors and write in a loop, like posix::write().  If possible,
26
  // use posix::write directly, but that introduces some dependency problems.
27
28
  if (write(fd, s->data_, len(s)) < 0) {
29
    assert(0);
30
  }
31
  if (write(fd, "\n", 1) < 0) {
32
    assert(0);
33
  }
34
}
35
#endif
36
37
0
BigStr* JoinBytes(List<int>* byte_list) {
38
0
  int n = len(byte_list);
39
0
  BigStr* result = NewStr(n);
40
0
  for (int i = 0; i < n; ++i) {
41
0
    result->data_[i] = byte_list->at(i);
42
0
  }
43
0
  return result;
44
0
}
45
46
// For BashArray
47
0
void BigIntSort(List<mops::BigInt>* keys) {
48
0
  keys->sort();
49
0
}
50
51
class MutableStr : public BigStr {};
52
53
82
MutableStr* NewMutableStr(int n) {
54
  // In order for everything to work, MutableStr must be identical in layout to
55
  // BigStr. One easy way to achieve this is for MutableStr to have no members
56
  // and to inherit from BigStr.
57
82
  static_assert(sizeof(MutableStr) == sizeof(BigStr),
58
82
                "BigStr and MutableStr must have same size");
59
82
  return reinterpret_cast<MutableStr*>(NewStr(n));
60
82
}
61
62
0
Tuple2<BigStr*, BigStr*> split_once(BigStr* s, BigStr* delim) {
63
0
  DCHECK(len(delim) == 1);
64
65
0
  const char* start = s->data_;  // note: this pointer may move
66
0
  char c = delim->data_[0];
67
0
  int length = len(s);
68
69
0
  const char* p = static_cast<const char*>(memchr(start, c, length));
70
71
0
  if (p) {
72
0
    int len1 = p - start;
73
0
    int len2 = length - len1 - 1;  // -1 for delim
74
75
0
    BigStr* s1 = nullptr;
76
0
    BigStr* s2 = nullptr;
77
    // Allocate together to avoid 's' moving in between
78
0
    s1 = NewStr(len1);
79
0
    s2 = NewStr(len2);
80
81
0
    memcpy(s1->data_, s->data_, len1);
82
0
    memcpy(s2->data_, s->data_ + len1 + 1, len2);
83
84
0
    return Tuple2<BigStr*, BigStr*>(s1, s2);
85
0
  } else {
86
0
    return Tuple2<BigStr*, BigStr*>(s, nullptr);
87
0
  }
88
0
}
89
90
LineReader* gStdin;
91
92
0
LineReader* open(BigStr* path) {
93
  // TODO: Don't use C I/O; use POSIX I/O!
94
0
  FILE* f = fopen(path->data_, "r");
95
0
  if (f == nullptr) {
96
0
    throw Alloc<IOError>(errno);
97
0
  }
98
99
0
  return reinterpret_cast<LineReader*>(Alloc<CFile>(f));
100
0
}
101
102
0
BigStr* CFile::readline() {
103
0
  char* line = nullptr;
104
0
  size_t allocated_size = 0;  // unused
105
106
  // Reset errno because we turn the EOF error into empty string (like Python).
107
0
  errno = 0;
108
0
  ssize_t len = getline(&line, &allocated_size, f_);
109
  // log("getline = %d", len);
110
0
  if (len < 0) {
111
    // Reset EOF flag so the next readline() will get a line.
112
0
    clearerr(f_);
113
114
    // man page says the buffer should be freed even if getline fails
115
0
    free(line);
116
117
0
    if (errno != 0) {  // Unexpected error
118
      // log("getline() error: %s", strerror(errno));
119
0
      throw Alloc<IOError>(errno);
120
0
    }
121
0
    return kEmptyString;  // Indicate EOF with empty string, like Python
122
0
  }
123
124
  // Note: getline() NUL-terminates the buffer
125
0
  BigStr* result = ::StrFromC(line, len);
126
0
  free(line);
127
0
  return result;
128
0
}
129
130
0
bool CFile::isatty() {
131
0
  return ::isatty(fileno(f_));
132
0
}
133
134
// Problem: most BigStr methods like index() and slice() COPY so they have a
135
// NUL terminator.
136
// log("%s") falls back on sprintf, so it expects a NUL terminator.
137
// It would be easier for us to just share.
138
0
BigStr* BufLineReader::readline() {
139
0
  BigStr* line = nullptr;
140
141
0
  int str_len = len(s_);
142
0
  if (pos_ == str_len) {
143
0
    return kEmptyString;
144
0
  }
145
146
0
  int orig_pos = pos_;
147
0
  const char* p = strchr(s_->data_ + pos_, '\n');
148
  // log("pos_ = %s", pos_);
149
0
  int line_len;
150
0
  if (p) {
151
0
    int new_pos = p - s_->data_;
152
0
    line_len = new_pos - pos_ + 1;  // past newline char
153
0
    pos_ = new_pos + 1;
154
0
  } else {             // leftover line
155
0
    if (pos_ == 0) {   // The string has no newlines at all -- just return it
156
0
      pos_ = str_len;  // advance to the end
157
0
      return s_;
158
0
    } else {
159
0
      line_len = str_len - pos_;
160
0
      pos_ = str_len;  // advance to the end
161
0
    }
162
0
  }
163
164
0
  line = NewStr(line_len);
165
0
  memcpy(line->data_, s_->data_ + orig_pos, line_len);
166
0
  DCHECK(line->data_[line_len] == '\0');
167
0
  return line;
168
0
}
169
170
Writer* gStdout;
171
Writer* gStderr;
172
173
//
174
// CFileWriter
175
//
176
177
0
void CFile::write(BigStr* s) {
178
  // Writes can be short!
179
0
  int n = len(s);
180
0
  int num_written = ::fwrite(s->data_, sizeof(char), n, f_);
181
  // Similar to CPython fileobject.c
182
0
  if (num_written != n) {
183
0
    throw Alloc<IOError>(errno);
184
0
  }
185
0
}
186
187
0
void CFile::flush() {
188
0
  if (::fflush(f_) != 0) {
189
0
    throw Alloc<IOError>(errno);
190
0
  }
191
0
}
192
193
0
void CFile::close() {
194
0
  if (::fclose(f_) != 0) {
195
0
    throw Alloc<IOError>(errno);
196
0
  }
197
0
}
198
199
//
200
// BufWriter
201
//
202
203
184
void BufWriter::EnsureMoreSpace(int n) {
204
184
  if (str_ == nullptr) {
205
    // TODO: we could make the default capacity big enough for a line, e.g. 128
206
    // capacity: 128 -> 256 -> 512
207
44
    str_ = NewMutableStr(n);
208
44
    return;
209
44
  }
210
211
140
  int current_cap = len(str_);
212
140
  DCHECK(current_cap >= len_);
213
214
0
  int new_cap = len_ + n;
215
216
140
  if (current_cap < new_cap) {
217
38
    auto* s = NewMutableStr(std::max(current_cap * 2, new_cap));
218
38
    memcpy(s->data_, str_->data_, len_);
219
38
    s->data_[len_] = '\0';
220
38
    str_ = s;
221
38
  }
222
140
}
223
224
96
uint8_t* BufWriter::LengthPointer() {
225
  // start + len
226
96
  return reinterpret_cast<uint8_t*>(str_->data_) + len_;
227
96
}
228
229
96
uint8_t* BufWriter::CapacityPointer() {
230
  // start + capacity
231
96
  return reinterpret_cast<uint8_t*>(str_->data_) + str_->len_;
232
96
}
233
234
84
void BufWriter::SetLengthFrom(uint8_t* length_ptr) {
235
84
  uint8_t* begin = reinterpret_cast<uint8_t*>(str_->data_);
236
84
  DCHECK(length_ptr >= begin);  // we should have written some data
237
238
  // Set the length, e.g. so we know where to resume writing from
239
0
  len_ = length_ptr - begin;
240
  // printf("SET LEN to %d\n", len_);
241
84
}
242
243
12
void BufWriter::Truncate(int length) {
244
12
  len_ = length;
245
12
}
246
247
100
void BufWriter::WriteRaw(char* s, int n) {
248
100
  DCHECK(is_valid_);  // Can't write() after getvalue()
249
250
  // write('') is a no-op, so don't create Buf if we don't need to
251
100
  if (n == 0) {
252
0
    return;
253
0
  }
254
255
100
  EnsureMoreSpace(n);
256
257
  // Append the contents to the buffer
258
100
  memcpy(str_->data_ + len_, s, n);
259
100
  len_ += n;
260
100
  str_->data_[len_] = '\0';
261
100
}
262
263
100
void BufWriter::WriteConst(const char* c_string) {
264
  // meant for short strings like '"'
265
100
  WriteRaw(const_cast<char*>(c_string), strlen(c_string));
266
100
}
267
268
0
void BufWriter::write(BigStr* s) {
269
0
  WriteRaw(s->data_, len(s));
270
0
}
271
272
0
void BufWriter::write_spaces(int n) {
273
0
  DCHECK(n >= 0);
274
0
  if (n == 0) {
275
0
    return;
276
0
  }
277
278
0
  EnsureMoreSpace(n);
279
280
0
  char* dest = str_->data_ + len_;
281
0
  for (int i = 0; i < n; ++i) {
282
0
    dest[i] = ' ';
283
0
  }
284
0
  len_ += n;
285
0
  str_->data_[len_] = '\0';
286
0
}
287
288
44
BigStr* BufWriter::getvalue() {
289
44
  DCHECK(is_valid_);  // Check for two INVALID getvalue() in a row
290
0
  is_valid_ = false;
291
292
44
  if (str_ == nullptr) {  // if no write() methods are called, the result is ""
293
0
    return kEmptyString;
294
44
  } else {
295
44
    BigStr* s = str_;
296
44
    s->MaybeShrink(len_);
297
44
    str_ = nullptr;
298
44
    len_ = -1;
299
44
    return s;
300
44
  }
301
44
}
302
303
}  // namespace mylib