Skip to content
This repository was archived by the owner on Nov 1, 2021. It is now read-only.

Commit 30660a8

Browse files
committed
Add new code migrator support for migrating existing Objective-C code to use
the new Objective-C NSArray/NSDictionary/NSNumber literal syntax. This introduces a new library, libEdit, which provides a new way to support migration of code that improves on the original ARC migrator. We now believe that most of its functionality can be refactored into the existing libraries, and thus this new library may shortly disappear. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@152141 91177308-0d34-0410-b5e6-96231b3b80d8
1 parent 2a25396 commit 30660a8

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

64 files changed

+3748
-99
lines changed

include/clang-c/Index.h

+14
Original file line numberDiff line numberDiff line change
@@ -3982,6 +3982,20 @@ typedef void *CXRemapping;
39823982
*/
39833983
CINDEX_LINKAGE CXRemapping clang_getRemappings(const char *path);
39843984

3985+
/**
3986+
* \brief Retrieve a remapping.
3987+
*
3988+
* \param filePaths pointer to an array of file paths containing remapping info.
3989+
*
3990+
* \param numFiles number of file paths.
3991+
*
3992+
* \returns the requested remapping. This remapping must be freed
3993+
* via a call to \c clang_remap_dispose(). Can return NULL if an error occurred.
3994+
*/
3995+
CINDEX_LINKAGE
3996+
CXRemapping clang_getRemappingsFromFileList(const char **filePaths,
3997+
unsigned numFiles);
3998+
39853999
/**
39864000
* \brief Determine the number of remappings.
39874001
*/

include/clang/ARCMigrate/ARCMT.h

+9
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,15 @@ bool getFileRemappings(std::vector<std::pair<std::string,std::string> > &remap,
7676
StringRef outputDir,
7777
DiagnosticConsumer *DiagClient);
7878

79+
/// \brief Get the set of file remappings from a list of files with remapping
80+
/// info.
81+
///
82+
/// \returns false if no error is produced, true otherwise.
83+
bool getFileRemappingsFromFileList(
84+
std::vector<std::pair<std::string,std::string> > &remap,
85+
ArrayRef<StringRef> remapFiles,
86+
DiagnosticConsumer *DiagClient);
87+
7988
typedef void (*TransformFn)(MigrationPass &pass);
8089

8190
std::vector<TransformFn> getAllTransformations(LangOptions::GCMode OrigGCMode,

include/clang/ARCMigrate/ARCMTActions.h

+26
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
#define LLVM_CLANG_ARCMIGRATE_ARCMT_ACTION_H
1212

1313
#include "clang/Frontend/FrontendAction.h"
14+
#include "clang/ARCMigrate/FileRemapper.h"
1415
#include "llvm/ADT/OwningPtr.h"
1516

1617
namespace clang {
@@ -32,6 +33,14 @@ class ModifyAction : public WrapperFrontendAction {
3233
ModifyAction(FrontendAction *WrappedAction);
3334
};
3435

36+
class MigrateSourceAction : public ASTFrontendAction {
37+
FileRemapper Remapper;
38+
protected:
39+
virtual bool BeginInvocation(CompilerInstance &CI);
40+
virtual ASTConsumer *CreateASTConsumer(CompilerInstance &CI,
41+
StringRef InFile);
42+
};
43+
3544
class MigrateAction : public WrapperFrontendAction {
3645
std::string MigrateDir;
3746
std::string PlistOut;
@@ -45,6 +54,23 @@ class MigrateAction : public WrapperFrontendAction {
4554
bool emitPremigrationARCErrors);
4655
};
4756

57+
/// \brief Migrates to modern ObjC syntax.
58+
class ObjCMigrateAction : public WrapperFrontendAction {
59+
std::string MigrateDir;
60+
bool MigrateLiterals;
61+
bool MigrateSubscripting;
62+
FileRemapper Remapper;
63+
CompilerInstance *CompInst;
64+
public:
65+
ObjCMigrateAction(FrontendAction *WrappedAction, StringRef migrateDir,
66+
bool migrateLiterals,
67+
bool migrateSubscripting);
68+
69+
protected:
70+
virtual ASTConsumer *CreateASTConsumer(CompilerInstance &CI,StringRef InFile);
71+
virtual bool BeginInvocation(CompilerInstance &CI);
72+
};
73+
4874
}
4975
}
5076

