Skip to content

Commit c3d4436

Browse files
Introduce a Target class and use it to define generic 32 and 64bit variants.
1 parent 2bc7a77 commit c3d4436

File tree

5 files changed

+144
-9
lines changed

5 files changed

+144
-9
lines changed

python/.style.yapf .style.yapf

File renamed without changes.

pytest/Compiler/constants32.py

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
# RUN: %PYTHON %s | npcomp-opt -split-input-file | FileCheck %s --dump-input=fail
2+
3+
# Subset of constant tests which verify against a GenericTarget32.
4+
5+
from npcomp.compiler.frontend import *
6+
from npcomp.compiler.target import *
7+
8+
9+
def import_global(f):
10+
fe = ImportFrontend(target_factory=GenericTarget32)
11+
fe.import_global_function(f)
12+
print("// -----")
13+
print(fe.ir_module.to_asm())
14+
return f
15+
16+
17+
# CHECK-LABEL: func @integer_constants
18+
@import_global
19+
def integer_constants():
20+
# CHECK: %[[A:.*]] = constant 100 : i32
21+
a = 100
22+
return a
23+
24+
25+
# CHECK-LABEL: func @float_constants
26+
@import_global
27+
def float_constants():
28+
# CHECK: %[[A:.*]] = constant 2.200000e+00 : f32
29+
a = 2.2
30+
return a

python/npcomp/compiler/frontend.py

+15-3
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414

1515
from . import logging
1616
from .importer import *
17+
from .target import *
1718

1819
__all__ = [
1920
"ImportFrontend",
@@ -32,12 +33,21 @@ def __init__(self, *args, **kwargs):
3233

3334
class ImportFrontend:
3435
"""Frontend for importing various entities into a Module."""
35-
36-
def __init__(self, ir_context: ir.MLIRContext = None):
36+
__slots__ = [
37+
"_ir_context",
38+
"_ir_module",
39+
"_helper",
40+
"_target_factory",
41+
]
42+
43+
def __init__(self,
44+
ir_context: ir.MLIRContext = None,
45+
target_factory: TargetFactory = GenericTarget64):
3746
self._ir_context = ir.MLIRContext() if not ir_context else ir_context
3847
self._ir_module = self._ir_context.new_module()
3948
self._helper = AllDialectHelper(self._ir_context,
4049
ir.OpBuilder(self._ir_context))
50+
self._target_factory = target_factory
4151

4252
@property
4353
def ir_context(self):
@@ -66,6 +76,7 @@ def import_global_function(self, f):
6676
h = self.ir_h
6777
ir_c = self.ir_context
6878
ir_m = self.ir_module
79+
target = self._target_factory(h)
6980
filename = inspect.getsourcefile(f)
7081
source_lines, start_lineno = inspect.getsourcelines(f)
7182
source = "".join(source_lines)
@@ -94,7 +105,8 @@ def import_global_function(self, f):
94105
fctx = FunctionContext(ir_c=ir_c,
95106
ir_f=ir_f,
96107
ir_h=h,
97-
filename_ident=filename_ident)
108+
filename_ident=filename_ident,
109+
target=target)
98110
for f_arg, ir_arg in zip(f_params, ir_f.first_block.args):
99111
fctx.map_local_name(f_arg, ir_arg)
100112

python/npcomp/compiler/importer.py

+10-6
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
from _npcomp.mlir import ir
1111

1212
from . import logging
13+
from .target import *
1314

1415
__all__ = [
1516
"FunctionContext",
@@ -24,14 +25,16 @@ class FunctionContext:
2425
"ir_c",
2526
"ir_f",
2627
"ir_h",
28+
"target",
2729
"filename_ident",
2830
"local_name_value_map",
2931
]
3032

31-
def __init__(self, ir_c, ir_f, ir_h, filename_ident):
33+
def __init__(self, ir_c, ir_f, ir_h, target, filename_ident):
3234
self.ir_c = ir_c
3335
self.ir_f = ir_f
3436
self.ir_h = ir_h
37+
self.target = target
3538
self.filename_ident = filename_ident
3639
self.local_name_value_map = dict()
3740

