12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
3c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines#include "clang/Config/config.h"
450398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang-c/Index.h"
5db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison#include "clang-c/CXCompilationDatabase.h"
6651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines#include "clang-c/BuildSystem.h"
76bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines#include "clang-c/Documentation.h"
81e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor#include <ctype.h>
90c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor#include <stdlib.h>
1089922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff#include <stdio.h>
11af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff#include <string.h>
12f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor#include <assert.h>
13af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff
14f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
15f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/parser.h>
16f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/relaxng.h>
17f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/xmlerror.h>
18f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
19f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
20d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#ifdef _WIN32
21d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <direct.h>
22d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#else
23d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <unistd.h>
24d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#endif
25d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
264967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarextern int indextest_core_main(int argc, const char **argv);
274967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
280d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
290d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Utility functions.                                                         */
300d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
310d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
322e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#ifdef _MSC_VER
332e06fc877a633abea3b40a64950c7316dac29ca8John Thompsonchar *basename(const char* path)
342e06fc877a633abea3b40a64950c7316dac29ca8John Thompson{
352e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base1 = (char*)strrchr(path, '/');
362e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base2 = (char*)strrchr(path, '\\');
372e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    if (base1 && base2)
382e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return((base1 > base2) ? base1 + 1 : base2 + 1);
392e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base1)
402e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base1 + 1);
412e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base2)
422e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base2 + 1);
432e06fc877a633abea3b40a64950c7316dac29ca8John Thompson
442e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    return((char*)path);
452e06fc877a633abea3b40a64950c7316dac29ca8John Thompson}
46db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonchar *dirname(char* path)
47db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison{
48db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base1 = (char*)strrchr(path, '/');
49db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base2 = (char*)strrchr(path, '\\');
50db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (base1 && base2)
51db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (base1 > base2)
52db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base1 = 0;
53db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        else
54db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base2 = 0;
55db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base1)
560fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base1 = 0;
57db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base2)
580fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base2 = 0;
59db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
60db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return path;
61db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
622e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#else
63ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroffextern char *basename(const char *);
64db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonextern char *dirname(char *);
652e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#endif
66ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroff
6745ba9a1b31110568d0d362c8d31c6133cf9011b7Douglas Gregor/** \brief Return the default parsing options. */
6844c181aec37789f25f6c15543c164416f72e562aDouglas Gregorstatic unsigned getDefaultParsingOptions() {
6944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  unsigned options = CXTranslationUnit_DetailedPreprocessingRecord;
7044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
7144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (getenv("CINDEXTEST_EDITING"))
72b1c031be513705d924038f497279b9b599868ba1Douglas Gregor    options |= clang_defaultEditingTranslationUnitOptions();
7387c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  if (getenv("CINDEXTEST_COMPLETION_CACHING"))
7487c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    options |= CXTranslationUnit_CacheCompletionResults;
75dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis  if (getenv("CINDEXTEST_COMPLETION_NO_CACHING"))
76dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis    options &= ~CXTranslationUnit_CacheCompletionResults;
776a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen  if (getenv("CINDEXTEST_SKIP_FUNCTION_BODIES"))
786a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen    options |= CXTranslationUnit_SkipFunctionBodies;
79d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
80d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    options |= CXTranslationUnit_IncludeBriefCommentsInCodeCompletion;
8187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (getenv("CINDEXTEST_CREATE_PREAMBLE_ON_FIRST_PARSE"))
8287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    options |= CXTranslationUnit_CreatePreambleOnFirstParse;
834967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (getenv("CINDEXTEST_KEEP_GOING"))
844967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    options |= CXTranslationUnit_KeepGoing;
8587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
8644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  return options;
8744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor}
8844c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
89651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines/** \brief Returns 0 in case of success, non-zero in case of a failure. */
90bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU);
91bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
92651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic void describeLibclangFailure(enum CXErrorCode Err) {
93651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  switch (Err) {
94651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Success:
95651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Success\n");
96651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
97651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
98651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Failure:
99651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure (no details available)\n");
100651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
101651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
102651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Crashed:
103651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: libclang crashed\n");
104651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
105651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
106651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_InvalidArguments:
107651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: invalid arguments passed to a libclang routine\n");
108651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
109651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
110651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_ASTReadError:
111651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: AST deserialization error occurred\n");
112651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
113651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
114651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
115651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
11651b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbarstatic void PrintExtent(FILE *out, unsigned begin_line, unsigned begin_column,
11751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar                        unsigned end_line, unsigned end_column) {
11851b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  fprintf(out, "[%d:%d - %d:%d]", begin_line, begin_column,
119d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar          end_line, end_column);
12051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar}
12151b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar
1221c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenekstatic unsigned CreateTranslationUnit(CXIndex Idx, const char *file,
1231c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                      CXTranslationUnit *TU) {
124651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err = clang_createTranslationUnit2(Idx, file, TU);
125651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1261c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Unable to load translation unit from '%s'!\n", file);
127651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
128651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    *TU = 0;
1291c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 0;
130e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1311c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 1;
1321c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
1331c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1344db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorvoid free_remapped_files(struct CXUnsavedFile *unsaved_files,
1354db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int num_unsaved_files) {
1364db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (i = 0; i != num_unsaved_files; ++i) {
1384db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Filename);
1394db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Contents);
1404db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
141653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor  free(unsaved_files);
1424db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1434db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
144651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files_with_opt(const char *opt_name,
145651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int argc, const char **argv,
146651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int start_arg,
147651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         struct CXUnsavedFile **unsaved_files,
148651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int *num_unsaved_files) {
1494db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int arg;
151651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int prefix_len = strlen(opt_name);
152651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int arg_indices[20];
1534db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files = 0;
1544db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *num_unsaved_files = 0;
155e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1564db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  /* Count the number of remapped files. */
1574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg; arg < argc; ++arg) {
158651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (strncmp(argv[arg], opt_name, prefix_len))
159651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      continue;
160e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
161651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    assert(*num_unsaved_files < (int)(sizeof(arg_indices)/sizeof(int)));
162651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    arg_indices[*num_unsaved_files] = arg;
1634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    ++*num_unsaved_files;
1644db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
165e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  if (*num_unsaved_files == 0)
1674db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return 0;
168e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files
170653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor    = (struct CXUnsavedFile *)malloc(sizeof(struct CXUnsavedFile) *
171653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor                                     *num_unsaved_files);
172651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i != *num_unsaved_files; ++i) {
1734db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    struct CXUnsavedFile *unsaved = *unsaved_files + i;
174651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    const char *arg_string = argv[arg_indices[i]] + prefix_len;
1754db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    int filename_len;
1764db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *filename;
1774db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *contents;
1784db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    FILE *to_file;
179651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    const char *sep = strchr(arg_string, ',');
180651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (!sep) {
181e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr,
182651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              "error: %sfrom:to argument is missing comma\n", opt_name);
1834db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1844db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1854db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1864db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1874db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
188e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1894db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Open the file that we're remapping to. */
190651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    to_file = fopen(sep + 1, "rb");
1914db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!to_file) {
1924db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: cannot open file %s that we are remapping to\n",
193651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              sep + 1);
1944db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1954db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1964db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1974db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1984db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
199e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2004db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Determine the length of the file we're remapping to. */
2014db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_END);
2024db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Length = ftell(to_file);
2034db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_SET);
204e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2054db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Read the contents of the file we're remapping to. */
2064db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents = (char *)malloc(unsaved->Length + 1);
2074db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (fread(contents, 1, unsaved->Length, to_file) != unsaved->Length) {
2084db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: unexpected %s reading 'to' file %s\n",
209651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              (feof(to_file) ? "EOF" : "error"), sep + 1);
2104db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fclose(to_file);
2114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
212e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith      free(contents);
2134db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
2144db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
2154db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
2164db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
2174db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents[unsaved->Length] = 0;
2184db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Contents = contents;
219e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2204db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Close the file. */
2214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fclose(to_file);
222e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2234db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Copy the file name that we're remapping from. */
224651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    filename_len = sep - arg_string;
2254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename = (char *)malloc(filename_len + 1);
2264db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    memcpy(filename, arg_string, filename_len);
2274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename[filename_len] = 0;
2284db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Filename = filename;
2294db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
230e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2314db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return 0;
2324db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
2334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
234651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files(int argc, const char **argv, int start_arg,
235651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                struct CXUnsavedFile **unsaved_files,
236651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                int *num_unsaved_files) {
237651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return parse_remapped_files_with_opt("-remap-file=", argc, argv, start_arg,
238651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files, num_unsaved_files);
239651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
240651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
241651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files_with_try(int try_idx,
242651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int argc, const char **argv,
243651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int start_arg,
244651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         struct CXUnsavedFile **unsaved_files,
245651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int *num_unsaved_files) {
246651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  struct CXUnsavedFile *unsaved_files_no_try_idx;
247651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int num_unsaved_files_no_try_idx;
248651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  struct CXUnsavedFile *unsaved_files_try_idx;
249651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int num_unsaved_files_try_idx;
250651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int ret;
251651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  char opt_name[32];
252651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
253651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  ret = parse_remapped_files(argc, argv, start_arg,
254651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &unsaved_files_no_try_idx, &num_unsaved_files_no_try_idx);
255651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (ret)
256651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return ret;
257651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
258651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  sprintf(opt_name, "-remap-file-%d=", try_idx);
259651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  ret = parse_remapped_files_with_opt(opt_name, argc, argv, start_arg,
260651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &unsaved_files_try_idx, &num_unsaved_files_try_idx);
261651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (ret)
262651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return ret;
263651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
26487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (num_unsaved_files_no_try_idx == 0) {
26587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    *unsaved_files = unsaved_files_try_idx;
26687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    *num_unsaved_files = num_unsaved_files_try_idx;
26787d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return 0;
26887d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  }
26987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (num_unsaved_files_try_idx == 0) {
27087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    *unsaved_files = unsaved_files_no_try_idx;
27187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    *num_unsaved_files = num_unsaved_files_no_try_idx;
27287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return 0;
27387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  }
27487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
275651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  *num_unsaved_files = num_unsaved_files_no_try_idx + num_unsaved_files_try_idx;
276651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  *unsaved_files
277651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    = (struct CXUnsavedFile *)realloc(unsaved_files_no_try_idx,
278651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                      sizeof(struct CXUnsavedFile) *
279651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                        *num_unsaved_files);
280651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  memcpy(*unsaved_files + num_unsaved_files_no_try_idx,
281651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines         unsaved_files_try_idx, sizeof(struct CXUnsavedFile) *
282651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines            num_unsaved_files_try_idx);
283651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  free(unsaved_files_try_idx);
284651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return 0;
285651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
286651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
287f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic const char *parse_comments_schema(int argc, const char **argv) {
288f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentsSchemaArg = "-comments-xml-schema=";
289f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile = NULL;
290f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
291f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (argc == 0)
292f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return CommentSchemaFile;
293f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
294f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!strncmp(argv[0], CommentsSchemaArg, strlen(CommentsSchemaArg)))
295f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    CommentSchemaFile = argv[0] + strlen(CommentsSchemaArg);
296f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
297f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  return CommentSchemaFile;
298f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
299f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
3000d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3010d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Pretty-printing.                                                           */
3020d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3030d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
304ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic const char *FileCheckPrefix = "CHECK";
305ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
306ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCString(const char *CStr) {
3072d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  if (CStr != NULL && CStr[0] != '\0') {
3082d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    for ( ; *CStr; ++CStr) {
3092d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      const char C = *CStr;
3102d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      switch (C) {
3112d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\n': printf("\\n"); break;
3122d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\r': printf("\\r"); break;
3132d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\t': printf("\\t"); break;
3142d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\v': printf("\\v"); break;
3152d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\f': printf("\\f"); break;
3162d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        default:   putchar(C);    break;
3172d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      }
3182d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    }
3192d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  }
320ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
321ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
322ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCStringWithPrefix(const char *Prefix, const char *CStr) {
323ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(" %s=[", Prefix);
324ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(CStr);
3252d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  printf("]");
3262d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko}
3272d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko
328ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringAndDispose(CXString Str) {
329ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(clang_getCString(Str));
330ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
331ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
332ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
333f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCXStringWithPrefix(const char *Prefix, CXString Str) {
334f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
335f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
336f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
337ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringWithPrefixAndDispose(const char *Prefix,
338ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                              CXString Str) {
339ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
340ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
341ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
342ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
343430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregorstatic void PrintRange(CXSourceRange R, const char *str) {
344430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  CXFile begin_file, end_file;
345430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  unsigned begin_line, begin_column, end_line, end_column;
346430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
347430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeStart(R),
348430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &begin_file, &begin_line, &begin_column, 0);
349430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeEnd(R),
350430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &end_file, &end_line, &end_column, 0);
351430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (!begin_file || !end_file)
352430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    return;
353430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
354ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis  if (str)
355ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis    printf(" %s=", str);
356430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintExtent(stdout, begin_line, begin_column, end_line, end_column);
357430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
358430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
359358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregorint want_display_name = 0;
360358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
361cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregorstatic void printVersion(const char *Prefix, CXVersion Version) {
362cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Major < 0)
363cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
364cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf("%s%d", Prefix, Version.Major);
365cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
366cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Minor < 0)
367cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
368cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Minor);
369cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
370cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Subminor < 0)
371cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
372cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Subminor);
373cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor}
374cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
375ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostruct CommentASTDumpingContext {
376ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  int IndentLevel;
377ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko};
378ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
379ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXCommentInternal(struct CommentASTDumpingContext *Ctx,
380ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                  CXComment Comment) {
3815ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned i;
3825ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned e;
3835ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  enum CXCommentKind Kind = clang_Comment_getKind(Comment);
3845ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko
385ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel++;
3865ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  for (i = 0, e = Ctx->IndentLevel; i != e; ++i)
387ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("  ");
388ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
389ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("(");
390ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  switch (Kind) {
391ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Null:
392ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Null");
393ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
394ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Text:
395ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Text");
396ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose("Text",
397ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                      clang_TextComment_getText(Comment));
398ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
399ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
400ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
401ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
402ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
403ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_InlineCommand:
404ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_InlineCommand");
405ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
406ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
407ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_InlineCommandComment_getCommandName(Comment));
4082d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    switch (clang_InlineCommandComment_getRenderKind(Comment)) {
4092d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Normal:
4102d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderNormal");
4112d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4122d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Bold:
4132d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderBold");
4142d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4152d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Monospaced:
4162d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderMonospaced");
4172d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4182d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Emphasized:
4192d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderEmphasized");
4202d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4212d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    }
4225ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_InlineCommandComment_getNumArgs(Comment);
423ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
424ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
425ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
426ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_InlineCommandComment_getArgText(Comment, i));
427ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
428ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
429ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
430ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
4315ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  case CXComment_HTMLStartTag: {
4325ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned NumAttrs;
433ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLStartTag");
434ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
435ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
436ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
4375ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    NumAttrs = clang_HTMLStartTag_getNumAttrs(Comment);
438ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (NumAttrs != 0) {
439ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Attrs:");
4405ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko      for (i = 0; i != NumAttrs; ++i) {
441ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf(" ");
442ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrName(Comment, i));
443ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf("=");
444ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrValue(Comment, i));
445ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      }
446ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
447ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_HTMLStartTagComment_isSelfClosing(Comment))
448ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" SelfClosing");
449ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
450ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
451ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
4525ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  }
453ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_HTMLEndTag:
454ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLEndTag");
455ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
456ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
457ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
458ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
459ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
460ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
461ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Paragraph:
462ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Paragraph");
463ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
464ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
465ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
466ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_BlockCommand:
467ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_BlockCommand");
468ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
469ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
470ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
4715ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_BlockCommandComment_getNumArgs(Comment);
472ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
473ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
474ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
475ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_BlockCommandComment_getArgText(Comment, i));
476ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
477ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
478ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_ParamCommand:
479ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_ParamCommand");
480ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    switch (clang_ParamCommandComment_getDirection(Comment)) {
481ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_In:
482ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in");
483ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
484ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_Out:
485ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" out");
486ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
487ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_InOut:
488ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in,out");
489ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
490ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
491ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isDirectionExplicit(Comment))
492ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" explicitly");
493ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
494ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" implicitly");
495ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
496ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "ParamName",
497ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_ParamCommandComment_getParamName(Comment));
498ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isParamIndexValid(Comment))
499ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=%u", clang_ParamCommandComment_getParamIndex(Comment));
500ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
501ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=Invalid");
502ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
50396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko  case CXComment_TParamCommand:
50496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    printf("CXComment_TParamCommand");
50596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    PrintCXStringWithPrefixAndDispose(
50696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        "ParamName",
50796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        clang_TParamCommandComment_getParamName(Comment));
50896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    if (clang_TParamCommandComment_isParamPositionValid(Comment)) {
50996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition={");
51096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      for (i = 0, e = clang_TParamCommandComment_getDepth(Comment);
51196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko           i != e; ++i) {
51296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        printf("%u", clang_TParamCommandComment_getIndex(Comment, i));
51396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        if (i != e - 1)
51496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko          printf(", ");
51596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      }
51696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf("}");
51796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    } else
51896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition=Invalid");
51996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    break;
520ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockCommand:
521ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockCommand");
522ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
523ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
524ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
525ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
526ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockLine:
527ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockLine");
528ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
529ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
530ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimBlockLineComment_getText(Comment));
531ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
532ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimLine:
533ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimLine");
534ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
535ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
536ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimLineComment_getText(Comment));
537ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
538ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_FullComment:
539ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_FullComment");
540ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
541ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
542ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  if (Kind != CXComment_Null) {
543ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const unsigned NumChildren = clang_Comment_getNumChildren(Comment);
5445ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned i;
5455ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0; i != NumChildren; ++i) {
546ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf("\n// %s: ", FileCheckPrefix);
547ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXCommentInternal(Ctx, clang_Comment_getChild(Comment, i));
548ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
549ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
550ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(")");
551ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel--;
552ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
553ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
554ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXComment(CXComment Comment) {
555ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  struct CommentASTDumpingContext Ctx;
556ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx.IndentLevel = 1;
557ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("\n// %s:  CommentAST=[\n// %s:", FileCheckPrefix, FileCheckPrefix);
558ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  DumpCXCommentInternal(&Ctx, Comment);
559ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("]");
560ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
561ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
5626bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void ValidateCommentXML(const char *Str, const char *CommentSchemaFile) {
563f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
564f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGParserCtxtPtr RNGParser;
565f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGPtr Schema;
566f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlDocPtr Doc;
567f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGValidCtxtPtr ValidationCtxt;
568f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  int status;
569f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5706bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (!CommentSchemaFile)
571f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
572f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5736bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  RNGParser = xmlRelaxNGNewParserCtxt(CommentSchemaFile);
5746bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (!RNGParser) {
575f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
576f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
577f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
5786bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  Schema = xmlRelaxNGParse(RNGParser);
579f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
580f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  Doc = xmlParseDoc((const xmlChar *) Str);
581f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
582f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!Doc) {
583f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
584f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not well-formed XML: %s]", Error->message);
585f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
586f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
587f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5886bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  ValidationCtxt = xmlRelaxNGNewValidCtxt(Schema);
589f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  status = xmlRelaxNGValidateDoc(ValidationCtxt, Doc);
590f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!status)
591f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLValid");
592f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  else if (status > 0) {
593f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
594f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not vaild XML: %s]", Error->message);
595f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  } else
596f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
597f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
598f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGFreeValidCtxt(ValidationCtxt);
599f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlFreeDoc(Doc);
6006bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  xmlRelaxNGFree(Schema);
6016bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  xmlRelaxNGFreeParserCtxt(RNGParser);
602f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
603f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
604f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
605e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenkostatic void PrintCursorComments(CXCursor Cursor,
6066bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                const char *CommentSchemaFile) {
607ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
608ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString RawComment;
609ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *RawCommentCString;
610ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString BriefComment;
611ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *BriefCommentCString;
612ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
613ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawComment = clang_Cursor_getRawCommentText(Cursor);
614ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawCommentCString = clang_getCString(RawComment);
615ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (RawCommentCString != NULL && RawCommentCString[0] != '\0') {
616ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCStringWithPrefix("RawComment", RawCommentCString);
617ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintRange(clang_Cursor_getCommentRange(Cursor), "RawCommentRange");
618ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
619ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefComment = clang_Cursor_getBriefCommentText(Cursor);
620ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefCommentCString = clang_getCString(BriefComment);
621ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      if (BriefCommentCString != NULL && BriefCommentCString[0] != '\0')
622ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCStringWithPrefix("BriefComment", BriefCommentCString);
623ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      clang_disposeString(BriefComment);
624ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
625ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    clang_disposeString(RawComment);
626ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
627ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
628ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
62910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXComment Comment = clang_Cursor_getParsedComment(Cursor);
630ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_getKind(Comment) != CXComment_Null) {
631ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringWithPrefixAndDispose("FullCommentAsHTML",
632ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                        clang_FullComment_getAsHTML(Comment));
633f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      {
634f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        CXString XML;
635e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko        XML = clang_FullComment_getAsXML(Comment);
636f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCXStringWithPrefix("FullCommentAsXML", XML);
6376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        ValidateCommentXML(clang_getCString(XML), CommentSchemaFile);
638f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        clang_disposeString(XML);
639f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      }
640f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
641ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXComment(Comment);
642ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
643ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
644ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
645ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
646b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidistypedef struct {
647b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned line;
648b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned col;
649b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis} LineCol;
650b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
651b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidisstatic int lineCol_cmp(const void *p1, const void *p2) {
652b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *lhs = p1;
653b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *rhs = p2;
654b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  if (lhs->line != rhs->line)
655b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis    return (int)lhs->line - (int)rhs->line;
656b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  return (int)lhs->col - (int)rhs->col;
657b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis}
658b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
6596bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void PrintCursor(CXCursor Cursor, const char *CommentSchemaFile) {
660aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
661e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (clang_isInvalid(Cursor.kind)) {
66210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString ks = clang_getCursorKindSpelling(Cursor.kind);
663e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("Invalid Cursor => %s", clang_getCString(ks));
664e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
665e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
666699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  else {
667e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString string, ks;
668c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    CXCursor Referenced;
6691db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
670e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    CXCursor SpecializationOf;
6719f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    CXCursor *overridden;
6729f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned num_overridden;
673430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    unsigned RefNameRangeNr;
674430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange CursorExtent;
675430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange RefNameRange;
676cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysUnavailable;
677cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysDeprecated;
678cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString UnavailableMessage;
679cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString DeprecatedMessage;
680cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXPlatformAvailability PlatformAvailability[2];
681cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int NumPlatformAvailability;
682cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int I;
683aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
684e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    ks = clang_getCursorKindSpelling(Cursor.kind);
685358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    string = want_display_name? clang_getCursorDisplayName(Cursor)
686358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              : clang_getCursorSpelling(Cursor);
687e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("%s=%s", clang_getCString(ks),
688e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                    clang_getCString(string));
689e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
690ef0cef6cec8de5fc60e469a93436eed7212e0dc2Steve Naroff    clang_disposeString(string);
691e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
692c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    Referenced = clang_getCursorReferenced(Cursor);
693c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    if (!clang_equalCursors(Referenced, clang_getNullCursor())) {
6941f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (clang_getCursorKind(Referenced) == CXCursor_OverloadedDeclRef) {
6951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        unsigned I, N = clang_getNumOverloadedDecls(Referenced);
6961f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("[");
6971f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        for (I = 0; I != N; ++I) {
69810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXCursor Ovl = clang_getOverloadedDecl(Referenced, I);
6991f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          CXSourceLocation Loc;
7001f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          if (I)
7011f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor            printf(", ");
7021f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
7031f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          Loc = clang_getCursorLocation(Ovl);
704a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor          clang_getSpellingLocation(Loc, 0, &line, &column, 0);
7051f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          printf("%d:%d", line, column);
7061f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        }
7071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("]");
7081f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      } else {
70910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation Loc = clang_getCursorLocation(Referenced);
710a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
7111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf(":%d:%d", line, column);
7121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
713c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
714b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
715b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (clang_isCursorDefinition(Cursor))
716b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      printf(" (Definition)");
71758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
71858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    switch (clang_getCursorAvailability(Cursor)) {
71958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Available:
72058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
72158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
72258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Deprecated:
72358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (deprecated)");
72458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
72558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
72658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_NotAvailable:
72758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (unavailable)");
72858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
729d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
730d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen      case CXAvailability_NotAccessible:
731d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        printf(" (inaccessible)");
732d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        break;
73358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
73495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
735cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    NumPlatformAvailability
736cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      = clang_getCursorPlatformAvailability(Cursor,
737cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysDeprecated,
738cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &DeprecatedMessage,
739cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysUnavailable,
740cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &UnavailableMessage,
741cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            PlatformAvailability, 2);
742cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    if (AlwaysUnavailable) {
743cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always unavailable: \"%s\")",
744cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(UnavailableMessage));
745cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else if (AlwaysDeprecated) {
746cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always deprecated: \"%s\")",
747cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(DeprecatedMessage));
748cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else {
749cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      for (I = 0; I != NumPlatformAvailability; ++I) {
750cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (I >= 2)
751cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          break;
752cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
753cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf("  (%s", clang_getCString(PlatformAvailability[I].Platform));
754cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (PlatformAvailability[I].Unavailable)
755cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", unavailable");
756cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        else {
757cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", introduced=", PlatformAvailability[I].Introduced);
758cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", deprecated=", PlatformAvailability[I].Deprecated);
759cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", obsoleted=", PlatformAvailability[I].Obsoleted);
760cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        }
761cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (clang_getCString(PlatformAvailability[I].Message)[0])
762cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", message=\"%s\"",
763cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                 clang_getCString(PlatformAvailability[I].Message));
764cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf(")");
765cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      }
766cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
767cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    for (I = 0; I != NumPlatformAvailability; ++I) {
768cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      if (I >= 2)
769cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        break;
770cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      clang_disposeCXPlatformAvailability(PlatformAvailability + I);
771cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
772cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
773cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(DeprecatedMessage);
774cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(UnavailableMessage);
7754967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
7764967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (clang_CXXConstructor_isDefaultConstructor(Cursor))
7774967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" (default constructor)");
7784967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
7794967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (clang_CXXConstructor_isMoveConstructor(Cursor))
7804967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" (move constructor)");
7814967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (clang_CXXConstructor_isCopyConstructor(Cursor))
7824967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" (copy constructor)");
7834967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (clang_CXXConstructor_isConvertingConstructor(Cursor))
7844967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" (converting constructor)");
78587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    if (clang_CXXField_isMutable(Cursor))
78687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar      printf(" (mutable)");
7874967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (clang_CXXMethod_isDefaulted(Cursor))
7884967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" (defaulted)");
789b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isStatic(Cursor))
790b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (static)");
791b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isVirtual(Cursor))
792b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (virtual)");
7936bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    if (clang_CXXMethod_isConst(Cursor))
7946bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      printf(" (const)");
795c965f76e02d536d875b1e9e3ea96f39452c6bfa2Dmitri Gribenko    if (clang_CXXMethod_isPureVirtual(Cursor))
796c965f76e02d536d875b1e9e3ea96f39452c6bfa2Dmitri Gribenko      printf(" (pure)");
79780e1acaeb7040548cb494e609b120b134e22a193Argyrios Kyrtzidis    if (clang_Cursor_isVariadic(Cursor))
79880e1acaeb7040548cb494e609b120b134e22a193Argyrios Kyrtzidis      printf(" (variadic)");
799514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis    if (clang_Cursor_isObjCOptional(Cursor))
800514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis      printf(" (@optional)");
801514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis
80295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
80310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType T =
80410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        clang_getCanonicalType(clang_getIBOutletCollectionType(Cursor));
80510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString S = clang_getTypeKindSpelling(T.kind);
80695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      printf(" [IBOutletCollection=%s]", clang_getCString(S));
80795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      clang_disposeString(S);
80895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    }
8093064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
8103064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
8113064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
8123064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      unsigned isVirtual = clang_isVirtualBase(Cursor);
8133064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      const char *accessStr = 0;
8143064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
8153064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      switch (access) {
8163064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXInvalidAccessSpecifier:
8173064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "invalid"; break;
8183064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPublic:
8193064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "public"; break;
8203064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXProtected:
8213064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "protected"; break;
8223064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPrivate:
8233064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "private"; break;
8243064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      }
8253064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
8263064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      printf(" [access=%s isVirtual=%s]", accessStr,
8273064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek             isVirtual ? "true" : "false");
8283064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
829176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
830e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    SpecializationOf = clang_getSpecializedCursorTemplate(Cursor);
831e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    if (!clang_equalCursors(SpecializationOf, clang_getNullCursor())) {
83210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXSourceLocation Loc = clang_getCursorLocation(SpecializationOf);
83310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString Name = clang_getCursorSpelling(SpecializationOf);
834a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      clang_getSpellingLocation(Loc, 0, &line, &column, 0);
835176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      printf(" [Specialization of %s:%d:%d]",
836e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
837e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
838176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
839176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      if (Cursor.kind == CXCursor_FunctionDecl) {
840176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        /* Collect the template parameter kinds from the base template. */
841176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        unsigned NumTemplateArgs = clang_Cursor_getNumTemplateArguments(Cursor);
842176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        unsigned I;
843176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        for (I = 0; I < NumTemplateArgs; I++) {
844176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          enum CXTemplateArgumentKind TAK =
845176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              clang_Cursor_getTemplateArgumentKind(Cursor, I);
846176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          switch(TAK) {
847176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            case CXTemplateArgumentKind_Type:
848176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              {
849176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                CXType T = clang_Cursor_getTemplateArgumentType(Cursor, I);
850176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                CXString S = clang_getTypeSpelling(T);
851176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                printf(" [Template arg %d: kind: %d, type: %s]",
852176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                       I, TAK, clang_getCString(S));
853176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                clang_disposeString(S);
854176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              }
855176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              break;
856176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            case CXTemplateArgumentKind_Integral:
857176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              printf(" [Template arg %d: kind: %d, intval: %lld]",
858176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                     I, TAK, clang_Cursor_getTemplateArgumentValue(Cursor, I));
859176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              break;
860176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            default:
861176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              printf(" [Template arg %d: kind: %d]\n", I, TAK);
862176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          }
863176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        }
864176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      }
865e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
8669f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
8679f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
8689f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
8699f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
870b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      LineCol lineCols[50];
871b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      assert(num_overridden <= 50);
8729f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
8739f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
87410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
875a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
876b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].line = line;
877b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].col = column;
878b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      }
879642214936430d2258318f2022184c1709dfa16d3Michael Liao      /* Make the order of the override list deterministic. */
880b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      qsort(lineCols, num_overridden, sizeof(LineCol), lineCol_cmp);
881b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      for (I = 0; I != num_overridden; ++I) {
8829f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
8839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
884b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        printf("@%d:%d", lineCols[I].line, lineCols[I].col);
8859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
8869f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
8879f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
8889f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
889ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
890ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
89110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXFile File = clang_getIncludedFile(Cursor);
89210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString Included = clang_getFileName(File);
893ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
894ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
895dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
896dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
897dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
898ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
899430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
900430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
901430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
902430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
903430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
904430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
905430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
906430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
907430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
908430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
909430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
910430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
911430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
912430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
913430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
914430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
915430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
916430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
917430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
918430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
919aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
9206bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursorComments(Cursor, CommentSchemaFile);
9219ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis
9229ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    {
9239ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      unsigned PropAttrs = clang_Cursor_getObjCPropertyAttributes(Cursor, 0);
9249ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      if (PropAttrs != CXObjCPropertyAttr_noattr) {
9259ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf(" [");
9269ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        #define PRINT_PROP_ATTR(A) \
9279ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis          if (PropAttrs & CXObjCPropertyAttr_##A) printf(#A ",")
9289ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readonly);
9299ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(getter);
9309ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(assign);
9319ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readwrite);
9329ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(retain);
9339ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(copy);
9349ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(nonatomic);
9359ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(setter);
9369ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(atomic);
9379ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(weak);
9389ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(strong);
9399ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(unsafe_unretained);
9404967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        PRINT_PROP_ATTR(class);
9419ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf("]");
9429ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      }
9439ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    }
94438dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis
94538dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    {
94638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      unsigned QT = clang_Cursor_getObjCDeclQualifiers(Cursor);
94738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      if (QT != CXObjCDeclQualifier_None) {
94838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf(" [");
94938dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        #define PRINT_OBJC_QUAL(A) \
95038dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis          if (QT & CXObjCDeclQualifier_##A) printf(#A ",")
95138dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(In);
95238dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Inout);
95338dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Out);
95438dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Bycopy);
95538dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Byref);
95638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Oneway);
95738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf("]");
95838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      }
95938dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    }
960699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
961af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
96289922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
963e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
96410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
96574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
9661db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
967b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
9681db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
969e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
97074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
97174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
97274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
97374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
974e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
97574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
97674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
97774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
9789298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
9799298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
9800d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
981ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
982ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
983ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
984ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
985ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
986a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
987a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
9885352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
989274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
9900a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
991aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
992aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
9930a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
994f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
9950a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
9965352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
997e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
998274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
999274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
1000274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
1001f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
1002a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
1003a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
10040a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
10050a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
1006e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
10070a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
10083739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
10090a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
1010473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
101110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
101210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation start = clang_getRangeStart(range);
101310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation end = clang_getRangeEnd(range);
1014473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
1015473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
1016a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
1017a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
1018a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
1019473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
1020473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
1021473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
10220a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
1023473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
1024473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
1025473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
10260a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
10270a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
10280a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
10290a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
1030436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
1031473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
1032473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
10330a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
10340a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
10350a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
1036473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
103751c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
10380a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
1039473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
104051c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
10415352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
10425352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
10437473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
10447473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
10457473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
10467473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
10477473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
1048a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
10497473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
10507473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
10517473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
10527473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
10537473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
10547473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
10557473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
10567473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
10577473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
1058a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
1059a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
106059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
1061b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
10624e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
106310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
10643c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
10654e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
1066f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
106759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
106859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
10694e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
107059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
107159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
10724e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
107359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
1074f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
107559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
107659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
1077ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
1078e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
10790d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
10800d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1081a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
108210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceRange extent = clang_getCursorExtent(C);
108310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  PrintRange(extent, "Extent");
1084fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
1085fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1086f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko/* Data used by the visitors. */
1087f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
1088e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
1089e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
10906bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  const char *CommentSchemaFile;
1091e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
1092fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1093625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1094e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
1095e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
1096e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
1097e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1098e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
109910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
11001db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
1101a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
1102fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
11031db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
11046bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursor(Cursor, Data->CommentSchemaFile);
1105a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
110604b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    if (clang_isDeclaration(Cursor.kind)) {
110704b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
110804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      const char *accessStr = 0;
110904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
111004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      switch (access) {
111104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXInvalidAccessSpecifier: break;
111204b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPublic:
111304b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "public"; break;
111404b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXProtected:
111504b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "protected"; break;
111604b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPrivate:
111704b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "private"; break;
111804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      }
111904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
112004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      if (accessStr)
112104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        printf(" [access=%s]", accessStr);
112204b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    }
1123e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
1124e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1125625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
1126e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1127e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
1128625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1129c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
1130e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
1131e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
1132e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
1133625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
1134625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
1135625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
1136e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1137625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1138b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
1139b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
1140e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
1141625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1142625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
1143625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
1144625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
1145625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
1146625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
1147625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
1148625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1149625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
115098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
11511db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
115274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
1153e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1154625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
1155625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
1156625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
1157625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
1158625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
1159625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
1160e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
116198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
1162a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
1163e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
11641db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
1165e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
116610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXSourceLocation RefLoc
116710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        = clang_getLocation(Data->TU, file, curLine, curColumn);
1168b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
116998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
117098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
117198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
117298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
117398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
11746bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintCursor(Ref, Data->CommentSchemaFile);
117598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
117698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
11774ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
117874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
1179625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
11802d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
1181e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1182e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
118389922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
118450398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
11857d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11867d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
11877d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11887d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1189e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
1190e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
1191e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1192e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
119310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString USR = clang_getCursorUSR(C);
119410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    const char *cstr = clang_getCString(USR);
1195e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
11967d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
1197e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
11987d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
1199e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
1200e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
1201a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
12027d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
12037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
1204e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1205e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1206e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1207e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1208e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
12097d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
12107d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
12117d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
121216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
121316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
121416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
121516b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
121616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
1217e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
121816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
121974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
122074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
122174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
1222e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
122374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
1224e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
122516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
122616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
122716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
1228a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
1229a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
123074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
1231e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
123274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
123316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
123416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
123516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
123616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
123716b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
1238e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
123916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
124016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
124116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
12423bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
12433bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
12443bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12453bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
12463bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
12473bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
12483bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12493bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
12503bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
12513bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12523bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
12533bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
1254c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
1255c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
1256c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
1257c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
12583bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12593bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12603bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
1261f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
12623bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
12633bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12643bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12653bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
12663bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
12673bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12683bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
126987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar/* Visibility testing.                                                        */
127087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar/******************************************************************************/
127187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
127287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainarstatic enum CXChildVisitResult PrintVisibility(CXCursor cursor, CXCursor p,
127387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar                                               CXClientData d) {
127487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  const char *visibility = 0;
127587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
127687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (clang_isInvalid(clang_getCursorKind(cursor)))
127787d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return CXChildVisit_Recurse;
127887d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
127987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  switch (clang_getCursorVisibility(cursor)) {
128087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    case CXVisibility_Invalid: break;
128187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    case CXVisibility_Hidden: visibility = "Hidden"; break;
128287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    case CXVisibility_Protected: visibility = "Protected"; break;
128387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    case CXVisibility_Default: visibility = "Default"; break;
128487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  }
128587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
128687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (visibility) {
128787d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    PrintCursor(cursor, NULL);
128887d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    printf("visibility=%s\n", visibility);
128987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  }
129087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
129187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  return CXChildVisit_Recurse;
129287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar}
129387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
129487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar/******************************************************************************/
12958e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
12968e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
12978e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1298ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic void PrintTypeAndTypeKind(CXType T, const char *Format) {
1299ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  CXString TypeSpelling, TypeKindSpelling;
1300ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
1301ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeSpelling = clang_getTypeSpelling(T);
1302ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeKindSpelling = clang_getTypeKindSpelling(T.kind);
1303ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  printf(Format,
1304ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeSpelling),
1305ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeKindSpelling));
1306ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeSpelling);
1307ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeKindSpelling);
1308ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko}
1309ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
131058878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainarstatic enum CXVisitorResult FieldVisitor(CXCursor C,
131158878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar                                         CXClientData client_data) {
131258878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    (*(int *) client_data)+=1;
131358878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    return CXVisit_Continue;
131458878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar}
131558878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar
1316ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic enum CXChildVisitResult PrintType(CXCursor cursor, CXCursor p,
1317ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                         CXClientData d) {
13188e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
131910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXType T = clang_getCursorType(cursor);
1320659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    enum CXRefQualifierKind RQ = clang_Type_getCXXRefQualifier(T);
1321f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
1322ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1323e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
1324e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
1325e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
1326e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
1327e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
1328e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
1329659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_LValue)
1330659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" lvalue-ref-qualifier");
1331659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_RValue)
1332659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" rvalue-ref-qualifier");
1333e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
133404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
133510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType CT = clang_getCanonicalType(T);
133604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
1337ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(CT, " [canonicaltype=%s] [canonicaltypekind=%s]");
133804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
133904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1340e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
134104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
134210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType RT = clang_getCursorResultType(cursor);
134304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
1344ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(RT, " [resulttype=%s] [resulttypekind=%s]");
134504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
134604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1347d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1348d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1349651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumArgs = clang_Cursor_getNumArguments(cursor);
1350651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumArgs != -1 && NumArgs != 0) {
135147f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1352d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
1353651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumArgs; ++i) {
135410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1355d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1356ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko            PrintTypeAndTypeKind(T, " [%s] [%s]");
1357d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1358d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1359d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1360d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1361d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
1362651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    /* Print the template argument types if they exist. */
1363651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    {
1364651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumTArgs = clang_Type_getNumTemplateArguments(T);
1365651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumTArgs != -1 && NumTArgs != 0) {
1366651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        int i;
1367651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf(" [templateargs/%d=", NumTArgs);
1368651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumTArgs; ++i) {
1369651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXType TArg = clang_Type_getTemplateArgumentAsType(T, i);
1370651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          if (TArg.kind != CXType_Invalid) {
1371651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines            PrintTypeAndTypeKind(TArg, " [type=%s] [typekind=%s]");
1372651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          }
1373651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        }
1374651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf("]");
1375651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
1376651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
13773ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
13783ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
13796bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    /* Print the pointee type. */
13806bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    {
13816bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      CXType PT = clang_getPointeeType(T);
13826bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      if (PT.kind != CXType_Invalid) {
13836bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintTypeAndTypeKind(PT, " [pointeetype=%s] [pointeekind=%s]");
13846bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      }
13856bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    }
138658878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    /* Print the number of fields if they exist. */
138758878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    {
138858878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar      int numFields = 0;
138958878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar      if (clang_Type_visitFields(T, FieldVisitor, &numFields)){
139058878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        if (numFields != 0) {
139158878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar          printf(" [nbFields=%d]", numFields);
139258878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        }
139358878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        /* Print if it is an anonymous record. */
139458878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        {
139558878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar          unsigned isAnon = clang_Cursor_isAnonymous(cursor);
139658878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar          if (isAnon != 0) {
139758878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar            printf(" [isAnon=%d]", isAnon);
139858878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar          }
139958878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        }
140058878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar      }
140158878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    }
140204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
14038e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
14048e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
14058e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
14068e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
14078e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1408411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidisstatic enum CXChildVisitResult PrintTypeSize(CXCursor cursor, CXCursor p,
1409411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                             CXClientData d) {
1410411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  CXType T;
1411411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  enum CXCursorKind K = clang_getCursorKind(cursor);
1412411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  if (clang_isInvalid(K))
1413411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return CXChildVisit_Recurse;
1414411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  T = clang_getCursorType(cursor);
1415411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintCursor(cursor, NULL);
1416411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1417411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type sizeof if applicable. */
1418411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1419411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Size = clang_Type_getSizeOf(T);
1420411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Size >= 0 || Size < -1 ) {
1421411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [sizeof=%lld]", Size);
1422411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1423411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1424411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type alignof if applicable. */
1425411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1426411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Align = clang_Type_getAlignOf(T);
1427411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Align >= 0 || Align < -1) {
1428411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [alignof=%lld]", Align);
1429411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1430411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1431411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the record field offset if applicable. */
1432411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
14336bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString FieldSpelling = clang_getCursorSpelling(cursor);
14346bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *FieldName = clang_getCString(FieldSpelling);
143558878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    /* recurse to get the first parent record that is not anonymous. */
143658878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar    unsigned RecordIsAnonymous = 0;
14376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    if (clang_getCursorKind(cursor) == CXCursor_FieldDecl) {
14384967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      CXCursor Record;
14394967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      CXCursor Parent = p;
1440411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      do {
144158878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        Record = Parent;
144258878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        Parent = clang_getCursorSemanticParent(Record);
144358878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        RecordIsAnonymous = clang_Cursor_isAnonymous(Record);
144458878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        /* Recurse as long as the parent is a CXType_Record and the Record
144558878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar           is anonymous */
144658878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar      } while ( clang_getCursorType(Parent).kind == CXType_Record &&
144758878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar                RecordIsAnonymous > 0);
1448411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      {
144958878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        long long Offset = clang_Type_getOffsetOf(clang_getCursorType(Record),
1450411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                                  FieldName);
145158878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        long long Offset2 = clang_Cursor_getOffsetOfField(cursor);
145258878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        if (Offset == Offset2){
145358878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar            printf(" [offsetof=%lld]", Offset);
145458878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        } else {
145558878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar            /* Offsets will be different in anonymous records. */
145658878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar            printf(" [offsetof=%lld/%lld]", Offset, Offset2);
145758878f85ab89b13e9eea4af3ccf055e42c557bc8Pirama Arumuga Nainar        }
1458411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      }
1459411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
14606bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(FieldSpelling);
1461411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1462411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print if its a bitfield */
1463411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1464411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    int IsBitfield = clang_Cursor_isBitField(cursor);
1465411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (IsBitfield)
1466411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [BitFieldSize=%d]", clang_getFieldDeclBitWidth(cursor));
1467411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1468411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  printf("\n");
1469411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  return CXChildVisit_Recurse;
1470411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis}
1471411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis
14721eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
1473176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/* Mangling testing.                                                          */
1474176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/******************************************************************************/
1475176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
1476176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hinesstatic enum CXChildVisitResult PrintMangledName(CXCursor cursor, CXCursor p,
1477176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                                                CXClientData d) {
1478176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  CXString MangledName;
147987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (clang_isUnexposed(clang_getCursorKind(cursor)))
148087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return CXChildVisit_Recurse;
1481176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  PrintCursor(cursor, NULL);
1482176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  MangledName = clang_Cursor_getMangling(cursor);
1483176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  printf(" [mangled=%s]\n", clang_getCString(MangledName));
1484176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  clang_disposeString(MangledName);
1485176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  return CXChildVisit_Continue;
1486176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines}
1487176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
148887d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainarstatic enum CXChildVisitResult PrintManglings(CXCursor cursor, CXCursor p,
148987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar                                              CXClientData d) {
149087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  unsigned I, E;
149187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  CXStringSet *Manglings = NULL;
149287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (clang_isUnexposed(clang_getCursorKind(cursor)))
149387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return CXChildVisit_Recurse;
149487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (!clang_isDeclaration(clang_getCursorKind(cursor)))
149587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return CXChildVisit_Recurse;
149687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  if (clang_getCursorKind(cursor) == CXCursor_ParmDecl)
149787d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return CXChildVisit_Continue;
149887d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  PrintCursor(cursor, NULL);
149987d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  Manglings = clang_Cursor_getCXXManglings(cursor);
150087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  for (I = 0, E = Manglings->Count; I < E; ++I)
150187d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    printf(" [mangled=%s]", clang_getCString(Manglings->Strings[I]));
150287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  clang_disposeStringSet(Manglings);
150387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  printf("\n");
150487d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  return CXChildVisit_Recurse;
150587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar}
150687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar
1507176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/******************************************************************************/
15081eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/* Bitwidth testing.                                                          */
15091eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
15101eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
15111eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenkostatic enum CXChildVisitResult PrintBitWidth(CXCursor cursor, CXCursor p,
15121eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                             CXClientData d) {
151302c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  int Bitwidth;
15141eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (clang_getCursorKind(cursor) != CXCursor_FieldDecl)
15151eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return CXChildVisit_Recurse;
15161eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
151702c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  Bitwidth = clang_getFieldDeclBitWidth(cursor);
15181eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (Bitwidth >= 0) {
15191eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    PrintCursor(cursor, NULL);
15201eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    printf(" bitwidth=%d\n", Bitwidth);
15211eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  }
15221eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
15231eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  return CXChildVisit_Recurse;
15241eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko}
15258e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
15268e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
15274967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar/* Type declaration testing                                                   */
15284967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar/******************************************************************************/
15294967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
15304967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic enum CXChildVisitResult PrintTypeDeclaration(CXCursor cursor, CXCursor p,
15314967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                                             CXClientData d) {
15324967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXCursor typeDeclaration = clang_getTypeDeclaration(clang_getCursorType(cursor));
15334967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
15344967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (clang_isDeclaration(typeDeclaration.kind)) {
15354967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    PrintCursor(cursor, NULL);
15364967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    PrintTypeAndTypeKind(clang_getCursorType(typeDeclaration), " [typedeclaration=%s] [typekind=%s]\n");
15374967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
15384967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
15394967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  return CXChildVisit_Recurse;
15404967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar}
15414967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
15424967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar/******************************************************************************/
15437d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
15447d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
15457d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1546625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
154798271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1548ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1549f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             PostVisitTU PV,
1550f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             const char *CommentSchemaFile) {
1551e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1552fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1553e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1554e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1555e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1556e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1557e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1558e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1559e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1560e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1561e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1562358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1563358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1564358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1565358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1566358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1567b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1568e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1569e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1570e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1571e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1572e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1573e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1574e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1575e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1576e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1577e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1578e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1579e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1580e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
15816bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    Data.CommentSchemaFile = CommentSchemaFile;
1582e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1583e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1584e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1585ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1586ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1587e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1588a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
158916ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
159016ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
159116ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
159216ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
159316ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
15940d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
15950d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
15960d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
15970d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1598fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1599ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1600ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1601625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1602625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1603020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1604e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
16050a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1606fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                          /* displayDiagnostics=*/1);
1607e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1608020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1609020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1610625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1611020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1612625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1613f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV, NULL);
1614020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1615020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1616625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1617625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1618ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1619ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1620ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1621ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1622ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
1623f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
16244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
16254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
1626651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
16274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
16284967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  unsigned Repeats = 0;
16294967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  unsigned I;
1630651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1631ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1632358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1633358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
1634cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1635ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1636f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if ((CommentSchemaFile = parse_comments_schema(argc, argv))) {
1637f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argc--;
1638f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argv++;
1639f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
1640f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
1641020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1642020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
16434db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1644020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
16454db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
16464967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (getenv("CINDEXTEST_EDITING"))
16474967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    Repeats = 5;
16484967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
1649651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1650651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files,
1651651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files,
1652651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files, num_unsaved_files,
1653651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
1654651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1655ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1656651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1657abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1658020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1659ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1660ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1661ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
16624967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  for (I = 0; I != Repeats; ++I) {
16634967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (checkForErrors(TU) != 0)
16644967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      return -1;
16654967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
16664967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (Repeats > 1) {
16674967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
16684967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                                         clang_defaultReparseOptions(TU));
16694967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      if (Err != CXError_Success) {
16704967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        describeLibclangFailure(Err);
16714967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        free_remapped_files(unsaved_files, num_unsaved_files);
16724967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        clang_disposeIndex(Idx);
16734967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        return 1;
16744967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      }
16754967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
16764967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
16774967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
1678f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV,
1679f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             CommentSchemaFile);
16804db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1681020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
16824db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1683ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1684ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1685abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1686abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1687abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1688abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1689abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1690abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1691abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1692651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int compiler_arg_idx = 0;
1693651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
1694651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int result, i;
1695abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
169640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
169740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1698abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1699abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1700abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1701cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1702abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1703abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1704abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1705abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1706abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1707651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1708651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i < argc; ++i) {
1709651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (strcmp(argv[i], "--") == 0)
1710651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      break;
1711651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
1712651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (i < argc)
1713651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = i+1;
1714651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (num_unsaved_files > compiler_arg_idx)
1715651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = num_unsaved_files;
1716abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1717c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1718c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
1719651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1720651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + compiler_arg_idx,
1721651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - compiler_arg_idx,
1722651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
1723651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1724abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1725651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1726abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1727abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1728abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1729abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1730abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1731bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1732bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1733bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
173440098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
173540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
173640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
173740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
173840098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1739abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
1740651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    free_remapped_files(unsaved_files, num_unsaved_files);
1741651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (parse_remapped_files_with_try(trial, argc, argv, 0,
1742651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                      &unsaved_files, &num_unsaved_files)) {
1743651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeTranslationUnit(TU);
1744651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeIndex(Idx);
1745651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      return -1;
1746651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
1747651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1748651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    Err = clang_reparseTranslationUnit(
1749651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        TU,
1750651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? num_unsaved_files : 0,
1751651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? unsaved_files : 0,
1752651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_defaultReparseOptions(TU));
1753651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Err != CXError_Success) {
1754c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1755651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      describeLibclangFailure(Err);
1756abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1757abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1758abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1759abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1760abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1761bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1762bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1763bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1764abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1765abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1766f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV, NULL);
1767bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1768abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1769abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1770abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1771abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1772abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
17730d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
17741c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
17751c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
17761c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1777dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
17781c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
17791d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
17801d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
17819096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
17821d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
17831d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
178451b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
178551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1786f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
17871c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
17881c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
17891c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
17901d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
17911d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
17921c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
17931c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
17941c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
179510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXCursor prevCursor = clang_getNullCursor();
1796b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
17972389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
17988f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1799e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18000a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
1801fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
18021c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
18031c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
18041c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1805e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18061c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
18071c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1808e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18091c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
18101c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
1811176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines    clang_disposeTranslationUnit(TU);
18121c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
18131c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1814e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1815b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
18162389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
18172389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
18182389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
18192389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
18202389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
18212389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
18222389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
18232389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
18242389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
18252389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
18262389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
18272389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
18282389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
18292389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
18302389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
18312389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1832dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1833d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
18342389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
18352389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1836a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
18372389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
18382389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1839a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
18402389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
18411c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1842e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18431c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
18444f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
18454f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
18461c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
18471c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
18481c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
18491c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
185032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
18510d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
18520d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
18530c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
18540c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
18550c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1856e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1857fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1858fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
185988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1860fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1861fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1862fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1863fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
18640c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1865fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1866fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1867fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1868fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1869fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1870fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
18710c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
18720c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
18730c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1874fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1875fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
187688d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1877fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1878fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1879fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1880e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1881fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1882fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1883fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1884e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1885fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1886fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
188788d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1888fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1889fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1890fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1891fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1892fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1893e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1894fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1895e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1896fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1897fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1898fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
18990c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1900fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1901fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1902fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1903e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1904fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1905fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1906fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1907fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1908fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
190988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1910fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1911fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1912fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
19130c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
19140c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
19150c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
19160c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
19170c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
19180c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
19190c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
19200c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
19210c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
19220c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
19230c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
19240c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
19250c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
19260c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
19270c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
19280c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
19290c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
19300c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
19310c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
19320c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
19330c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1934ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
193501dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
193601dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
193701dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
193801dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
193901dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
19400c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1941e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
19420c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
19430c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
19440c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1945dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1946dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1947dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1948dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1949dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1950dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1951dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1952dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1953dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1954dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1955dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1956dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1957dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1958dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1959dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1960dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1961dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1962dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1963dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1964dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1965dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1966dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1967dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1968dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1969dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
19706bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_string(CXCompletionString completion_string,
19716bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
1972f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1973e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
19743ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
19750c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
19762ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
19772ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
19780c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
19793ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1980e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
19813ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
19823ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
19833ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1984e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
19853ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
19863ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
19873ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
19885a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
19895a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
19905a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
19915a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
19925a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
19933ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1994e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1995d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
19962ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1997e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
19980c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
19992ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
20002ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
20010c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
20022ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
20033ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
20043ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
20056bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_result(CXCompletionResult *completion_result,
20066bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
200710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
20086164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
2009ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
2010ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
2011d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
20124967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXString Annotation;
2013d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
2014ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
2015e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
2016e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
2017e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
20183ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
201958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
202012e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
202158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
202258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
202358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
202458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
202558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
202658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
202758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
202858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
202958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
203058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
203158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
2032d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
2033d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
2034d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
2035d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
203658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
20376164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
20386164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
20396164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
20406164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
20416164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
20426164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
20436164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
20446164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
20456164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
20464967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      Annotation =
20474967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar          clang_getCompletionAnnotation(completion_result->CompletionString, i);
20484967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      fprintf(file, "\"%s\"", clang_getCString(Annotation));
20494967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      clang_disposeString(Annotation);
20506164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
20516164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
20526164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
20536164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
2054ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
2055ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
2056ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
2057ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
205810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
2059ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
2060ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
2061ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
2062ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
2063ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
2064ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
2065ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
2066d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
2067d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
2068d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
2069d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
2070d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
2071d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
2072d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
2073d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
2074ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
207558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
20760c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
20770c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
20783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
20793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
20803da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
20813da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
20823da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20833da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
20843da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
20853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20863da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
20873da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
20883da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20893da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
20903da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
20913da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20923da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
20933da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
20943da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20953da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
20963da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
20973da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20983da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
20993da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
21003da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21013da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
21023da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
21033da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21043da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
21053da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
21063da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21073da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
21083da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
21093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21103da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
21113da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
21123da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21133da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
21143da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
21153da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21163da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
21173da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
21183da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21193da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
21203da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
21213da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
21233da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
21243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21253da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
21263da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
21273da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21283da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
21293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
21303da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21313da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
21323da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
21333da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21343da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
21353da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
21363da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21373da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
21383da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
21393da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21403da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
21413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
21423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21433da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
21443da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
21453da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21463da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
21473da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
21483da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
21493da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
21503da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
21511982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
21520c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
21530c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
21540c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
21550c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
2156f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
2157f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
2158735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2159735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
2160ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
2161651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2162651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXTranslationUnit TU;
216332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
216432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
216532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
216632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
216732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
2168d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
2169d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
2170df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
21711982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
21721982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
21731982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
21741982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
21751982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
2176e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2177fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
2178f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
21790c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2180735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
2181735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
2182735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
218332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
218432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
218532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
218632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
2187651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2188651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, 0,
2189651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
2190651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2,
2191651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
2192651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
219332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
2194651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
219532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
219632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
219708bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
2198651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_reparseTranslationUnit(TU, 0, 0,
2199651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                     clang_defaultReparseOptions(TU));
2200651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2201651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
220287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    fprintf(stderr, "Unable to reparse translation unit!\n");
2203651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2204651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
220508bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
220608bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
2207651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
220832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
220932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
221032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
221132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
221232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
221332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
22142de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
22152de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
221632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
221732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
221832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
2219936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
2220ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
2221e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
22223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
2223e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
22240a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
22250a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
22261e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
22271e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
22281e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
22291e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
22301982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
22311982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
22321e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
2233a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
2234a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
2235a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
2236a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
2237a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
2238a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
22393da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
22403da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
22413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
22423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
22430a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
22440a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
2245e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2246e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
2247e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
2248e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
2249e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
2250e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
2251e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
2252e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2253e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
2254e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
2255e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2256e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
2257e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
2258e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2259e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2260e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
2261e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
2262e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
2263e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
2264e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
22650a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
22660a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
22670a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
22680a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
22690a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
22700a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
22710a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
2272ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
2273ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
2274df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
22750c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
22760c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
2277e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2278735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2279735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
2280f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
22810c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
22820c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2283f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
2284f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
2285f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
2286f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
2287f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
2288f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
22894967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainartypedef void (*cursor_handler_t)(CXCursor cursor);
22904967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
22914967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic int inspect_cursor_at(int argc, const char **argv,
22924967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                             const char *locations_flag,
22934967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                             cursor_handler_t handler) {
2294f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
2295f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
2296f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2297f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
2298651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2299f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
2300f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
2301f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
2302f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
23038e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
2304bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
23058e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2306e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
23074967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  while (strstr(argv[NumLocations+1], locations_flag) == argv[NumLocations+1])
2308f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
2309e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2310f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
2311f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
2312f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
2313f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
2314f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
23154967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    const char *input = argv[Loc + 1] + strlen(locations_flag);
2316e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2317e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
2318fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
2319f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
2320f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2321e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2322e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2323f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
2324f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2325e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
23268e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
23278e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
23288e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
23298e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
23308e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
23318e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
2332651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2333651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argv + num_unsaved_files + 1 + NumLocations,
2334651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argc - num_unsaved_files - 2 - NumLocations,
2335651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   unsaved_files,
2336651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   Repeats > 1? 0 : num_unsaved_files,
2337651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   getDefaultParsingOptions(), &TU);
2338651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2339f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
2340651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2341f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2342f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2343e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2344dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2345dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2346dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2347bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
2348651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2349651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2350651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2351651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2352651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2353651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2354651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2355651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
23568e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2357dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2358dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2359dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
23608e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
23618e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
23628e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
23638e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
23648e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
23658e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
23668e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
23678e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
23688e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
2369dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2370dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2371dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2372dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
23738e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
23744967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        handler(Cursor);
23758e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
23768e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
23778e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2378f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
23798e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2380a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2381f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
2382f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
2383f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
2384f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2385f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
2386f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
2387f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
23884967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic void inspect_print_cursor(CXCursor Cursor) {
23894967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
23904967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXCompletionString completionString = clang_getCursorCompletionString(
23914967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                                                                  Cursor);
23924967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
23934967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXString Spelling;
23944967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  const char *cspell;
23954967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  unsigned line, column;
23964967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
23974967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("%d:%d ", line, column);
23984967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursor(Cursor, NULL);
23994967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursorExtent(Cursor);
24004967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  Spelling = clang_getCursorSpelling(Cursor);
24014967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  cspell = clang_getCString(Spelling);
24024967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (cspell && strlen(cspell) != 0) {
24034967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    unsigned pieceIndex;
24044967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Spelling=%s (", cspell);
24054967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    for (pieceIndex = 0; ; ++pieceIndex) {
24064967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      CXSourceRange range =
24074967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
24084967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      if (clang_Range_isNull(range))
24094967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        break;
24104967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      PrintRange(range, 0);
24114967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24124967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(")");
24134967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
24144967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_disposeString(Spelling);
24154967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
24164967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Selector index=%d",
24174967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar           clang_Cursor_getObjCSelectorIndex(Cursor));
24184967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (clang_Cursor_isDynamicCall(Cursor))
24194967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Dynamic-call");
24204967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (Cursor.kind == CXCursor_ObjCMessageExpr) {
24214967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    CXType T = clang_Cursor_getReceiverType(Cursor);
24224967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    CXString S = clang_getTypeKindSpelling(T.kind);
24234967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Receiver-type=%s", clang_getCString(S));
24244967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    clang_disposeString(S);
24254967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
24264967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
24274967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  {
24284967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    CXModule mod = clang_Cursor_getModule(Cursor);
24294967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    CXFile astFile;
24304967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    CXString name, astFilename;
24314967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    unsigned i, numHeaders;
24324967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    if (mod) {
24334967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      astFile = clang_Module_getASTFile(mod);
24344967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      astFilename = clang_getFileName(astFile);
24354967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      name = clang_Module_getFullName(mod);
24364967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      numHeaders = clang_Module_getNumTopLevelHeaders(TU, mod);
24374967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf(" ModuleName=%s (%s) system=%d Headers(%d):",
24384967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar             clang_getCString(name), clang_getCString(astFilename),
24394967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar             clang_Module_isSystem(mod), numHeaders);
24404967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      clang_disposeString(name);
24414967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      clang_disposeString(astFilename);
24424967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      for (i = 0; i < numHeaders; ++i) {
24434967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        CXFile file = clang_Module_getTopLevelHeader(TU, mod, i);
24444967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        CXString filename = clang_getFileName(file);
24454967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        printf("\n%s", clang_getCString(filename));
24464967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        clang_disposeString(filename);
24474967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      }
24484967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24494967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
24504967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
24514967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (completionString != NULL) {
24524967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf("\nCompletion string: ");
24534967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    print_completion_string(completionString, stdout);
24544967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
24554967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("\n");
24564967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar}
24574967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
24584967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic void display_evaluate_results(CXEvalResult result) {
24594967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  switch (clang_EvalResult_getKind(result)) {
24604967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    case CXEval_Int:
24614967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    {
24624967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      int val = clang_EvalResult_getAsInt(result);
24634967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Kind: Int , Value: %d", val);
24644967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24654967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24664967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    case CXEval_Float:
24674967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    {
24684967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      double val = clang_EvalResult_getAsDouble(result);
24694967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Kind: Float , Value: %f", val);
24704967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24714967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24724967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    case CXEval_ObjCStrLiteral:
24734967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    {
24744967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      const char* str = clang_EvalResult_getAsStr(result);
24754967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Kind: ObjCString , Value: %s", str);
24764967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24774967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24784967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    case CXEval_StrLiteral:
24794967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    {
24804967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      const char* str = clang_EvalResult_getAsStr(result);
24814967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Kind: CString , Value: %s", str);
24824967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24834967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24844967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    case CXEval_CFStr:
24854967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    {
24864967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      const char* str = clang_EvalResult_getAsStr(result);
24874967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Kind: CFString , Value: %s", str);
24884967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24894967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24904967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    default:
24914967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      printf("Unexposed");
24924967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      break;
24934967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
24944967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar}
24954967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
24964967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic void inspect_evaluate_cursor(CXCursor Cursor) {
24974967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
24984967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXString Spelling;
24994967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  const char *cspell;
25004967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  unsigned line, column;
25014967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXEvalResult ER;
25024967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
25034967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
25044967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("%d:%d ", line, column);
25054967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursor(Cursor, NULL);
25064967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursorExtent(Cursor);
25074967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  Spelling = clang_getCursorSpelling(Cursor);
25084967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  cspell = clang_getCString(Spelling);
25094967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (cspell && strlen(cspell) != 0) {
25104967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    unsigned pieceIndex;
25114967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Spelling=%s (", cspell);
25124967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    for (pieceIndex = 0; ; ++pieceIndex) {
25134967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      CXSourceRange range =
25144967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar         clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
25154967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      if (clang_Range_isNull(range))
25164967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        break;
25174967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      PrintRange(range, 0);
25184967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
25194967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(")");
25204967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
25214967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_disposeString(Spelling);
25224967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
25234967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  ER = clang_Cursor_Evaluate(Cursor);
25244967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (!ER) {
25254967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf("Not Evaluatable");
25264967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  } else {
25274967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    display_evaluate_results(ER);
25284967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    clang_EvalResult_dispose(ER);
25294967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
25304967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("\n");
25314967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar}
25324967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
25334967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainarstatic void inspect_macroinfo_cursor(CXCursor Cursor) {
25344967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
25354967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  CXString Spelling;
25364967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  const char *cspell;
25374967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  unsigned line, column;
25384967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
25394967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("%d:%d ", line, column);
25404967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursor(Cursor, NULL);
25414967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  PrintCursorExtent(Cursor);
25424967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  Spelling = clang_getCursorSpelling(Cursor);
25434967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  cspell = clang_getCString(Spelling);
25444967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (cspell && strlen(cspell) != 0) {
25454967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    unsigned pieceIndex;
25464967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(" Spelling=%s (", cspell);
25474967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    for (pieceIndex = 0; ; ++pieceIndex) {
25484967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      CXSourceRange range =
25494967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar         clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
25504967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      if (clang_Range_isNull(range))
25514967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar        break;
25524967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar      PrintRange(range, 0);
25534967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    }
25544967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf(")");
25554967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
25564967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  clang_disposeString(Spelling);
25574967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
25584967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (clang_Cursor_isMacroBuiltin(Cursor)) {
25594967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf("[builtin macro]");
25604967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  } else if (clang_Cursor_isMacroFunctionLike(Cursor)) {
25614967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    printf("[function macro]");
25624967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  }
25634967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  printf("\n");
25644967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar}
25654967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
2566aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
2567aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2568aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
2569aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
2570aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2571f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
2572aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
2573aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
2574aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
2575aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2576aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2577aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
2578aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
2579aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
2580aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2581aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
2582651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2583aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
2584aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
2585aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
2586aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
2587aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
2588aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
2589aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2590aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
2591aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
2592aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
2593aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2594aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
2595aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
2596aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
2597aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
2598aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
2599aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
2600aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2601aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
2602aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
2603aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
2604aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2605aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2606aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2607aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
2608aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2609aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2610aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2611aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
2612aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2613aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2614aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2615aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2616651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2617651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 1 + NumLocations,
2618651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2 - NumLocations,
2619651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
2620651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    Repeats > 1? 0 : num_unsaved_files,
2621651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
2622651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2623aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2624651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2625651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2626aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2627aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2628aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2629dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2630dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2631dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2632aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2633651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2634651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2635651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2636651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2637651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2638651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2639651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2640651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2641aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2642dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2643dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2644dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
2645aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2646aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
2647aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
2648aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
2649aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
2650aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2651aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
2652aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
2653aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
2654dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2655dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2656dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2657dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2658aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
265926fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
2660f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
2661aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
2662aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
2663aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
2664dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2665dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2666dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
2667aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
2668aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2669aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2670aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2671aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
2672aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2673aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
2674aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
2675aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2676aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
2677aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2678aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2679ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic enum CXVisitorResult findFileIncludesVisit(void *context,
2680ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2681ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintCursor(cursor, NULL);
2682ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintRange(range, "");
2683ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  printf("\n");
2684ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return CXVisit_Continue;
2685ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2686ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2687ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic int find_file_includes_in(int argc, const char **argv) {
2688ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXIndex CIdx;
2689ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2690ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  int num_unsaved_files = 0;
2691651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2692ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXTranslationUnit TU;
2693ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  const char **Filenames = 0;
2694ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned NumFilenames = 0;
2695ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned Repeats = 1;
2696ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned I, FI;
2697ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2698ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Count the number of locations. */
2699ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  while (strstr(argv[NumFilenames+1], "-file-includes-in=") == argv[NumFilenames+1])
2700ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    ++NumFilenames;
2701ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2702ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the locations. */
2703ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  assert(NumFilenames > 0 && "Unable to count filenames?");
2704ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  Filenames = (const char **)malloc(NumFilenames * sizeof(const char *));
2705ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I < NumFilenames; ++I) {
2706ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    const char *input = argv[I + 1] + strlen("-file-includes-in=");
2707ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    /* Copy the file name. */
2708ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Filenames[I] = input;
2709ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2710ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2711ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumFilenames + 1, &unsaved_files,
2712ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                           &num_unsaved_files))
2713ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2714ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2715ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2716ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Repeats = 2;
2717ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2718ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2719ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2720ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2721651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
2722651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CIdx, argv[argc - 1],
2723651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argv + num_unsaved_files + 1 + NumFilenames,
2724651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argc - num_unsaved_files - 2 - NumFilenames,
2725651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files,
2726651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Repeats > 1 ? 0 : num_unsaved_files, getDefaultParsingOptions(), &TU);
2727651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2728651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2729ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2730651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2731651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2732ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2733ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2734ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2735ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2736ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2737ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2738ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2739651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2740651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2741651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2742651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2743651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2744651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2745651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2746651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2747ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2748ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2749ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2750ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      return -1;
2751ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2752ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    for (FI = 0; FI < NumFilenames; ++FI) {
2753ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Filenames[FI]);
2754ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (!file)
2755ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        continue;
2756ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2757ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2758ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        return -1;
2759ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2760ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (I + 1 == Repeats) {
2761ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        CXCursorAndRangeVisitor visitor = { 0, findFileIncludesVisit };
2762ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        clang_findIncludesInFile(TU, file, visitor);
2763ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2764ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2765ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis          return -1;
2766ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      }
2767ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2768ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2769ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2770ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintDiagnostics(TU);
2771ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2772ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
27735256c1fd315104f7b1542bb231f16dbad8a8d64bArgyrios Kyrtzidis  free((void *)Filenames);
2774ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2775ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return 0;
2776ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2777ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
277811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis#define MAX_IMPORTED_ASTFILES 200
277911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
278011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidistypedef struct {
278111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  char **filenames;
278211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned num_files;
278311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis} ImportedASTFilesData;
278411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
278511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic ImportedASTFilesData *importedASTs_create() {
278611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *p;
278711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p = malloc(sizeof(ImportedASTFilesData));
278811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames = malloc(MAX_IMPORTED_ASTFILES * sizeof(const char *));
278911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->num_files = 0;
279011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  return p;
279111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
279211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
279311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTs_dispose(ImportedASTFilesData *p) {
279411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
279511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (!p)
279611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return;
279711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
279811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
279911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    free(p->filenames[i]);
280011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p->filenames);
280111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p);
280211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
280311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
280411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTS_insert(ImportedASTFilesData *p, const char *file) {
280511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
280611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p && file);
280711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
280811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    if (strcmp(file, p->filenames[i]) == 0)
280911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis      return;
281011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p->num_files + 1 < MAX_IMPORTED_ASTFILES);
281111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames[p->num_files++] = strdup(file);
281211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
281311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
28146bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinestypedef struct IndexDataStringList_ {
28156bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  struct IndexDataStringList_ *next;
28166bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  char data[1]; /* Dynamically sized. */
28176bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines} IndexDataStringList;
28186bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
28194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
28204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
28214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
2822dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
28236f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
282413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
282511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
28266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *strings;
28276bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXTranslationUnit TU;
28284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
28294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
28306bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void free_client_data(IndexData *index_data) {
28316bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node = index_data->strings;
28326bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  while (node) {
28336bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    IndexDataStringList *next = node->next;
28346bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    free(node);
28356bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    node = next;
28366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
28376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = NULL;
28386bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines}
28396bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
28404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
28414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
28424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
28434e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
28444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
28454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
28464e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
28474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
28484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
28494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2851dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
285210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString filename = clang_getFileName((CXFile)file);
28534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
28544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
28554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
285713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
285813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
28594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
286013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
2861dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
28624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
286313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
28644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
286513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
28664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
28674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
28688003fd613724847bba834ae41aca6f446af1f818Argyrios Kyrtzidis    printf("<invalid>");
28694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
28704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2871c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2872c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2873c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2874c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
28754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
28764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
287713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
287813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
287913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
288013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
288113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
288213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
288313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
28844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
28854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
28864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
28874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
28884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
289013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
289113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
289213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
289313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
289413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
289513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
289613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
289713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
289813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
289913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
29006bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer makeClientContainer(CXClientData *client_data,
29016bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                                const CXIdxEntityInfo *info,
29026ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
29036bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexData *index_data;
29046bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node;
29054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
29064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2907dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
29084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
29094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2910dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
29114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
29124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
29134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
29156bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
29166bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node =
29176bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      (IndexDataStringList *)malloc(sizeof(IndexDataStringList) + strlen(name) +
29186bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    digitCount(line) + digitCount(column) + 2);
29196bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  newStr = node->data;
29204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
29216bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
29226bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  /* Remember string so it can be freed later. */
29236bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data = (IndexData *)client_data;
29246bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node->next = index_data->strings;
29256bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = node;
29266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
29276ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
29284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29302957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
29312957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
29322957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
29333e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
29343e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
29353e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
29363e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
29374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2939dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2940dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2941dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2942dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2943dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2944dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2945dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2946dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2947dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2948dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2949dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2950c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2951c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2952dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2953dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2954dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2955dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2956dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2957dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
29582957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
29592957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
29602957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
29612957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
29622957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
29632957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
29642957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
29652957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
29662957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
296735adca009c65a9578b91ab0bdca13691624e13afDavid Blaikie  case CXIdxEntity_CXXInterface: return "c++-__interface";
29682957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
29692957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
29702957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
29712957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
29722957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
29732957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
29742957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
29752957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
29762957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
29772957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
29782957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
29792957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2980dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
29816ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
29826ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2983dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2984dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2985838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2986838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2987838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2988838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2989838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2990838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2991838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2992838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2993838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2994838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2995838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2996dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2997dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
29986ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
29994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
30004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
3001643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
30024e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
30034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
30044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3005c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
3006c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
3007c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
3008c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
3009c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
3010dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
30114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
30124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
30134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
30142957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
30152957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
3016dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
3017dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
3018c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
3019643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
3020643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
3021643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
3022643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
3023f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
3024643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
30254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
30264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3027b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
3028b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
3029b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
3030b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
3031f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
3032b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
303313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
3034b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
3035b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
3036c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
3037c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
3038c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
3039c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
3040c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
3041c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
3042c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
3043f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(ProtoInfo->protocols[i]->cursor, NULL);
3044c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
304513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
3046c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
3047c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
3048c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
3049c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
30504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
3051996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
30524e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
30534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
3054996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
3055996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
30564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
30574e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
30584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
30594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3060996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
3061996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
3062996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
3063996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
3064996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
3065996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
3066996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
3067996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
3068996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
3069996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
3070996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
3071996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
3072dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
30734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
30744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3075dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
3076dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
3077dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
307862d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
307962d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
3080dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
3081dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
3082dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
308362d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
308413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
308513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
308613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
3087dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
3088dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
3089dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
3090dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
3091dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
30924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
30934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3094dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
30956ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
30964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
30976bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXModule Mod;
30984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
30994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
31004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
310166042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
3102dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
31034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
31044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
310513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
31066bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf(" | isImport: %d | isAngled: %d | isModule: %d",
31078d7a24e94b58676e57fd3f47353cbdbc59917d81Argyrios Kyrtzidis         info->isImport, info->isAngled, info->isModuleImport);
31086bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
31096bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  Mod = clang_getModuleForFile(index_data->TU, (CXFile)info->file);
31106bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (Mod) {
31116bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString str = clang_Module_getFullName(Mod);
31126bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *cstr = clang_getCString(str);
31136bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    printf(" | module: %s", cstr);
31146bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(str);
31156bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
31166bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
31176bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf("\n");
3118dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
3119dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
31204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
31214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31222c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidisstatic CXIdxClientFile index_importedASTFile(CXClientData client_data,
31232c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis                                         const CXIdxImportedASTFileInfo *info) {
31242c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  IndexData *index_data;
31252c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
31262c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCheck(index_data);
31272c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
312811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (index_data->importedASTs) {
312910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString filename = clang_getFileName(info->file);
313011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    importedASTS_insert(index_data->importedASTs, clang_getCString(filename));
313111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    clang_disposeString(filename);
313211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
313311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
31342c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printf("[importedASTFile]: ");
31352c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
3136134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  if (info->module) {
313710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString name = clang_Module_getFullName(info->module);
3138134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | loc: ");
3139134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printCXIndexLoc(info->loc, client_data);
3140134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | name: \"%s\"", clang_getCString(name));
3141134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | isImplicit: %d\n", info->isImplicit);
3142134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    clang_disposeString(name);
3143900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis  } else {
31443c5527e939c454bfd3e16354908e0c4315be5d2fNAKAMURA Takumi    /* PCH file, the rest are not relevant. */
3145900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis    printf("\n");
3146134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  }
31472c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
31482c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
31492c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis}
31502c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
31516bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer
31526bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesindex_startedTranslationUnit(CXClientData client_data, void *reserved) {
31534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
31544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
31554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
31564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
315766042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
3158dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
31594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
31604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31616ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
31622957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
3163dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
31646ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
31656ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
3166c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
3167792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
3168b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
3169b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
3170dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
31714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3172dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
3173dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
3174f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
3175dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
317613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
3177b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
3178b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
3179b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
3180b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
3181dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
3182c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
3183838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (info->flags & CXIdxDeclFlag_Skipped) {
3184838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    assert(!info->isContainer);
3185838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: skipped");
3186838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  } else {
3187838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: %d", info->isContainer);
3188838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  }
3189c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
31904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3191b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
319287adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
3193b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
3194f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
3195b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
3196b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
3197b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
3198dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
3199dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
3200dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
3201dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
3202dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
3203dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
3204dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
3205dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
3206dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
3207dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
3208dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
3209dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
3210dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
3211dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
32124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
32136ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
3214dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
3215dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
321621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
3217f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(CatInfo->classCursor, NULL);
321821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
321913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
3220dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
3221dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
32224e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
32236ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
32246ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
3225b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
32266ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
32276ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
32284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
32294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3230c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
3231c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
32326ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
32334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3234792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
3235792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
3236792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
3237792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
3238792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
3239792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
3240792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
3241792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
3242792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
3243792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
3244792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
3245b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
3246b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
3247b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
3248b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
3249b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
3250b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
3251b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
32522957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
32536bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_index_setClientContainer(
32546bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        info->declAsContainer,
32556bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        makeClientContainer(client_data, info->entityInfo, info->loc));
32564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
32574e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
32584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
32596ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
32606bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printEntityInfo("[indexEntityReference]", client_data,
32616bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                  info->referencedEntity);
32624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
3263f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
32644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
326513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
3266dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
32674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
32684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
3269c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
3270aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
3271aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
3272b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
3273aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
32744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
32754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
32764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
32776f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
32786f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
32796f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
32806f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
32816f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
32826f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
32834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
32846f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
32854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
3286dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
32874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
32882c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_importedASTFile,
32894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
3290dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
32914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
32924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
32934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
329422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
329522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
329622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
329722490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
329822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
329922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
330022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
3301838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_DISABLE_SKIPPARSEDBODIES"))
3302838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SkipParsedBodiesInSession;
330322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
330422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
330522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
330622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
3307d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_args(int num_args, const char **args,
3308d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              CXIndexAction idxAction,
3309d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              ImportedASTFilesData *importedASTs,
3310d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              const char *check_prefix) {
3311d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3312d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3313d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3314d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3315d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (num_args == 0) {
3316d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
3317d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3318d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3319d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3320d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3321d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3322d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3323d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3324d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3325d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
33266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
33276bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = NULL;
3328d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3329d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3330d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
3331d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
3332d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 0, args, num_args, 0, 0, 0,
3333d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 getDefaultParsingOptions());
3334651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (result != CXError_Success)
3335651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(result);
3336651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3337d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3338d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3339d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
33406bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3341d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3342d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3343d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3344d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_ast_file(const char *ast_file,
3345d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndex Idx,
3346d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndexAction idxAction,
3347d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          ImportedASTFilesData *importedASTs,
3348d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          const char *check_prefix) {
3349d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXTranslationUnit TU;
3350d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3351d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3352d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3353d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3354d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, ast_file, &TU))
3355d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3356d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3357d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3358d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3359d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3360d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3361d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3362d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
33636bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
33646bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = TU;
3365d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3366d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3367d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
3368d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
3369d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      index_opts, TU);
3370d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3371d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3372d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3373d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
33746bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3375d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3376d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3377d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
337811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic int index_file(int argc, const char **argv, int full) {
33794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
33802957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
33812957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
3382d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
3383dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
33844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
33854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
33864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
33874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
33884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
33894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
33904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
33914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
33924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
33934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
33942957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3395fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
33962957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
33972957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
33982957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
3399d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3400d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs = 0;
340111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full)
3402d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    importedASTs = importedASTs_create();
3403d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3404d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_compile_args(argc, argv, idxAction, importedASTs, check_prefix);
3405d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (result != 0)
3406d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    goto finished;
3407dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
340811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full) {
340911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    unsigned i;
3410d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    for (i = 0; i < importedASTs->num_files && result == 0; ++i) {
3411d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      result = index_ast_file(importedASTs->filenames[i], Idx, idxAction,
3412d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              importedASTs, check_prefix);
341311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    }
341411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
341521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
341611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisfinished:
3417d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs_dispose(importedASTs);
34182957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
34192957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
342021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
342121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
342221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
342321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
3424d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
342521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
34262957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
342721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
342821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
342921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
343021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
343121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
343221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
343321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
343421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
343521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
343621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
343721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3438d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3439fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
3440d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
3441d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return 1;
3442d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3443d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3444d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3445d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_ast_file(argv[0], Idx, idxAction,
3446d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          /*importedASTs=*/0, check_prefix);
3447d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3448d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
3449d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3450d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3451d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3452d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3453d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_db(int argc, const char **argv) {
3454d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
3455d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndex Idx;
3456d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndexAction idxAction;
3457d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int errorCode = 0;
3458d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3459d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  check_prefix = 0;
3460d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 0) {
3461d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
3462d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
3463d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      ++argv;
3464d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      --argc;
3465d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3466d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3467d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
346821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
3469d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compilation database\n");
347021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
347121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
347221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
347321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3474fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
347521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
347621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
347721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
3478d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
347921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3480d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  {
3481d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *database = argv[0];
3482d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase db = 0;
3483d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommands CCmds = 0;
3484d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommand CCmd;
3485d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase_Error ec;
3486d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString wd;
3487d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#define MAX_COMPILE_ARGS 512
3488d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString cxargs[MAX_COMPILE_ARGS];
3489d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *args[MAX_COMPILE_ARGS];
3490d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *tmp;
3491d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    unsigned len;
3492d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *buildDir;
3493d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    int i, a, numCmds, numArgs;
3494d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3495d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    len = strlen(database);
3496d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    tmp = (char *) malloc(len+1);
3497d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    memcpy(tmp, database, len+1);
3498d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    buildDir = dirname(tmp);
3499d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3500d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3501d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3502d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (db) {
3503d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3504d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (ec!=CXCompilationDatabase_NoError) {
3505d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("unexpected error %d code while loading compilation database\n", ec);
3506d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3507d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3508d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
350921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
35102bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      if (chdir(buildDir) != 0) {
35112bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        printf("Could not chdir to %s\n", buildDir);
35122bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        errorCode = -1;
35132bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        goto cdb_end;
35142bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      }
351521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
35162bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      CCmds = clang_CompilationDatabase_getAllCompileCommands(db);
3517d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (!CCmds) {
3518d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("compilation db is empty\n");
3519d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3520d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3521d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3522d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3523d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      numCmds = clang_CompileCommands_getSize(CCmds);
3524d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3525d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (numCmds==0) {
3526d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        fprintf(stderr, "should not get an empty compileCommand set\n");
3527d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3528d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3529d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3530d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3531d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      for (i=0; i<numCmds && errorCode == 0; ++i) {
3532d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        CCmd = clang_CompileCommands_getCommand(CCmds, i);
3533d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3534d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        wd = clang_CompileCommand_getDirectory(CCmd);
35352bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        if (chdir(clang_getCString(wd)) != 0) {
35362bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          printf("Could not chdir to %s\n", clang_getCString(wd));
35372bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          errorCode = -1;
35382bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          goto cdb_end;
35392bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        }
3540d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        clang_disposeString(wd);
3541d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3542d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        numArgs = clang_CompileCommand_getNumArgs(CCmd);
3543d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        if (numArgs > MAX_COMPILE_ARGS){
3544d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          fprintf(stderr, "got more compile arguments than maximum\n");
3545d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          errorCode = -1;
3546d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          goto cdb_end;
3547d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3548d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a) {
3549d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          cxargs[a] = clang_CompileCommand_getArg(CCmd, a);
3550d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          args[a] = clang_getCString(cxargs[a]);
3551d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3552d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3553d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = index_compile_args(numArgs, args, idxAction,
3554d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                       /*importedASTs=*/0, check_prefix);
3555d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3556d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a)
3557d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          clang_disposeString(cxargs[a]);
3558d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3559d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    } else {
3560d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      printf("database loading failed with error code %d.\n", ec);
3561d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      errorCode = -1;
3562d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3563d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3564d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  cdb_end:
3565d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompileCommands_dispose(CCmds);
3566d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompilationDatabase_dispose(db);
3567d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    free(tmp);
3568d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3569d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
35704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
35712957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
35722957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3573d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return errorCode;
35744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
35754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3576fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
3577fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
3578fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
3579fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
3580fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
3581fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
3582fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
3583fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
3584fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
3585fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
3586fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
3587fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
3588fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
3589fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
3590fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
3591fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
3592651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXSourceRangeList *skipped_ranges = 0;
3593651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
3594fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
3595fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3596fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
3597fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
3598fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
3599fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
3600fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3601e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
3602e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
3603fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3604e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3605fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
36060a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
3607651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
3608651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
3609651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 3,
3610651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
3611651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    num_unsaved_files,
3612651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
3613651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
3614fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
3615651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
3616fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
3617fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
3618fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
3619fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3620e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
3621fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
3622fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3623e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
3624e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
3625e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
3626e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3627dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3628ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
3629ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
3630651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
3631651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
3632651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
3633ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
3634651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
3635ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
3636ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
3637ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
3638ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
3639ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3640ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
3641dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3642dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3643dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3644dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3645dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3646fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
3647fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
3648fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
3649fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3650fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
3651fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3652fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3653fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
3654fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
3655e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
3656fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
3657fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3658e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3659fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3660fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3661fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
3662fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
3663e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
3664fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
3665fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3666e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3667fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3668fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3669fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
3670fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
3671dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3672dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3673dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3674dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3675dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3676dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3677fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
3678fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
3679dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3680dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3681dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3682dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3683dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3684dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3685651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  skipped_ranges = clang_getSkippedRanges(TU, file);
3686651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i != skipped_ranges->count; ++i) {
3687651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    unsigned start_line, start_column, end_line, end_column;
3688651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeStart(skipped_ranges->ranges[i]),
3689651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &start_line, &start_column, 0);
3690651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeEnd(skipped_ranges->ranges[i]),
3691651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &end_line, &end_column, 0);
3692651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("Skipping: ");
3693651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    PrintExtent(stdout, start_line, start_column, end_line, end_column);
3694651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("\n");
3695651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
3696651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  clang_disposeSourceRangeList(skipped_ranges);
3697651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3698fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
3699fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
370010f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
370110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
3702fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
3703fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3704fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
3705fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
3706fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
3707fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
3708fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
3709fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
3710fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
3711a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
3712a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
3713a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
3714a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
371551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
3716342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
371751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
37180045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
37190045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
3720f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      PrintCursor(cursors[i], NULL);
37210045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
37220045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
3723fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3724fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
372593f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
3726fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3727fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
3728a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
3729fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
3730fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
3731fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
3732fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
3733fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
3734fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
3735fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3736db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
3737db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
3738db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
3739db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
3740db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
3741db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
3742db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
3743db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
3744db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
3745db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
3746db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
3747db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
3748db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
3749db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3750db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
3751db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
3752db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
3753db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
3754db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3755c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3756db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3757db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
3758db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3759db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
3760db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
3761db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
3762db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
3763db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3764db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3765db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
3766db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
3767c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
3768db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3769db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
3770db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
3771db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3772db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3773db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3774db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3775c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
3776db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3777db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
3778db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
3779db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
3780db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3781db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3782db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3783db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3784db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
3785c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
3786db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3787c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
3788db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
3789db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
3790db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3791db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
3792c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
3793db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
3794db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
3795c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
3796db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
3797db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
3798db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
3799db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
3800db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3801db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3802c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
3803db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3804db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
3805db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
3806db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3807c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
3808db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
3809db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
3810db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
3811db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
3812db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3813db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
3814db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
3815db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3816db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
3817db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
3818db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
38190d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3820f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
3821f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
3822f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3823f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
3824f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
3825f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3826f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3827f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3828f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
3829f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
3830f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3831f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3832f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
3833f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
3834f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3835f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3836f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3837f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
3838f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
3839f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
3840f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
3841f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3842f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3843f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
3844f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
3845f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
3846f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
3847f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
3848f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
3849f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
3850f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
3851f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
3852f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3853f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3854f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
3855f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
3856f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
3857f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
3858f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
3859f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
3860f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
3861f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3862f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
3863f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3864f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3865f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3866f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3867a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3868ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3869f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
3870f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3871f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3872f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3873f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3874f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3875f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3876f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
3877f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
3878f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3879f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
3880f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
3881f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3882f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3883f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3884f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3885f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
3886f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
3887f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
3888f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
3889f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
3890f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
3891f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
3892f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3893f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3894a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
3895ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3896f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
3897f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3898f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
3899f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3900f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3901f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3902f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
3903f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
3904f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3905f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
3906f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
3907f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3908f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3909f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3910f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
3911f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3912f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
3913f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
3914f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3915f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3916f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3917f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
3918f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3919f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
3920f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3921f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3922f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3923f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3924a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3925ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3926f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
3927f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3928f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3929f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3930f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3931f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3932f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
3933f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3934f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3935f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
3936f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3937f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3938f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
3939f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
3940f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
3941f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3942f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3943f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3944f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3945f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3946f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
3947f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
3948f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
3949f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
3950f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3951f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
3952f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
3953f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
3954f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3955f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3956f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3957f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
3958f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
3959f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
3960f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
3961f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
3962f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
3963f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3964f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
3965f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3966f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3967f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
3968f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
3969f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3970f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
3971f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3972f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3973f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
3974f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
3975f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
3976f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
3977f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
3978f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
3979f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
3980f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
3981f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3982f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
3983f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
3984f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3985f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3986f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
3987f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3988f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3989f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3990f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
39910d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
39920d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
39937ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
39947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
39957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
39967ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
39977ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
3998651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
399908aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
40007ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
4001fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnostics=*/1);
40027ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
40037ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
40047ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
40057ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
40067ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
4007651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
4008651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
4009651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Idx, 0, argv + num_unsaved_files, argc - num_unsaved_files,
4010651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files, num_unsaved_files,
4011651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CXTranslationUnit_Incomplete |
4012651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_DetailedPreprocessingRecord |
4013651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_ForSerialization,
4014651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &TU);
4015651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
40167ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
4017651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
40187ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
4019651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
40207ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
40217ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
40227ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
40237ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
402439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
402539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
402639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
402739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
402839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
402939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
403039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
403139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
403239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
403339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
403439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
403539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
403639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
403739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
403839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
403939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
404039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
404139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
404239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
404339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
404439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
404539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
404639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
404739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
40487ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
40497ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
40507ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
405139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
40527ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
40537ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
40547ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
4055153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
4056153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
4057153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4058153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
4059153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
4060153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
4061153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
4062153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
4063153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
4064153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4065153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
4066153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4067153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4068153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
4069153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
4070153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
4071153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
4072153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
4073153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
4074153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
4075153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4076153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
4077153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4078153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4079153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
4080a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
4081a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
4082a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
4083a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
4084153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
4085a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
4086153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
4087153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4088153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4089153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4090153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
4091153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
4092153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
4093153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
4094153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4095153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
4096153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
4097153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4098153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
4099153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
4100153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4101153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4102153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
4103153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
4104153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4105153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
4106153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
410710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange SR = clang_getDiagnosticRange(D, i);
4108153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
4109153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
4110153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4111153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
4112153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
4113153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
4114153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
4115153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
4116153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
4117153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4118153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4119153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4120153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
4121153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
41223739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
4123153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
4124153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
4125153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
4126153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
4127153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4128153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
4129153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
4130153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
4131153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
4132153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
4133153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
4134153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
4135153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4136153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4137153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4138153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
41399190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
41409190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
4141153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
4142153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
4143153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
41449190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
4145153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
4146153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
4147153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
4148153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
414978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
4150153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
415178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
4152153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4153153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
4154153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
4155153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
4156153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
4157153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
4158153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4159153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
4160153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4161153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
4162153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
4163153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
4164153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
4165153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
4166153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
4167153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4168153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
4169153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
4170153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
4171153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
4172153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
4173153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
417478d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
417578d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
417678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
417778d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
417878d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
417978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
4180153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
4181153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4182153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
4183153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
4184153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4185a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
4186153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
4187153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4188153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
4189153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
4190153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
41916bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(DiagCat);
4192153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4193153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4194153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4195153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
4196153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
4197153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
4198153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
4199153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4200153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
4201153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
4202153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
4203153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
4204153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
4205153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
4206153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
4207153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
4208153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4209153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
4210a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
4211a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
4212153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
4213153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
4214153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4215153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4216651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int perform_print_build_session_timestamp(void) {
4217651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  printf("%lld\n", clang_getBuildSessionTimestamp());
4218651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return 0;
4219651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
4220651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
4221153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
42227ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
42237ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
4224f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
4225e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
42267d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
4227e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
42287d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
42297d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
423059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
423159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
42327d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
42337d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
42347d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
4235f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
4236f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
42370d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
42381982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
4239f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
42404967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    "       c-index-test -evaluate-cursor-at=<site> <compiler arguments>\n"
42414967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    "       c-index-test -get-macro-info-cursor-at=<site> <compiler arguments>\n"
4242ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-refs-at=<site> <compiler arguments>\n"
4243ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-includes-in=<filename> <compiler arguments>\n");
42443584972866f39d42aa4029586278022d89451bd9NAKAMURA Takumi  fprintf(stderr,
42454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
424611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    "       c-index-test -index-file-full [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
424721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
4248d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    "       c-index-test -index-compile-db [-check-prefix=<FileCheck prefix>] <compilation database>\n"
42491d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
425026fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
425126fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
4252fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
4253fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
42547d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
42557d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
42561982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
4257f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
425859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
425959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
4260abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
4261abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
42621982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
426359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
426459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
426516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
426616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
42674e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
426853513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
42694e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
427087d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    "       c-index-test -test-print-visibility {<args>}*\n"
4271ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    "       c-index-test -test-print-type {<args>}*\n"
4272411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    "       c-index-test -test-print-type-size {<args>}*\n"
42731eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    "       c-index-test -test-print-bitwidth {<args>}*\n"
42744967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    "       c-index-test -test-print-type-declaration {<args>}*\n"
427559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
42767ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
4277153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
4278153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
4279db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
4280db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
4281651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    "       c-index-test -print-build-session-timestamp\n");
4282651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  fprintf(stderr,
4283153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
4284caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
42857d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
42860d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
42870d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
42880d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
42890d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
42900d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
42910d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
4292625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
4293625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
4294f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
4295f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
42966edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
42976edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
42986edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
42990a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
4300153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
4301153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
4302f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
43031982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
43041982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
43051982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
4306f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
43074967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    return inspect_cursor_at(argc, argv, "-cursor-at=", inspect_print_cursor);
43084967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (argc > 2 && strstr(argv[1], "-evaluate-cursor-at=") == argv[1])
43094967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    return inspect_cursor_at(argc, argv, "-evaluate-cursor-at=",
43104967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                             inspect_evaluate_cursor);
43114967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (argc > 2 && strstr(argv[1], "-get-macro-info-cursor-at=") == argv[1])
43124967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    return inspect_cursor_at(argc, argv, "-get-macro-info-cursor-at=",
43134967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                             inspect_macroinfo_cursor);
4314aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
4315aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
4316ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-includes-in=") == argv[1])
4317ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return find_file_includes_in(argc, argv);
43184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
431911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/0);
432011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file-full") == 0)
432111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/1);
432221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
432321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
4324d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-compile-db") == 0)
4325d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return index_compile_db(argc - 2, argv + 2);
43267d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
4327e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
43287d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
4329ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
4330ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
43317d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
4332abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
4333abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
4334abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
4335abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
4336abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
4337abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
4338abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
4339abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
43407d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
4341e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
434259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
434359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
434459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
434559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
434659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
43477d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
434859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
434959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
43507d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
43517d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
43521d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
43531d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
4354fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
4355fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
435616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
435716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
435816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
435916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
436016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
436116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
43623bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
43633bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
43643bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
436587d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  else if (argc > 2 && strcmp(argv[1], "-test-print-visibility") == 0)
436687d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return perform_test_load_source(argc - 2, argv + 2, "all", PrintVisibility,
436787d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar                                    NULL);
4368ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-type") == 0)
43698e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
4370ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                    PrintType, 0);
4371411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  else if (argc > 2 && strcmp(argv[1], "-test-print-type-size") == 0)
4372411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return perform_test_load_source(argc - 2, argv + 2, "all",
4373411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                    PrintTypeSize, 0);
43744967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  else if (argc > 2 && strcmp(argv[1], "-test-print-type-declaration") == 0)
43754967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    return perform_test_load_source(argc - 2, argv + 2, "all",
43764967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                                    PrintTypeDeclaration, 0);
43771eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-bitwidth") == 0)
43781eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return perform_test_load_source(argc - 2, argv + 2, "all",
43791eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                    PrintBitWidth, 0);
4380176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  else if (argc > 2 && strcmp(argv[1], "-test-print-mangle") == 0)
4381176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines    return perform_test_load_tu(argv[2], "all", NULL, PrintMangledName, NULL);
438287d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar  else if (argc > 2 && strcmp(argv[1], "-test-print-manglings") == 0)
438387d948ecccffea9e9e37d0d053b246e2d6d6c47bPirama Arumuga Nainar    return perform_test_load_tu(argv[2], "all", NULL, PrintManglings, NULL);
4384f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
4385f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
4386f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
4387f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
4388f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
4389f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
4390f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
4391f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
4392f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
4393f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
43947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
43957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
4396db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
4397db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
4398651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  else if (argc == 2 && strcmp(argv[1], "-print-build-session-timestamp") == 0)
4399651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return perform_print_build_session_timestamp();
4400db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
4401f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
440250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
440350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
44046edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
44056edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
44066edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
44076edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
44086edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
44096edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
44106edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
44116edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
44124967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  int (*main_func)(int argc, const char **argv);
44136edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
44146edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
44156edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
44166edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
441784294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
44186edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
44194967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  client_data->result = client_data->main_func(client_data->argc,
44204967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar                                               client_data->argv);
4421c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines}
4422c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines
4423c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hinesstatic void flush_atexit(void) {
4424c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  /* stdout, and surprisingly even stderr, are not always flushed on process
4425c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines   * and thread exit, particularly when the system is under heavy load. */
4426c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  fflush(stdout);
4427c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  fflush(stderr);
44286edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
44296edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
44306edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
4431d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer  thread_info client_data;
4432d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer
4433c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  atexit(flush_atexit);
4434c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines
4435f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
4436f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  LIBXML_TEST_VERSION
4437f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
4438f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
44394967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  client_data.main_func = cindextest_main;
44406edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
44416edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
44424967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
44434967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (argc > 1 && strcmp(argv[1], "core") == 0)
44444967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    client_data.main_func = indextest_core_main;
44454967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
44464967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar  if (getenv("CINDEXTEST_NOTHREADS"))
44474967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar    return client_data.main_func(client_data.argc, client_data.argv);
44484967a710c84587c654b56c828382219c3937dacbPirama Arumuga Nainar
4449a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
44506edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
44516edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
4452