jdk-24/test/hotspot/gtest/utilities/test_ostream.cpp
2024-06-07 07:34:58 +00:00

428 lines
14 KiB
C++

/*
* Copyright (c) 2019, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2019 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/
#include "precompiled.hpp"
#include "memory/resourceArea.hpp"
#include "runtime/os.hpp"
#include "utilities/globalDefinitions.hpp"
#include "utilities/defaultStream.hpp"
#include "utilities/ostream.hpp"
#include "unittest.hpp"
static size_t print_lorem(outputStream* st) {
// Create a ResourceMark just to make sure the stream does not use ResourceArea
ResourceMark rm;
static const char* const lorem = "Lorem ipsum dolor sit amet, consectetur adipiscing elit, "
"sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Lacinia at quis "
"risus sed vulputate odio ut enim blandit. Amet risus nullam eget felis eget. Viverra "
"orci sagittis eu volutpat odio facilisis mauris sit. Erat velit scelerisque in dictum non.";
static const size_t len_lorem = strlen(lorem);
// Randomly alternate between short and long writes at a ratio of 9:1.
const bool short_write = (os::random() % 10) > 0;
const size_t len = os::random() % (short_write ? 10 : len_lorem);
st->write(lorem, len);
return len;
}
static void test_stringStream_is_zero_terminated(const stringStream* ss) {
ASSERT_EQ(ss->base()[ss->size()], '\0');
}
static void do_test_stringStream(stringStream* ss, size_t expected_cap) {
test_stringStream_is_zero_terminated(ss);
size_t written = 0;
for (int i = 0; i < 1000; i ++) {
written += print_lorem(ss);
if (expected_cap > 0 && written >= expected_cap) {
ASSERT_EQ(ss->size(), expected_cap - 1);
} else {
ASSERT_EQ(ss->size(), written);
}
// Internal buffer should always be zero-terminated.
test_stringStream_is_zero_terminated(ss);
}
// Reset should zero terminate too
ss->reset();
ASSERT_EQ(ss->size(), (size_t)0);
test_stringStream_is_zero_terminated(ss);
}
TEST_VM(ostream, stringStream_dynamic_start_with_internal_buffer) {
stringStream ss;
do_test_stringStream(&ss, 0);
ss.reset();
do_test_stringStream(&ss, 0);
}
TEST_VM(ostream, stringStream_dynamic_start_with_malloced_buffer) {
stringStream ss(128);
do_test_stringStream(&ss, 0);
ss.reset();
do_test_stringStream(&ss, 0);
}
TEST_VM(ostream, stringStream_static) {
char buffer[128 + 1];
char* canary_at = buffer + sizeof(buffer) - 1;
*canary_at = 'X';
size_t stream_buf_size = sizeof(buffer) - 1;
stringStream ss(buffer, stream_buf_size);
do_test_stringStream(&ss, stream_buf_size);
ASSERT_EQ(*canary_at, 'X'); // canary
}
TEST_VM(ostream, bufferedStream_dynamic_small) {
bufferedStream bs(1); // small to excercise realloc.
size_t written = 0;
// The max cap imposed is 100M, we should be safely below this in this test.
for (int i = 0; i < 10; i ++) {
written += print_lorem(&bs);
ASSERT_EQ(bs.size(), written);
}
}
static void test_autoindent(bool on) {
stringStream ss;
ss.set_autoindent(on);
{
streamIndentor si(&ss, 5);
ss.print("ABC");
ss.print("DEF");
ss.cr();
ss.print_cr("0123");
{
streamIndentor si(&ss, 5);
ss.print_cr("4567");
ss.print_raw("89AB");
ss.print_raw("CDEXXXX", 3);
ss.print_raw_cr("XYZ");
}
ss.print("%u", 100);
ss.print_raw("KB");
ss.cr();
}
ss.print("end");
if (on) {
EXPECT_STREQ(ss.base(),
" ABCDEF\n"
" 0123\n"
" 4567\n"
" 89ABCDEXYZ\n"
" 100KB\n"
"end"
);
} else {
// no autoindent: calls should work as always without indentation
EXPECT_STREQ(ss.base(),
"ABCDEF\n"
"0123\n"
"4567\n"
"89ABCDEXYZ\n"
"100KB\n"
"end"
);
}
}
TEST_VM(ostream, autoindent_on) { test_autoindent(true); }
TEST_VM(ostream, autoindent_off) { test_autoindent(false); }
/* Activate to manually test bufferedStream dynamic cap.
TEST_VM(ostream, bufferedStream_dynamic_large) {
bufferedStream bs(1); // small to excercise realloc.
size_t written = 0;
// The max cap imposed is 100M. Writing this much should safely hit it.
// Note that this will assert in debug builds which is the expected behavior.
size_t expected_cap_at = 100 * M;
for (int i = 0; i < 10000000; i ++) {
written += print_lorem(&bs);
if (written < expected_cap_at) {
ASSERT_EQ(bs.size(), written);
} else {
ASSERT_EQ(bs.size(), expected_cap_at - 1);
}
}
}
*/
// Test helper for do_vsnprintf
class outputStream::TestSupport : AllStatic {
// Shared constants and variables for all subtests.
static const size_t buflen = 11;
static char buffer[buflen];
static const size_t max_len = buflen - 1;
static size_t result_len;
static const char* result;
static void reset() {
result_len = 0;
result = nullptr;
buffer[0] = '\0';
}
static const char* test(char* buf, size_t len, bool add_cr,
size_t& rlen, const char* format, ...) {
va_list ap;
va_start(ap, format);
const char* res = do_vsnprintf(buf, len, format, ap, add_cr, rlen);
va_end(ap);
return res;
}
public:
// Case set 1: constant string with no format specifiers
static void test_constant_string() {
reset();
// Case 1-1: no cr, no truncation, excess capacity
{
const char* str = "012345678";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, false, result_len, str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 1-2: no cr, no truncation, exact capacity
{
const char* str = "0123456789";
size_t initial_len = strlen(str);
ASSERT_EQ(initial_len, max_len);
result = test(buffer, buflen, false, result_len, str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 1-3: no cr, no truncation, exceeds capacity
{
const char* str = "0123456789A";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len > max_len);
result = test(buffer, buflen, false, result_len, str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len);
}
reset();
// Case 1-4: add cr, no truncation, excess capacity
{
const char* str = "01234567";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 1-5: add cr, no truncation, exact capacity
{
const char* str = "012345678";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 1-6: add cr, truncation
{
const char* str = "0123456789";
size_t initial_len = strlen(str);
ASSERT_EQ(initial_len, max_len);
::printf("Truncation warning expected: requires %d\n", (int)(initial_len + 1 + 1));
result = test(buffer, buflen, true, result_len, str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len);
ASSERT_TRUE(result_len <= max_len);
}
}
// Case set 2: "%s" string
static void test_percent_s_string() {
reset();
// Case 2-1: no cr, no truncation, excess capacity
{
const char* str = "012345678";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, false, result_len, "%s", str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 2-2: no cr, no truncation, exact capacity
{
const char* str = "0123456789";
size_t initial_len = strlen(str);
ASSERT_EQ(initial_len, max_len);
result = test(buffer, buflen, false, result_len, "%s", str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 2-3: no cr, no truncation, exceeds capacity
{
const char* str = "0123456789A";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len > max_len);
result = test(buffer, buflen, false, result_len, "%s", str);
ASSERT_EQ(result, str);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len);
}
reset();
// Case 2-4: add cr, no truncation, excess capacity
{
const char* str = "01234567";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, "%s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 2-5: add cr, no truncation, exact capacity
{
const char* str = "012345678";
size_t initial_len = strlen(str);
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, "%s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 2-6: add cr, truncation
{
const char* str = "0123456789";
size_t initial_len = strlen(str);
ASSERT_EQ(initial_len, max_len);
::printf("Truncation warning expected: requires %d\n", (int)(initial_len + 1 + 1));
result = test(buffer, buflen, true, result_len, "%s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len);
ASSERT_TRUE(result_len <= max_len);
}
}
// Case set 3: " %s" string - the space means we avoid the pass-through optimization and use vsnprintf
static void test_general_string() {
reset();
// Case 3-1: no cr, no truncation, excess capacity
{
const char* str = "01234567";
size_t initial_len = strlen(str) + 1;
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, false, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 3-2: no cr, no truncation, exact capacity
{
const char* str = "012345678";
size_t initial_len = strlen(str) + 1;
ASSERT_EQ(initial_len, max_len);
result = test(buffer, buflen, false, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 3-3: no cr, truncation
{
const char* str = "0123456789";
size_t initial_len = strlen(str) + 1;
ASSERT_TRUE(initial_len > max_len);
::printf("Truncation warning expected: requires %d\n", (int)(initial_len + 1));
result = test(buffer, buflen, false, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
}
reset();
// Case 3-4: add cr, no truncation, excess capacity
{
const char* str = "0123456";
size_t initial_len = strlen(str) + 1;
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 3-5: add cr, no truncation, exact capacity
{
const char* str = "01234567";
size_t initial_len = strlen(str) + 1;
ASSERT_TRUE(initial_len < max_len);
result = test(buffer, buflen, true, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len + 1);
ASSERT_TRUE(result_len <= max_len);
}
reset();
// Case 3-6: add cr, truncation
{
const char* str = "012345678";
size_t initial_len = strlen(str) + 1;
ASSERT_EQ(initial_len, max_len);
::printf("Truncation warning expected: requires %d\n", (int)(initial_len + 1 + 1));
result = test(buffer, buflen, true, result_len, " %s", str);
ASSERT_EQ(result, buffer);
ASSERT_EQ(strlen(result), result_len);
ASSERT_EQ(result_len, initial_len);
ASSERT_TRUE(result_len <= max_len);
}
}
};
const size_t outputStream::TestSupport::max_len;
char outputStream::TestSupport::buffer[outputStream::TestSupport::buflen];
size_t outputStream::TestSupport::result_len = 0;
const char* outputStream::TestSupport::result = nullptr;
TEST_VM(ostream, do_vsnprintf_buffering) {
outputStream::TestSupport::test_constant_string();
outputStream::TestSupport::test_percent_s_string();
outputStream::TestSupport::test_general_string();
}