include/clang/ARCMigrate/FileRemapper.h

+6-3
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ namespace clang {
2424
class FileManager;
2525
class FileEntry;
2626
class DiagnosticsEngine;
27-
class CompilerInvocation;
27+
class PreprocessorOptions;
2828

2929
namespace arcmt {
3030

@@ -44,17 +44,20 @@ class FileRemapper {
4444

4545
bool initFromDisk(StringRef outputDir, DiagnosticsEngine &Diag,
4646
bool ignoreIfFilesChanged);
47+
bool initFromFile(StringRef filePath, DiagnosticsEngine &Diag,
48+
bool ignoreIfFilesChanged);
4749
bool flushToDisk(StringRef outputDir, DiagnosticsEngine &Diag);
50+
bool flushToFile(StringRef outputPath, DiagnosticsEngine &Diag);
4851

4952
bool overwriteOriginal(DiagnosticsEngine &Diag,
5053
StringRef outputDir = StringRef());
5154

5255
void remap(StringRef filePath, llvm::MemoryBuffer *memBuf);
5356
void remap(StringRef filePath, StringRef newPath);
5457

55-
void applyMappings(CompilerInvocation &CI) const;
58+
void applyMappings(PreprocessorOptions &PPOpts) const;
5659

57-
void transferMappingsAndClear(CompilerInvocation &CI);
60+
void transferMappingsAndClear(PreprocessorOptions &PPOpts);
5861

5962
void clear(StringRef outputDir = StringRef());
6063

include/clang/Basic/Diagnostic.h

+23-2
Original file line numberDiff line numberDiff line change
@@ -50,13 +50,19 @@ class FixItHint {
5050
/// insertion hint.
5151
CharSourceRange RemoveRange;
5252

53+
/// \brief Code in the specific range that should be inserted in the insertion
54+
/// location.
55+
CharSourceRange InsertFromRange;
56+
5357
/// \brief The actual code to insert at the insertion location, as a
5458
/// string.
5559
std::string CodeToInsert;
5660

61+
bool BeforePreviousInsertions;
62+
5763
/// \brief Empty code modification hint, indicating that no code
5864
/// modification is known.
59-
FixItHint() : RemoveRange() { }
65+
FixItHint() : BeforePreviousInsertions(false) { }
6066

6167
bool isNull() const {
6268
return !RemoveRange.isValid();
@@ -65,11 +71,26 @@ class FixItHint {
6571
/// \brief Create a code modification hint that inserts the given
6672
/// code string at a specific location.
6773
static FixItHint CreateInsertion(SourceLocation InsertionLoc,
68-
StringRef Code) {
74+
StringRef Code,
75+
bool BeforePreviousInsertions = false) {
6976
FixItHint Hint;
7077
Hint.RemoveRange =
7178
CharSourceRange(SourceRange(InsertionLoc, InsertionLoc), false);
7279
Hint.CodeToInsert = Code;
80+
Hint.BeforePreviousInsertions = BeforePreviousInsertions;
81+
return Hint;
82+
}
83+
84+
/// \brief Create a code modification hint that inserts the given
85+
/// code from \arg FromRange at a specific location.
86+
static FixItHint CreateInsertionFromRange(SourceLocation InsertionLoc,
87+
CharSourceRange FromRange,
88+
bool BeforePreviousInsertions = false) {
89+
FixItHint Hint;
90+
Hint.RemoveRange =
91+
CharSourceRange(SourceRange(InsertionLoc, InsertionLoc), false);
92+
Hint.InsertFromRange = FromRange;
93+
Hint.BeforePreviousInsertions = BeforePreviousInsertions;
7394
return Hint;
7495
}
7596

include/clang/Driver/Action.h

+12
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ class Action {
3939
PreprocessJobClass,
4040
PrecompileJobClass,
4141
AnalyzeJobClass,
42+
MigrateJobClass,
4243
CompileJobClass,
4344
AssembleJobClass,
4445
LinkJobClass,
@@ -171,6 +172,17 @@ class AnalyzeJobAction : public JobAction {
171172
static bool classof(const AnalyzeJobAction *) { return true; }
172173
};
173174

175+
class MigrateJobAction : public JobAction {
176+
virtual void anchor();
177+
public:
178+
MigrateJobAction(Action *Input, types::ID OutputType);
179+
180+
static bool classof(const Action *A) {
181+
return A->getKind() == MigrateJobClass;
182+
}
183+
static bool classof(const MigrateJobAction *) { return true; }
184+
};
185+
174186
class CompileJobAction : public JobAction {
175187
virtual void anchor();
176188
public:

include/clang/Driver/CC1Options.td

+9-2
Original file line numberDiff line numberDiff line change
@@ -433,21 +433,28 @@ def rewrite_objc : Flag<"-rewrite-objc">,
433433
HelpText<"Rewrite ObjC into C (code rewriter example)">;
434434
def rewrite_macros : Flag<"-rewrite-macros">,
435435
HelpText<"Expand macros without full preprocessing">;
436+
def migrate : Flag<"-migrate">,
437+
HelpText<"Migrate source code">;
436438
}
437439

440+
def mt_migrate_directory : Separate<"-mt-migrate-directory">,
441+
HelpText<"Directory for temporary files produced during ARC or ObjC migration">;
438442
def arcmt_check : Flag<"-arcmt-check">,
439443
HelpText<"Check for ARC migration issues that need manual handling">;
440444
def arcmt_modify : Flag<"-arcmt-modify">,
441445
HelpText<"Apply modifications to files to conform to ARC">;
442446
def arcmt_migrate : Flag<"-arcmt-migrate">,
443447
HelpText<"Apply modifications and produces temporary files that conform to ARC">;
444-
def arcmt_migrate_directory : Separate<"-arcmt-migrate-directory">,
445-
HelpText<"Directory for temporary files produced during ARC migration">;
446448
def arcmt_migrate_report_output : Separate<"-arcmt-migrate-report-output">,
447449
HelpText<"Output path for the plist report">;
448450
def arcmt_migrate_emit_arc_errors : Flag<"-arcmt-migrate-emit-errors">,
449451
HelpText<"Emit ARC errors even if the migrator can fix them">;
450452

453+
def objcmt_migrate_literals : Flag<"-objcmt-migrate-literals">,
454+
HelpText<"Enable migration to modern ObjC literals">;
455+
def objcmt_migrate_subscripting : Flag<"-objcmt-migrate-subscripting">,
456+
HelpText<"Enable migration to modern ObjC subscripting">;
457+
451458
def working_directory : JoinedOrSeparate<"-working-directory">,
452459
HelpText<"Resolve file paths relative to the specified directory">;
453460
def working_directory_EQ : Joined<"-working-directory=">,

include/clang/Driver/Options.td

+10-2
Original file line numberDiff line numberDiff line change
@@ -118,13 +118,21 @@ def ccc_arrmt_check : Flag<"-ccc-arrmt-check">, Alias<ccc_arcmt_check>;
118118
def ccc_arrmt_modify : Flag<"-ccc-arrmt-modify">, Alias<ccc_arcmt_modify>;
119119
def ccc_arcmt_migrate : Separate<"-ccc-arcmt-migrate">, CCCDriverOpt,
120120
HelpText<"Apply modifications and produces temporary files that conform to ARC">;
121-
def ccc_arcmt_migrate_EQ : Joined<"-ccc-arcmt-migrate=">, CCCDriverOpt,
122-
Alias<ccc_arcmt_migrate>;
123121
def arcmt_migrate_report_output : Separate<"-arcmt-migrate-report-output">,
124122
HelpText<"Output path for the plist report">;
125123
def arcmt_migrate_emit_arc_errors : Flag<"-arcmt-migrate-emit-errors">,
126124
HelpText<"Emit ARC errors even if the migrator can fix them">;
127125

126+
def _migrate : Flag<"--migrate">, Flags<[DriverOption]>,
127+
HelpText<"Run the migrator">;
128+
def ccc_objcmt_migrate : Separate<"-ccc-objcmt-migrate">, CCCDriverOpt,
129+
HelpText<"Apply modifications and produces temporary files to migrate to "
130+
"modern ObjC syntax">;
131+
def objcmt_migrate_literals : Flag<"-objcmt-migrate-literals">,
132+
HelpText<"Enable migration to modern ObjC literals">;
133+
def objcmt_migrate_subscripting : Flag<"-objcmt-migrate-subscripting">,
134+
HelpText<"Enable migration to modern ObjC subscripting">;
135+
128136
// Make sure all other -ccc- options are rejected.
129137
def ccc_ : Joined<"-ccc-">, Group<ccc_Group>, Flags<[Unsupported]>;
130138

include/clang/Driver/Types.def

+1
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,7 @@ TYPE("lto-bc", LTO_BC, INVALID, "o", "")
8282
TYPE("ast", AST, INVALID, "ast", "u")
8383
TYPE("plist", Plist, INVALID, "plist", "")
8484
TYPE("rewritten-objc", RewrittenObjC,INVALID, "cpp", "")
85+
TYPE("remap", Remap, INVALID, "remap", "")
8586
TYPE("precompiled-header", PCH, INVALID, "gch", "A")
8687
TYPE("object", Object, INVALID, "o", "")
8788
TYPE("treelang", Treelang, INVALID, 0, "u")

include/clang/Edit/Commit.h

+140
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,140 @@
1+
//===----- Commit.h - A unit of edits ---------------------------*- C++ -*-===//
2+
//
3+
// The LLVM Compiler Infrastructure
4+
//
5+
// This file is distributed under the University of Illinois Open Source
6+
// License. See LICENSE.TXT for details.
7+
//
8+
//===----------------------------------------------------------------------===//
9+
10+
#ifndef LLVM_CLANG_EDIT_COMMIT_H
11+
#define LLVM_CLANG_EDIT_COMMIT_H
12+
13+
#include "clang/Edit/FileOffset.h"
14+
#include "llvm/ADT/StringRef.h"
15+
#include "llvm/ADT/SmallVector.h"
16+
17+
namespace clang {
18+
class LangOptions;
19+
class PreprocessingRecord;
20+
21+
namespace edit {
22+
class EditedSource;
23+
24+
class Commit {
25+
public:
26+
enum EditKind {
27+
Act_Insert,
28+
Act_InsertFromRange,
29+
Act_Remove
30+
};
31+
32+
struct Edit {
33+
EditKind Kind;
34+
StringRef Text;
35+
SourceLocation OrigLoc;
36+
FileOffset Offset;
37+
FileOffset InsertFromRangeOffs;
38+
unsigned Length;
39+
bool BeforePrev;
40+
41+
SourceLocation getFileLocation(SourceManager &SM) const;
42+
CharSourceRange getFileRange(SourceManager &SM) const;
43+
CharSourceRange getInsertFromRange(SourceManager &SM) const;
44+
};
45+
46+
private:
47+
const SourceManager &SourceMgr;
48+
const LangOptions &LangOpts;
49+
const PreprocessingRecord *PPRec;
50+
EditedSource *Editor;
51+
52+
bool IsCommitable;
53+
SmallVector<Edit, 8> CachedEdits;
54+
55+
public:
56+
explicit Commit(EditedSource &Editor);
57+
Commit(const SourceManager &SM, const LangOptions &LangOpts,
58+
const PreprocessingRecord *PPRec = 0)
59+
: SourceMgr(SM), LangOpts(LangOpts), PPRec(PPRec), Editor(0),
60+
IsCommitable(true) { }
61+
62+
bool isCommitable() const { return IsCommitable; }
63+
64+
bool insert(SourceLocation loc, StringRef text, bool afterToken = false,
65+
bool beforePreviousInsertions = false);
66+
bool insertAfterToken(SourceLocation loc, StringRef text,
67+
bool beforePreviousInsertions = false) {
68+
return insert(loc, text, /*afterToken=*/true, beforePreviousInsertions);
69+
}
70+
bool insertBefore(SourceLocation loc, StringRef text) {
71+
return insert(loc, text, /*afterToken=*/false,
72+
/*beforePreviousInsertions=*/true);
73+
}
74+
bool insertFromRange(SourceLocation loc, CharSourceRange range,
75+
bool afterToken = false,
76+
bool beforePreviousInsertions = false);
77+
bool insertWrap(StringRef before, CharSourceRange range, StringRef after);
78+
79+
bool remove(CharSourceRange range);
80+
81+
bool replace(CharSourceRange range, StringRef text);
82+
bool replaceWithInner(CharSourceRange range, CharSourceRange innerRange);
83+
bool replaceText(SourceLocation loc, StringRef text,
84+
StringRef replacementText);
85+
86+
bool insertFromRange(SourceLocation loc, SourceRange TokenRange,
87+
bool afterToken = false,
88+
bool beforePreviousInsertions = false) {
89+
return insertFromRange(loc, CharSourceRange::getTokenRange(TokenRange),
90+
afterToken, beforePreviousInsertions);
91+
}
92+
bool insertWrap(StringRef before, SourceRange TokenRange, StringRef after) {
93+
return insertWrap(before, CharSourceRange::getTokenRange(TokenRange), after);
94+
}
95+
bool remove(SourceRange TokenRange) {
96+
return remove(CharSourceRange::getTokenRange(TokenRange));
97+
}
98+
bool replace(SourceRange TokenRange, StringRef text) {
99+
return replace(CharSourceRange::getTokenRange(TokenRange), text);
100+
}
101+
bool replaceWithInner(SourceRange TokenRange, SourceRange TokenInnerRange) {
102+
return replaceWithInner(CharSourceRange::getTokenRange(TokenRange),
103+
CharSourceRange::getTokenRange(TokenInnerRange));
104+
}
105+
106+
typedef SmallVector<Edit, 8>::const_iterator edit_iterator;
107+
edit_iterator edit_begin() const { return CachedEdits.begin(); }
108+
edit_iterator edit_end() const { return CachedEdits.end(); }
109+
110+
private:
111+
void addInsert(SourceLocation OrigLoc,
112+
FileOffset Offs, StringRef text, bool beforePreviousInsertions);
113+
void addInsertFromRange(SourceLocation OrigLoc, FileOffset Offs,
114+
FileOffset RangeOffs, unsigned RangeLen,
115+
bool beforePreviousInsertions);
116+
void addRemove(SourceLocation OrigLoc, FileOffset Offs, unsigned Len);
117+
118+
bool canInsert(SourceLocation loc, FileOffset &Offset);
119+
bool canInsertAfterToken(SourceLocation loc, FileOffset &Offset,
120+
SourceLocation &AfterLoc);
121+
bool canInsertInOffset(SourceLocation OrigLoc, FileOffset Offs);
122+
bool canRemoveRange(CharSourceRange range, FileOffset &Offs, unsigned &Len);
123+
bool canReplaceText(SourceLocation loc, StringRef text,
124+
FileOffset &Offs, unsigned &Len);
125+
126+
void commitInsert(FileOffset offset, StringRef text,
127+
bool beforePreviousInsertions);
128+
void commitRemove(FileOffset offset, unsigned length);
129+
130+
bool isAtStartOfMacroExpansion(SourceLocation loc,
131+
SourceLocation *MacroBegin = 0) const;
132+
bool isAtEndOfMacroExpansion(SourceLocation loc,
133+
SourceLocation *MacroEnd = 0) const;
134+
};
135+
136+
}
137+
138+
} // end namespace clang
139+
140+
#endif

0 commit comments

Comments
 (0)