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
|
#include "ModuleCache.h"
#include "module.h"
#include "printutils.h"
#include "boosty.h"
#include "openscad.h"
#include <stdio.h>
#include <sstream>
#include <sys/stat.h>
ModuleCache *ModuleCache::inst = NULL;
Module *ModuleCache::evaluate(const std::string &filename)
{
Module *cached = NULL;
struct stat st;
memset(&st, 0, sizeof(struct stat));
stat(filename.c_str(), &st);
std::stringstream idstream;
idstream << std::hex << st.st_mtime << "." << st.st_size;
std::string cache_id = idstream.str();
if (this->entries.find(filename) != this->entries.end() &&
this->entries[filename].cache_id == cache_id) {
#ifdef DEBUG
PRINTB("Using cached library: %s (%s)", filename % cache_id);
#endif
PRINTB("%s", this->entries[filename].msg);
cached = &(*this->entries[filename].module);
}
if (cached) {
cached->handleDependencies();
return cached;
}
else {
if (this->entries.find(filename) != this->entries.end()) {
PRINTB("Recompiling cached library: %s (%s)", filename % cache_id);
}
else {
PRINTB("Compiling library '%s'.", filename);
}
}
FILE *fp = fopen(filename.c_str(), "rt");
if (!fp) {
fprintf(stderr, "WARNING: Can't open library file '%s'\n", filename.c_str());
return NULL;
}
std::stringstream text;
char buffer[513];
int ret;
while ((ret = fread(buffer, 1, 512, fp)) > 0) {
buffer[ret] = 0;
text << buffer;
}
fclose(fp);
print_messages_push();
cache_entry e = { NULL, cache_id, std::string("WARNING: Library `") + filename + "' tries to recursively use itself!" };
if (this->entries.find(filename) != this->entries.end())
delete this->entries[filename].module;
this->entries[filename] = e;
std::string pathname = boosty::stringy(fs::path(filename).parent_path());
Module *lib_mod = dynamic_cast<Module*>(parse(text.str().c_str(), pathname.c_str(), 0));
if (lib_mod) {
this->entries[filename].module = lib_mod;
this->entries[filename].msg = print_messages_stack.back();
lib_mod->handleDependencies();
} else {
this->entries.erase(filename);
}
print_messages_pop();
return lib_mod;
}
void ModuleCache::clear()
{
this->entries.clear();
}
|