diff --git a/divine/cc/link.hpp b/divine/cc/link.hpp
new file mode 100644
index 0000000000000000000000000000000000000000..473600aac11baa2cb8c7fcddd980364d21144c3a
--- /dev/null
+++ b/divine/cc/link.hpp
@@ -0,0 +1,119 @@
+// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 4 -*-
+
+/*
+ * (c) 2017 Jan Horáček <me@apophis.cz>
+ * (c) 2017-2019 Zuzana Baranová <xbaranov@fi.muni.cz>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+ 
+#include <memory>
+#include <divine/cc/cc1.hpp>
+#include <divine/cc/filetype.hpp>
+#include <divine/vm/xg-code.hpp>
+
+DIVINE_RELAX_WARNINGS
+#include <llvm/Bitcode/BitcodeReader.h>
+#include <llvm/IR/Function.h>
+#include <llvm/IR/GlobalVariable.h>
+#include <llvm/IR/Module.h>
+#include <llvm/IR/Verifier.h>
+#include <llvm/Object/IRObjectFile.h>
+#include <llvm/Support/MemoryBuffer.h>
+DIVINE_UNRELAX_WARNINGS
+
+#include <brick-string>
+using namespace llvm;
+
+namespace divine::cc
+{
+    using PairedFiles = std::vector< std::pair< std::string, std::string > >;
+
+    bool whitelisted( llvm::Function &f )
+    {
+        using brick::string::startsWith;
+        using vm::xg::hypercall;
+
+        auto n = f.getName();
+        return hypercall( &f ) != vm::lx::NotHypercall ||
+               startsWith( n, "__dios_" ) ||
+               startsWith( n, "_ZN6__dios" ) ||
+               startsWith( n, "_Unwind_" ) ||
+               n == "setjmp" || n == "longjmp";
+    }
+
+    bool whitelisted( llvm::GlobalVariable &gv )
+    {
+        return brick::string::startsWith( gv.getName(), "__md_" );
+    }
+
+    template < typename Driver, bool link_dios >
+    std::unique_ptr< llvm::Module > link_bitcode( PairedFiles& files, cc::CC1& clang,
+                                                  std::vector< std::string > libSearchPath )
+    {
+        auto drv = std::make_unique< Driver >( clang.context() );
+        for( auto path : libSearchPath )
+            drv->addDirectory( path );
+
+        for ( auto file : files )
+        {
+            if ( !is_object_type( file.second ) )
+            {
+                if ( file.first != "lib" )
+                    continue;
+                else
+                {
+                    drv->linkLib( file.second, libSearchPath );
+                    continue;
+                }
+            }
+
+            ErrorOr< std::unique_ptr< MemoryBuffer > > buf = MemoryBuffer::getFile( file.second );
+            if ( !buf ) throw cc::CompileError( "Error parsing file " + file.second + " into MemoryBuffer" );
+
+            if ( is_type( file.second, FileType::Archive ) )
+            {
+                drv->linkArchive( std::move( buf.get() ) , clang.context() );
+                continue;
+            }
+
+            auto bc = llvm::object::IRObjectFile::findBitcodeInMemBuffer( (*buf.get()).getMemBufferRef() );
+            if ( !bc ) std::cerr << "No .llvmbc section found in file " << file.second << "." << std::endl;
+
+            auto expected_m = llvm::parseBitcodeFile( bc.get(), *clang.context().get() );
+            if ( !expected_m )
+                std::cerr << "Error parsing bitcode." << std::endl;
+            auto m = std::move( expected_m.get() );
+            m->setTargetTriple( "x86_64-unknown-none-elf" );
+            verifyModule( *m );
+            drv->link( std::move( m ) );
+        }
+
+        if constexpr ( link_dios )
+            drv->linkLibs( Driver::defaultDIVINELibs );
+
+        auto m = drv->takeLinked();
+
+        for ( auto& func : *m )
+            if ( func.isDeclaration() && !whitelisted( func ) )
+                throw cc::CompileError( "Symbol undefined (function): " + func.getName().str() );
+
+        for ( auto& val : m->globals() )
+            if ( auto G = dyn_cast< llvm::GlobalVariable >( &val ) )
+                if ( !G->hasInitializer() && !whitelisted( *G ) )
+                    throw cc::CompileError( "Symbol undefined (global variable): " + G->getName().str() );
+
+        verifyModule( *m );
+        return m;
+    }
+}
diff --git a/tools/divcc.cpp b/tools/divcc.cpp
index ba4507f30ed5185894d53b3af72ce3de1400df28..639150eb2b39c22d9d9fc4010d8a60a58806ed0c 100644
--- a/tools/divcc.cpp
+++ b/tools/divcc.cpp
@@ -20,11 +20,11 @@
 #include <divine/cc/cc1.hpp>
 #include <divine/cc/codegen.hpp>
 #include <divine/cc/filetype.hpp>
+#include <divine/cc/link.hpp>
 #include <divine/cc/options.hpp>
 #include <divine/rt/dios-cc.hpp>
 #include <divine/rt/runtime.hpp>
 #include <divine/ui/version.hpp>
-#include <divine/vm/xg-code.hpp>
 
 DIVINE_RELAX_WARNINGS
 #include "llvm/Target/TargetMachine.h"
@@ -68,84 +68,6 @@ void addSection( std::string filepath, std::string sectionName, const std::strin
                         + ", objcopy exited with " + to_string( r ) );
 }
 
-bool whitelisted( llvm::Function &f )
-{
-    using brick::string::startsWith;
-    using vm::xg::hypercall;
-
-    auto n = f.getName();
-    return hypercall( &f ) != vm::lx::NotHypercall ||
-           startsWith( n, "__dios_" ) ||
-           startsWith( n, "_ZN6__dios" ) ||
-           startsWith( n, "_Unwind_" ) ||
-           n == "setjmp" || n == "longjmp";
-}
-
-bool whitelisted( llvm::GlobalVariable &gv )
-{
-    return brick::string::startsWith( gv.getName(), "__md_" );
-}
-
-template < typename Driver, bool link_dios >
-std::unique_ptr< llvm::Module > link_bitcode( PairedFiles& files, cc::CC1& clang,
-                                              std::vector< std::string > libSearchPath )
-{
-    auto drv = std::make_unique< Driver >( clang.context() );
-    for( auto path : libSearchPath )
-        drv->addDirectory( path );
-
-    for ( auto file : files )
-    {
-        if ( !is_object_type( file.second ) )
-        {
-            if ( file.first != "lib" )
-                continue;
-            else
-            {
-                drv->linkLib( file.second, libSearchPath );
-                continue;
-            }
-        }
-
-        ErrorOr< std::unique_ptr< MemoryBuffer > > buf = MemoryBuffer::getFile( file.second );
-        if ( !buf ) throw cc::CompileError( "Error parsing file " + file.second + " into MemoryBuffer" );
-
-        if ( is_type( file.second, FileType::Archive ) )
-        {
-            drv->linkArchive( std::move( buf.get() ) , clang.context() );
-            continue;
-        }
-
-        auto bc = llvm::object::IRObjectFile::findBitcodeInMemBuffer( (*buf.get()).getMemBufferRef() );
-        if ( !bc ) std::cerr << "No .llvmbc section found in file " << file.second << "." << std::endl;
-
-        auto expected_m = llvm::parseBitcodeFile( bc.get(), *clang.context().get() );
-        if ( !expected_m )
-            std::cerr << "Error parsing bitcode." << std::endl;
-        auto m = std::move( expected_m.get() );
-        m->setTargetTriple( "x86_64-unknown-none-elf" );
-        verifyModule( *m );
-        drv->link( std::move( m ) );
-    }
-
-    if constexpr ( link_dios )
-        drv->linkLibs( Driver::defaultDIVINELibs );
-
-    auto m = drv->takeLinked();
-
-    for ( auto& func : *m )
-        if ( func.isDeclaration() && !whitelisted( func ) )
-            throw cc::CompileError( "Symbol undefined (function): " + func.getName().str() );
-
-    for ( auto& val : m->globals() )
-        if ( auto G = dyn_cast< llvm::GlobalVariable >( &val ) )
-            if ( !G->hasInitializer() && !whitelisted( *G ) )
-                throw cc::CompileError( "Symbol undefined (global variable): " + G->getName().str() );
-
-    verifyModule( *m );
-    return m;
-}
-
 int compile( cc::ParsedOpts& po, cc::CC1& clang, PairedFiles& objFiles )
 {
     for ( auto file : objFiles )
@@ -243,7 +165,7 @@ int compile_and_link( cc::ParsedOpts& po, cc::CC1& clang, PairedFiles& objFiles,
         if ( !linked )
             throw cc::CompileError( "lld failed, not linked" );
     }
-    std::unique_ptr< llvm::Module > mod = link_bitcode< rt::DiosCC, true >( objFiles, clang, po.libSearchPath );
+    std::unique_ptr< llvm::Module > mod = cc::link_bitcode< rt::DiosCC, true >( objFiles, clang, po.libSearchPath );
     std::string file_out = po.outputFile != "" ? po.outputFile : "a.out";
 
     addSection( file_out, cc::llvm_section_name, clang.serializeModule( *mod ) );