Skip to content

Commit 1ab76cb

Browse files
authored
Issue #274: Port the fixup-linkage script. (#497)
* Issue #274: Port the fixup-linkage script. This rewrites the fixup-linkage.pl Perl script in C++. The interface is changed slightly so that the pre- and post-processed LLVM files are kept when `--save-temps` is specified. * run clang-format * Remove perl script. * Move space to replacement string.
1 parent af2354c commit 1ab76cb

File tree

7 files changed

+104
-73
lines changed

7 files changed

+104
-73
lines changed

test/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ if (NOT CUDAQ_DISABLE_CPP_FRONTEND)
2727
cudaq-quake
2828
cudaq-opt
2929
cudaq-translate
30+
fixup-linkage
3031

3132
CircuitCheck
3233
FileCheck)

tools/CMakeLists.txt

+3-2
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,8 @@ add_subdirectory(cudaq-opt)
1010
add_subdirectory(cudaq-translate)
1111

1212
if (NOT CUDAQ_DISABLE_CPP_FRONTEND)
13+
add_subdirectory(cudaq-lsp-server)
1314
add_subdirectory(cudaq-quake)
15+
add_subdirectory(fixup-linkage)
1416
add_subdirectory(nvqpp)
15-
add_subdirectory(cudaq-lsp-server)
16-
endif()
17+
endif()

tools/fixup-linkage/CMakeLists.txt

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
# ============================================================================ #
2+
# Copyright (c) 2022 - 2023 NVIDIA Corporation & Affiliates. #
3+
# All rights reserved. #
4+
# #
5+
# This source code and the accompanying materials are made available under #
6+
# the terms of the Apache License 2.0 which accompanies this distribution. #
7+
# ============================================================================ #
8+
9+
add_llvm_executable(fixup-linkage fixup-linkage.cpp)
10+
11+
install(TARGETS fixup-linkage DESTINATION bin)

tools/fixup-linkage/fixup-linkage.cpp

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
/*******************************************************************************
2+
* Copyright (c) 2022 - 2023 NVIDIA Corporation & Affiliates. *
3+
* All rights reserved. *
4+
* *
5+
* This source code and the accompanying materials are made available under *
6+
* the terms of the Apache License 2.0 which accompanies this distribution. *
7+
******************************************************************************/
8+
9+
/// The fixup-linkage tool is used to rewrite the LLVM IR produced by clang for
10+
/// the classical compute code such that it can be linked correctly with the
11+
/// LLVM IR that is generated for the quantum code. This avoids linker errors
12+
/// such as "duplicate symbol definition".
13+
14+
#include <fstream>
15+
#include <iostream>
16+
#include <regex>
17+
18+
int main(int argc, char *argv[]) {
19+
if (argc != 4) {
20+
std::cerr << "Usage:\n\tfixup-linkage <Quake-file> <LLVM-file> <output>\n";
21+
return 1;
22+
}
23+
24+
// 1. Look for all the mangled kernel names. These will be found in the
25+
// mangled_name_map in the quake file. Add these names to `funcs`.
26+
std::ifstream modFile(argv[1]);
27+
std::string line;
28+
std::vector<std::string> funcs;
29+
{
30+
std::regex mapRegex{"quake\\.mangled_name_map[^\"]*"};
31+
std::regex stringRegex{"\"(.*?)\""};
32+
while (std::getline(modFile, line) && funcs.empty()) {
33+
auto funcsBegin =
34+
std::sregex_iterator(line.begin(), line.end(), mapRegex);
35+
auto rgxEnd = std::sregex_iterator();
36+
if (funcsBegin == rgxEnd)
37+
continue;
38+
auto names = line.substr(funcsBegin->str().size() - 1);
39+
auto namesBegin =
40+
std::sregex_iterator(names.begin(), names.end(), stringRegex);
41+
for (std::sregex_iterator i = namesBegin; i != rgxEnd; ++i) {
42+
auto s = i->str();
43+
funcs.push_back(s.substr(1, s.size() - 2));
44+
}
45+
}
46+
modFile.close();
47+
if (funcs.empty()) {
48+
std::cerr << "No mangled name map in the quake file.\n";
49+
return 1;
50+
}
51+
}
52+
53+
// 2. Scan the LLVM file looking for the mangled kernel names. Where these
54+
// kernels are defined, they have their linkage modified to `linkonce_odr` if
55+
// that is not already the linkage. This change will prevent the duplicate
56+
// symbols defined error from the linker.
57+
std::ifstream llFile(argv[2]);
58+
std::ofstream outFile(argv[3]);
59+
std::regex filterRegex("^define (dso_local|internal) ");
60+
auto rgxEnd = std::sregex_iterator();
61+
while (std::getline(llFile, line)) {
62+
auto iter = std::sregex_iterator(line.begin(), line.end(), filterRegex);
63+
if (iter == rgxEnd) {
64+
outFile << line << std::endl;
65+
continue;
66+
}
67+
bool replaced = false;
68+
for (auto fn : funcs) {
69+
auto pos = line.find(fn);
70+
if (pos == std::string::npos)
71+
continue;
72+
auto ms = (*iter)[1].str();
73+
pos = (ms == "internal") ? sizeof("define internal")
74+
: sizeof("define dso_local");
75+
outFile << "define linkonce_odr dso_preemptable " << line.substr(pos)
76+
<< std::endl;
77+
replaced = true;
78+
break;
79+
}
80+
if (!replaced)
81+
outFile << line << std::endl;
82+
}
83+
llFile.close();
84+
outFile.close();
85+
return 0;
86+
}

tools/nvqpp/CMakeLists.txt

-3
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,8 @@ endif()
1818

1919
configure_file("nvq++.in"
2020
"${CUDAQ_BINARY_DIR}/bin/nvq++" @ONLY)
21-
configure_file("fixup-linkage.pl"
22-
"${CUDAQ_BINARY_DIR}/bin/fixup-linkage.pl" @ONLY)
2321

2422
install(PROGRAMS ${CUDAQ_BINARY_DIR}/bin/nvq++
25-
${CUDAQ_BINARY_DIR}/bin/fixup-linkage.pl
2623
DESTINATION bin)
2724
install(FILES backendConfig.cpp DESTINATION targets/)
2825
file(COPY backendConfig.cpp DESTINATION ${CMAKE_BINARY_DIR}/targets/)

tools/nvqpp/fixup-linkage.pl

-66
This file was deleted.

tools/nvqpp/nvq++.in

+3-2
Original file line numberDiff line numberDiff line change
@@ -520,8 +520,9 @@ for i in ${SRCS}; do
520520
fi
521521

522522
# Rewrite internal linkages so we can override the function.
523-
export PERL_BADLANG=0
524-
run ${install_dir}/bin/fixup-linkage.pl ${file}.qke ${file}.ll
523+
mv ${file}.ll ${file}.pre.ll
524+
TMPFILES="${TMPFILES} ${file}.pre.ll"
525+
run ${install_dir}/bin/fixup-linkage ${file}.qke ${file}.pre.ll ${file}.ll
525526

526527
# Lower our LLVM to object files
527528
run ${LLC} --relocation-model=pic --filetype=obj -O2 ${QUAKELL_FILE} -o ${file}.qke.o

0 commit comments

Comments
 (0)