2

我们在 Windows XP 上使用 VC 6.0、Purify、PC-Lint 和 Quantify 开发了一个控制台应用程序。VC6 无法在 Windows 7 和 8 上运行。如果我们要升级到 Windows 8,我已经查看了我们的开发环境选项。我们的应用程序是标准 C++ 控制台应用程序。几乎我们所有的用户都在 Linux 上。有没有人有使用 VC++ Pro 2013 或 2012 进行跨平台 C++ 开发的经验?具体来说,它是否可以做内存边界检查、内存泄漏检查和代码性能分析(每个函数需要多少时间)?

4

1 回答 1

3

Visual c++ 2013 可以做 Purify 和 Quantify 做的事情吗?

好吧,它并不是“一个人可以做其他人所做的一切”。它更像是一个十字路口,并使用它们来获得最大的覆盖范围。

Purify 是一个运行时检查器,因此它的性能通常优于 Visual Studio 的内置内存检查工具。但是 Purify 不做静态分析,所以你需要使用 Visual Studio。它的一个大伙伴关系。


我查看了我们的开发环境选项......我们的应用程序是 [跨平台] 标准 C++ 控制台应用程序。

Bike Shedding 开始了……这是一个很好的机会,因为您正在编写可移植的代码。如此多的人编写在一个平台上运行的代码,他们失去了来自其他平台上其他工具的诊断。

下面的所有内容都是免费的(Visual Studio Enterprise 下的 Enterprise Code Analysis 除外),如果您可以干净利落地完成该过程,那么您将拥有相当可靠的代码。

视窗

使用 Visual Studio(任何版本)并打开警告。它们包括/WAll/W4。如果您有 Visual Studio Enterprise,请务必添加 Enterprise Code Analysis 或添加/analyze开关。

您可以在 Visual Studio 中获得基本的内存检查。我不确定最新的 Purity 是如何使用它的。(我不使用它,因为我编写跨平台代码并使用 Linux 进行繁重的内存检查)。

对于 Windows 平台,您还应该做其他事情。您可以在 OWASP 的C-Based Toolchain Hardening中找到有关开发工具链的讨论。

Linux

请务必支持 GCC、Clang 和 ICC。使用它们时,请务必发出警告,包括-Wall-Wextra-Wconversion。GCC 是中流砥柱,您的代码可能在它上面运行良好。ICC 是英特尔的编译器,它对删除未定义的行为毫不留情。如果您的代码在 ICC 下中断,可能是因为编译器/优化器删除了一些未定义的行为(请参阅下面的 Clang 的未定义清理程序,了解如何定位有问题的代码)。

Clang 3.3 的消毒剂确实令人眼前一亮(Clang 3.2 及以下版本没有它们)。请务必使用-fsanitize=addressand运行-fsanitize=undefined。sanitizers 添加运行时检查器并在执行期间查找违规行为。您进行的自我测试越多越好。Clang 的控制代码生成中提供了完整的消毒剂列表。

Clang 3.3 配方如下。它们包括如何获取 Clang、热构建 Clang 以及如何使用 santizer 执行测试。

用 GCC、Clang 和 ICC 编译完成后,在 Valgrind 下运行程序。Valgrind 是另一个动态内存检查器。

对于 Linux 平台,您还应该做其他事情。您可以在 OWASP 的C-Based Toolchain Hardening中找到有关开发工具链的讨论。


要下载并使用最新版本构建 Clang 3.3:

wget http://llvm.org/releases/3.3/llvm-3.3.src.tar.gz
wget http://llvm.org/releases/3.3/cfe-3.3.src.tar.gz
wget http://llvm.org/releases/3.3/compiler-rt-3.3.src.tar.gz
# wget http://llvm.org/releases/3.3/lldb-3.3.src.tar.gz
tar xvf llvm-3.3.src.tar.gz
cd llvm-3.3.src/tools
tar xvf ../../cfe-3.3.src.tar.gz
mv cfe-3.3.src clang
# tar xvf ../../lldb-3.3.src.tar.gz
# mv lldb-3.3.src/ lldb
cd ..
cd projects
tar xvf ../../compiler-rt-3.3.src.tar.gz
mv compiler-rt-3.3.src/ compiler-rt
cd ..
./configure --enable-optimized --prefix=/usr/local
make -j4

# Pause to wait for the password prompt
read -p "Press [Enter] key to install..."

# Begin install
sudo make install

# Install does not copy asan_symbolize.py
sudo cp projects/compiler-rt/lib/asan/scripts/asan_symbolize.py /usr/local/bin

