blob: f20ebb4fb5dd36c25b3f0b6a5b07889c9efe858a [file] [log] [blame]
Lei Zhang6fa3f8a2016-03-31 17:26:31 -04001// Copyright (c) 2015-2016 The Khronos Group Inc.
2//
David Neto9fc86582016-09-01 15:33:59 -04003// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
Lei Zhang6fa3f8a2016-03-31 17:26:31 -04006//
David Neto9fc86582016-09-01 15:33:59 -04007// http://www.apache.org/licenses/LICENSE-2.0
Lei Zhang6fa3f8a2016-03-31 17:26:31 -04008//
David Neto9fc86582016-09-01 15:33:59 -04009// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040014
dan sinclaireda2cfb2018-08-03 15:06:09 -040015#include "source/spirv_target_env.h"
16
Ryan Harrison9150cd42021-01-14 16:45:18 -050017#include <cassert>
Dejan Mircevski27030392016-05-11 09:48:52 -040018#include <cstring>
David Neto63f57d92019-05-07 12:27:18 -040019#include <string>
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040020
dan sinclaireda2cfb2018-08-03 15:06:09 -040021#include "source/spirv_constant.h"
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040022#include "spirv-tools/libspirv.h"
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040023
24const char* spvTargetEnvDescription(spv_target_env env) {
25 switch (env) {
26 case SPV_ENV_UNIVERSAL_1_0:
27 return "SPIR-V 1.0";
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040028 case SPV_ENV_VULKAN_1_0:
29 return "SPIR-V 1.0 (under Vulkan 1.0 semantics)";
Dejan Mircevskicb3c49e2016-04-07 14:41:34 -040030 case SPV_ENV_UNIVERSAL_1_1:
31 return "SPIR-V 1.1";
Pierre Moreau12447d82017-11-30 00:49:23 +010032 case SPV_ENV_OPENCL_1_2:
33 return "SPIR-V 1.0 (under OpenCL 1.2 Full Profile semantics)";
34 case SPV_ENV_OPENCL_EMBEDDED_1_2:
35 return "SPIR-V 1.0 (under OpenCL 1.2 Embedded Profile semantics)";
36 case SPV_ENV_OPENCL_2_0:
37 return "SPIR-V 1.0 (under OpenCL 2.0 Full Profile semantics)";
38 case SPV_ENV_OPENCL_EMBEDDED_2_0:
39 return "SPIR-V 1.0 (under OpenCL 2.0 Embedded Profile semantics)";
David Netoc2967012016-08-05 18:19:30 -040040 case SPV_ENV_OPENCL_2_1:
Pierre Moreau12447d82017-11-30 00:49:23 +010041 return "SPIR-V 1.0 (under OpenCL 2.1 Full Profile semantics)";
42 case SPV_ENV_OPENCL_EMBEDDED_2_1:
43 return "SPIR-V 1.0 (under OpenCL 2.1 Embedded Profile semantics)";
David Netoc2967012016-08-05 18:19:30 -040044 case SPV_ENV_OPENCL_2_2:
Andrey Tuganov95843d72018-03-27 12:01:21 -040045 return "SPIR-V 1.2 (under OpenCL 2.2 Full Profile semantics)";
Pierre Moreau12447d82017-11-30 00:49:23 +010046 case SPV_ENV_OPENCL_EMBEDDED_2_2:
Andrey Tuganov95843d72018-03-27 12:01:21 -040047 return "SPIR-V 1.2 (under OpenCL 2.2 Embedded Profile semantics)";
David Netoc2967012016-08-05 18:19:30 -040048 case SPV_ENV_OPENGL_4_0:
Alan Baker42840d12018-04-06 14:15:27 -040049 return "SPIR-V 1.0 (under OpenGL 4.0 semantics)";
David Netoc2967012016-08-05 18:19:30 -040050 case SPV_ENV_OPENGL_4_1:
Alan Baker42840d12018-04-06 14:15:27 -040051 return "SPIR-V 1.0 (under OpenGL 4.1 semantics)";
David Netoc2967012016-08-05 18:19:30 -040052 case SPV_ENV_OPENGL_4_2:
Alan Baker42840d12018-04-06 14:15:27 -040053 return "SPIR-V 1.0 (under OpenGL 4.2 semantics)";
David Netoc2967012016-08-05 18:19:30 -040054 case SPV_ENV_OPENGL_4_3:
Alan Baker42840d12018-04-06 14:15:27 -040055 return "SPIR-V 1.0 (under OpenGL 4.3 semantics)";
David Netoc2967012016-08-05 18:19:30 -040056 case SPV_ENV_OPENGL_4_5:
Alan Baker42840d12018-04-06 14:15:27 -040057 return "SPIR-V 1.0 (under OpenGL 4.5 semantics)";
David Netodbc20492017-03-14 12:43:41 -040058 case SPV_ENV_UNIVERSAL_1_2:
59 return "SPIR-V 1.2";
David Neto00fa3932018-02-09 14:29:02 -050060 case SPV_ENV_UNIVERSAL_1_3:
61 return "SPIR-V 1.3";
62 case SPV_ENV_VULKAN_1_1:
63 return "SPIR-V 1.3 (under Vulkan 1.1 semantics)";
Corentin Wallezba602c92018-06-20 13:29:38 -040064 case SPV_ENV_WEBGPU_0:
Ryan Harrison9150cd42021-01-14 16:45:18 -050065 assert(false);
66 break;
David Neto63f57d92019-05-07 12:27:18 -040067 case SPV_ENV_UNIVERSAL_1_4:
68 return "SPIR-V 1.4";
69 case SPV_ENV_VULKAN_1_1_SPIRV_1_4:
70 return "SPIR-V 1.4 (under Vulkan 1.1 semantics)";
alan-baker5a48c0d2019-09-13 14:59:02 -040071 case SPV_ENV_UNIVERSAL_1_5:
72 return "SPIR-V 1.5";
David Netod46a1b02019-08-28 18:49:33 -040073 case SPV_ENV_VULKAN_1_2:
74 return "SPIR-V 1.5 (under Vulkan 1.2 semantics)";
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040075 }
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040076 return "";
77}
78
79uint32_t spvVersionForTargetEnv(spv_target_env env) {
80 switch (env) {
81 case SPV_ENV_UNIVERSAL_1_0:
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040082 case SPV_ENV_VULKAN_1_0:
Pierre Moreau12447d82017-11-30 00:49:23 +010083 case SPV_ENV_OPENCL_1_2:
84 case SPV_ENV_OPENCL_EMBEDDED_1_2:
85 case SPV_ENV_OPENCL_2_0:
86 case SPV_ENV_OPENCL_EMBEDDED_2_0:
David Netoc2967012016-08-05 18:19:30 -040087 case SPV_ENV_OPENCL_2_1:
Pierre Moreau12447d82017-11-30 00:49:23 +010088 case SPV_ENV_OPENCL_EMBEDDED_2_1:
David Netoc2967012016-08-05 18:19:30 -040089 case SPV_ENV_OPENGL_4_0:
90 case SPV_ENV_OPENGL_4_1:
91 case SPV_ENV_OPENGL_4_2:
92 case SPV_ENV_OPENGL_4_3:
93 case SPV_ENV_OPENGL_4_5:
Lei Zhang6fa3f8a2016-03-31 17:26:31 -040094 return SPV_SPIRV_VERSION_WORD(1, 0);
Dejan Mircevskicb3c49e2016-04-07 14:41:34 -040095 case SPV_ENV_UNIVERSAL_1_1:
96 return SPV_SPIRV_VERSION_WORD(1, 1);
David Netodbc20492017-03-14 12:43:41 -040097 case SPV_ENV_UNIVERSAL_1_2:
98 case SPV_ENV_OPENCL_2_2:
Pierre Moreau12447d82017-11-30 00:49:23 +010099 case SPV_ENV_OPENCL_EMBEDDED_2_2:
David Netodbc20492017-03-14 12:43:41 -0400100 return SPV_SPIRV_VERSION_WORD(1, 2);
David Neto00fa3932018-02-09 14:29:02 -0500101 case SPV_ENV_UNIVERSAL_1_3:
102 case SPV_ENV_VULKAN_1_1:
103 return SPV_SPIRV_VERSION_WORD(1, 3);
Ryan Harrison9150cd42021-01-14 16:45:18 -0500104 case SPV_ENV_WEBGPU_0:
105 assert(false);
106 break;
David Neto63f57d92019-05-07 12:27:18 -0400107 case SPV_ENV_UNIVERSAL_1_4:
108 case SPV_ENV_VULKAN_1_1_SPIRV_1_4:
109 return SPV_SPIRV_VERSION_WORD(1, 4);
alan-baker5a48c0d2019-09-13 14:59:02 -0400110 case SPV_ENV_UNIVERSAL_1_5:
David Netod46a1b02019-08-28 18:49:33 -0400111 case SPV_ENV_VULKAN_1_2:
alan-baker5a48c0d2019-09-13 14:59:02 -0400112 return SPV_SPIRV_VERSION_WORD(1, 5);
Lei Zhang6fa3f8a2016-03-31 17:26:31 -0400113 }
Lei Zhang6fa3f8a2016-03-31 17:26:31 -0400114 return SPV_SPIRV_VERSION_WORD(0, 0);
115}
Dejan Mircevski27030392016-05-11 09:48:52 -0400116
Kévin Petitdf86bb42019-06-21 13:47:27 +0100117static const std::pair<const char*, spv_target_env> spvTargetEnvNameMap[] = {
Kévin Petitbec7e032019-06-20 14:41:28 +0100118 {"vulkan1.1spv1.4", SPV_ENV_VULKAN_1_1_SPIRV_1_4},
119 {"vulkan1.0", SPV_ENV_VULKAN_1_0},
120 {"vulkan1.1", SPV_ENV_VULKAN_1_1},
David Netod46a1b02019-08-28 18:49:33 -0400121 {"vulkan1.2", SPV_ENV_VULKAN_1_2},
Kévin Petitbec7e032019-06-20 14:41:28 +0100122 {"spv1.0", SPV_ENV_UNIVERSAL_1_0},
123 {"spv1.1", SPV_ENV_UNIVERSAL_1_1},
124 {"spv1.2", SPV_ENV_UNIVERSAL_1_2},
125 {"spv1.3", SPV_ENV_UNIVERSAL_1_3},
126 {"spv1.4", SPV_ENV_UNIVERSAL_1_4},
alan-baker5a48c0d2019-09-13 14:59:02 -0400127 {"spv1.5", SPV_ENV_UNIVERSAL_1_5},
Kévin Petitbec7e032019-06-20 14:41:28 +0100128 {"opencl1.2embedded", SPV_ENV_OPENCL_EMBEDDED_1_2},
129 {"opencl1.2", SPV_ENV_OPENCL_1_2},
130 {"opencl2.0embedded", SPV_ENV_OPENCL_EMBEDDED_2_0},
131 {"opencl2.0", SPV_ENV_OPENCL_2_0},
132 {"opencl2.1embedded", SPV_ENV_OPENCL_EMBEDDED_2_1},
133 {"opencl2.1", SPV_ENV_OPENCL_2_1},
134 {"opencl2.2embedded", SPV_ENV_OPENCL_EMBEDDED_2_2},
135 {"opencl2.2", SPV_ENV_OPENCL_2_2},
136 {"opengl4.0", SPV_ENV_OPENGL_4_0},
137 {"opengl4.1", SPV_ENV_OPENGL_4_1},
138 {"opengl4.2", SPV_ENV_OPENGL_4_2},
139 {"opengl4.3", SPV_ENV_OPENGL_4_3},
140 {"opengl4.5", SPV_ENV_OPENGL_4_5},
Kévin Petitbec7e032019-06-20 14:41:28 +0100141};
142
Dejan Mircevski27030392016-05-11 09:48:52 -0400143bool spvParseTargetEnv(const char* s, spv_target_env* env) {
Kévin Petitdf86bb42019-06-21 13:47:27 +0100144 auto match = [s](const char* b) {
145 return s && (0 == strncmp(s, b, strlen(b)));
146 };
147 for (auto& name_env : spvTargetEnvNameMap) {
148 if (match(name_env.first)) {
149 if (env) {
150 *env = name_env.second;
151 }
152 return true;
Kévin Petitbec7e032019-06-20 14:41:28 +0100153 }
Kévin Petitbec7e032019-06-20 14:41:28 +0100154 }
155 if (env) *env = SPV_ENV_UNIVERSAL_1_0;
156 return false;
Dejan Mircevski27030392016-05-11 09:48:52 -0400157}
Andrey Tuganovbdc78372018-01-23 15:02:27 -0500158
David Netobb236c32020-01-23 17:20:32 -0500159#define VULKAN_VER(MAJOR, MINOR) ((MAJOR << 22) | (MINOR << 12))
160#define SPIRV_VER(MAJOR, MINOR) ((MAJOR << 16) | (MINOR << 8))
161
162struct VulkanEnv {
163 spv_target_env vulkan_env;
164 uint32_t vulkan_ver;
165 uint32_t spirv_ver;
166};
167// Maps each Vulkan target environment enum to the Vulkan version, and the
168// maximum supported SPIR-V version for that Vulkan environment.
169// Keep this ordered from least capable to most capable.
170static const VulkanEnv ordered_vulkan_envs[] = {
171 {SPV_ENV_VULKAN_1_0, VULKAN_VER(1, 0), SPIRV_VER(1, 0)},
172 {SPV_ENV_VULKAN_1_1, VULKAN_VER(1, 1), SPIRV_VER(1, 3)},
173 {SPV_ENV_VULKAN_1_1_SPIRV_1_4, VULKAN_VER(1, 1), SPIRV_VER(1, 4)},
174 {SPV_ENV_VULKAN_1_2, VULKAN_VER(1, 2), SPIRV_VER(1, 5)}};
175
176bool spvParseVulkanEnv(uint32_t vulkan_ver, uint32_t spirv_ver,
177 spv_target_env* env) {
178 for (auto triple : ordered_vulkan_envs) {
179 if (triple.vulkan_ver >= vulkan_ver && triple.spirv_ver >= spirv_ver) {
180 *env = triple.vulkan_env;
181 return true;
182 }
183 }
184 return false;
185}
186
Andrey Tuganovbdc78372018-01-23 15:02:27 -0500187bool spvIsVulkanEnv(spv_target_env env) {
188 switch (env) {
189 case SPV_ENV_UNIVERSAL_1_0:
190 case SPV_ENV_OPENCL_1_2:
191 case SPV_ENV_OPENCL_EMBEDDED_1_2:
192 case SPV_ENV_OPENCL_2_0:
193 case SPV_ENV_OPENCL_EMBEDDED_2_0:
194 case SPV_ENV_OPENCL_2_1:
195 case SPV_ENV_OPENCL_EMBEDDED_2_1:
196 case SPV_ENV_OPENGL_4_0:
197 case SPV_ENV_OPENGL_4_1:
198 case SPV_ENV_OPENGL_4_2:
199 case SPV_ENV_OPENGL_4_3:
200 case SPV_ENV_OPENGL_4_5:
201 case SPV_ENV_UNIVERSAL_1_1:
202 case SPV_ENV_UNIVERSAL_1_2:
203 case SPV_ENV_OPENCL_2_2:
204 case SPV_ENV_OPENCL_EMBEDDED_2_2:
David Neto00fa3932018-02-09 14:29:02 -0500205 case SPV_ENV_UNIVERSAL_1_3:
David Neto63f57d92019-05-07 12:27:18 -0400206 case SPV_ENV_UNIVERSAL_1_4:
alan-baker5a48c0d2019-09-13 14:59:02 -0400207 case SPV_ENV_UNIVERSAL_1_5:
Andrey Tuganovbdc78372018-01-23 15:02:27 -0500208 return false;
209 case SPV_ENV_VULKAN_1_0:
David Neto00fa3932018-02-09 14:29:02 -0500210 case SPV_ENV_VULKAN_1_1:
David Neto63f57d92019-05-07 12:27:18 -0400211 case SPV_ENV_VULKAN_1_1_SPIRV_1_4:
David Netod46a1b02019-08-28 18:49:33 -0400212 case SPV_ENV_VULKAN_1_2:
Andrey Tuganovbdc78372018-01-23 15:02:27 -0500213 return true;
Ryan Harrison9150cd42021-01-14 16:45:18 -0500214 case SPV_ENV_WEBGPU_0:
215 assert(false);
216 break;
Andrey Tuganovbdc78372018-01-23 15:02:27 -0500217 }
218 return false;
219}
Ben Ashbaughd3f88b02018-10-09 08:33:01 -0700220
221bool spvIsOpenCLEnv(spv_target_env env) {
222 switch (env) {
223 case SPV_ENV_UNIVERSAL_1_0:
224 case SPV_ENV_VULKAN_1_0:
225 case SPV_ENV_UNIVERSAL_1_1:
226 case SPV_ENV_OPENGL_4_0:
227 case SPV_ENV_OPENGL_4_1:
228 case SPV_ENV_OPENGL_4_2:
229 case SPV_ENV_OPENGL_4_3:
230 case SPV_ENV_OPENGL_4_5:
231 case SPV_ENV_UNIVERSAL_1_2:
232 case SPV_ENV_UNIVERSAL_1_3:
233 case SPV_ENV_VULKAN_1_1:
David Neto63f57d92019-05-07 12:27:18 -0400234 case SPV_ENV_UNIVERSAL_1_4:
235 case SPV_ENV_VULKAN_1_1_SPIRV_1_4:
alan-baker5a48c0d2019-09-13 14:59:02 -0400236 case SPV_ENV_UNIVERSAL_1_5:
David Netod46a1b02019-08-28 18:49:33 -0400237 case SPV_ENV_VULKAN_1_2:
Ben Ashbaughd3f88b02018-10-09 08:33:01 -0700238 return false;
239 case SPV_ENV_OPENCL_1_2:
240 case SPV_ENV_OPENCL_EMBEDDED_1_2:
241 case SPV_ENV_OPENCL_2_0:
242 case SPV_ENV_OPENCL_EMBEDDED_2_0:
243 case SPV_ENV_OPENCL_EMBEDDED_2_1:
244 case SPV_ENV_OPENCL_EMBEDDED_2_2:
245 case SPV_ENV_OPENCL_2_1:
246 case SPV_ENV_OPENCL_2_2:
247 return true;
Ryan Harrison8cd2a9d2018-11-19 14:32:18 -0500248 case SPV_ENV_WEBGPU_0:
Ryan Harrison9150cd42021-01-14 16:45:18 -0500249 assert(false);
250 break;
Ryan Harrison8cd2a9d2018-11-19 14:32:18 -0500251 }
252 return false;
253}
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500254
Caio Marcelo de Oliveira Filho9702d472019-07-02 05:11:20 -0700255bool spvIsOpenGLEnv(spv_target_env env) {
256 switch (env) {
257 case SPV_ENV_UNIVERSAL_1_0:
258 case SPV_ENV_VULKAN_1_0:
259 case SPV_ENV_UNIVERSAL_1_1:
260 case SPV_ENV_UNIVERSAL_1_2:
261 case SPV_ENV_UNIVERSAL_1_3:
262 case SPV_ENV_VULKAN_1_1:
263 case SPV_ENV_OPENCL_1_2:
264 case SPV_ENV_OPENCL_EMBEDDED_1_2:
265 case SPV_ENV_OPENCL_2_0:
266 case SPV_ENV_OPENCL_EMBEDDED_2_0:
267 case SPV_ENV_OPENCL_EMBEDDED_2_1:
268 case SPV_ENV_OPENCL_EMBEDDED_2_2:
269 case SPV_ENV_OPENCL_2_1:
270 case SPV_ENV_OPENCL_2_2:
Caio Marcelo de Oliveira Filho9702d472019-07-02 05:11:20 -0700271 case SPV_ENV_UNIVERSAL_1_4:
272 case SPV_ENV_VULKAN_1_1_SPIRV_1_4:
alan-baker5a48c0d2019-09-13 14:59:02 -0400273 case SPV_ENV_UNIVERSAL_1_5:
David Netod46a1b02019-08-28 18:49:33 -0400274 case SPV_ENV_VULKAN_1_2:
Caio Marcelo de Oliveira Filho9702d472019-07-02 05:11:20 -0700275 return false;
276 case SPV_ENV_OPENGL_4_0:
277 case SPV_ENV_OPENGL_4_1:
278 case SPV_ENV_OPENGL_4_2:
279 case SPV_ENV_OPENGL_4_3:
280 case SPV_ENV_OPENGL_4_5:
281 return true;
Ryan Harrison9150cd42021-01-14 16:45:18 -0500282 case SPV_ENV_WEBGPU_0:
283 assert(false);
284 break;
Caio Marcelo de Oliveira Filho9702d472019-07-02 05:11:20 -0700285 }
286 return false;
287}
288
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500289std::string spvLogStringForEnv(spv_target_env env) {
290 switch (env) {
291 case SPV_ENV_OPENCL_1_2:
292 case SPV_ENV_OPENCL_2_0:
293 case SPV_ENV_OPENCL_2_1:
294 case SPV_ENV_OPENCL_2_2:
295 case SPV_ENV_OPENCL_EMBEDDED_1_2:
296 case SPV_ENV_OPENCL_EMBEDDED_2_0:
297 case SPV_ENV_OPENCL_EMBEDDED_2_1:
298 case SPV_ENV_OPENCL_EMBEDDED_2_2: {
299 return "OpenCL";
300 }
301 case SPV_ENV_OPENGL_4_0:
302 case SPV_ENV_OPENGL_4_1:
303 case SPV_ENV_OPENGL_4_2:
304 case SPV_ENV_OPENGL_4_3:
305 case SPV_ENV_OPENGL_4_5: {
306 return "OpenGL";
307 }
308 case SPV_ENV_VULKAN_1_0:
David Neto63f57d92019-05-07 12:27:18 -0400309 case SPV_ENV_VULKAN_1_1:
310 case SPV_ENV_VULKAN_1_1_SPIRV_1_4: {
David Netod46a1b02019-08-28 18:49:33 -0400311 case SPV_ENV_VULKAN_1_2:
312 return "Vulkan";
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500313 }
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500314 case SPV_ENV_UNIVERSAL_1_0:
315 case SPV_ENV_UNIVERSAL_1_1:
316 case SPV_ENV_UNIVERSAL_1_2:
David Neto63f57d92019-05-07 12:27:18 -0400317 case SPV_ENV_UNIVERSAL_1_3:
alan-baker5a48c0d2019-09-13 14:59:02 -0400318 case SPV_ENV_UNIVERSAL_1_4:
319 case SPV_ENV_UNIVERSAL_1_5: {
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500320 return "Universal";
321 }
Ryan Harrison9150cd42021-01-14 16:45:18 -0500322 case SPV_ENV_WEBGPU_0:
323 assert(false);
324 break;
Ryan Harrison3a3ad2e2019-01-22 15:18:14 -0500325 }
326 return "Unknown";
327}
Kévin Petitbec7e032019-06-20 14:41:28 +0100328
329std::string spvTargetEnvList(const int pad, const int wrap) {
330 std::string ret;
331 size_t max_line_len = wrap - pad; // The first line isn't padded
332 std::string line;
333 std::string sep = "";
334
Kévin Petitdf86bb42019-06-21 13:47:27 +0100335 for (auto& name_env : spvTargetEnvNameMap) {
336 std::string word = sep + name_env.first;
Kévin Petitbec7e032019-06-20 14:41:28 +0100337 if (line.length() + word.length() > max_line_len) {
338 // Adding one word wouldn't fit, commit the line in progress and
339 // start a new one.
340 ret += line + "\n";
341 line.assign(pad, ' ');
342 // The first line is done. The max length now comprises the
343 // padding.
344 max_line_len = wrap;
345 }
346 line += word;
347 sep = "|";
348 }
349
350 ret += line;
351
352 return ret;
353}