summaryrefslogtreecommitdiffstats
path: root/clang/lib/CodeGen/CodeGenModule.cpp
diff options
context:
space:
mode:
authorEli Friedman <eli.friedman@gmail.com>2008-05-26 12:59:39 +0000
committerEli Friedman <eli.friedman@gmail.com>2008-05-26 12:59:39 +0000
commitdf649f3da55f6efebfec2114c6392961d506f9f8 (patch)
tree10c5db5495e8e6e7534c8c0b49fb6911ea2c3499 /clang/lib/CodeGen/CodeGenModule.cpp
parentd3f21d165f89eae9021d1f3026455754233ce6db (diff)
downloadbcm5719-llvm-df649f3da55f6efebfec2114c6392961d506f9f8.tar.gz
bcm5719-llvm-df649f3da55f6efebfec2114c6392961d506f9f8.zip
Emit memmove, not memcpy, for structure copies; this is unfortunately
required for correctness in cases of copying a struct to itself or to an overlapping struct (itself for cases like *a = *a, and overlapping is possible with unions). Hopefully, this won't end up being a perf issue; LLVM *should* be able to optimize memmove to memcpy in a lot of cases, and for small copies the generated code *should* be mostly comparable. (In reality, LLVM is currently horrible at optimizing memmove, but that's a bug, not a fundamental issue.) gcc currently generates wrong code; that's http://gcc.gnu.org/bugzilla/show_bug.cgi?id=32667. llvm-svn: 51566
Diffstat (limited to 'clang/lib/CodeGen/CodeGenModule.cpp')
-rw-r--r--clang/lib/CodeGen/CodeGenModule.cpp14
1 files changed, 13 insertions, 1 deletions
diff --git a/clang/lib/CodeGen/CodeGenModule.cpp b/clang/lib/CodeGen/CodeGenModule.cpp
index ec61d667eed..329599c1166 100644
--- a/clang/lib/CodeGen/CodeGenModule.cpp
+++ b/clang/lib/CodeGen/CodeGenModule.cpp
@@ -35,7 +35,8 @@ CodeGenModule::CodeGenModule(ASTContext &C, const LangOptions &LO,
llvm::Module &M, const llvm::TargetData &TD,
Diagnostic &diags, bool GenerateDebugInfo)
: Context(C), Features(LO), TheModule(M), TheTargetData(TD), Diags(diags),
- Types(C, M, TD), MemCpyFn(0), MemSetFn(0), CFConstantStringClassRef(0) {
+ Types(C, M, TD), MemCpyFn(0), MemMoveFn(0), MemSetFn(0),
+ CFConstantStringClassRef(0) {
//TODO: Make this selectable at runtime
Runtime = CreateObjCRuntime(M,
getTypes().ConvertType(getContext().IntTy),
@@ -589,6 +590,17 @@ llvm::Function *CodeGenModule::getMemCpyFn() {
return MemCpyFn = getIntrinsic(IID);
}
+llvm::Function *CodeGenModule::getMemMoveFn() {
+ if (MemMoveFn) return MemMoveFn;
+ llvm::Intrinsic::ID IID;
+ switch (Context.Target.getPointerWidth(0)) {
+ default: assert(0 && "Unknown ptr width");
+ case 32: IID = llvm::Intrinsic::memmove_i32; break;
+ case 64: IID = llvm::Intrinsic::memmove_i64; break;
+ }
+ return MemMoveFn = getIntrinsic(IID);
+}
+
llvm::Function *CodeGenModule::getMemSetFn() {
if (MemSetFn) return MemSetFn;
llvm::Intrinsic::ID IID;
OpenPOWER on IntegriCloud