2015-09-09 00:13:04 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include "sysdeps.h"
|
|
|
|
|
2015-12-05 06:00:26 +00:00
|
|
|
#include <android-base/test_utils.h>
|
2015-10-28 21:29:51 +00:00
|
|
|
|
2015-09-09 00:13:04 +00:00
|
|
|
TEST(sysdeps_win32, adb_getenv) {
|
|
|
|
// Insert all test env vars before first call to adb_getenv() which will
|
|
|
|
// read the env var block only once.
|
|
|
|
ASSERT_EQ(0, _putenv("SYSDEPS_WIN32_TEST_UPPERCASE=1"));
|
|
|
|
ASSERT_EQ(0, _putenv("sysdeps_win32_test_lowercase=2"));
|
|
|
|
ASSERT_EQ(0, _putenv("Sysdeps_Win32_Test_MixedCase=3"));
|
|
|
|
|
|
|
|
// UTF-16 value
|
|
|
|
ASSERT_EQ(0, _wputenv(L"SYSDEPS_WIN32_TEST_UNICODE=\u00a1\u0048\u006f\u006c"
|
|
|
|
L"\u0061\u0021\u03b1\u03b2\u03b3\u0061\u006d\u0062"
|
|
|
|
L"\u0075\u006c\u014d\u043f\u0440\u0438\u0432\u0435"
|
|
|
|
L"\u0442"));
|
|
|
|
|
|
|
|
// Search for non-existant env vars.
|
|
|
|
EXPECT_STREQ(nullptr, adb_getenv("SYSDEPS_WIN32_TEST_NONEXISTANT"));
|
|
|
|
|
|
|
|
// Search for existing env vars.
|
|
|
|
|
|
|
|
// There is no test for an env var with a value of a zero-length string
|
|
|
|
// because _putenv() does not support inserting such an env var.
|
|
|
|
|
|
|
|
// Search for env var that is uppercase.
|
|
|
|
EXPECT_STREQ("1", adb_getenv("SYSDEPS_WIN32_TEST_UPPERCASE"));
|
|
|
|
EXPECT_STREQ("1", adb_getenv("sysdeps_win32_test_uppercase"));
|
|
|
|
EXPECT_STREQ("1", adb_getenv("Sysdeps_Win32_Test_Uppercase"));
|
|
|
|
|
|
|
|
// Search for env var that is lowercase.
|
|
|
|
EXPECT_STREQ("2", adb_getenv("SYSDEPS_WIN32_TEST_LOWERCASE"));
|
|
|
|
EXPECT_STREQ("2", adb_getenv("sysdeps_win32_test_lowercase"));
|
|
|
|
EXPECT_STREQ("2", adb_getenv("Sysdeps_Win32_Test_Lowercase"));
|
|
|
|
|
|
|
|
// Search for env var that is mixed-case.
|
|
|
|
EXPECT_STREQ("3", adb_getenv("SYSDEPS_WIN32_TEST_MIXEDCASE"));
|
|
|
|
EXPECT_STREQ("3", adb_getenv("sysdeps_win32_test_mixedcase"));
|
|
|
|
EXPECT_STREQ("3", adb_getenv("Sysdeps_Win32_Test_MixedCase"));
|
|
|
|
|
|
|
|
// Check that UTF-16 was converted to UTF-8.
|
|
|
|
EXPECT_STREQ("\xc2\xa1\x48\x6f\x6c\x61\x21\xce\xb1\xce\xb2\xce\xb3\x61\x6d"
|
|
|
|
"\x62\x75\x6c\xc5\x8d\xd0\xbf\xd1\x80\xd0\xb8\xd0\xb2\xd0\xb5"
|
|
|
|
"\xd1\x82",
|
|
|
|
adb_getenv("SYSDEPS_WIN32_TEST_UNICODE"));
|
|
|
|
|
|
|
|
// Check an env var that should always be set.
|
|
|
|
const char* path_val = adb_getenv("PATH");
|
|
|
|
EXPECT_NE(nullptr, path_val);
|
|
|
|
if (path_val != nullptr) {
|
2016-01-15 20:14:24 +00:00
|
|
|
EXPECT_GT(strlen(path_val), 0U);
|
2015-09-09 00:13:04 +00:00
|
|
|
}
|
|
|
|
}
|
2015-10-18 23:45:09 +00:00
|
|
|
|
2015-10-28 21:29:51 +00:00
|
|
|
TEST(sysdeps_win32, unix_isatty) {
|
|
|
|
// stdin and stdout should be consoles. Use CONIN$ and CONOUT$ special files
|
|
|
|
// so that we can test this even if stdin/stdout have been redirected. Read
|
|
|
|
// permissions are required for unix_isatty().
|
|
|
|
int conin_fd = unix_open("CONIN$", O_RDONLY);
|
|
|
|
int conout_fd = unix_open("CONOUT$", O_RDWR);
|
|
|
|
for (const int fd : {conin_fd, conout_fd}) {
|
|
|
|
EXPECT_TRUE(fd >= 0);
|
|
|
|
EXPECT_EQ(1, unix_isatty(fd));
|
|
|
|
EXPECT_EQ(0, unix_close(fd));
|
|
|
|
}
|
|
|
|
|
|
|
|
// nul returns 1 from isatty(), make sure unix_isatty() corrects that.
|
|
|
|
for (auto flags : {O_RDONLY, O_RDWR}) {
|
|
|
|
int nul_fd = unix_open("nul", flags);
|
|
|
|
EXPECT_TRUE(nul_fd >= 0);
|
|
|
|
EXPECT_EQ(0, unix_isatty(nul_fd));
|
|
|
|
EXPECT_EQ(0, unix_close(nul_fd));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check a real file, both read-write and read-only.
|
|
|
|
TemporaryFile temp_file;
|
|
|
|
EXPECT_TRUE(temp_file.fd >= 0);
|
|
|
|
EXPECT_EQ(0, unix_isatty(temp_file.fd));
|
|
|
|
|
|
|
|
int temp_file_ro_fd = unix_open(temp_file.path, O_RDONLY);
|
|
|
|
EXPECT_TRUE(temp_file_ro_fd >= 0);
|
|
|
|
EXPECT_EQ(0, unix_isatty(temp_file_ro_fd));
|
|
|
|
EXPECT_EQ(0, unix_close(temp_file_ro_fd));
|
|
|
|
|
|
|
|
// Check a real OS pipe.
|
|
|
|
int pipe_fds[2];
|
|
|
|
EXPECT_EQ(0, _pipe(pipe_fds, 64, _O_BINARY));
|
|
|
|
EXPECT_EQ(0, unix_isatty(pipe_fds[0]));
|
|
|
|
EXPECT_EQ(0, unix_isatty(pipe_fds[1]));
|
|
|
|
EXPECT_EQ(0, _close(pipe_fds[0]));
|
|
|
|
EXPECT_EQ(0, _close(pipe_fds[1]));
|
|
|
|
|
|
|
|
// Make sure an invalid FD is handled correctly.
|
|
|
|
EXPECT_EQ(0, unix_isatty(-1));
|
|
|
|
}
|
adb: win32: handle incomplete UTF-8 in console output, other fixes
Previously, the various adb_printf, adb_fwrite, etc. functions did not
correctly handle the case of the passed buffer ending with an incomplete
UTF-8 sequence. This is fixed by buffering up incomplete UTF-8 sequences
in g_console_output_buffer (protected by the mutex
g_console_output_buffer) and outputting it later once the full sequence
is available.
A unittest for the main worker function, ParseCompleteUTF8(), was added
to adb_test.
Other fixes:
- Fix the return value of number-of-chars written to be number of UTF-8
bytes instead of number of UTF-16 characters.
- Don't overwrite errno in success cases of various adb_printf, etc.
functions. This might be excessive, but might be useful in the case
when these functions are used for debugging/tracing.
- Add missing UTF-8 stdio functions that aren't currently used by adb,
but might be in the future: vprintf, putc, putchar, puts.
- stdin_raw_init: If we can't get the console handle, don't call
SetConsoleMode(). Not a big deal, but this will prevent erroneous
trace output.
Change-Id: I8730e8af92882c42b884ad921b39a17b54465085
Signed-off-by: Spencer Low <CompareAndSwap@gmail.com>
2015-11-16 00:29:36 +00:00
|
|
|
|
|
|
|
void TestParseCompleteUTF8(const char* buf, const size_t buf_size,
|
|
|
|
const size_t expected_complete_bytes,
|
|
|
|
const std::vector<char>& expected_remaining_bytes) {
|
|
|
|
std::vector<char> remaining_bytes;
|
|
|
|
const size_t complete_bytes = internal::ParseCompleteUTF8(buf, buf + buf_size,
|
|
|
|
&remaining_bytes);
|
|
|
|
EXPECT_EQ(expected_complete_bytes, complete_bytes);
|
|
|
|
EXPECT_EQ(expected_remaining_bytes, remaining_bytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(sysdeps_win32, ParseCompleteUTF8) {
|
|
|
|
const std::vector<std::vector<char>> multi_byte_sequences = {
|
|
|
|
{ '\xc2', '\xa9' }, // 2 byte UTF-8 sequence
|
|
|
|
{ '\xe1', '\xb4', '\xa8' }, // 3 byte UTF-8 sequence
|
|
|
|
{ '\xf0', '\x9f', '\x98', '\x80' }, // 4 byte UTF-8 sequence
|
|
|
|
};
|
|
|
|
std::vector<std::vector<char>> all_sequences = {
|
|
|
|
{}, // 0 bytes
|
|
|
|
{ '\0' }, // NULL byte
|
|
|
|
{ 'a' }, // 1 byte UTF-8 sequence
|
|
|
|
};
|
|
|
|
all_sequences.insert(all_sequences.end(), multi_byte_sequences.begin(),
|
|
|
|
multi_byte_sequences.end());
|
|
|
|
|
|
|
|
// Vary a prefix of bytes in front of the sequence that we're actually interested in parsing.
|
|
|
|
for (const auto& prefix : all_sequences) {
|
|
|
|
// Parse (prefix + one byte of the sequence at a time)
|
|
|
|
for (const auto& seq : multi_byte_sequences) {
|
|
|
|
std::vector<char> buffer(prefix);
|
|
|
|
|
|
|
|
// For every byte of the sequence except the last
|
|
|
|
for (size_t i = 0; i < seq.size() - 1; ++i) {
|
|
|
|
buffer.push_back(seq[i]);
|
|
|
|
|
|
|
|
// When parsing an incomplete UTF-8 sequence, the amount of the buffer preceding
|
|
|
|
// the start of the incomplete UTF-8 sequence is valid. The remaining bytes are the
|
|
|
|
// bytes of the incomplete UTF-8 sequence.
|
|
|
|
TestParseCompleteUTF8(buffer.data(), buffer.size(), prefix.size(),
|
|
|
|
std::vector<char>(seq.begin(), seq.begin() + i + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
// For the last byte of the sequence
|
|
|
|
buffer.push_back(seq.back());
|
|
|
|
TestParseCompleteUTF8(buffer.data(), buffer.size(), buffer.size(), std::vector<char>());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse (prefix (aka sequence) + invalid trailing bytes) to verify that the invalid
|
|
|
|
// trailing bytes are immediately "returned" to prevent them from being stuck in some
|
|
|
|
// buffer.
|
|
|
|
std::vector<char> buffer(prefix);
|
|
|
|
for (size_t i = 0; i < 8; ++i) {
|
|
|
|
buffer.push_back(0x80); // trailing byte
|
|
|
|
TestParseCompleteUTF8(buffer.data(), buffer.size(), buffer.size(), std::vector<char>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|