TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
pla_file.hpp
1#ifndef LIBTEDDY_DETAILS_PLA_FILE_HPP
2#define LIBTEDDY_DETAILS_PLA_FILE_HPP
3
4#include <libteddy/details/debug.hpp>
5#include <libteddy/details/tools.hpp>
6#include <libteddy/details/types.hpp>
7
8#include <cassert>
9#include <cctype>
10#include <cstdint>
11#include <fstream>
12#include <optional>
13#include <string>
14#include <unordered_map>
15#include <vector>
16
17namespace teddy
18{
23{
24public:
25 static constexpr std::uint8_t DontCare = 0b11;
26
27public:
28 bool_cube(int32 size);
29
30 auto size () const -> int32;
31 auto get (int32 index) const -> int32;
32 auto set (int32 index, int32 value) -> void;
33
34private:
35 struct byte
36 {
37 std::uint8_t b0 : 2;
38 std::uint8_t b1 : 2;
39 std::uint8_t b2 : 2;
40 std::uint8_t b3 : 2;
41 };
42
43private:
44 int32 size_;
45 std::vector<byte> values_;
46};
47
52{
53public:
60 static auto load_file (std::string const& path) -> std::optional<pla_file>;
61
62public:
66 struct pla_line
67 {
68 bool_cube cube_;
69 bool_cube fVals_;
70 };
71
72public:
77 [[nodiscard]] auto get_variable_count () const -> int32;
78
83 [[nodiscard]] auto get_function_count () const -> int32;
84
92 [[nodiscard]] auto get_line_count () const -> int64;
93
98 [[nodiscard]] auto get_lines () const& -> std::vector<pla_line> const&;
99
104 [[nodiscard]] auto get_lines () && -> std::vector<pla_line>;
105
111 [[nodiscard]] auto get_input_labels (
112 ) const& -> std::vector<std::string> const&;
113
119 [[nodiscard]] auto get_input_labels () && -> std::vector<std::string>;
120
125 [[nodiscard]] auto get_output_labels (
126 ) const& -> std::vector<std::string> const&;
127
132 [[nodiscard]] auto get_output_labels () && -> std::vector<std::string>;
133
134private:
135 pla_file(
136 std::vector<pla_line> lines,
137 std::vector<std::string> inputLabels,
138 std::vector<std::string> outputLabels
139 );
140
141private:
142 std::vector<pla_line> lines_;
143 std::vector<std::string> inputLabels_;
144 std::vector<std::string> outputLabels_;
145};
146
147// bool_cube definitions:
148
149inline bool_cube::bool_cube(int32 const size) :
150 size_(size),
151 values_(as_usize(size / 4 + 1), byte {0, 0, 0, 0})
152{
153}
154
155inline auto bool_cube::size() const -> int32
156{
157 return size_;
158}
159
160inline auto bool_cube::get(int32 const i) const -> int32
161{
162 int32 const byteIndex = i / 4;
163 std::size_t const uByteIndex = as_uindex(byteIndex);
164
165 assert(byteIndex >= 0 && byteIndex < ssize(values_));
166
167 switch (i % 4)
168 {
169 case 0:
170 return values_[uByteIndex].b0;
171 case 1:
172 return values_[uByteIndex].b1;
173 case 2:
174 return values_[uByteIndex].b2;
175 case 3:
176 return values_[uByteIndex].b3;
177 }
178 return -1;
179}
180
181inline auto bool_cube::set(int32 const index, int32 const value) -> void
182{
183 int32 const byteIndex = index / 4;
184 std::size_t const uByteIndex = as_uindex(byteIndex);
185
186 assert((byteIndex >= 0 && byteIndex < ssize(values_)));
187 assert(value == 0 || value == 1 || value == DontCare);
188
189 switch (index % 4)
190 {
191 case 0:
192 values_[uByteIndex].b0 = value & 0b11;
193 break;
194 case 1:
195 values_[uByteIndex].b1 = value & 0b11;
196 break;
197 case 2:
198 values_[uByteIndex].b2 = value & 0b11;
199 break;
200 case 3:
201 values_[uByteIndex].b3 = value & 0b11;
202 break;
203 }
204}
205
206// pla_file definitions:
207
208inline auto pla_file::load_file(std::string const& path)
209 -> std::optional<pla_file>
210{
211 auto constexpr to_words = [] (std::string const& str)
212 {
213 std::vector<std::string> words;
214 auto strIt = str.begin();
215 auto const endIt = str.end();
216
217 while (strIt != endIt)
218 {
219 auto const wordBegin = utils::find_if_not(strIt, endIt, ::isspace);
220 auto const wordEnd = utils::find_if(wordBegin, endIt, ::isspace);
221 if (wordBegin != wordEnd)
222 {
223 words.emplace_back(wordBegin, wordEnd);
224 }
225 strIt = wordEnd;
226 }
227
228 return words;
229 };
230
231 auto ifst = std::ifstream(path);
232 if (not ifst.is_open())
233 {
234 return std::nullopt;
235 }
236
237 // Read options.
238 auto options = std::unordered_map<std::string, std::string>();
239 auto line = std::string();
240 while (std::getline(ifst, line))
241 {
242 // Skip leading spaces.
243 auto const first
244 = utils::find_if_not(line.begin(), line.end(), ::isspace);
245 auto const last = line.end();
246 if (first == last)
247 {
248 // Skip empty line.
249 continue;
250 }
251
252 if (*first == '#')
253 {
254 // Skip comment.
255 continue;
256 }
257
258 if (*first != '.')
259 {
260 // Not an option.
261 break;
262 }
263
264 // Split into (key, val) pair on the first space.
265 auto const keyLast = utils::find_if(first, last, ::isspace);
266 auto const valFirst
267 = keyLast == last
268 ? last
269 : utils::find_if_not(keyLast + 1, last, ::isspace);
270 std::string key(first, keyLast);
271 if (valFirst != last)
272 {
273 auto valLast = last;
274 while (::isspace(*(valLast - 1)))
275 {
276 --valLast;
277 }
278 std::string val(valFirst, valLast);
279 options.emplace(
280 static_cast<std::string&&>(key),
281 static_cast<std::string&&>(val)
282 );
283 }
284 else
285 {
286 options.emplace(static_cast<std::string&&>(key), std::string());
287 }
288 }
289
290 // Parse header.
291 auto const optionsEnd = options.end();
292 auto const varCountIt = options.find(".i");
293 auto const fCountIt = options.find(".o");
294 auto const lineCountIt = options.find(".p");
295 if (varCountIt == optionsEnd || fCountIt == optionsEnd
296 || lineCountIt == optionsEnd)
297 {
298 return std::nullopt;
299 }
300
301 std::optional<int32> varCount = utils::parse<int32>(varCountIt->second);
302 std::optional<int32> fCount = utils::parse<int32>(fCountIt->second);
303 std::optional<int32> lineCount = utils::parse<int64>(lineCountIt->second);
304
305 if (not varCount || not fCount || not lineCount)
306 {
307 return std::nullopt;
308 }
309
310 // Read data.
311 std::vector<pla_file::pla_line> lines;
312 lines.reserve(as_usize(*lineCount));
313 do
314 {
315 auto const first
316 = utils::find_if_not(line.begin(), line.end(), ::isspace);
317 auto const last = line.end();
318 if (first == last)
319 {
320 // Skip empty line.
321 continue;
322 }
323
324 if (*first == '.')
325 {
326 // This can only be the .e line.
327 break;
328 }
329
330 // Split on the first space.
331 auto const varsLast = utils::find_if(first, last, ::isspace);
332 if (varsLast == last)
333 {
334 return std::nullopt;
335 }
336 auto const fsFirst = utils::find_if_not(varsLast + 1, last, ::isspace);
337 auto const fsLast = utils::find_if(fsFirst, last, ::isspace);
338 std::string const varsStr(first, varsLast);
339 std::string const fStr(fsFirst, fsLast);
340
341 if (ssize(varsStr) != *varCount || ssize(fStr) != *fCount)
342 {
343 return std::nullopt;
344 }
345
346 // Parse variables into cube.
347 bool_cube variables(*varCount);
348 for (auto i = 0; i < *varCount; ++i)
349 {
350 switch (varsStr[as_uindex(i)])
351 {
352 case '0':
353 variables.set(i, 0);
354 break;
355 case '1':
356 variables.set(i, 1);
357 break;
358 case '~':
359 case '-':
360 variables.set(i, bool_cube::DontCare);
361 break;
362 default:
363 return std::nullopt;
364 }
365 }
366
367 // Parse functions into cube.
368 bool_cube functions(*fCount);
369 for (auto i = 0; i < *fCount; ++i)
370 {
371 switch (fStr[as_uindex(i)])
372 {
373 case '0':
374 functions.set(i, 0);
375 break;
376 case '1':
377 functions.set(i, 1);
378 break;
379 case '-':
380 case '~':
381 functions.set(i, bool_cube::DontCare);
382 break;
383 default:
384 return std::nullopt;
385 }
386 }
387
388 lines.push_back(pla_file::pla_line {variables, functions});
389
390 } while (std::getline(ifst, line));
391
392 // Read labels.
393 auto const inLbIt = options.find(".ilb");
394 auto const ouLbIt = options.find(".ob");
395 std::vector<std::string> inputLabels;
396 if (inLbIt != options.end())
397 {
398 inputLabels = to_words(inLbIt->second);
399 }
400
401 std::vector<std::string> outputLabels;
402 if (ouLbIt != options.end())
403 {
404 outputLabels = to_words(ouLbIt->second);
405 }
406
407 return pla_file(
408 static_cast<std::vector<pla_file::pla_line>&&>(lines),
409 static_cast<std::vector<std::string>&&>(inputLabels),
410 static_cast<std::vector<std::string>&&>(outputLabels)
411 );
412}
413
414inline pla_file::pla_file(
415 std::vector<pla_line> lines,
416 std::vector<std::string> inputLabels,
417 std::vector<std::string> outputLabels
418) :
419 lines_(static_cast<std::vector<pla_file::pla_line>&&>(lines)),
420 inputLabels_(static_cast<std::vector<std::string>&&>(inputLabels)),
421 outputLabels_(static_cast<std::vector<std::string>&&>(outputLabels))
422{
423}
424
425inline auto pla_file::get_variable_count() const -> int32
426{
427 return lines_.front().cube_.size();
428}
429
430inline auto pla_file::get_function_count() const -> int32
431{
432 return lines_.front().fVals_.size();
433}
434
435inline auto pla_file::get_line_count() const -> int64
436{
437 return static_cast<int64>(lines_.size());
438}
439
440inline auto pla_file::get_lines() const& -> std::vector<pla_line> const&
441{
442 return lines_;
443}
444
445inline auto pla_file::get_lines() && -> std::vector<pla_line>
446{
447 return std::move(lines_);
448}
449
451) const& -> std::vector<std::string> const&
452{
453 return inputLabels_;
454}
455
456inline auto pla_file::get_input_labels() && -> std::vector<std::string>
457{
458 return std::move(inputLabels_);
459}
460
462) const& -> std::vector<std::string> const&
463{
464 return outputLabels_;
465}
466
467inline auto pla_file::get_output_labels() && -> std::vector<std::string>
468{
469 return std::move(outputLabels_);
470}
471} // namespace teddy
472
473#endif
Bool cube.
Definition pla_file.hpp:23
Representation of a PLA file.
Definition pla_file.hpp:52
auto get_line_count() const -> int64
Returns number of lines in the file.
Definition pla_file.hpp:435
auto get_output_labels() const &-> std::vector< std::string > const &
Return reference to a vector holding labels of functions.
Definition pla_file.hpp:461
auto get_lines() const &-> std::vector< pla_line > const &
Returns reference to a vector holding lines.
Definition pla_file.hpp:440
static auto load_file(std::string const &path) -> std::optional< pla_file >
Loads PLA file from a file at given path.
Definition pla_file.hpp:208
auto get_function_count() const -> int32
Returns number of functions in the file.
Definition pla_file.hpp:430
auto get_input_labels() const &-> std::vector< std::string > const &
Returns reference to a vector holding labels of input variables.
Definition pla_file.hpp:450
auto get_variable_count() const -> int32
Returns number of variables in the file.
Definition pla_file.hpp:425
Represents one line of a PLA file.
Definition pla_file.hpp:67