From 2786608743727506c8ab614905e4032e66db1427 Mon Sep 17 00:00:00 2001 From: Christopher Ferris Date: Thu, 2 Aug 2018 15:21:37 -0700 Subject: [PATCH] Fix incorrect left shifts. Bug: 112142060 Test: New unit tests pass. Change-Id: I0c47b22582f0bf75ab503364a337c7de4de3ec43 --- libunwindstack/Android.bp | 1 + libunwindstack/RegsInfo.h | 20 ++++--- libunwindstack/tests/RegsInfoTest.cpp | 85 +++++++++++++++++++++++++++ 3 files changed, 97 insertions(+), 9 deletions(-) create mode 100644 libunwindstack/tests/RegsInfoTest.cpp diff --git a/libunwindstack/Android.bp b/libunwindstack/Android.bp index df5da650c..be2145d03 100644 --- a/libunwindstack/Android.bp +++ b/libunwindstack/Android.bp @@ -190,6 +190,7 @@ cc_test { "tests/MemoryRangeTest.cpp", "tests/MemoryRemoteTest.cpp", "tests/MemoryTest.cpp", + "tests/RegsInfoTest.cpp", "tests/RegsIterateTest.cpp", "tests/RegsStepIfSignalHandlerTest.cpp", "tests/RegsTest.cpp", diff --git a/libunwindstack/RegsInfo.h b/libunwindstack/RegsInfo.h index 47825f534..e6dd33c1b 100644 --- a/libunwindstack/RegsInfo.h +++ b/libunwindstack/RegsInfo.h @@ -25,11 +25,13 @@ namespace unwindstack { template struct RegsInfo { + static constexpr size_t MAX_REGISTERS = 64; + RegsInfo(RegsImpl* regs) : regs(regs) {} RegsImpl* regs = nullptr; uint64_t saved_reg_map = 0; - AddressType saved_regs[64]; + AddressType saved_regs[MAX_REGISTERS]; inline AddressType Get(uint32_t reg) { if (IsSaved(reg)) { @@ -39,23 +41,23 @@ struct RegsInfo { } inline AddressType* Save(uint32_t reg) { - if (reg > sizeof(saved_regs) / sizeof(AddressType)) { - // This should never happen as since all currently supported - // architectures have the total number of registers < 64. + if (reg > MAX_REGISTERS) { + // This should never happen since all currently supported + // architectures have < 64 total registers. abort(); } - saved_reg_map |= 1 << reg; + saved_reg_map |= 1ULL << reg; saved_regs[reg] = (*regs)[reg]; return &(*regs)[reg]; } inline bool IsSaved(uint32_t reg) { - if (reg > sizeof(saved_regs) / sizeof(AddressType)) { - // This should never happen as since all currently supported - // architectures have the total number of registers < 64. + if (reg > MAX_REGISTERS) { + // This should never happen since all currently supported + // architectures have < 64 total registers. abort(); } - return saved_reg_map & (1 << reg); + return saved_reg_map & (1ULL << reg); } inline uint16_t Total() { return regs->total_regs(); } diff --git a/libunwindstack/tests/RegsInfoTest.cpp b/libunwindstack/tests/RegsInfoTest.cpp new file mode 100644 index 000000000..052b5bf05 --- /dev/null +++ b/libunwindstack/tests/RegsInfoTest.cpp @@ -0,0 +1,85 @@ +/* + * Copyright (C) 2018 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 + +#include + +#include + +#include "RegsFake.h" +#include "RegsInfo.h" + +namespace unwindstack { + +TEST(RegsInfoTest, single_uint32_t) { + RegsImplFake regs(10); + RegsInfo info(®s); + + regs[1] = 0x100; + ASSERT_FALSE(info.IsSaved(1)); + ASSERT_EQ(0x100U, info.Get(1)); + ASSERT_EQ(10, info.Total()); + + uint32_t* value = info.Save(1); + ASSERT_EQ(value, ®s[1]); + regs[1] = 0x200; + ASSERT_TRUE(info.IsSaved(1)); + ASSERT_EQ(0x100U, info.Get(1)); + ASSERT_EQ(0x200U, regs[1]); +} + +TEST(RegsInfoTest, single_uint64_t) { + RegsImplFake regs(20); + RegsInfo info(®s); + + regs[3] = 0x300; + ASSERT_FALSE(info.IsSaved(3)); + ASSERT_EQ(0x300U, info.Get(3)); + ASSERT_EQ(20, info.Total()); + + uint64_t* value = info.Save(3); + ASSERT_EQ(value, ®s[3]); + regs[3] = 0x400; + ASSERT_TRUE(info.IsSaved(3)); + ASSERT_EQ(0x300U, info.Get(3)); + ASSERT_EQ(0x400U, regs[3]); +} + +TEST(RegsInfoTest, all) { + RegsImplFake regs(64); + RegsInfo info(®s); + + for (uint32_t i = 0; i < 64; i++) { + regs[i] = i * 0x100; + ASSERT_EQ(i * 0x100, info.Get(i)) << "Reg " + std::to_string(i) + " failed."; + } + + for (uint32_t i = 0; i < 64; i++) { + ASSERT_FALSE(info.IsSaved(i)) << "Reg " + std::to_string(i) + " failed."; + uint64_t* reg = info.Save(i); + ASSERT_EQ(reg, ®s[i]) << "Reg " + std::to_string(i) + " failed."; + *reg = i * 0x1000 + 0x100; + ASSERT_EQ(i * 0x1000 + 0x100, regs[i]) << "Reg " + std::to_string(i) + " failed."; + } + + for (uint32_t i = 0; i < 64; i++) { + ASSERT_TRUE(info.IsSaved(i)) << "Reg " + std::to_string(i) + " failed."; + ASSERT_EQ(i * 0x100, info.Get(i)) << "Reg " + std::to_string(i) + " failed."; + } +} + +} // namespace unwindstack