1
0
Fork 0
mirror of https://we.phorge.it/source/arcanist.git synced 2024-11-26 08:42:40 +01:00

Modernize ArcanistCppcheckLinter.

Summary: Ref T2039. Make the `ArcanistCppcheckLinter` compatible with `.arclint`.

Test Plan: Wrote and executed unit tests.

Reviewers: epriestley, #blessed_reviewers

Reviewed By: epriestley, #blessed_reviewers

Subscribers: epriestley, Korvin

Maniphest Tasks: T2039

Differential Revision: https://secure.phabricator.com/D9068
This commit is contained in:
Joshua Spence 2014-05-12 04:30:36 -07:00 committed by epriestley
parent 54c377448d
commit 315314425e
7 changed files with 142 additions and 98 deletions

View file

@ -45,6 +45,7 @@ phutil_register_library_map(array(
'ArcanistConfigurationManager' => 'configuration/ArcanistConfigurationManager.php', 'ArcanistConfigurationManager' => 'configuration/ArcanistConfigurationManager.php',
'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php', 'ArcanistCoverWorkflow' => 'workflow/ArcanistCoverWorkflow.php',
'ArcanistCppcheckLinter' => 'lint/linter/ArcanistCppcheckLinter.php', 'ArcanistCppcheckLinter' => 'lint/linter/ArcanistCppcheckLinter.php',
'ArcanistCppcheckLinterTestCase' => 'lint/linter/__tests__/ArcanistCppcheckLinterTestCase.php',
'ArcanistCpplintLinter' => 'lint/linter/ArcanistCpplintLinter.php', 'ArcanistCpplintLinter' => 'lint/linter/ArcanistCpplintLinter.php',
'ArcanistCpplintLinterTestCase' => 'lint/linter/__tests__/ArcanistCpplintLinterTestCase.php', 'ArcanistCpplintLinterTestCase' => 'lint/linter/__tests__/ArcanistCpplintLinterTestCase.php',
'ArcanistDiffChange' => 'parser/diff/ArcanistDiffChange.php', 'ArcanistDiffChange' => 'parser/diff/ArcanistDiffChange.php',
@ -225,7 +226,8 @@ phutil_register_library_map(array(
'ArcanistConduitLinter' => 'ArcanistLinter', 'ArcanistConduitLinter' => 'ArcanistLinter',
'ArcanistConfigurationDrivenLintEngine' => 'ArcanistLintEngine', 'ArcanistConfigurationDrivenLintEngine' => 'ArcanistLintEngine',
'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow', 'ArcanistCoverWorkflow' => 'ArcanistBaseWorkflow',
'ArcanistCppcheckLinter' => 'ArcanistLinter', 'ArcanistCppcheckLinter' => 'ArcanistExternalLinter',
'ArcanistCppcheckLinterTestCase' => 'ArcanistArcanistLinterTestCase',
'ArcanistCpplintLinter' => 'ArcanistExternalLinter', 'ArcanistCpplintLinter' => 'ArcanistExternalLinter',
'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase', 'ArcanistCpplintLinterTestCase' => 'ArcanistArcanistLinterTestCase',
'ArcanistDiffParserTestCase' => 'ArcanistTestCase', 'ArcanistDiffParserTestCase' => 'ArcanistTestCase',

View file

@ -1,121 +1,110 @@
<?php <?php
/** /**
* Uses cppcheck to do basic checks in a cpp file * Uses Cppcheck to do basic checks in a C++ file.
*
* You can get it here:
* http://cppcheck.sourceforge.net/
*
* @group linter
*/ */
final class ArcanistCppcheckLinter extends ArcanistLinter { final class ArcanistCppcheckLinter extends ArcanistExternalLinter {
public function getInfoName() {
return 'Cppcheck';
}
public function getInfoURI() {
return 'http://cppcheck.sourceforge.net';
}
public function getInfoDescription() {
return pht('Use `cppcheck` to perform static analysis on C/C++ code.');
}
public function getLinterName() { public function getLinterName() {
return 'cppcheck'; return 'cppcheck';
} }
public function getLintOptions() { public function getLinterConfigurationName() {
$config = $this->getEngine()->getConfigurationManager(); return 'cppcheck';
// You will for sure want some options. The below default tends to be ok
return $config->getConfigFromAnySource(
'lint.cppcheck.options',
'-j2 --inconclusive --enable=performance,style,portability,information');
} }
public function getLintPath() { public function getDefaultBinary() {
$config = $this->getEngine()->getConfigurationManager(); $prefix = $this->getDeprecatedConfiguration('lint.cppcheck.prefix');
$prefix = $config->getConfigFromAnySource('lint.cppcheck.prefix'); $bin = $this->getDeprecatedConfiguration('lint.cppcheck.bin', 'cppcheck');
$bin = $config->getConfigFromAnySource('lint.cppcheck.bin', 'cppcheck');
if ($prefix !== null) { if ($prefix) {
if (!Filesystem::pathExists($prefix.'/'.$bin)) { return $prefix.'/'.$bin;
throw new ArcanistUsageException( } else {
"Unable to find cppcheck binary in a specified directory. Make ". return $bin;
"sure that 'lint.cppcheck.prefix' and 'lint.cppcheck.bin' keys are ". }
"set correctly. If you'd rather use a copy of cppcheck installed ".
"globally, you can just remove these keys from your .arcconfig.");
} }
return csprintf("%s/%s", $prefix, $bin); public function getVersion() {
list($stdout) = execx('%C --version', $this->getExecutableCommand());
$matches = array();
$regex = '/^Cppcheck (?P<version>\d+\.\d+)$/';
if (preg_match($regex, $stdout, $matches)) {
return $matches['version'];
} else {
return false;
}
} }
// Look for globally installed cppcheck public function getInstallInstructions() {
list($err) = exec_manual('which %s', $bin); return pht('Install Cpplint using `apt-get install cpplint` or similar.');
if ($err) { }
throw new ArcanistUsageException(
"cppcheck does not appear to be installed on this system. Install ". protected function getMandatoryFlags() {
"it (from http://cppcheck.sourceforge.net/) or configure ". return array(
"'lint.cppcheck.prefix' in your .arcconfig to point to the ". '--quiet',
"directory where it resides." '--inline-suppr',
'--xml',
'--xml-version=2',
); );
} }
return $bin; protected function getDefaultFlags() {
} return $this->getDeprecatedConfiguration(
'lint.cppcheck.options',
public function lintPath($path) { array('-j2', '--enable=performance,style,portability,information'));
$bin = $this->getLintPath();
$options = $this->getLintOptions();
list($rc, $stdout, $stderr) = exec_manual(
"%C %C --inline-suppr --xml-version=2 -q %s",
$bin,
$options,
$this->getEngine()->getFilePathOnDisk($path));
if ($rc === 1) {
throw new Exception("cppcheck failed to run correctly:\n".$stderr);
} }
protected function parseLinterOutput($path, $err, $stdout, $stderr) {
$dom = new DOMDocument(); $dom = new DOMDocument();
libxml_clear_errors(); $ok = @$dom->loadXML($stderr);
if ($dom->loadXML($stderr) === false || libxml_get_errors()) {
throw new ArcanistUsageException('cppcheck Linter failed to load ' .
'output. Something happened when running cppcheck. ' .
"Output:\n$stderr" .
"\nTry running lint with --trace flag to get more details.");
}
if (!$ok) {
return false;
}
$errors = $dom->getElementsByTagName('error'); $errors = $dom->getElementsByTagName('error');
$messages = array();
foreach ($errors as $error) { foreach ($errors as $error) {
$loc_node = $error->getElementsByTagName('location'); foreach ($error->getElementsByTagName('location') as $location) {
if (!$loc_node) {
continue;
}
$location = $loc_node->item(0);
if (!$location) {
continue;
}
$file = $location->getAttribute('file');
if ($file != Filesystem::resolvePath($path)) {
continue;
}
$line = $location->getAttribute('line');
$id = $error->getAttribute('id');
$severity = $error->getAttribute('severity');
$msg = $error->getAttribute('msg');
$inconclusive = $error->getAttribute('inconclusive');
$verbose_msg = $error->getAttribute('verbose');
$severity_code = ArcanistLintSeverity::SEVERITY_WARNING;
if ($inconclusive) {
$severity_code = ArcanistLintSeverity::SEVERITY_ADVICE;
} else if (stripos($severity, 'error') !== false) {
$severity_code = ArcanistLintSeverity::SEVERITY_ERROR;
}
$message = new ArcanistLintMessage(); $message = new ArcanistLintMessage();
$message->setPath($path); $message->setPath($location->getAttribute('file'));
$message->setLine($line); $message->setLine($location->getAttribute('line'));
$message->setCode($severity); $message->setCode('Cppcheck');
$message->setName($id); $message->setName($error->getAttribute('id'));
$message->setDescription($msg); $message->setDescription($error->getAttribute('msg'));
$message->setSeverity($severity_code);
$this->addLintMessage($message); switch ($error->getAttribute('severity')) {
case 'error':
$message->setSeverity(ArcanistLintSeverity::SEVERITY_ERROR);
break;
default:
if ($error->getAttribute('inconclusive')) {
$message->setSeverity(ArcanistLintSeverity::SEVERITY_ADVICE);
} else {
$message->setSeverity(ArcanistLintSeverity::SEVERITY_WARNING);
}
break;
}
$messages[] = $message;
} }
} }
return $messages;
}
} }

View file

@ -0,0 +1,12 @@
<?php
final class ArcanistCppcheckLinterTestCase
extends ArcanistArcanistLinterTestCase {
public function testCppcheckLint() {
return $this->executeTestsInDirectory(
dirname(__FILE__).'/cppcheck/',
new ArcanistCppcheckLinter());
}
}

View file

@ -0,0 +1,9 @@
int main()
{
char a[10];
a[10] = 0;
return 0;
}
~~~~~~~~~~
error:4:
warning:4:

View file

@ -0,0 +1,7 @@
void f() {
char arr[5];
// cppcheck-suppress arrayIndexOutOfBounds
// cppcheck-suppress unreadVariable
arr[10] = 0;
}
~~~~~~~~~~

View file

@ -0,0 +1,5 @@
int main()
{
return 0;
}
~~~~~~~~~~

View file

@ -0,0 +1,20 @@
/**
* Taken from http://www.slideshare.net/zblair/cppcheck-10316379
*/
void foo(char* str) {
char* buf = new char[8];
strcpy(buf, str);
FILE* file = fopen("out.txt", "w");
if (!file)
return;
for (char* c = buf; *c; ++c)
fputc((int)*c, file);
delete buf;
}
~~~~~~~~~~
error:10:
error:15:
error:16: