2 // Copyright 2012 Francisco Jerez
4 // Permission is hereby granted, free of charge, to any person obtaining a
5 // copy of this software and associated documentation files (the "Software"),
6 // to deal in the Software without restriction, including without limitation
7 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 // and/or sell copies of the Software, and to permit persons to whom the
9 // Software is furnished to do so, subject to the following conditions:
11 // The above copyright notice and this permission notice shall be included in
12 // all copies or substantial portions of the Software.
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 // OTHER DEALINGS IN THE SOFTWARE.
23 #include "core/program.hpp"
24 #include "llvm/invocation.hpp"
25 #include "tgsi/invocation.hpp"
27 using namespace clover
;
29 program::program(clover::context
&ctx
, const std::string
&source
) :
30 has_source(true), context(ctx
), _source(source
), _kernel_ref_counter(0) {
33 program::program(clover::context
&ctx
,
34 const ref_vector
<device
> &devs
,
35 const std::vector
<module
> &binaries
) :
36 has_source(false), context(ctx
),
37 _devices(devs
), _kernel_ref_counter(0) {
38 for_each([&](device
&dev
, const module
&bin
) {
39 _builds
[&dev
] = { bin
};
45 program::compile(const ref_vector
<device
> &devs
, const std::string
&opts
,
46 const header_map
&headers
) {
50 for (auto &dev
: devs
) {
54 const module m
= (dev
.ir_format() == PIPE_SHADER_IR_TGSI
?
55 tgsi::compile_program(_source
, log
) :
56 llvm::compile_program(_source
, headers
, dev
,
58 _builds
[&dev
] = { m
, opts
, log
};
60 _builds
[&dev
] = { module(), opts
, log
};
68 program::link(const ref_vector
<device
> &devs
, const std::string
&opts
,
69 const ref_vector
<program
> &progs
) {
72 for (auto &dev
: devs
) {
73 const std::vector
<module
> ms
= map([&](const program
&prog
) {
74 return prog
.build(dev
).binary
;
76 std::string log
= _builds
[&dev
].log
;
79 const module m
= (dev
.ir_format() == PIPE_SHADER_IR_TGSI
?
80 tgsi::link_program(ms
) :
81 llvm::link_program(ms
, dev
, opts
, log
));
82 _builds
[&dev
] = { m
, opts
, log
};
84 _builds
[&dev
] = { module(), opts
, log
};
91 program::source() const {
96 program::devices() const {
97 return map(evals(), _devices
);
101 program::build::status() const {
102 if (!binary
.secs
.empty())
103 return CL_BUILD_SUCCESS
;
105 return CL_BUILD_ERROR
;
107 return CL_BUILD_NONE
;
110 cl_program_binary_type
111 program::build::binary_type() const {
112 if (any_of(type_equals(module::section::text_intermediate
), binary
.secs
))
113 return CL_PROGRAM_BINARY_TYPE_COMPILED_OBJECT
;
114 else if (any_of(type_equals(module::section::text_library
), binary
.secs
))
115 return CL_PROGRAM_BINARY_TYPE_LIBRARY
;
116 else if (any_of(type_equals(module::section::text_executable
), binary
.secs
))
117 return CL_PROGRAM_BINARY_TYPE_EXECUTABLE
;
119 return CL_PROGRAM_BINARY_TYPE_NONE
;
122 const struct program::build
&
123 program::build(const device
&dev
) const {
124 static const struct build null
;
125 return _builds
.count(&dev
) ? _builds
.find(&dev
)->second
: null
;
128 const std::vector
<module::symbol
> &
129 program::symbols() const {
131 throw error(CL_INVALID_PROGRAM_EXECUTABLE
);
133 return _builds
.begin()->second
.binary
.syms
;
137 program::kernel_ref_count() const {
138 return _kernel_ref_counter
.ref_count();