view tools/binding/llvm-typemonitor.cpp @ 1491:360a8e8eea51

Teach stack promotion to walk the CFG when a potential reuse of an allocation is found to see if it can actually happen instead of just assuming it will. This allows it to catch cases like {{{ int i; Foo f; while (cond(i)) f = new Foo(i++); return f.value; }}} where it previously wouldn't because a phi using the allocation would appear in the condition block to propagate it to the use after the loop.
author Frits van Bommel <fvbommel wxs.nl>
date Thu, 11 Jun 2009 02:04:44 +0200
parents 4ff9ab0d472c
children
line wrap: on
line source

/// Support for callbacks when an abstract type becomes more concrete.

#include "llvm/Support/Streams.h"
#include "llvm/Type.h"
#include "llvm-c/Core.h"

using namespace llvm;

extern "C" typedef int (*RefineCallback)(void *handle, LLVMTypeRef newT);

class TypeMonitor : AbstractTypeUser {
    void *handle_;
    RefineCallback callback_;
    
    void onRefineType(const Type* oldT, const Type* newT) {
        callback_(handle_, wrap(newT));
        oldT->removeAbstractTypeUser(this);
        delete this;
    }
    
    public:
    
    TypeMonitor(Type* T, void *handle, RefineCallback callback)
    : handle_(handle), callback_(callback) {
        T->addAbstractTypeUser(this);
    }
    
    virtual void refineAbstractType(const DerivedType *OldTy, const Type *NewTy) {
        onRefineType(OldTy, NewTy);
    }
    
    virtual void typeBecameConcrete(const DerivedType *AbsTy) {
        onRefineType(AbsTy, AbsTy);
    }
    
    virtual void dump() const {
        cerr << "<TypeMonitor>";
    }
};

extern "C" void LLVMRegisterAbstractTypeCallback(LLVMTypeRef T,
                                                 void *handle,
                                                 RefineCallback callback)
{
    new TypeMonitor(unwrap(T), handle, callback);
}