c-index-test.c revision e4330a302ac20b41b9800267ebd4b5b01f8553f8
12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang-c/Index.h"
4db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison#include "clang-c/CXCompilationDatabase.h"
5f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include "llvm/Config/config.h"
61e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor#include <ctype.h>
70c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor#include <stdlib.h>
889922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff#include <stdio.h>
9af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff#include <string.h>
10f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor#include <assert.h>
11af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff
12f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
13f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/parser.h>
14f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/relaxng.h>
15f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/xmlerror.h>
16f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
17f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
180d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
190d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Utility functions.                                                         */
200d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
210d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
222e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#ifdef _MSC_VER
232e06fc877a633abea3b40a64950c7316dac29ca8John Thompsonchar *basename(const char* path)
242e06fc877a633abea3b40a64950c7316dac29ca8John Thompson{
252e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base1 = (char*)strrchr(path, '/');
262e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base2 = (char*)strrchr(path, '\\');
272e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    if (base1 && base2)
282e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return((base1 > base2) ? base1 + 1 : base2 + 1);
292e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base1)
302e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base1 + 1);
312e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base2)
322e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base2 + 1);
332e06fc877a633abea3b40a64950c7316dac29ca8John Thompson
342e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    return((char*)path);
352e06fc877a633abea3b40a64950c7316dac29ca8John Thompson}
36db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonchar *dirname(char* path)
37db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison{
38db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base1 = (char*)strrchr(path, '/');
39db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base2 = (char*)strrchr(path, '\\');
40db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (base1 && base2)
41db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (base1 > base2)
42db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base1 = 0;
43db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        else
44db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base2 = 0;
45db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base1)
460fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base1 = 0;
47db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base2)
480fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base2 = 0;
49db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
50db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return path;
51db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
522e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#else
53ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroffextern char *basename(const char *);
54db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonextern char *dirname(char *);
552e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#endif
56ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroff
5745ba9a1b31110568d0d362c8d31c6133cf9011b7Douglas Gregor/** \brief Return the default parsing options. */
5844c181aec37789f25f6c15543c164416f72e562aDouglas Gregorstatic unsigned getDefaultParsingOptions() {
5944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  unsigned options = CXTranslationUnit_DetailedPreprocessingRecord;
6044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
6144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (getenv("CINDEXTEST_EDITING"))
62b1c031be513705d924038f497279b9b599868ba1Douglas Gregor    options |= clang_defaultEditingTranslationUnitOptions();
6387c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  if (getenv("CINDEXTEST_COMPLETION_CACHING"))
6487c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    options |= CXTranslationUnit_CacheCompletionResults;
65dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis  if (getenv("CINDEXTEST_COMPLETION_NO_CACHING"))
66dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis    options &= ~CXTranslationUnit_CacheCompletionResults;
676a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen  if (getenv("CINDEXTEST_SKIP_FUNCTION_BODIES"))
686a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen    options |= CXTranslationUnit_SkipFunctionBodies;
69d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
70d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    options |= CXTranslationUnit_IncludeBriefCommentsInCodeCompletion;
7144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
7244c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  return options;
7344c181aec37789f25f6c15543c164416f72e562aDouglas Gregor}
7444c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
75bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU);
76bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
7751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbarstatic void PrintExtent(FILE *out, unsigned begin_line, unsigned begin_column,
7851b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar                        unsigned end_line, unsigned end_column) {
7951b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  fprintf(out, "[%d:%d - %d:%d]", begin_line, begin_column,
80d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar          end_line, end_column);
8151b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar}
8251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar
831c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenekstatic unsigned CreateTranslationUnit(CXIndex Idx, const char *file,
841c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                      CXTranslationUnit *TU) {
85e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
86a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  *TU = clang_createTranslationUnit(Idx, file);
876be2a22979e6a4af575af32cccdf6a6190a8edcaDan Gohman  if (!*TU) {
881c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Unable to load translation unit from '%s'!\n", file);
891c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 0;
90e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
911c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 1;
921c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
931c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
944db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorvoid free_remapped_files(struct CXUnsavedFile *unsaved_files,
954db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int num_unsaved_files) {
964db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
974db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (i = 0; i != num_unsaved_files; ++i) {
984db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Filename);
994db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Contents);
1004db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
101653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor  free(unsaved_files);
1024db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1034db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1044db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorint parse_remapped_files(int argc, const char **argv, int start_arg,
1054db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         struct CXUnsavedFile **unsaved_files,
1064db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int *num_unsaved_files) {
1074db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1084db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int arg;
1094db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int prefix_len = strlen("-remap-file=");
1104db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files = 0;
1114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *num_unsaved_files = 0;
112e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1134db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  /* Count the number of remapped files. */
1144db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg; arg < argc; ++arg) {
1154db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (strncmp(argv[arg], "-remap-file=", prefix_len))
1164db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      break;
117e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1184db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    ++*num_unsaved_files;
1194db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
120e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  if (*num_unsaved_files == 0)
1224db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return 0;
123e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files
125653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor    = (struct CXUnsavedFile *)malloc(sizeof(struct CXUnsavedFile) *
126653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor                                     *num_unsaved_files);
1274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg, i = 0; i != *num_unsaved_files; ++i, ++arg) {
1284db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    struct CXUnsavedFile *unsaved = *unsaved_files + i;
1294db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *arg_string = argv[arg] + prefix_len;
1304db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    int filename_len;
1314db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *filename;
1324db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *contents;
1334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    FILE *to_file;
1344db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *semi = strchr(arg_string, ';');
1354db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!semi) {
136e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr,
1374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              "error: -remap-file=from;to argument is missing semicolon\n");
1384db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1394db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1404db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1414db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1424db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
143e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1444db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Open the file that we're remapping to. */
145c44fe4bf2962bfcb323e93a63ba4a72d67839cdeFrancois Pichet    to_file = fopen(semi + 1, "rb");
1464db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!to_file) {
1474db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: cannot open file %s that we are remapping to\n",
1484db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              semi + 1);
1494db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1514db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1524db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1534db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
154e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1554db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Determine the length of the file we're remapping to. */
1564db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_END);
1574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Length = ftell(to_file);
1584db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_SET);
159e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1604db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Read the contents of the file we're remapping to. */
1614db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents = (char *)malloc(unsaved->Length + 1);
1624db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (fread(contents, 1, unsaved->Length, to_file) != unsaved->Length) {
1634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: unexpected %s reading 'to' file %s\n",
1644db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              (feof(to_file) ? "EOF" : "error"), semi + 1);
1654db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fclose(to_file);
1664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
167e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith      free(contents);
1684db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1704db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1714db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
1724db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents[unsaved->Length] = 0;
1734db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Contents = contents;
174e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1754db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Close the file. */
1764db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fclose(to_file);
177e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1784db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Copy the file name that we're remapping from. */
1794db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename_len = semi - arg_string;
1804db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename = (char *)malloc(filename_len + 1);
1814db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    memcpy(filename, arg_string, filename_len);
1824db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename[filename_len] = 0;
1834db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Filename = filename;
1844db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
185e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1864db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return 0;
1874db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1884db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
189f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic const char *parse_comments_schema(int argc, const char **argv) {
190f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentsSchemaArg = "-comments-xml-schema=";
191f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile = NULL;
192f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
193f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (argc == 0)
194f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return CommentSchemaFile;
195f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
196f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!strncmp(argv[0], CommentsSchemaArg, strlen(CommentsSchemaArg)))
197f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    CommentSchemaFile = argv[0] + strlen(CommentsSchemaArg);
198f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
199f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  return CommentSchemaFile;
200f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
201f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
2020d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2030d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Pretty-printing.                                                           */
2040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2050d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
206ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic const char *FileCheckPrefix = "CHECK";
207ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
208ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCString(const char *CStr) {
2092d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  if (CStr != NULL && CStr[0] != '\0') {
2102d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    for ( ; *CStr; ++CStr) {
2112d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      const char C = *CStr;
2122d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      switch (C) {
2132d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\n': printf("\\n"); break;
2142d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\r': printf("\\r"); break;
2152d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\t': printf("\\t"); break;
2162d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\v': printf("\\v"); break;
2172d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\f': printf("\\f"); break;
2182d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        default:   putchar(C);    break;
2192d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      }
2202d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    }
2212d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  }
222ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
223ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
224ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCStringWithPrefix(const char *Prefix, const char *CStr) {
225ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(" %s=[", Prefix);
226ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(CStr);
2272d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  printf("]");
2282d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko}
2292d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko
230ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringAndDispose(CXString Str) {
231ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(clang_getCString(Str));
232ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
233ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
234ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
235f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCXStringWithPrefix(const char *Prefix, CXString Str) {
236f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
237f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
238f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
239ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringWithPrefixAndDispose(const char *Prefix,
240ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                              CXString Str) {
241ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
242ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
243ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
244ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
245430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregorstatic void PrintRange(CXSourceRange R, const char *str) {
246430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  CXFile begin_file, end_file;
247430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  unsigned begin_line, begin_column, end_line, end_column;
248430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
249430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeStart(R),
250430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &begin_file, &begin_line, &begin_column, 0);
251430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeEnd(R),
252430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &end_file, &end_line, &end_column, 0);
253430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (!begin_file || !end_file)
254430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    return;
255430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
256ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis  if (str)
257ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis    printf(" %s=", str);
258430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintExtent(stdout, begin_line, begin_column, end_line, end_column);
259430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
260430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
261358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregorint want_display_name = 0;
262358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
263cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregorstatic void printVersion(const char *Prefix, CXVersion Version) {
264cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Major < 0)
265cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
266cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf("%s%d", Prefix, Version.Major);
267cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
268cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Minor < 0)
269cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
270cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Minor);
271cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
272cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Subminor < 0)
273cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
274cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Subminor);
275cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor}
276cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
277ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostruct CommentASTDumpingContext {
278ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  int IndentLevel;
279ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko};
280ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
281ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXCommentInternal(struct CommentASTDumpingContext *Ctx,
282ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                  CXComment Comment) {
2835ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned i;
2845ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned e;
2855ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  enum CXCommentKind Kind = clang_Comment_getKind(Comment);
2865ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko
287ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel++;
2885ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  for (i = 0, e = Ctx->IndentLevel; i != e; ++i)
289ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("  ");
290ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
291ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("(");
292ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  switch (Kind) {
293ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Null:
294ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Null");
295ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
296ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Text:
297ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Text");
298ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose("Text",
299ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                      clang_TextComment_getText(Comment));
300ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
301ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
302ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
303ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
304ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
305ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_InlineCommand:
306ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_InlineCommand");
307ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
308ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
309ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_InlineCommandComment_getCommandName(Comment));
3102d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    switch (clang_InlineCommandComment_getRenderKind(Comment)) {
3112d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Normal:
3122d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderNormal");
3132d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3142d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Bold:
3152d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderBold");
3162d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3172d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Monospaced:
3182d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderMonospaced");
3192d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3202d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Emphasized:
3212d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderEmphasized");
3222d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3232d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    }
3245ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_InlineCommandComment_getNumArgs(Comment);
325ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
326ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
327ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
328ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_InlineCommandComment_getArgText(Comment, i));
329ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
330ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
331ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
332ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3335ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  case CXComment_HTMLStartTag: {
3345ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned NumAttrs;
335ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLStartTag");
336ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
337ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
338ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
3395ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    NumAttrs = clang_HTMLStartTag_getNumAttrs(Comment);
340ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (NumAttrs != 0) {
341ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Attrs:");
3425ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko      for (i = 0; i != NumAttrs; ++i) {
343ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf(" ");
344ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrName(Comment, i));
345ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf("=");
346ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrValue(Comment, i));
347ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      }
348ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
349ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_HTMLStartTagComment_isSelfClosing(Comment))
350ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" SelfClosing");
351ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
352ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
353ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3545ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  }
355ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_HTMLEndTag:
356ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLEndTag");
357ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
358ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
359ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
360ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
361ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
362ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
363ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Paragraph:
364ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Paragraph");
365ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
366ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
367ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
368ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_BlockCommand:
369ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_BlockCommand");
370ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
371ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
372ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
3735ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_BlockCommandComment_getNumArgs(Comment);
374ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
375ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
376ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
377ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_BlockCommandComment_getArgText(Comment, i));
378ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
379ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
380ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_ParamCommand:
381ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_ParamCommand");
382ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    switch (clang_ParamCommandComment_getDirection(Comment)) {
383ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_In:
384ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in");
385ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
386ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_Out:
387ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" out");
388ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
389ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_InOut:
390ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in,out");
391ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
392ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
393ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isDirectionExplicit(Comment))
394ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" explicitly");
395ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
396ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" implicitly");
397ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
398ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "ParamName",
399ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_ParamCommandComment_getParamName(Comment));
400ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isParamIndexValid(Comment))
401ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=%u", clang_ParamCommandComment_getParamIndex(Comment));
402ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
403ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=Invalid");
404ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
40596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko  case CXComment_TParamCommand:
40696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    printf("CXComment_TParamCommand");
40796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    PrintCXStringWithPrefixAndDispose(
40896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        "ParamName",
40996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        clang_TParamCommandComment_getParamName(Comment));
41096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    if (clang_TParamCommandComment_isParamPositionValid(Comment)) {
41196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition={");
41296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      for (i = 0, e = clang_TParamCommandComment_getDepth(Comment);
41396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko           i != e; ++i) {
41496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        printf("%u", clang_TParamCommandComment_getIndex(Comment, i));
41596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        if (i != e - 1)
41696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko          printf(", ");
41796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      }
41896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf("}");
41996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    } else
42096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition=Invalid");
42196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    break;
422ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockCommand:
423ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockCommand");
424ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
425ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
426ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
427ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
428ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockLine:
429ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockLine");
430ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
431ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
432ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimBlockLineComment_getText(Comment));
433ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
434ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimLine:
435ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimLine");
436ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
437ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
438ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimLineComment_getText(Comment));
439ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
440ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_FullComment:
441ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_FullComment");
442ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
443ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
444ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  if (Kind != CXComment_Null) {
445ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const unsigned NumChildren = clang_Comment_getNumChildren(Comment);
4465ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned i;
4475ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0; i != NumChildren; ++i) {
448ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf("\n// %s: ", FileCheckPrefix);
449ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXCommentInternal(Ctx, clang_Comment_getChild(Comment, i));
450ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
451ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
452ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(")");
453ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel--;
454ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
455ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
456ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXComment(CXComment Comment) {
457ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  struct CommentASTDumpingContext Ctx;
458ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx.IndentLevel = 1;
459ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("\n// %s:  CommentAST=[\n// %s:", FileCheckPrefix, FileCheckPrefix);
460ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  DumpCXCommentInternal(&Ctx, Comment);
461ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("]");
462ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
463ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
464f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
465f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
466f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
467f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGParserCtxtPtr RNGParser;
468f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGPtr Schema;
469f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
470f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko} CommentXMLValidationData;
471f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
472f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void ValidateCommentXML(const char *Str,
473f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                               CommentXMLValidationData *ValidationData) {
474f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
475f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlDocPtr Doc;
476f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGValidCtxtPtr ValidationCtxt;
477f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  int status;
478f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
479f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData || !ValidationData->CommentSchemaFile)
480f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
481f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
482f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData->RNGParser) {
483f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    ValidationData->RNGParser =
484f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        xmlRelaxNGNewParserCtxt(ValidationData->CommentSchemaFile);
485f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    ValidationData->Schema = xmlRelaxNGParse(ValidationData->RNGParser);
486f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
487f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData->RNGParser) {
488f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
489f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
490f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
491f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
492f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  Doc = xmlParseDoc((const xmlChar *) Str);
493f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
494f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!Doc) {
495f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
496f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not well-formed XML: %s]", Error->message);
497f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
498f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
499f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
500f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  ValidationCtxt = xmlRelaxNGNewValidCtxt(ValidationData->Schema);
501f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  status = xmlRelaxNGValidateDoc(ValidationCtxt, Doc);
502f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!status)
503f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLValid");
504f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  else if (status > 0) {
505f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
506f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not vaild XML: %s]", Error->message);
507f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  } else
508f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
509f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
510f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGFreeValidCtxt(ValidationCtxt);
511f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlFreeDoc(Doc);
512f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
513f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
514f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
515e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenkostatic void PrintCursorComments(CXCursor Cursor,
516f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                                CommentXMLValidationData *ValidationData) {
517ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
518ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString RawComment;
519ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *RawCommentCString;
520ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString BriefComment;
521ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *BriefCommentCString;
522ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
523ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawComment = clang_Cursor_getRawCommentText(Cursor);
524ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawCommentCString = clang_getCString(RawComment);
525ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (RawCommentCString != NULL && RawCommentCString[0] != '\0') {
526ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCStringWithPrefix("RawComment", RawCommentCString);
527ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintRange(clang_Cursor_getCommentRange(Cursor), "RawCommentRange");
528ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
529ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefComment = clang_Cursor_getBriefCommentText(Cursor);
530ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefCommentCString = clang_getCString(BriefComment);
531ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      if (BriefCommentCString != NULL && BriefCommentCString[0] != '\0')
532ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCStringWithPrefix("BriefComment", BriefCommentCString);
533ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      clang_disposeString(BriefComment);
534ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
535ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    clang_disposeString(RawComment);
536ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
537ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
538ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
539ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXComment Comment = clang_Cursor_getParsedComment(Cursor);
540ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_getKind(Comment) != CXComment_Null) {
541ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringWithPrefixAndDispose("FullCommentAsHTML",
542ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                        clang_FullComment_getAsHTML(Comment));
543f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      {
544f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        CXString XML;
545e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko        XML = clang_FullComment_getAsXML(Comment);
546f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCXStringWithPrefix("FullCommentAsXML", XML);
547f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        ValidateCommentXML(clang_getCString(XML), ValidationData);
548f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        clang_disposeString(XML);
549f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      }
550f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
551ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXComment(Comment);
552ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
553ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
554ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
555ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
556b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidistypedef struct {
557b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned line;
558b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned col;
559b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis} LineCol;
560b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
561b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidisstatic int lineCol_cmp(const void *p1, const void *p2) {
562b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *lhs = p1;
563b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *rhs = p2;
564b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  if (lhs->line != rhs->line)
565b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis    return (int)lhs->line - (int)rhs->line;
566b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  return (int)lhs->col - (int)rhs->col;
567b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis}
568b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
569f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCursor(CXCursor Cursor,
570f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                        CommentXMLValidationData *ValidationData) {
571aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
572e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (clang_isInvalid(Cursor.kind)) {
573e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString ks = clang_getCursorKindSpelling(Cursor.kind);
574e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("Invalid Cursor => %s", clang_getCString(ks));
575e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
576e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
577699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  else {
578e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString string, ks;
579c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    CXCursor Referenced;
5801db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
581e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    CXCursor SpecializationOf;
5829f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    CXCursor *overridden;
5839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned num_overridden;
584430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    unsigned RefNameRangeNr;
585430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange CursorExtent;
586430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange RefNameRange;
587cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysUnavailable;
588cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysDeprecated;
589cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString UnavailableMessage;
590cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString DeprecatedMessage;
591cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXPlatformAvailability PlatformAvailability[2];
592cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int NumPlatformAvailability;
593cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int I;
594aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
595e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    ks = clang_getCursorKindSpelling(Cursor.kind);
596358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    string = want_display_name? clang_getCursorDisplayName(Cursor)
597358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              : clang_getCursorSpelling(Cursor);
598e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("%s=%s", clang_getCString(ks),
599e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                    clang_getCString(string));
600e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
601ef0cef6cec8de5fc60e469a93436eed7212e0dc2Steve Naroff    clang_disposeString(string);
602e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
603c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    Referenced = clang_getCursorReferenced(Cursor);
604c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    if (!clang_equalCursors(Referenced, clang_getNullCursor())) {
6051f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (clang_getCursorKind(Referenced) == CXCursor_OverloadedDeclRef) {
6061f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        unsigned I, N = clang_getNumOverloadedDecls(Referenced);
6071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("[");
6081f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        for (I = 0; I != N; ++I) {
6091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          CXCursor Ovl = clang_getOverloadedDecl(Referenced, I);
6101f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          CXSourceLocation Loc;
6111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          if (I)
6121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor            printf(", ");
6131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
6141f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          Loc = clang_getCursorLocation(Ovl);
615a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor          clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6161f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          printf("%d:%d", line, column);
6171f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        }
6181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("]");
6191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      } else {
6201f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(Referenced);
621a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6221f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf(":%d:%d", line, column);
6231f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
624c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
625b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
626b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (clang_isCursorDefinition(Cursor))
627b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      printf(" (Definition)");
62858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
62958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    switch (clang_getCursorAvailability(Cursor)) {
63058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Available:
63158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
63258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
63358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Deprecated:
63458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (deprecated)");
63558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
63658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
63758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_NotAvailable:
63858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (unavailable)");
63958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
640d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
641d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen      case CXAvailability_NotAccessible:
642d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        printf(" (inaccessible)");
643d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        break;
64458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
64595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
646cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    NumPlatformAvailability
647cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      = clang_getCursorPlatformAvailability(Cursor,
648cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysDeprecated,
649cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &DeprecatedMessage,
650cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysUnavailable,
651cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &UnavailableMessage,
652cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            PlatformAvailability, 2);
653cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    if (AlwaysUnavailable) {
654cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always unavailable: \"%s\")",
655cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(UnavailableMessage));
656cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else if (AlwaysDeprecated) {
657cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always deprecated: \"%s\")",
658cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(DeprecatedMessage));
659cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else {
660cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      for (I = 0; I != NumPlatformAvailability; ++I) {
661cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (I >= 2)
662cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          break;
663cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
664cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf("  (%s", clang_getCString(PlatformAvailability[I].Platform));
665cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (PlatformAvailability[I].Unavailable)
666cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", unavailable");
667cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        else {
668cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", introduced=", PlatformAvailability[I].Introduced);
669cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", deprecated=", PlatformAvailability[I].Deprecated);
670cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", obsoleted=", PlatformAvailability[I].Obsoleted);
671cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        }
672cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (clang_getCString(PlatformAvailability[I].Message)[0])
673cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", message=\"%s\"",
674cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                 clang_getCString(PlatformAvailability[I].Message));
675cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf(")");
676cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      }
677cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
678cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    for (I = 0; I != NumPlatformAvailability; ++I) {
679cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      if (I >= 2)
680cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        break;
681cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      clang_disposeCXPlatformAvailability(PlatformAvailability + I);
682cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
683cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
684cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(DeprecatedMessage);
685cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(UnavailableMessage);
686cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
687b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isStatic(Cursor))
688b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (static)");
689b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isVirtual(Cursor))
690b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (virtual)");
691b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor
69295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
69395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXType T =
69495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek        clang_getCanonicalType(clang_getIBOutletCollectionType(Cursor));
69595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXString S = clang_getTypeKindSpelling(T.kind);
69695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      printf(" [IBOutletCollection=%s]", clang_getCString(S));
69795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      clang_disposeString(S);
69895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    }
6993064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7003064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
7013064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
7023064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      unsigned isVirtual = clang_isVirtualBase(Cursor);
7033064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      const char *accessStr = 0;
7043064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7053064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      switch (access) {
7063064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXInvalidAccessSpecifier:
7073064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "invalid"; break;
7083064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPublic:
7093064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "public"; break;
7103064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXProtected:
7113064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "protected"; break;
7123064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPrivate:
7133064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "private"; break;
7143064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      }
7153064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7163064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      printf(" [access=%s isVirtual=%s]", accessStr,
7173064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek             isVirtual ? "true" : "false");
7183064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
719e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor
720e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    SpecializationOf = clang_getSpecializedCursorTemplate(Cursor);
721e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    if (!clang_equalCursors(SpecializationOf, clang_getNullCursor())) {
722e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXSourceLocation Loc = clang_getCursorLocation(SpecializationOf);
723e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXString Name = clang_getCursorSpelling(SpecializationOf);
724a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      clang_getSpellingLocation(Loc, 0, &line, &column, 0);
725e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      printf(" [Specialization of %s:%d:%d]",
726e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
727e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
728e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
7299f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
7309f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
7319f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
7329f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
733b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      LineCol lineCols[50];
734b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      assert(num_overridden <= 50);
7359f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
7369f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
7379f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
738a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
739b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].line = line;
740b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].col = column;
741b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      }
742642214936430d2258318f2022184c1709dfa16d3Michael Liao      /* Make the order of the override list deterministic. */
743b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      qsort(lineCols, num_overridden, sizeof(LineCol), lineCol_cmp);
744b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      for (I = 0; I != num_overridden; ++I) {
7459f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
7469f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
747b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        printf("@%d:%d", lineCols[I].line, lineCols[I].col);
7489f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
7499f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
7509f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
7519f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
752ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
753ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
754ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXFile File = clang_getIncludedFile(Cursor);
755ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXString Included = clang_getFileName(File);
756ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
757ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
758dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
759dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
760dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
761ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
762430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
763430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
764430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
765430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
766430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
767430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
768430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
769430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
770430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
771430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
772430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
773430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
774430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
775430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
776430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
777430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
778430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
779430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
780430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
781430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
782aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
783e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko    PrintCursorComments(Cursor, ValidationData);
784699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
785af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
78689922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
787e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
7881db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
78974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
7901db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
791b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
7921db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
793e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
79474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
79574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
79674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
79774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
798e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
79974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
80074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
80174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
8029298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
8039298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
8040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
805ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
806ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
807ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
808ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
809ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
810a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
811a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
8125352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
813274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
8140a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
815aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
816aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
8170a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
818f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
8190a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
8205352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
821e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
822274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
823274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
824274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
825f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
826a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
827a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
8280a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
8290a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
830e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
8310a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
8323739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
8330a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
834473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
835473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
836473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation start = clang_getRangeStart(range);
837473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation end = clang_getRangeEnd(range);
838473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
839473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
840a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
841a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
842a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
843473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
844473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
845473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
8460a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
847473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
848473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
849473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
8500a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
8510a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
8520a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
8530a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
854436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
855473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
856473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
8570a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
8580a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
8590a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
860473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
86151c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
8620a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      break;
8630a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
864473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
86551c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
8665352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
8675352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
8687473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
8697473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
8707473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
8717473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
8727473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
873a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
8747473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
8757473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
8767473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
8777473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
8787473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
8797473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
8807473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
8817473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
8827473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
883a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
884a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
88559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
886b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
8874e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
888f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
8893c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
8904e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
891f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
89259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
89359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
8944e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
89559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
89659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
8974e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
89859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
899f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
90059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
90159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
902ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
903e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
9040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
9050d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
906a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
907a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  CXSourceRange extent = clang_getCursorExtent(C);
908430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintRange(extent, "Extent");
909fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
910fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
911f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko/* Data used by the visitors. */
912f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
913e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
914e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
915f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  CommentXMLValidationData ValidationData;
916e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
917fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
918625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
919e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
920e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
921e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
922e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
923e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
92498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
9251db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
926a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
927fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
9281db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
929f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Cursor, &Data->ValidationData);
930a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
931e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
932e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
933625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
934e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
935e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
936625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
937c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
938e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
939e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
940e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
941625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
942625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
943625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
944e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
945625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
946b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
947b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
948e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
949625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
950625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
951625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
952625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
953625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
954625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
955625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
956625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
957625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
95898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
9591db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
96074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
961e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
962625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
963625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
964625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
965625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
966625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
967625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
968e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
96998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
970a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
971e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
9721db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
973e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
974b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      CXSourceLocation RefLoc
975b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor        = clang_getLocation(Data->TU, file, curLine, curColumn);
976b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
97798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
97898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
97998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
98098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
98198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
982f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Ref, &Data->ValidationData);
98398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
98498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
9854ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
98674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
987625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
9882d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
989e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
990e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
99189922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
99250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
9937d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
9947d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
9957d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
9967d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
997e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
998e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
999e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1000e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
1001cf84aa46def41cccf4dd4c51cd0543b70c11e4ebTed Kremenek    CXString USR = clang_getCursorUSR(C);
1002e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    const char *cstr = clang_getCString(USR);
1003e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
10047d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
1005e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
10067d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
1007e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
1008e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
1009a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
10107d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
10117d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
1012e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1013e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1014e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1015e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1016e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
10177d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
10187d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
10197d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
102016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
102116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
102216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
102316b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
102416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
1025e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
102616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
102774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
102874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
102974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
1030e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
103174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
1032e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
103316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
103416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
103516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
1036a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
1037a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
103874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
1039e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
104074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
104116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
104216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
104316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
104416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
104516b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
1046e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
104716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
104816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
104916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
10503bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
10513bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
10523bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10533bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
10543bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
10553bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
10563bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10573bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
10583bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
10593bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10603bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
10613bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
1062c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
1063c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
1064c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
1065c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
10663bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
10673bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10683bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
1069f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
10703bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
10713bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
10723bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10733bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
10743bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
10753bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10763bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
10778e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
10788e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
10798e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
10808e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenekstatic enum CXChildVisitResult PrintTypeKind(CXCursor cursor, CXCursor p,
10818e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                             CXClientData d) {
10828e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
10838e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXType T = clang_getCursorType(cursor);
10848e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXString S = clang_getTypeKindSpelling(T.kind);
1085f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
10868e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf(" typekind=%s", clang_getCString(S));
1087e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
1088e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
1089e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
1090e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
1091e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
1092e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
10938e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    clang_disposeString(S);
1094e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
109504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
109604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      CXType CT = clang_getCanonicalType(T);
109704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
109804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString CS = clang_getTypeKindSpelling(CT.kind);
109904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [canonical=%s]", clang_getCString(CS));
110004c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(CS);
110104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
110204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1103e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
110404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
11059a140845438c2fc31e7d48a6dedbc695f4c83c68Ted Kremenek      CXType RT = clang_getCursorResultType(cursor);
110604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
110704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString RS = clang_getTypeKindSpelling(RT.kind);
110804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [result=%s]", clang_getCString(RS));
110904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(RS);
111004c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
111104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1112d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1113d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1114d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      int numArgs = clang_Cursor_getNumArguments(cursor);
1115d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      if (numArgs != -1 && numArgs != 0) {
111647f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1117d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
111847f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        for (i = 0; i < numArgs; ++i) {
1119d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1120d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1121d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            CXString S = clang_getTypeKindSpelling(T.kind);
1122d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            printf(" %s", clang_getCString(S));
1123d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            clang_disposeString(S);
1124d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1125d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1126d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1127d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1128d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
11293ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
11303ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
113104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
11328e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
11338e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
11348e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
11358e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
11368e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
11378e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
11388e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
11397d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
11407d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11417d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1142625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
114398271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1144ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1145f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             PostVisitTU PV,
1146f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             const char *CommentSchemaFile) {
1147e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1148fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1149e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1150e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1151e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1152e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1153e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1154e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1155e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1156e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1157e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1158358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1159358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1160358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1161358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1162358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1163b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1164e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1165e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1166e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1167e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1168e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1169e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1170e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1171e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1172e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1173e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1174e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1175e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1176e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
1177f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.CommentSchemaFile = CommentSchemaFile;
1178f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
1179f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.RNGParser = NULL;
1180f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.Schema = NULL;
1181f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
1182e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1183e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1184e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1185ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1186ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1187e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1188a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
118916ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
119016ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
119116ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
119216ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
119316ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
11940d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
11950d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
11960d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
11970d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1198fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1199ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1200ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1201625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1202625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1203020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1204e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
12050a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
12060a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          /* displayDiagnosics=*/1);
1207e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1208020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1209020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1210625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1211020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1212625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1213f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV, NULL);
1214020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1215020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1216625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1217625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1218ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1219ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1220ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1221ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1222ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
1223f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
12244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
12254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
12264db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
1227abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1228ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1229358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1230358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
12314814fb560322daf009fda6226cffccb10bd44620Douglas Gregor                          /* displayDiagnosics=*/0);
1232ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1233f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if ((CommentSchemaFile = parse_comments_schema(argc, argv))) {
1234f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argc--;
1235f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argv++;
1236f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
1237f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
1238020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1239020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
12404db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1241020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
12424db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1243dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
1244dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files,
1245dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files,
1246dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files, num_unsaved_files,
1247dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
1248ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  if (!TU) {
1249ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1250abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1251020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1252ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1253ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1254ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1255f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV,
1256f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             CommentSchemaFile);
12574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1258020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
12594db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1260ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1261ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1262abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1263abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1264abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1265abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1266abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1267abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1268abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1269abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int result;
1270abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
127140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
127240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1273abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1274abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1275abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
12761aa27307c462baaa9e5fda14ff6797dd39fe8b84Douglas Gregor                          /* displayDiagnosics=*/0);
1277abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1278abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1279abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1280abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1281abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1282abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1283c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1284c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
128544c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
128644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argv + num_unsaved_files,
128744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argc - num_unsaved_files,
1288c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar                                  0, 0, getDefaultParsingOptions());
1289abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU) {
1290abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1291abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1292abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1293abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1294abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1295abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1296bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1297bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1298bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
129940098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
130040098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
130140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
130240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
130340098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1304abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
130540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    if (clang_reparseTranslationUnit(TU,
130640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? num_unsaved_files : 0,
130740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? unsaved_files : 0,
1308e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
1309c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1310abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1311abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1312abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1313abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1314abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1315bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1316bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1317bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1318abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1319abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1320f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV, NULL);
1321bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1322abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1323abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1324abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1325abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1326abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
13270d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
13281c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
13291c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
13301c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1331dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
13321c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
13331d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
13341d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
13359096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
13361d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
13371d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
133851b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
133951b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1340f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
13411c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
13421c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
13431c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
13441d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
13451d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
13461c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
13471c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
13481c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
13492389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  CXCursor prevCursor = clang_getNullCursor();
1350b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
13512389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
13528f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1353e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13540a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
13550a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                                /* displayDiagnosics=*/1))) {
13561c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
13571c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
13581c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1359e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13601c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
13611c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1362e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13631c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
13641c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
13651c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
13661c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1367e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1368b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
13692389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
13702389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
13712389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
13722389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
13732389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
13742389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
13752389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
13762389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
13772389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
13782389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
13792389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
13802389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
13812389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
13822389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
13832389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
13842389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1385dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1386d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
13872389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
13882389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1389a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
13902389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
13912389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1392a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
13932389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
13941c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1395e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13961c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
13974f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
13984f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
13991c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
14001c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
14011c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
14021c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
140332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
14040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
14050d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
14060c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
14070c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
14080c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1409e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1410fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1411fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
141288d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1413fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1414fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1415fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1416fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
14170c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1418fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1419fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1420fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1421fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1422fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1423fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
14240c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
14250c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
14260c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1427fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1428fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
142988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1430fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1431fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1432fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1433e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1434fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1435fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1436fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1437e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1438fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1439fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
144088d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1441fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1442fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1443fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1444fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1445fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1446e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1447fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1448e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1449fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1451fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
14520c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1453fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1456e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1457fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1458fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1459fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1460fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1461fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
146288d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1463fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1464fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1465fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
14660c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
14670c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
14680c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
14690c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
14700c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
14710c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
14720c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
14730c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
14740c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
14750c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
14760c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
14770c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
14780c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
14790c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
14800c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
14810c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
14820c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
14830c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
14840c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
14850c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
14860c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1487ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
148801dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
148901dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
149001dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
149101dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
149201dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
14930c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1494e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
14950c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
14960c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
14970c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1498dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1499dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1500dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1501dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1502dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1503dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1504dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1505dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1506dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1507dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1508dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1509dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1510dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1511dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1512dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1513dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1514dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1515dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1516dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1517dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1518dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1519dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1520dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1521dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1522dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
15233ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_string(CXCompletionString completion_string, FILE *file) {
1524f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1525e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15263ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
15270c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
15282ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
15292ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
15300c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
15313ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1532e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15333ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
15343ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
15353ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1536e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
15373ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
15383ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
15393ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
15405a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
15415a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
15425a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
15435a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
15445a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
15453ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1546e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1547d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
15482ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1549e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
15500c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
15512ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
15522ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
15530c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
15542ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
15553ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
15563ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
15573ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_result(CXCompletionResult *completion_result,
15583ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                             CXClientData client_data) {
15593ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  FILE *file = (FILE *)client_data;
1560e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
15616164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
1562ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
1563ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
1564d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
1565d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
1566ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
1567e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
1568e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
1569e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15703ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
157158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
157212e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
157358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
157458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
157558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
157658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
157758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
157858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
157958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
158058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
158158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
158258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
158358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
1584d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
1585d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
1586d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
1587d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
158858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
15896164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
15906164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
15916164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
15926164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
15936164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
15946164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
15956164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
15966164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
15976164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
15986164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      fprintf(file, "\"%s\"",
15996164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen              clang_getCString(clang_getCompletionAnnotation(
16006164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen                                 completion_result->CompletionString, i)));
16016164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
16026164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
16036164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
16046164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
1605ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
1606ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
1607ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
1608ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
1609ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
1610ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
1611ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
1612ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
1613ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
1614ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
1615ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
1616ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
1617d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
1618d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
1619d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
1620d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
1621d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
1622d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
1623d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
1624d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
1625ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
162658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
16270c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
16280c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
16293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
16303da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
16313da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
16323da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
16333da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16343da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
16353da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
16363da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16373da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
16383da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
16393da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16403da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
16413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
16423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16433da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
16443da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
16453da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16463da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
16473da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
16483da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16493da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
16503da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
16513da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16523da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
16533da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
16543da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16553da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
16563da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
16573da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16583da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
16593da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
16603da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16613da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
16623da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
16633da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16643da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
16653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
16663da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16673da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
16683da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
16693da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16703da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
16713da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
16723da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16733da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
16743da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
16753da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16763da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
16773da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
16783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
16803da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
16813da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16823da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
16833da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
16843da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
16863da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
16873da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16883da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
16893da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
16903da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16913da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
16923da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
16933da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16943da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
16953da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
16963da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16973da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
16983da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
16993da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17003da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
17013da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
17021e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregorint my_stricmp(const char *s1, const char *s2) {
17031e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  while (*s1 && *s2) {
17046d5552131a4637f9bbe8c93386648e9bbb2c30feNAKAMURA Takumi    int c1 = tolower((unsigned char)*s1), c2 = tolower((unsigned char)*s2);
17051e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (c1 < c2)
17061e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return -1;
17071e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    else if (c1 > c2)
17081e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return 1;
17091e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17101e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s1;
17111e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s2;
17121e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  }
17131e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17141e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  if (*s1)
17151e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return 1;
17161e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  else if (*s2)
17171e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return -1;
17181e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  return 0;
17191e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor}
17201e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17211982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
17220c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
17230c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
17240c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
17250c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
1726f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
1727f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
1728735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1729735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
1730ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
173125d9b00ab0b128d651d993c38726a00cd9969124Dawn Perchik  CXTranslationUnit TU = 0;
173232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
173332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
173432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
173532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
173632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
1737d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
1738d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
1739df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
17401982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
17411982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
17421982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
17431982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
17441982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
1745e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
1746fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
1747f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
17480c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1749735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
1750735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
1751735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
175232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
175332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
175432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
175532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
175632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
175732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, 0,
175832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argv + num_unsaved_files + 2,
175932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argc - num_unsaved_files - 2,
176032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  0, 0, getDefaultParsingOptions());
176132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (!TU) {
176232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
176332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
176432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
176508bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
176608bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  if (clang_reparseTranslationUnit(TU, 0, 0, clang_defaultReparseOptions(TU))) {
176708bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    fprintf(stderr, "Unable to reparse translation init!\n");
176808bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
176908bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
177032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
177132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
177232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
177332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
177432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
177532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
177632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
17772de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
17782de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
177932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
178032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
178132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
1782936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
1783ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
1784e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
17853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
1786e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
17870a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
17880a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
17891e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
17901e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
17911e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
17921e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17931982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
17941982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
17951e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
1796a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
1797a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
1798a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
1799a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
1800a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
1801a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
18023da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
18033da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
18043da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
18053da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
18060a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
18070a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
1808e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1809e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
1810e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
1811e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
1812e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
1813e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
1814e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
1815e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1816e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
1817e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
1818e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1819e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
1820e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
1821e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1822e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1823e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
1824e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
1825e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
1826e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
1827e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
18280a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
18290a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
18300a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
18310a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
18320a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
18330a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
18340a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
1835ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
1836ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
1837df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
18380c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
18390c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
1840e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1841735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1842735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
1843f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
18440c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
18450c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1846f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
1847f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
1848f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
1849f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
1850f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
1851f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
1852aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int inspect_cursor_at(int argc, const char **argv) {
1853f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
1854f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
1855f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1856f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
1857f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
1858f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
1859f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
1860f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
18618e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
1862bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
18638e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1864e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
1865f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  while (strstr(argv[NumLocations+1], "-cursor-at=") == argv[NumLocations+1])
1866f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
1867e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1868f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
1869f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
1870f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
1871f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
1872f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
1873f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    const char *input = argv[Loc + 1] + strlen("-cursor-at=");
1874e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
1875e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
1876fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
1877f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
1878f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1879e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1880e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
1881f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
1882f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1883e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18848e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
18858e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
18868e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
18878e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
18888e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
18898e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
18908e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
18918e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  argv + num_unsaved_files + 1 + NumLocations,
1892f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  argc - num_unsaved_files - 2 - NumLocations,
18938e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  unsaved_files,
18948e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  Repeats > 1? 0 : num_unsaved_files,
18958e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  getDefaultParsingOptions());
18968e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1897f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (!TU) {
1898f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
1899f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1900f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1901e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1902dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1903dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
1904dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1905bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
19068e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    if (Repeats > 1 &&
19078e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
19088e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
19098e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      clang_disposeTranslationUnit(TU);
19108e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      return 1;
19118e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
1912dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1913dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1914dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
19158e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
19168e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
19178e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
19188e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
19198e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
19208e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
19218e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
19228e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
19238e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
1924dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1925dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
1926dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
1927dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
19288e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
19298fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        CXCompletionString completionString = clang_getCursorCompletionString(
19308fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor                                                                        Cursor);
193166373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
193266373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXString Spelling;
193366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        const char *cspell;
193466373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        unsigned line, column;
193566373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
193666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        printf("%d:%d ", line, column);
1937f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
193866373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        PrintCursorExtent(Cursor);
193966373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        Spelling = clang_getCursorSpelling(Cursor);
194066373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        cspell = clang_getCString(Spelling);
1941ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        if (cspell && strlen(cspell) != 0) {
1942ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          unsigned pieceIndex;
1943ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(" Spelling=%s (", cspell);
1944ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          for (pieceIndex = 0; ; ++pieceIndex) {
19456c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer            CXSourceRange range =
19466c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer              clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
1947ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            if (clang_Range_isNull(range))
1948ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis              break;
1949ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            PrintRange(range, 0);
1950ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          }
1951ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(")");
1952ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        }
195366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_disposeString(Spelling);
195434ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis        if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
195534ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis          printf(" Selector index=%d",clang_Cursor_getObjCSelectorIndex(Cursor));
1956f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis        if (clang_Cursor_isDynamicCall(Cursor))
1957f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis          printf(" Dynamic-call");
1958f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis
19598fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        if (completionString != NULL) {
19608fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          printf("\nCompletion string: ");
19618fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          print_completion_string(completionString, stdout);
19628fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        }
19638e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        printf("\n");
19648e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
19658e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
19668e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
1967f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
19688e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1969a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
1970f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
1971f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
1972f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
1973f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1974f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
1975f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
1976f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
1977aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
1978aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
1979aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
1980aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
1981aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1982f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
1983aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
1984aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
1985aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
1986aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
1987aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1988aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
1989aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
1990aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
1991aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
1992aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
1993aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
1994aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
1995aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
1996aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
1997aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
1998aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
1999aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2000aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
2001aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
2002aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
2003aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2004aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
2005aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
2006aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
2007aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
2008aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
2009aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
2010aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2011aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
2012aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
2013aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
2014aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2015aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2016aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2017aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
2018aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2019aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2020aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2021aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
2022aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2023aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2024aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2025aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2026aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
2027aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argv + num_unsaved_files + 1 + NumLocations,
2028aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argc - num_unsaved_files - 2 - NumLocations,
2029aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  unsaved_files,
2030aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  Repeats > 1? 0 : num_unsaved_files,
2031aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  getDefaultParsingOptions());
2032aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2033aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (!TU) {
2034aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2035aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2036aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2037aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2038dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2039dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2040dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2041aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2042aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Repeats > 1 &&
2043aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2044aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                     clang_defaultReparseOptions(TU))) {
2045aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      clang_disposeTranslationUnit(TU);
2046aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return 1;
2047aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2048dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2049dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2050dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
2051aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2052aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
2053aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
2054aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
2055aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
2056aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2057aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
2058aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
2059aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
2060dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2061dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2062dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2063dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2064aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
206526fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
2066f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
2067aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
2068aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
2069aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
2070dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2071dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2072dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
2073aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
2074aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2075aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2076aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2077aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
2078aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2079aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
2080aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
2081aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2082aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
2083aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2084aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
20854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
20864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
20874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
2088dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
20896f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
209013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
20914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
20924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
20934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
20944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
20954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
20964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
20974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
20984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
20994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
21004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
21014e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
21024e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2104dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
21054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename = clang_getFileName((CXFile)file);
21064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
21074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
21084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
211013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
211113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
21124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
211313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
2114dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
21154e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
211613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
21174e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
211813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
21194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
21204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
21214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf("<null loc>");
21224e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
21234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2124c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2125c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2126c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2127c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
21284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
21294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
213013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
213113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
213213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
213313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
213413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
213513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
213613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
21374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
21384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
21394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
21404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
21414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
214313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
214413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
214513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
214613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
214713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
214813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
214913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
215013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
215113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
215213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
21536ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic CXIdxClientContainer makeClientContainer(const CXIdxEntityInfo *info,
21546ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
21554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
21564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2157dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
21584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
21594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2160dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
21614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
21624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
21634e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
21644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
2165f89bc0500c74a590ff9e99aab75bcc22d05bc760Argyrios Kyrtzidis  /* FIXME: free these.*/
216613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  newStr = (char *)malloc(strlen(name) +
216713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis                          digitCount(line) + digitCount(column) + 3);
21684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
21696ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
21704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
21722957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
21732957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
21742957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
21753e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
21763e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
21773e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
21783e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
21794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2181dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2182dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2183dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2184dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2185dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2186dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2187dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2188dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2189dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2190dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2191dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2192c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2193c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2194dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2195dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2196dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2197dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2198dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2199dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
22002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
22012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
22022957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
22032957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
22042957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
22052957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
22062957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
22072957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
22082957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
220935adca009c65a9578b91ab0bdca13691624e13afDavid Blaikie  case CXIdxEntity_CXXInterface: return "c++-__interface";
22102957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
22112957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
22122957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
22132957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
22142957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
22152957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
22162957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
22172957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
22182957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
22192957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
22202957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
22212957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2222dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
22236ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
22246ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2225dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2226dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2227838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2228838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2229838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2230838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2231838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2232838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2233838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2234838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2235838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2236838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2237838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2238dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2239dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
22406ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
22414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
22424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
2243643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
22444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
22454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
22464e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2247c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
2248c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
2249c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
2250c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
2251c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
2252dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
22534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
22544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
22554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22562957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
22572957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
2258dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
2259dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
2260c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
2261643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
2262643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
2263643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
2264643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
2265f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2266643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
22674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2269b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
2270b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
2271b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
2272b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
2273f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2274b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
227513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2276b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
2277b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
2278c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
2279c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
2280c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
2281c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
2282c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
2283c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
2284c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
2285f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(ProtoInfo->protocols[i]->cursor, NULL);
2286c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
228713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
2288c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
2289c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
2290c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
2291c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
22924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
2293996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
22944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
22954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
2296996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
2297996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
22984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
22994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
23004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
23014e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2302996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
2303996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
2304996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
2305996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
2306996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
2307996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
2308996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
2309996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
2310996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
2311996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
2312996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
2313996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
2314dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
23154e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23164e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2317dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
2318dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
2319dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
232062d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
232162d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
2322dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
2323dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
2324dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
232562d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
232613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
232713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
232813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
2329dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
2330dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
2331dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
2332dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2333dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
23344e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2336dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
23376ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
23384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
23394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
23404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
23414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
234266042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
2343dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
23444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
23454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
234613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
23474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | isImport: %d | isAngled: %d\n", info->isImport, info->isAngled);
2348dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2349dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
23504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23514e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2352dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientContainer index_startedTranslationUnit(CXClientData client_data,
23534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis                                                   void *reserved) {
23544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
23554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
23564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
23574e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
235866042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
2359dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
23604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23626ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
23632957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
2364dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
23656ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
23666ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
2367c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
2368792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
2369b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
2370b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
2371dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
23724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2373dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
2374dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
2375f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2376dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
237713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2378b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
2379b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
2380b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
2381b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
2382dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
2383c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
2384c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isContainer: %d", info->isContainer);
2385c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
23864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2387b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
238887adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
2389b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
2390f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2391b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
2392b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
2393b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
2394dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
2395dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
2396dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
2397dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
2398dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
2399dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
2400dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
2401dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
2402dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
2403dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
2404dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
2405dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
2406dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
2407dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
24084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24096ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
2410dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
2411dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
241221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
2413f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(CatInfo->classCursor, NULL);
241421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
241513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
2416dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
2417dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
24184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24196ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
24206ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
2421b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
24226ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
24236ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
24244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
24254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2426c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
2427c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
24286ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
24294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2430792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
2431792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
2432792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
2433792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2434792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2435792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
2436792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
2437792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2438792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2439792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
2440792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
2441b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
2442b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
2443b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
2444b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
2445b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
2446b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
2447b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
24482957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
24492957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    clang_index_setClientContainer(info->declAsContainer,
24502957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                              makeClientContainer(info->entityInfo, info->loc));
24514e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24524e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
24546ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
2455dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexEntityReference]", client_data, info->referencedEntity);
24564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
2457f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
24584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
245913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2460dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
24614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
24624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
2463c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
2464aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
2465aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
2466b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
2467aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
24684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
24694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24716f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
24726f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
24736f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
24746f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
24756f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
24766f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
24774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
24786f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
24794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
2480dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
24814e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
2482f89bc0500c74a590ff9e99aab75bcc22d05bc760Argyrios Kyrtzidis  0, /*importedASTFile*/
24834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
2484dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
24854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
24864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
24874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
248822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
248922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
249022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
249122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
249222490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
249322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
249422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
249522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
249622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
249722490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
249822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
24994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic int index_file(int argc, const char **argv) {
25004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
25012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
25022957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
25034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData index_data;
2504b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned index_opts;
2505dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
25064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
25084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
25094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
25104e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
25114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
25124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
25134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
25144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
25154e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25164e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc == 0) {
25174e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
25184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return -1;
25194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
25204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25212957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
25222957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
25232957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
25242957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
25252957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
25262957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = 0;
25272957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
25284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data.check_prefix = check_prefix;
25294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data.first_check_printed = 0;
2530dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  index_data.fail_for_error = 0;
25316f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data.abort = 0;
2532dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
253322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = getIndexOptions();
25342957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
25352957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
2536b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
2537c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis                                 0, argv, argc, 0, 0, 0, 0);
253821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (index_data.fail_for_error)
25392957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    result = -1;
254021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
25412957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
25422957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
254321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
254421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
254521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
254621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
254721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
25482957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
254921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXTranslationUnit TU;
255021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  const char *check_prefix;
255121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  IndexData index_data;
2552b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned index_opts;
255321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
255421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
255521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
255621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
255721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
255821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
255921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
256021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
256121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
256221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
256321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
256421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
256521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "no ast file\n");
256621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
256721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
256821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
256921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
257021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
257121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
257221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
257321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
25742957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = 0;
25752957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = 1;
257621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
257721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, argv[0], &TU))
25782957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    goto finished;
257921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
258021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
258121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.first_check_printed = 0;
258221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.fail_for_error = 0;
25836f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data.abort = 0;
258421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
258522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = getIndexOptions();
25862957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
25872957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
2588c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
25892957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                      index_opts, TU);
2590dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (index_data.fail_for_error)
25912957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    goto finished;
25924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25932957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  finished:
25942957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
25952957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
25962957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
2597dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return result;
25984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
25994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2600fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
2601fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
2602fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
2603fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
2604fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
2605fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
2606fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
2607fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
2608fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2609fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
2610fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
2611fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
2612fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
2613fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
2614fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
2615fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
2616fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
2617fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2618fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
2619fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2620fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
2621fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
2622fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2623e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
2624e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
2625fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2626e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2627fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
26280a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
2629dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
2630dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files + 2,
2631dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files - 3,
2632dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files,
2633dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  num_unsaved_files,
2634dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
2635fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!TU) {
2636fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
2637fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
2638fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
2639fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
2640fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2641e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
2642fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
2643fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2644e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
2645e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
2646e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
2647e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2648dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2649ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
2650ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
2651ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      if (clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2652ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       clang_defaultReparseOptions(TU))) {
2653ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
2654ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
2655ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
2656ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
2657ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
2658ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
2659ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
2660dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2661dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2662dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2663dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2664dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2665fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
2666fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
2667fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
2668fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2669fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
2670fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2671fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2672fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
2673fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
2674e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
2675fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
2676fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2677e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2678fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2679fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2680fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
2681fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
2682e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
2683fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
2684fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2685e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2686fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2687fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2688fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
2689fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
2690dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2691dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2692dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2693dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2694dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2695dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2696fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
2697fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
2698dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2699dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2700dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2701dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2702dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2703dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2704fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
2705fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
2706fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
2707fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
2708fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
2709fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2710fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
2711fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
2712fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
2713fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
2714fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
2715fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
2716fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
2717a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
2718a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
2719a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
2720a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
272151b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
2722342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
272351b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
27240045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
27250045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
2726f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      PrintCursor(cursors[i], NULL);
27270045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
27280045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
2729fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2730fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
273193f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
2732fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2733fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
2734a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2735fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
2736fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
2737fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
2738fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2739fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
2740fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
2741fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2742db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
2743db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
2744db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
2745db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
2746db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
2747db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
2748db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
2749db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
2750db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
2751db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
2752db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
2753db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
2754db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
2755db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2756db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
2757db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
2758db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
2759db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
2760db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2761c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
2762db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2763db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
2764db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2765db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
2766db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
2767db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
2768db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
2769db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
2770db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2771db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
2772db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
2773c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
2774db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2775db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
2776db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
2777db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
2778db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
2779db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2780db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2781c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
2782db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2783db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
2784db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
2785db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
2786db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
2787db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
2788db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2789db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2790db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
2791c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
2792db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2793c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
2794db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
2795db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
2796db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2797db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
2798c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
2799db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
2800db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
2801c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
2802db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
2803db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
2804db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
2805db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
2806db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2807db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2808c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
2809db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2810db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
2811db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
2812db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
2813c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
2814db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
2815db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
2816db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
2817db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
2818db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2819db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
2820db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
2821db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2822db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
2823db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
2824db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
28250d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2826f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
2827f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
2828f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2829f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
2830f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
2831f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
2832f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2833f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2834f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
2835f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
2836f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2837f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2838f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
2839f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
2840f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
2841f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2842f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2843f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
2844f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
2845f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
2846f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
2847f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2848f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2849f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
2850f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
2851f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
2852f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
2853f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
2854f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
2855f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
2856f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
2857f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
2858f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2859f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2860f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
2861f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
2862f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
2863f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
2864f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
2865f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
2866f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
2867f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2868f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
2869f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
2870f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
2871f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2872f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2873a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
2874ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2875f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
2876f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2877f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2878f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2879f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2880f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2881f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2882f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
2883f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
2884f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
2885f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
2886f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
2887f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
2888f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2889f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2890f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2891f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
2892f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
2893f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
2894f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
2895f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
2896f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
2897f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
2898f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2899f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2900a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
2901ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2902f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
2903f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2904f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
2905f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2906f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2907f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2908f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
2909f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
2910f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2911f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
2912f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
2913f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2914f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2915f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2916f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
2917f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
2918f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
2919f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
2920f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
2921f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2922f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2923f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
2924f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2925f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
2926f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
2927f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
2928f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2929f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2930a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
2931ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2932f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
2933f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2934f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2935f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2936f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2937f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2938f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
2939f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2940f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
2941f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
2942f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2943f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2944f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
2945f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
2946f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
2947f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
2948f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2949f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
2950f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2951f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2952f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
2953f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
2954f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
2955f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
2956f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2957f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
2958f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
2959f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
2960f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
2961f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2962f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2963f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
2964f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
2965f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
2966f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
2967f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
2968f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
2969f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2970f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
2971f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
2972f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2973f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
2974f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
2975f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2976f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
2977f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
2978f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2979f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
2980f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
2981f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
2982f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
2983f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
2984f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
2985f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
2986f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
2987f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
2988f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
2989f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
2990f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2991f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2992f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
2993f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
2994f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2995f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2996f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
29970d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
29980d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
29997ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
30007ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
30017ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
30027ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
30037ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
300408aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
30057ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
30067ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnosics=*/1);
30077ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
30087ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
30097ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
30107ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
30117ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
30127ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
30137ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
30147ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argv + num_unsaved_files,
30157ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argc - num_unsaved_files,
30167ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  unsaved_files,
30177ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  num_unsaved_files,
30187ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  CXTranslationUnit_Incomplete);
30197ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU) {
30207ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
30217ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
30227ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
30237ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
30247ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
30257ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
302639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
302739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
302839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
302939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
303039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
303139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
303239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
303339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
303439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
303539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
303639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
303739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
303839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
303939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
304039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
304139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
304239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
304339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
304439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
304539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
304639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
304739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
304839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
304939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
30507ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
30517ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
30527ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
305339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
30547ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
30557ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
30567ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3057153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
3058153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
3059153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3060153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
3061153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
3062153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
3063153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
3064153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
3065153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
3066153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3067153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
3068153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3069153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3070153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
3071153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
3072153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
3073153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
3074153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
3075153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
3076153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
3077153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3078153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
3079153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3080153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3081153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
3082a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
3083a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
3084a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
3085a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
3086153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
3087a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
3088153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
3089153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3090153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3091153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3092153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
3093153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
3094153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
3095153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
3096153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3097153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
3098153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
3099153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3100153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
3101153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
3102153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3103153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3104153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
3105153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
3106153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3107153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3108153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
3109153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange SR = clang_getDiagnosticRange(D, i);
3110153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
3111153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
3112153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3113153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3114153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
3115153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
3116153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
3117153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
3118153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3119153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3120153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3121153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3122153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
3123153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
31243739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
3125153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
3126153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
3127153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
3128153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
3129153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3130153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3131153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
3132153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
3133153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
3134153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
3135153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
3136153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
3137153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3138153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3139153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3140153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
31419190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
31429190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
3143153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
3144153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
3145153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
31469190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
3147153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3148153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
3149153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
3150153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
315178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
3152153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
315378d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
3154153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3155153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
3156153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
3157153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
3158153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
3159153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
3160153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3161153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3162153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3163153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
3164153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
3165153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
3166153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
3167153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
3168153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
3169153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3170153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
3171153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
3172153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
3173153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
3174153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
3175153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
317678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
317778d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
317878d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
317978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
318078d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
318178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
3182153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3183153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3184153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
3185153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
3186153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3187a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
3188153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
3189153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3190153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
3191153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
3192153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
3193153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3194153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3195153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3196153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
3197153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
3198153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
3199153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
3200153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3201153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
3202153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
3203153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
3204153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
3205153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
3206153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
3207153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
3208153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3209153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3210153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
3211a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
3212a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
3213153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
3214153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
3215153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3216153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3217153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
32187ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
32197ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3220f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
3221e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
32227d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
3223e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
32247d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
32257d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
322659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
322759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
32287d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
32297d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
32307d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
3231f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
3232f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
32330d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
32341982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
3235f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
3236aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    "       c-index-test -file-refs-at=<site> <compiler arguments>\n"
32374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
323821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
32391d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
324026fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
324126fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
3242fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
3243fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
32447d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
32457d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
32461982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
3247f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
324859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
324959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
3250abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
3251abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
32521982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
325359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
325459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
325516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
325616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
32574e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
325853513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
32594e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
326059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-print-typekind {<args>}*\n"
326159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
32627ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
3263153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
3264153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
3265db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
3266db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
3267153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
3268caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
32697d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
32700d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
32710d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
32720d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
32730d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
32740d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
32750d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
3276625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
3277625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
3278f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
3279f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
32806edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
32816edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32826edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
32830a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
3284153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
3285153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
3286f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
32871982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
32881982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
32891982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
3290f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
3291f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return inspect_cursor_at(argc, argv);
3292aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
3293aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
32944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
32954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return index_file(argc - 2, argv + 2);
329621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
329721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
32987d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
3299e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
33007d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
3301ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
3302ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
33037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
3304abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
3305abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
3306abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
3307abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
3308abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
3309abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
3310abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
3311abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
33127d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
3313e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
331459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
331559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
331659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
331759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
331859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
33197d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
332059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
332159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
33227d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
33237d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
33241d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
33251d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
3326fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
3327fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
332816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
332916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
333016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
333116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
333216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
333316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
33343bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
33353bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
33363bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
33378e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-typekind") == 0)
33388e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
33398e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                    PrintTypeKind, 0);
3340f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
3341f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
3342f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
3343f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
3344f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
3345f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
3346f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3347f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3348f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
3349f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
33507ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
33517ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
3352db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
3353db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
3354db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3355f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
335650398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
335750398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
33586edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
33596edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
33606edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
33616edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
33626edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
33636edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
33646edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
33656edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
33666edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
33676edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
33686edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
33696edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
337084294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
33716edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
33726edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data->result = cindextest_main(client_data->argc, client_data->argv);
33733be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#ifdef __CYGWIN__
33743be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi  fflush(stdout);  /* stdout is not flushed on Cygwin. */
33753be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#endif
33766edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
33776edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
33786edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
3379d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer  thread_info client_data;
3380d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer
3381f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
3382f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  LIBXML_TEST_VERSION
3383f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
3384f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
33856160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor  if (getenv("CINDEXTEST_NOTHREADS"))
33866160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor    return cindextest_main(argc, argv);
33876160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor
33886edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
33896edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
3390a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
33916edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
33926edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
3393