# Install does not install scan-build and scan-view
# Perform the copy, and/or put them on-path
sudo mkdir /usr/local/bin/scan-build
sudo cp -r tools/clang/tools/scan-build /usr/local/bin
sudo mkdir /usr/local/bin/scan-view
sudo cp -r tools/clang/tools/scan-view /usr/local/bin

要使用 Clang:

export CC=/usr/local/bin/clang
export CXX=/usr/local/bin/clang++
export CFLAGS="-g3 -fsanitize=undefined"
export CXXFLAGS="-g3 -fsanitize=undefined -fno-sanitize=vptr"

./configure

make

make check | /usr/local/bin/asan_symbolize.py

如果您遇到任何内存问题,它将类似于以下内容(取自Squid 3.3.9 Self Test Failures on Mac OS X 10.8):

==76794==ERROR: AddressSanitizer: global-buffer-overflow on address
0x000105ad50d2 at pc 0x105a364ab bp 0x7fff5a23f720 sp 0x7fff5a23f718
READ of size 19 at 0x000105ad50d2 thread T0
    #0 0x105a364aa in MemBuf::append MemBuf.cc:248
    #1 0x105a4ef57 in testHttpReply::testSanityCheckFirstLine
testHttpReply.cc:197
    #2 0x1068d71d1 in CppUnit::TestCaseMethodFunctor::operator()() const (in
libcppunit-1.12.1.dylib) + 33
    #3 0x1068cd9a3 in CppUnit::DefaultProtector::protect(CppUnit::Functor
const&, CppUnit::ProtectorContext const&) (in libcppunit-1.12.1.dylib) + 35
    #4 0x1068d4d88 in CppUnit::ProtectorChain::ProtectFunctor::operator()()
const (in libcppunit-1.12.1.dylib) + 24
    #5 0x1068d45e8 in CppUnit::ProtectorChain::protect(CppUnit::Functor const&,
CppUnit::ProtectorContext const&) (in libcppunit-1.12.1.dylib) + 456
    #6 0x1068dcf78 in CppUnit::TestResult::protect(CppUnit::Functor const&,
CppUnit::Test*, std::string const&) (in libcppunit-1.12.1.dylib) + 56
    #7 0x1068d6d1d in CppUnit::TestCase::run(CppUnit::TestResult*) (in
libcppunit-1.12.1.dylib) + 285
    #8 0x1068d77b6 in
CppUnit::TestComposite::doRunChildTests(CppUnit::TestResult*) (in
libcppunit-1.12.1.dylib) + 54
    #9 0x1068d76be in CppUnit::TestComposite::run(CppUnit::TestResult*) (in
libcppunit-1.12.1.dylib) + 30
    #10 0x1068d77b6 in
CppUnit::TestComposite::doRunChildTests(CppUnit::TestResult*) (in
libcppunit-1.12.1.dylib) + 54
    #11 0x1068d76be in CppUnit::TestComposite::run(CppUnit::TestResult*) (in
libcppunit-1.12.1.dylib) + 30
    #12 0x1068dcde1 in CppUnit::TestResult::runTest(CppUnit::Test*) (in
libcppunit-1.12.1.dylib) + 33
    #13 0x1068de9a5 in CppUnit::TestRunner::run(CppUnit::TestResult&,
std::string const&) (in libcppunit-1.12.1.dylib) + 53
    #14 0x105a55a97 in main testMain.cc:31
    #15 0x7fff90af07e0 in start (in libdyld.dylib) + 0
    #16 0x0
0x000105ad50d2 is located 46 bytes to the left of global variable '.str28' from
'tests/testHttpReply.cc' (0x105ad5100) of size 16
  '.str28' is ascii string 'HTTP/1.1 -000

0x000105ad50d2 is located 0 bytes to the right of global variable '.str27' from
'tests/testHttpReply.cc' (0x105ad50c0) of size 18
  '.str27' is ascii string 'HTTP/1.10 Okay

这是未定义的行为和非法转变的样子(取自 Postgres 的Clang 3.3 调查结果和非法转变)

make check
...

vacuuming database template1 ... localtime.c:127:20: runtime error:
left shift of negative value -1
pg_lzcompress.c:601:5: runtime error: left shift of negative value -68
pg_lzcompress.c:601:5: runtime error: left shift of negative value -68
pg_lzcompress.c:385:16: runtime error: left shift of negative value -68
pg_lzcompress.c:615:4: runtime error: left shift of negative value -68

如果你在 Intel 的 ICC 下失败了,那么一定要在 Clang with 下运行它,-fsanitize=undefined因为 ICC 会默默地删除任何有问题的代码。这是我发现找到有问题的代码的最简单方法。

于 2014-01-18T15:51:53.753 回答