@@ -93,7 +96,8 @@ def import_body(self):
9396
if not self._last_was_return:
9497
# Add a default terminator.
9598
none_value = ir_h.basicpy_singleton_op(ir_h.basicpy_NoneType).result
96-
none_cast = ir_h.basicpy_unknown_cast_op(ir_h.basicpy_UnknownType, none_value).result
99+
none_cast = ir_h.basicpy_unknown_cast_op(ir_h.basicpy_UnknownType,
100+
none_value).result
97101
ir_h.return_op([none_cast])
98102

99103
def visit_Assign(self, ast_node):
@@ -137,6 +141,8 @@ class ExpressionImporter(BaseNodeVisitor):
137141
def __init__(self, fctx):
138142
super().__init__(fctx)
139143
self.value = None
144+
self._int_type = fctx.target.impl_int_type
145+
self._float_type = fctx.target.impl_float_type
140146

141147
def visit(self, node):
142148
super().visit(node)
@@ -158,13 +164,11 @@ def emit_constant(self, value):
158164
elif value is None:
159165
self.value = ir_h.basicpy_singleton_op(ir_h.basicpy_NoneType).result
160166
elif isinstance(value, int):
161-
# TODO: Configurable type mapping
162-
ir_type = ir_h.i64_type
167+
ir_type = self._int_type
163168
ir_attr = ir_c.integer_attr(ir_type, value)
164169
self.value = ir_h.constant_op(ir_type, ir_attr).result
165170
elif isinstance(value, float):
166-
# TODO: Configurable type mapping
167-
ir_type = ir_h.f64_type
171+
ir_type = self._float_type
168172
ir_attr = ir_c.float_attr(ir_type, value)
169173
self.value = ir_h.constant_op(ir_type, ir_attr).result
170174
elif isinstance(value, str):

python/npcomp/compiler/target.py

+89
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
# Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
2+
# See https://llvm.org/LICENSE.txt for license information.
3+
# SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
4+
5+
from typing import *
6+
from _npcomp.mlir import ir
7+
8+
__all__ = [
9+
"GenericTarget32",
10+
"GenericTarget64",
11+
"Target",
12+
"TargetFactory",
13+
]
14+
15+
16+
class Target:
17+
"""
18+
Abstract class providing configuration and hooks for a specific compilation
19+
target.
20+
"""
21+
__slots__ = [
22+
"_mlir_helper",
23+
]
24+
25+
def __init__(self, mlir_helper: ir.DialectHelper):
26+
super().__init__()
27+
self._mlir_helper = mlir_helper
28+
29+
@property
30+
def mlir_helper(self):
31+
return self._mlir_helper
32+
33+
@property
34+
def mlir_context(self):
35+
return self._mlir_helper.context
36+
37+
@property
38+
def target_name(self) -> str:
39+
return NotImplementedError()
40+
41+
@property
42+
def impl_int_type(self) -> ir.Type:
43+
"""Gets the default int type for the backend for the Python 'int' type."""
44+
raise NotImplementedError()
45+
46+
@property
47+
def impl_float_type(self) -> ir.Type:
48+
"""Gets the implementation's type for the python 'float' type."""
49+
raise NotImplementedError()
50+
51+
52+
class GenericTarget64(Target):
53+
"""A generic 64 bit target."""
54+
55+
@property
56+
def target_name(self) -> str:
57+
return "generic64"
58+
59+
@property
60+
def impl_int_type(self) -> ir.Type:
61+
"""Gets the default int type for the backend for the Python 'int' type."""
62+
return self.mlir_helper.i64_type
63+
64+
@property
65+
def impl_float_type(self) -> ir.Type:
66+
"""Gets the implementation's type for the python 'float' type."""
67+
return self.mlir_helper.f64_type
68+
69+
70+
class GenericTarget32(Target):
71+
"""A generic 32 bit target (uses 32bit ints and floats)."""
72+
73+
@property
74+
def target_name(self) -> str:
75+
return "generic32"
76+
77+
@property
78+
def impl_int_type(self) -> ir.Type:
79+
"""Gets the default int type for the backend for the Python 'int' type."""
80+
return self.mlir_helper.i32_type
81+
82+
@property
83+
def impl_float_type(self) -> ir.Type:
84+
"""Gets the implementation's type for the python 'float' type."""
85+
return self.mlir_helper.f32_type
86+
87+
88+
# Factory for producing a target (matches the Target constructor).
89+
TargetFactory = Callable[[ir.DialectHelper], Target]

0 commit comments

Comments
 (0)