| // Copyright 2014 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "src/ostreams.h" |
| #include "src/objects.h" |
| #include "src/objects/string.h" |
| |
| #if V8_OS_WIN |
| #if _MSC_VER < 1900 |
| #define snprintf sprintf_s |
| #endif |
| #endif |
| |
| #if V8_OS_STARBOARD |
| #include "src/poems.h" |
| #endif |
| |
| namespace v8 { |
| namespace internal { |
| |
| #if !defined(COBALT) |
| OFStreamBase::OFStreamBase(FILE* f) : f_(f) {} |
| |
| |
| OFStreamBase::~OFStreamBase() {} |
| |
| |
| int OFStreamBase::sync() { |
| #if !V8_OS_STARBOARD |
| std::fflush(f_); |
| #endif |
| return 0; |
| } |
| |
| |
| OFStreamBase::int_type OFStreamBase::overflow(int_type c) { |
| #if V8_OS_STARBOARD |
| return c; |
| #else |
| return (c != EOF) ? std::fputc(c, f_) : c; |
| #endif |
| } |
| |
| |
| std::streamsize OFStreamBase::xsputn(const char* s, std::streamsize n) { |
| #if V8_OS_STARBOARD |
| return n; |
| #else |
| return static_cast<std::streamsize>( |
| std::fwrite(s, 1, static_cast<size_t>(n), f_)); |
| #endif |
| } |
| |
| |
| OFStream::OFStream(FILE* f) : std::ostream(nullptr), buf_(f) { |
| DCHECK_NOT_NULL(f); |
| rdbuf(&buf_); |
| } |
| #else |
| OFStream::OFStream(FILE* f) : std::ostream(nullptr) {} |
| #endif // !defined(COBALT) |
| |
| |
| OFStream::~OFStream() {} |
| |
| |
| namespace { |
| |
| // Locale-independent predicates. |
| bool IsPrint(uint16_t c) { return 0x20 <= c && c <= 0x7E; } |
| bool IsSpace(uint16_t c) { return (0x9 <= c && c <= 0xD) || c == 0x20; } |
| bool IsOK(uint16_t c) { return (IsPrint(c) || IsSpace(c)) && c != '\\'; } |
| |
| |
| std::ostream& PrintUC16(std::ostream& os, uint16_t c, bool (*pred)(uint16_t)) { |
| char buf[10]; |
| const char* format = pred(c) ? "%c" : (c <= 0xFF) ? "\\x%02x" : "\\u%04x"; |
| snprintf(buf, sizeof(buf), format, c); |
| return os << buf; |
| } |
| |
| std::ostream& PrintUC16ForJSON(std::ostream& os, uint16_t c, |
| bool (*pred)(uint16_t)) { |
| // JSON does not allow \x99; must use \u0099. |
| char buf[10]; |
| const char* format = pred(c) ? "%c" : "\\u%04x"; |
| snprintf(buf, sizeof(buf), format, c); |
| return os << buf; |
| } |
| |
| std::ostream& PrintUC32(std::ostream& os, int32_t c, bool (*pred)(uint16_t)) { |
| if (c <= String::kMaxUtf16CodeUnit) { |
| return PrintUC16(os, static_cast<uint16_t>(c), pred); |
| } |
| char buf[13]; |
| snprintf(buf, sizeof(buf), "\\u{%06x}", c); |
| return os << buf; |
| } |
| |
| } // namespace |
| |
| |
| std::ostream& operator<<(std::ostream& os, const AsReversiblyEscapedUC16& c) { |
| return PrintUC16(os, c.value, IsOK); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, const AsEscapedUC16ForJSON& c) { |
| if (c.value == '\n') return os << "\\n"; |
| if (c.value == '\r') return os << "\\r"; |
| if (c.value == '\t') return os << "\\t"; |
| if (c.value == '\"') return os << "\\\""; |
| return PrintUC16ForJSON(os, c.value, IsOK); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, const AsUC16& c) { |
| return PrintUC16(os, c.value, IsPrint); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, const AsUC32& c) { |
| return PrintUC32(os, c.value, IsPrint); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, const AsHex& hex) { |
| // Each byte uses up to two characters. Plus two characters for the prefix, |
| // plus null terminator. |
| DCHECK_GE(sizeof(hex.value) * 2, hex.min_width); |
| static constexpr size_t kMaxHexLength = 3 + sizeof(hex.value) * 2; |
| char buf[kMaxHexLength]; |
| snprintf(buf, kMaxHexLength, "%s%.*" PRIx64, hex.with_prefix ? "0x" : "", |
| hex.min_width, hex.value); |
| return os << buf; |
| } |
| |
| std::ostream& operator<<(std::ostream& os, const AsHexBytes& hex) { |
| uint8_t bytes = hex.min_bytes; |
| while (bytes < sizeof(hex.value) && (hex.value >> (bytes * 8) != 0)) ++bytes; |
| for (uint8_t b = 0; b < bytes; ++b) { |
| if (b) os << " "; |
| uint8_t printed_byte = |
| hex.byte_order == AsHexBytes::kLittleEndian ? b : bytes - b - 1; |
| os << AsHex((hex.value >> (8 * printed_byte)) & 0xFF, 2); |
| } |
| return os; |
| } |
| |
| } // namespace internal |
| } // namespace v8 |