-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathstructs.py
More file actions
159 lines (137 loc) · 6.28 KB
/
structs.py
File metadata and controls
159 lines (137 loc) · 6.28 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
from __future__ import annotations
from dataclasses import dataclass, field
from os import environ, system
from pathlib import Path
from sys import executable, platform as sys_platform
from sysconfig import get_path
from typing import Literal
from hatchling.builders.config import BuilderConfig
__all__ = (
"HatchCppBuildConfig",
"HatchCppLibrary",
"HatchCppPlatform",
"HatchCppBuildPlan",
)
Platform = Literal["linux", "darwin", "win32"]
CompilerToolchain = Literal["gcc", "clang", "msvc"]
PlatformDefaults = {
"linux": {"CC": "gcc", "CXX": "g++"},
"darwin": {"CC": "clang", "CXX": "clang++"},
"win32": {"CC": "cl", "CXX": "cl"},
}
@dataclass
class HatchCppBuildConfig(BuilderConfig):
"""Build config values for Hatch C++ Builder."""
toolchain: str | None = field(default="raw")
libraries: list[dict[str, str]] = field(default_factory=list)
verbose: bool | None = field(default=False)
# build_function: str | None = None
# build_kwargs: t.Mapping[str, str] = field(default_factory=dict)
# editable_build_kwargs: t.Mapping[str, str] = field(default_factory=dict)
# ensured_targets: list[str] = field(default_factory=list)
# skip_if_exists: list[str] = field(default_factory=list)
@dataclass
class HatchCppLibrary(object):
"""A C++ library."""
name: str
sources: list[str]
include_dirs: list[str] = field(default_factory=list)
library_dirs: list[str] = field(default_factory=list)
libraries: list[str] = field(default_factory=list)
extra_compile_args: list[str] = field(default_factory=list)
extra_link_args: list[str] = field(default_factory=list)
extra_objects: list[str] = field(default_factory=list)
define_macros: list[str] = field(default_factory=list)
undef_macros: list[str] = field(default_factory=list)
export_symbols: list[str] = field(default_factory=list)
depends: list[str] = field(default_factory=list)
@dataclass
class HatchCppPlatform(object):
cc: str
cxx: str
platform: Platform
toolchain: CompilerToolchain
@staticmethod
def default() -> HatchCppPlatform:
platform = environ.get("HATCH_CPP_PLATFORM", sys_platform)
CC = environ.get("CC", PlatformDefaults[platform]["CC"])
CXX = environ.get("CXX", PlatformDefaults[platform]["CXX"])
if "gcc" in CC and "g++" in CXX:
toolchain = "gcc"
elif "clang" in CC and "clang++" in CXX:
toolchain = "clang"
elif "cl" in CC and "cl" in CXX:
toolchain = "msvc"
else:
raise Exception(f"Unrecognized toolchain: {CC}, {CXX}")
return HatchCppPlatform(cc=CC, cxx=CXX, platform=platform, toolchain=toolchain)
def get_compile_flags(self, library: HatchCppLibrary) -> str:
flags = ""
if self.toolchain == "gcc":
flags = f"-I{get_path('include')}"
flags += " " + " ".join(f"-I{d}" for d in library.include_dirs)
flags += " -fPIC -shared"
flags += " " + " ".join(library.extra_compile_args)
flags += " " + " ".join(library.extra_link_args)
flags += " " + " ".join(library.extra_objects)
flags += " " + " ".join(f"-l{lib}" for lib in library.libraries)
flags += " " + " ".join(f"-L{lib}" for lib in library.library_dirs)
flags += " " + " ".join(f"-D{macro}" for macro in library.define_macros)
flags += " " + " ".join(f"-U{macro}" for macro in library.undef_macros)
flags += f" -o {library.name}.so"
elif self.toolchain == "clang":
flags = f"-I{get_path('include')} "
flags += " ".join(f"-I{d}" for d in library.include_dirs)
flags += " -undefined dynamic_lookup -fPIC -shared"
flags += " " + " ".join(library.extra_compile_args)
flags += " " + " ".join(library.extra_link_args)
flags += " " + " ".join(library.extra_objects)
flags += " " + " ".join(f"-l{lib}" for lib in library.libraries)
flags += " " + " ".join(f"-L{lib}" for lib in library.library_dirs)
flags += " " + " ".join(f"-D{macro}" for macro in library.define_macros)
flags += " " + " ".join(f"-U{macro}" for macro in library.undef_macros)
flags += f" -o {library.name}.so"
elif self.toolchain == "msvc":
flags = f"/I{get_path('include')} "
flags += " ".join(f"/I{d}" for d in library.include_dirs)
flags += " " + " ".join(library.extra_compile_args)
flags += " " + " ".join(library.extra_link_args)
flags += " " + " ".join(library.extra_objects)
flags += " " + " ".join(f"/D{macro}" for macro in library.define_macros)
flags += " " + " ".join(f"/U{macro}" for macro in library.undef_macros)
flags += " /EHsc /DWIN32 /LD"
flags += f" /Fo:{library.name}.obj"
flags += f" /Fe:{library.name}.pyd"
flags += " /link /DLL"
if (Path(executable).parent / "libs").exists():
flags += f" /LIBPATH:{str(Path(executable).parent / 'libs')}"
flags += " " + " ".join(f"{lib}.lib" for lib in library.libraries)
flags += " " + " ".join(f"/LIBPATH:{lib}" for lib in library.library_dirs)
# clean
while flags.count(" "):
flags = flags.replace(" ", " ")
return flags
def get_link_flags(self, library: HatchCppLibrary) -> str:
flags = ""
return flags
@dataclass
class HatchCppBuildPlan(object):
libraries: list[HatchCppLibrary] = field(default_factory=list)
platform: HatchCppPlatform = field(default_factory=HatchCppPlatform.default)
commands: list[str] = field(default_factory=list)
def generate(self):
self.commands = []
for library in self.libraries:
flags = self.platform.get_compile_flags(library)
self.commands.append(f"{self.platform.cc} {' '.join(library.sources)} {flags}")
return self.commands
def execute(self):
for command in self.commands:
system(command)
return self.commands
def cleanup(self):
if self.platform.platform == "win32":
for library in self.libraries:
temp_obj = Path(f"{library.name}.obj")
if temp_obj.exists():
temp_obj.unlink()