mycpp

Coverage Report

Created: 2024-07-09 17:10

/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
1
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
1
  gHeap.Init(50000);
16
1
}
17
18
5
void print_stderr(BigStr* s) {
19
5
  fputs(s->data_, stderr);  // prints until first NUL
20
5
  fputc('\n', stderr);
21
5
}
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
5
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
5
  static_assert(sizeof(MutableStr) == sizeof(BigStr),
58
5
                "BigStr and MutableStr must have same size");
59
5
  return reinterpret_cast<MutableStr*>(NewStr(n));
60
5
}
61
62
5
Tuple2<BigStr*, BigStr*> split_once(BigStr* s, BigStr* delim) {
63
5
  DCHECK(len(delim) == 1);
64
65
0
  const char* start = s->data_;  // note: this pointer may move
66
5
  char c = delim->data_[0];
67
5
  int length = len(s);
68
69
5
  const char* p = static_cast<const char*>(memchr(start, c, length));
70
71
5
  if (p) {
72
3
    int len1 = p - start;
73
3
    int len2 = length - len1 - 1;  // -1 for delim
74
75
3
    BigStr* s1 = nullptr;
76
3
    BigStr* s2 = nullptr;
77
    // Allocate together to avoid 's' moving in between
78
3
    s1 = NewStr(len1);
79
3
    s2 = NewStr(len2);
80
81
3
    memcpy(s1->data_, s->data_, len1);
82
3
    memcpy(s2->data_, s->data_ + len1 + 1, len2);
83
84
3
    return Tuple2<BigStr*, BigStr*>(s1, s2);
85
3
  } else {
86
2
    return Tuple2<BigStr*, BigStr*>(s, nullptr);
87
2
  }
88
5
}
89
90
LineReader* gStdin;
91
92
2
LineReader* open(BigStr* path) {
93
  // TODO: Don't use C I/O; use POSIX I/O!
94
2
  FILE* f = fopen(path->data_, "r");
95
2
  if (f == nullptr) {
96
0
    throw Alloc<IOError>(errno);
97
0
  }
98
99
2
  return reinterpret_cast<LineReader*>(Alloc<CFile>(f));
100
2
}
101
102
318
BigStr* CFile::readline() {
103
318
  char* line = nullptr;
104
318
  size_t allocated_size = 0;  // unused
105
106
  // Reset errno because we turn the EOF error into empty string (like Python).
107
318
  errno = 0;
108
318
  ssize_t len = getline(&line, &allocated_size, f_);
109
  // log("getline = %d", len);
110
318
  if (len < 0) {
111
    // Reset EOF flag so the next readline() will get a line.
112
1
    clearerr(f_);
113
114
    // man page says the buffer should be freed even if getline fails
115
1
    free(line);
116
117
1
    if (errno != 0) {  // Unexpected error
118
      // log("getline() error: %s", strerror(errno));
119
0
      throw Alloc<IOError>(errno);
120
0
    }
121
1
    return kEmptyString;  // Indicate EOF with empty string, like Python
122
1
  }
123
124
  // Note: getline() NUL-terminates the buffer
125
317
  BigStr* result = ::StrFromC(line, len);
126
317
  free(line);
127
317
  return result;
128
318
}
129
130
2
bool CFile::isatty() {
131
2
  return ::isatty(fileno(f_));
132
2
}
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
8
BigStr* BufLineReader::readline() {
139
8
  BigStr* line = nullptr;
140
141
8
  int str_len = len(s_);
142
8
  if (pos_ == str_len) {
143
4
    return kEmptyString;
144
4
  }
145
146
4
  int orig_pos = pos_;
147
4
  const char* p = strchr(s_->data_ + pos_, '\n');
148
  // log("pos_ = %s", pos_);
149
4
  int line_len;
150
4
  if (p) {
151
2
    int new_pos = p - s_->data_;
152
2
    line_len = new_pos - pos_ + 1;  // past newline char
153
2
    pos_ = new_pos + 1;
154
2
  } else {             // leftover line
155
2
    if (pos_ == 0) {   // The string has no newlines at all -- just return it
156
1
      pos_ = str_len;  // advance to the end
157
1
      return s_;
158
1
    } else {
159
1
      line_len = str_len - pos_;
160
1
      pos_ = str_len;  // advance to the end
161
1
    }
162
2
  }
163
164
3
  line = NewStr(line_len);
165
3
  memcpy(line->data_, s_->data_ + orig_pos, line_len);
166
3
  DCHECK(line->data_[line_len] == '\0');
167
0
  return line;
168
4
}
169
170
Writer* gStdout;
171
Writer* gStderr;
172
173
//
174
// CFileWriter
175
//
176
177
2
void CFile::write(BigStr* s) {
178
  // Writes can be short!
179
2
  int n = len(s);
180
2
  int num_written = ::fwrite(s->data_, sizeof(char), n, f_);
181
  // Similar to CPython fileobject.c
182
2
  if (num_written != n) {
183
0
    throw Alloc<IOError>(errno);
184
0
  }
185
2
}
186
187
1
void CFile::flush() {
188
1
  if (::fflush(f_) != 0) {
189
0
    throw Alloc<IOError>(errno);
190
0
  }
191
1
}
192
193
1
void CFile::close() {
194
1
  if (::fclose(f_) != 0) {
195
0
    throw Alloc<IOError>(errno);
196
0
  }
197
1
}
198
199
//
200
// BufWriter
201
//
202
203
5
void BufWriter::EnsureMoreSpace(int n) {
204
5
  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
3
    str_ = NewMutableStr(n);
208
3
    return;
209
3
  }
