summaryrefslogtreecommitdiffstats
path: root/mozglue/interposers/getline_interposer.cpp
blob: 902b52b7187bb89051ac27cac92c1fcb15facba6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

/*
 * Interposing getline because of
 * https://bugzilla.mozilla.org/show_bug.cgi?id=914190
 */

#ifdef __ANDROID__

#  include <cstdlib>
#  include <cstdio>
#  include <cerrno>

#  include <limits>

namespace {

// RAII on file locking.
class FileLocker {
  FILE* stream;

 public:
  explicit FileLocker(FILE* stream) : stream(stream) { flockfile(stream); }
  ~FileLocker() { funlockfile(stream); }
};

ssize_t internal_getdelim(char** __restrict lineptr, size_t* __restrict n,
                          int delim, FILE* __restrict stream) {
  constexpr size_t n_default = 64;
  constexpr size_t n_max =
      std::numeric_limits<ssize_t>::max() < std::numeric_limits<size_t>::max()
          ? (size_t)std::numeric_limits<ssize_t>::max() + 1
          : std::numeric_limits<size_t>::max();
  constexpr size_t n_limit = 2 * ((n_max - 1) / 3);

  if (!lineptr || !n || !stream) {
    errno = EINVAL;
    return -1;
  }

  // Lock the file so that we can us unlocked getc in the inner loop.
  FileLocker fl(stream);

  if (!*lineptr || *n == 0) {
    *n = n_default;
    if (auto* new_lineptr = reinterpret_cast<char*>(realloc(*lineptr, *n))) {
      *lineptr = new_lineptr;
    } else {
      errno = ENOMEM;
      return -1;
    }
  }

  ssize_t result;
  size_t cur_len = 0;

  while (true) {
    // Retrieve an extra char.
    int i = getc_unlocked(stream);
    if (i == EOF) {
      result = -1;
      break;
    }

    // Eventually grow the buffer.
    if (cur_len + 1 >= *n) {
      size_t needed = *n >= n_limit ? n_max : 3 * *n / 2 + 1;

      if (cur_len + 1 >= needed) {
        errno = EOVERFLOW;
        return -1;
      }

      if (auto* new_lineptr = (char*)realloc(*lineptr, needed)) {
        *lineptr = new_lineptr;
      } else {
        errno = ENOMEM;
        return -1;
      }
      *n = needed;
    }

    (*lineptr)[cur_len] = i;
    cur_len++;

    if (i == delim) break;
  }
  (*lineptr)[cur_len] = '\0';
  return cur_len ? cur_len : result;
}

}  // namespace

extern "C" {

MFBT_API ssize_t getline(char** __restrict lineptr, size_t* __restrict n,
                         FILE* __restrict stream) {
  return internal_getdelim(lineptr, n, '\n', stream);
}

MFBT_API ssize_t getdelim(char** __restrict lineptr, size_t* __restrict n,
                          int delim, FILE* __restrict stream) {
  return internal_getdelim(lineptr, n, delim, stream);
}

}  // extern "C"

#endif