210
211
2
  int current_cap = len(str_);
212
2
  DCHECK(current_cap >= len_);
213
214
0
  int new_cap = len_ + n;
215
216
2
  if (current_cap < new_cap) {
217
2
    auto* s = NewMutableStr(std::max(current_cap * 2, new_cap));
218
2
    memcpy(s->data_, str_->data_, len_);
219
2
    s->data_[len_] = '\0';
220
2
    str_ = s;
221
2
  }
222
2
}
223
224
0
uint8_t* BufWriter::LengthPointer() {
225
  // start + len
226
0
  return reinterpret_cast<uint8_t*>(str_->data_) + len_;
227
0
}
228
229
0
uint8_t* BufWriter::CapacityPointer() {
230
  // start + capacity
231
0
  return reinterpret_cast<uint8_t*>(str_->data_) + str_->len_;
232
0
}
233
234
0
void BufWriter::SetLengthFrom(uint8_t* length_ptr) {
235
0
  uint8_t* begin = reinterpret_cast<uint8_t*>(str_->data_);
236
0
  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
0
}
242
243
0
void BufWriter::Truncate(int length) {
244
0
  len_ = length;
245
0
}
246
247
4
void BufWriter::WriteRaw(char* s, int n) {
248
4
  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
4
  if (n == 0) {
252
0
    return;
253
0
  }
254
255
4
  EnsureMoreSpace(n);
256
257
  // Append the contents to the buffer
258
4
  memcpy(str_->data_ + len_, s, n);
259
4
  len_ += n;
260
4
  str_->data_[len_] = '\0';
261
4
}
262
263
0
void BufWriter::WriteConst(const char* c_string) {
264
  // meant for short strings like '"'
265
0
  WriteRaw(const_cast<char*>(c_string), strlen(c_string));
266
0
}
267
268
4
void BufWriter::write(BigStr* s) {
269
4
  WriteRaw(s->data_, len(s));
270
4
}
271
272
2
void BufWriter::write_spaces(int n) {
273
2
  if (n == 0) {
274
1
    return;
275
1
  }
276
277
1
  EnsureMoreSpace(n);
278
279
1
  char* dest = str_->data_ + len_;
280
4
  for (int i = 0; i < n; ++i) {
281
3
    dest[i] = ' ';
282
3
  }
283
1
  len_ += n;
284
1
  str_->data_[len_] = '\0';
285
1
}
286
287
4
BigStr* BufWriter::getvalue() {
288
4
  DCHECK(is_valid_);  // Check for two INVALID getvalue() in a row
289
0
  is_valid_ = false;
290
291
4
  if (str_ == nullptr) {  // if no write() methods are called, the result is ""
292
1
    return kEmptyString;
293
3
  } else {
294
3
    BigStr* s = str_;
295
3
    s->MaybeShrink(len_);
296
3
    str_ = nullptr;
297
3
    len_ = -1;
298
3
    return s;
299
3
  }
300
4
}
301
302
}  // namespace mylib