| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | # -*- python -*- | 
					
						
							|  |  |  | #============================================================================== | 
					
						
							|  |  |  | # Copyright (C) 2011 by Denys Duchier | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This program is free software: you can redistribute it and/or modify it under | 
					
						
							|  |  |  | # the terms of the GNU General Public License as published by the Free Software | 
					
						
							|  |  |  | # Foundation, either version 3 of the License, or (at your option) any later | 
					
						
							|  |  |  | # version. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | # This program is distributed in the hope that it will be useful, but WITHOUT | 
					
						
							|  |  |  | # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | 
					
						
							|  |  |  | # FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | 
					
						
							|  |  |  | # details. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | # You should have received a copy of the GNU General Public License along with | 
					
						
							|  |  |  | # this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | #============================================================================== | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NOTICE_CC = """// -*- c++ -*-
 | 
					
						
							|  |  |  | //============================================================================= | 
					
						
							|  |  |  | // Copyright (C) 2011 by Denys Duchier | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This program is free software: you can redistribute it and/or modify it | 
					
						
							|  |  |  | // under the terms of the GNU Lesser General Public License as published by the | 
					
						
							|  |  |  | // Free Software Foundation, either version 3 of the License, or (at your | 
					
						
							|  |  |  | // option) any later version. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | // This program is distributed in the hope that it will be useful, but WITHOUT | 
					
						
							|  |  |  | // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
					
						
							|  |  |  | // FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | 
					
						
							|  |  |  | // more details. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | //============================================================================= | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NOTICE_PROLOG="""%% -*- prolog -*-
 | 
					
						
							|  |  |  | %%============================================================================= | 
					
						
							|  |  |  | %% Copyright (C) 2011 by Denys Duchier | 
					
						
							|  |  |  | %% | 
					
						
							|  |  |  | %% This program is free software: you can redistribute it and/or modify it | 
					
						
							|  |  |  | %% under the terms of the GNU Lesser General Public License as published by the | 
					
						
							|  |  |  | %% Free Software Foundation, either version 3 of the License, or (at your | 
					
						
							|  |  |  | %% option) any later version. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | %% | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | %% This program is distributed in the hope that it will be useful, but WITHOUT | 
					
						
							|  |  |  | %% ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
					
						
							|  |  |  | %% FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | 
					
						
							|  |  |  | %% more details. | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | %% | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | %% You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | %% along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | %%============================================================================= | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def prolog_print_notice(): | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |     print(NOTICE_PROLOG) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def cc_print_notice(): | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |     print(NOTICE_CC) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  | def islegal(a): | 
					
						
							| 
									
										
										
										
											2017-11-17 23:56:51 +00:00
										 |  |  |     return a[0].islower() and a.replace("_","a").isalnum() or a[0] == '\'' and a[len(a)-1] == '\'' | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | class Type(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DEFAULT   = re.compile("""^(.+)=(.+)$""") | 
					
						
							|  |  |  |     CONST     = re.compile("""^const\\b(.+)$""") | 
					
						
							|  |  |  |     UNSIGNED  = re.compile("""^unsigned\\b(.+)$""") | 
					
						
							|  |  |  |     REFERENCE = re.compile("""^(.+)&$""") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, text): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         if isinstance(text, Type): | 
					
						
							|  |  |  |             self.clone_from(text) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         text = text.strip() | 
					
						
							|  |  |  |         m = self.DEFAULT.match(text) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             self.default = m.group(2).strip() | 
					
						
							|  |  |  |             text = m.group(1).strip() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.default = None | 
					
						
							|  |  |  |         m = self.CONST.match(text) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             self.const = True | 
					
						
							|  |  |  |             text = m.group(1).strip() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.const = False | 
					
						
							|  |  |  |         m = self.UNSIGNED.match(text) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             self.unsigned = True | 
					
						
							|  |  |  |             text = m.group(1).strip() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.unsigned = False | 
					
						
							|  |  |  |         m = self.REFERENCE.match(text) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             self.reference = True | 
					
						
							|  |  |  |             text = m.group(1).strip() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.reference = False | 
					
						
							|  |  |  |         self.type = text | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         l = [] | 
					
						
							|  |  |  |         if self.const: l.append("const ") | 
					
						
							|  |  |  |         if self.unsigned: l.append("unsigned ") | 
					
						
							|  |  |  |         l.append(self.type) | 
					
						
							|  |  |  |         if self.reference: l.append("&") | 
					
						
							|  |  |  |         if self.default is not None: | 
					
						
							|  |  |  |             l.append("=") | 
					
						
							|  |  |  |             l.append(self.default) | 
					
						
							|  |  |  |         return ''.join(l) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clone_from(self, other): | 
					
						
							|  |  |  |         self.const = other.const | 
					
						
							|  |  |  |         self.unsigned = other.unsigned | 
					
						
							|  |  |  |         self.type = other.type | 
					
						
							|  |  |  |         self.reference = other.reference | 
					
						
							|  |  |  |         self.default = other.default | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clone(self): | 
					
						
							|  |  |  |         return type(self)(self) | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Constraint(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DECL = re.compile("""^([^(]+)\\b(\w+)\((.*)\);$""") | 
					
						
							|  |  |  |     ARG  = re.compile("""((?:[^,<(]|<[^>]*>|\([^)]*\))+),?""") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, line): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         if isinstance(line, Constraint): | 
					
						
							|  |  |  |             self.clone_from(line) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         line = line.strip() | 
					
						
							|  |  |  |         m = self.DECL.match(line) | 
					
						
							|  |  |  |         self.rettype = Type(m.group(1).strip()) | 
					
						
							|  |  |  |         self.name = m.group(2) | 
					
						
							|  |  |  |         argtypes = [] | 
					
						
							|  |  |  |         for x in self.ARG.finditer(m.group(3).strip()): | 
					
						
							|  |  |  |             argtypes.append(Type(x.group(1))) | 
					
						
							|  |  |  |         self.argtypes = tuple(argtypes) | 
					
						
							|  |  |  |         self.api = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         l = [] | 
					
						
							|  |  |  |         l.append(str(self.rettype)) | 
					
						
							|  |  |  |         l.append(" ") | 
					
						
							|  |  |  |         l.append(self.name) | 
					
						
							|  |  |  |         sep = "(" | 
					
						
							|  |  |  |         for x in self.argtypes: | 
					
						
							|  |  |  |             l.append(sep) | 
					
						
							|  |  |  |             sep = ", " | 
					
						
							|  |  |  |             l.append(str(x)) | 
					
						
							|  |  |  |         l.append(")") | 
					
						
							|  |  |  |         if self.api is not None: | 
					
						
							|  |  |  |             l.append(" -> ") | 
					
						
							|  |  |  |             l.append(self.api) | 
					
						
							|  |  |  |         l.append(";") | 
					
						
							|  |  |  |         return ''.join(l) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clone_from(self, other): | 
					
						
							|  |  |  |         self.rettype = other.rettype.clone() | 
					
						
							|  |  |  |         self.name = other.name | 
					
						
							|  |  |  |         self.argtypes = tuple(t.clone() for t in other.argtypes) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clone(self): | 
					
						
							|  |  |  |         return type(self)(self) | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | COMMENT = re.compile("""^\\s*//.*$""") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def load_decls(filename): | 
					
						
							|  |  |  |     decls = [] | 
					
						
							|  |  |  |     for line in open(filename): | 
					
						
							|  |  |  |         line = line.strip() | 
					
						
							|  |  |  |         if not line: continue | 
					
						
							|  |  |  |         m = COMMENT.match(line) | 
					
						
							|  |  |  |         if m: continue | 
					
						
							|  |  |  |         decls.append(Constraint(line)) | 
					
						
							|  |  |  |     return decls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DeclsLoader(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.decls = load_decls(filename) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_decls(self): | 
					
						
							|  |  |  |         for con in self.decls: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print(str(con)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class PredGenerator(DeclsLoader): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |     OMIT = () | 
					
						
							|  |  |  |         # "VarBranchOptions", | 
					
						
							|  |  |  |         #     "ValBranchOptions", | 
					
						
							|  |  |  |         #     "TieBreakVarBranch<IntVarBranch>", | 
					
						
							|  |  |  |         #     "TieBreak<IntVarBranch>", | 
					
						
							|  |  |  |         #     "TieBreak<FloatVarBranch>", | 
					
						
							|  |  |  |         #     "TieBreak<SetVarBranch>", | 
					
						
							|  |  |  |         #     "TieBreak<BoolVarBranch>", | 
					
						
							|  |  |  |         #     "TieBreakVarBranchOptions", | 
					
						
							|  |  |  |         #     "TieBreakVarBranch<SetVarBranch>") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         super(PredGenerator, self).__init__(filename) | 
					
						
							|  |  |  |         self._change_home_to_space() | 
					
						
							|  |  |  |         self._change_intsharedarray_to_intargs() | 
					
						
							|  |  |  |         self._generate() | 
					
						
							|  |  |  |         self._number() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _change_home_to_space(self): | 
					
						
							|  |  |  |         for p in self.decls: | 
					
						
							|  |  |  |             for t in p.argtypes: | 
					
						
							|  |  |  |                 if t.type=="Home": | 
					
						
							|  |  |  |                     t.type="Space" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _change_intsharedarray_to_intargs(self): | 
					
						
							|  |  |  |         for p in self.decls: | 
					
						
							|  |  |  |             for t in p.argtypes: | 
					
						
							|  |  |  |                 if t.type=="IntSharedArray": | 
					
						
							|  |  |  |                     t.type="IntArgs" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _generate(self): | 
					
						
							|  |  |  |         # drop the constraints and optional arguments we can't handle | 
					
						
							|  |  |  |         preds = [] | 
					
						
							|  |  |  |         for con in self.decls: | 
					
						
							|  |  |  |             if self._con_ok(con): | 
					
						
							|  |  |  |                 con = con.clone() | 
					
						
							|  |  |  |                 con.argtypes = tuple(self._drop_deco(t) for t in con.argtypes | 
					
						
							|  |  |  |                                      if t.type not in self.OMIT) | 
					
						
							|  |  |  |                 preds.append(con) | 
					
						
							|  |  |  |         # for each pred that has an argument with a default produce | 
					
						
							|  |  |  |         # 2 preds (1 without, 1 with).  repeat until all defaults have | 
					
						
							|  |  |  |         # been removed. | 
					
						
							|  |  |  |         again = True | 
					
						
							|  |  |  |         while again: | 
					
						
							|  |  |  |             preds_ = [] | 
					
						
							|  |  |  |             again = False | 
					
						
							|  |  |  |             for con in preds: | 
					
						
							|  |  |  |                 i = self._defaulted(con.argtypes) | 
					
						
							|  |  |  |                 if i is None: | 
					
						
							|  |  |  |                     preds_.append(con) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     again = True | 
					
						
							|  |  |  |                     before = con.argtypes[:i] | 
					
						
							|  |  |  |                     # without the default argument | 
					
						
							|  |  |  |                     # and therefore without the args that follow | 
					
						
							|  |  |  |                     con1 = con.clone() | 
					
						
							|  |  |  |                     con1.argtypes = before | 
					
						
							|  |  |  |                     preds_.append(con1) | 
					
						
							|  |  |  |                     # with the default argument (not default anymore) | 
					
						
							|  |  |  |                     con2 = con.clone() | 
					
						
							|  |  |  |                     arg = con.argtypes[i].clone() | 
					
						
							|  |  |  |                     arg.default=None | 
					
						
							|  |  |  |                     after = con.argtypes[i+1:] | 
					
						
							|  |  |  |                     con2.argtypes = before + (arg,) + after | 
					
						
							|  |  |  |                     preds_.append(con2) | 
					
						
							|  |  |  |             preds = preds_ | 
					
						
							|  |  |  |         self.preds = preds | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _con_ok(self, con): | 
					
						
							|  |  |  |         for t in con.argtypes: | 
					
						
							|  |  |  |             if (t.type in self.OMIT) and (t.default is None): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _drop_deco(self, t): | 
					
						
							|  |  |  |         # drop const, ref, and unsigned indications | 
					
						
							|  |  |  |         t.const = False | 
					
						
							|  |  |  |         t.reference = False | 
					
						
							|  |  |  |         t.unsigned = False | 
					
						
							|  |  |  |         return t | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _defaulted(self, argtypes): | 
					
						
							|  |  |  |         i = 0 | 
					
						
							|  |  |  |         for x in argtypes: | 
					
						
							|  |  |  |             if x.default is not None: | 
					
						
							|  |  |  |                 return i | 
					
						
							|  |  |  |             i += 1 | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _number(self): | 
					
						
							|  |  |  |         i = 1 | 
					
						
							|  |  |  |         for x in self.preds: | 
					
						
							|  |  |  |             x.api = "%s_%d" % (x.name,i) | 
					
						
							|  |  |  |             i += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_preds(self): | 
					
						
							|  |  |  |         for p in self.preds: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print(str(p)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Cluster(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, arity): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object, object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.arity = arity | 
					
						
							|  |  |  |         self.preds = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # discriminating tree based on argument types | 
					
						
							|  |  |  | class DTree(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, i, preds, cluster): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object, object, object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.index = i | 
					
						
							|  |  |  |         self.cluster = cluster | 
					
						
							|  |  |  |         if len(preds) == 1 and len(preds[0].argtypes) == i: | 
					
						
							|  |  |  |             self.is_leaf = True | 
					
						
							|  |  |  |             self.pred = preds[0] | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         self.is_leaf = False | 
					
						
							|  |  |  |         # i is the index of the current arg | 
					
						
							|  |  |  |         # preds are the predicates to be indexed under this tree node | 
					
						
							|  |  |  |         dispatch = {} | 
					
						
							|  |  |  |         for p in preds: | 
					
						
							|  |  |  |             t = p.argtypes[i] | 
					
						
							|  |  |  |             d = dispatch.get(t.type, None) | 
					
						
							|  |  |  |             if d is None: | 
					
						
							|  |  |  |                 d = [] | 
					
						
							|  |  |  |                 dispatch[t.type] = d | 
					
						
							|  |  |  |             d.append(p) | 
					
						
							|  |  |  |         self.subtrees = tuple((t2,DTree(i+1,p2,cluster)) | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |                               for t2,p2 in dispatch.items()) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _generate_body(self, user_vars, lib_vars): | 
					
						
							|  |  |  |         if self.is_leaf: | 
					
						
							|  |  |  |             return PrologLiteral("gecode_constraint_%s(%s)" % (self.pred.api, ",".join(lib_vars))) | 
					
						
							|  |  |  |         X = user_vars[self.index] | 
					
						
							|  |  |  |         Y = lib_vars[self.index] | 
					
						
							|  |  |  |         return self._generate_dispatch(0, user_vars, lib_vars) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _generate_dispatch(self, i, user_vars, lib_vars): | 
					
						
							|  |  |  |         if i == len(self.subtrees): | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             ty = self.subtrees[i-1][0] | 
					
						
							|  |  |  |             name = self.cluster.name | 
					
						
							|  |  |  |             if not islegal(ty): | 
					
						
							|  |  |  |                 ty = "\'"+ty+"\'" | 
					
						
							|  |  |  |             return PrologLiteral("throw(error(type_error(%s(X%d)),gecode_argument_error(%s(%s),arg=%d)))"  % (ty, self.index, name, ",".join(user_vars), self.index+1)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         typ, dtree = self.subtrees[i] | 
					
						
							|  |  |  |         idx = self.index | 
					
						
							|  |  |  |         X = user_vars[idx] | 
					
						
							|  |  |  |         Y = lib_vars[idx] | 
					
						
							|  |  |  |         if typ=="Space": | 
					
						
							|  |  |  |             typ = "Space_or_Clause" | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         elif typ.startswith("std::function") and typ.endswith(")>"): | 
					
						
							|  |  |  |             typ = "std_function" | 
					
						
							|  |  |  |         elif typ.endswith(">"): | 
					
						
							|  |  |  |             sp = typ.split("<") | 
					
						
							|  |  |  |             if len(sp) > 1: | 
					
						
							|  |  |  |                 typ = sp[1].rstrip(">") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         return PrologIF( | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |                     PrologLiteral("is_%s(%s,%s)" % (typ,X,Y)), | 
					
						
							|  |  |  |                     dtree._generate_body(user_vars, lib_vars), | 
					
						
							|  |  |  |                     self._generate_dispatch(i+1, user_vars, lib_vars)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _cc_descriptors(self, name, argtypes): | 
					
						
							|  |  |  |         if self.is_leaf: | 
					
						
							|  |  |  |             return (CCDescriptor(name, argtypes, self.pred.api),) | 
					
						
							|  |  |  |         descriptors = [] | 
					
						
							|  |  |  |         for typ,dtree in self.subtrees: | 
					
						
							|  |  |  |             descriptors.extend(dtree._cc_descriptors(name,(argtypes+(typ,)))) | 
					
						
							|  |  |  |         return descriptors | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPConstraintGeneratorBase(PredGenerator): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         super(YAPConstraintGeneratorBase, self).__init__(filename) | 
					
						
							|  |  |  |         self._classify() | 
					
						
							|  |  |  |         self._dtreefy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # create clusters (predicate,arity) | 
					
						
							|  |  |  |     def _classify(self): | 
					
						
							|  |  |  |         clusters = {} | 
					
						
							|  |  |  |         for pred in self.preds: | 
					
						
							|  |  |  |             name = pred.name | 
					
						
							|  |  |  |             arity = len(pred.argtypes) | 
					
						
							|  |  |  |             key = (name,arity) | 
					
						
							|  |  |  |             cluster = clusters.get(key, None) | 
					
						
							|  |  |  |             if cluster is None: | 
					
						
							|  |  |  |                 cluster = Cluster(name, arity) | 
					
						
							|  |  |  |                 clusters[key] = cluster | 
					
						
							|  |  |  |             cluster.preds.append(pred) | 
					
						
							|  |  |  |         self.clusters = clusters | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # for each cluster, create a dtree | 
					
						
							|  |  |  |     def _dtreefy(self): | 
					
						
							|  |  |  |         dtrees = {} | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         for key, cluster in self.clusters.items(): | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             dtree = DTree(0, cluster.preds, cluster) | 
					
						
							|  |  |  |             dtrees[key] = dtree | 
					
						
							|  |  |  |         self.dtrees = dtrees | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _user_vars(self, arity): | 
					
						
							|  |  |  |         return tuple(("X%d" % i) for i in range(arity)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _lib_vars(self, arity): | 
					
						
							|  |  |  |         return tuple(("Y%d" % i) for i in range(arity)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPConstraintPrologGenerator(YAPConstraintGeneratorBase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         super(YAPConstraintPrologGenerator, self).__init__(filename) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _prolog_clauses(self): | 
					
						
							|  |  |  |         clauses = [] | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         for (name, arity), dtree in self.dtrees.items(): | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             user_vars = self._user_vars(arity) | 
					
						
							|  |  |  |             lib_vars = self._lib_vars(arity) | 
					
						
							|  |  |  |             head = "%s(%s)" % (name, ",".join(user_vars)) | 
					
						
							|  |  |  |             body = dtree._generate_body(user_vars, lib_vars) | 
					
						
							|  |  |  |             clause = PrologClause(head, body) | 
					
						
							|  |  |  |             clauses.append(clause) | 
					
						
							|  |  |  |         return clauses | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							|  |  |  |         out = OStream(sys.stdout) | 
					
						
							|  |  |  |         for clause in self._prolog_clauses(): | 
					
						
							|  |  |  |             clause.pp(out, 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPConstraintCCGenerator(YAPConstraintGeneratorBase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         super(YAPConstraintCCGenerator, self).__init__(filename) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cc_descriptors(self): | 
					
						
							|  |  |  |         descriptors = [] | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         for (name, arity), dtree in self.dtrees.items(): | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             descriptors.extend(dtree._cc_descriptors(name,())) | 
					
						
							|  |  |  |         return descriptors | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate_impl(self): | 
					
						
							|  |  |  |         for d in self._cc_descriptors(): | 
					
						
							|  |  |  |             d.generate_impl() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate_init(self): | 
					
						
							|  |  |  |         for d in self._cc_descriptors(): | 
					
						
							|  |  |  |             d.generate_init() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # output stream that keeps track of the current column | 
					
						
							|  |  |  | # to facilitate proper indentation | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class OStream(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, fd=sys.stdout): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.file = fd | 
					
						
							|  |  |  |         self.column = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def write(self, s): | 
					
						
							|  |  |  |         reset = False | 
					
						
							|  |  |  |         for x in s.split('\n'): | 
					
						
							|  |  |  |             if reset: | 
					
						
							|  |  |  |                 self.newline() | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 reset = True | 
					
						
							|  |  |  |             self.file.write(x) | 
					
						
							|  |  |  |             self.column += len(x) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def newline(self): | 
					
						
							|  |  |  |         self.file.write("\n") | 
					
						
							|  |  |  |         self.column = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def writeln(self, s=None): | 
					
						
							|  |  |  |         if s is not None: | 
					
						
							|  |  |  |             self.write(s) | 
					
						
							|  |  |  |         self.newline() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def indent_to(self, n): | 
					
						
							|  |  |  |         if n<self.column: | 
					
						
							|  |  |  |             self.newline() | 
					
						
							|  |  |  |         n = n - self.column | 
					
						
							|  |  |  |         while n>0: | 
					
						
							|  |  |  |             self.write(' ') | 
					
						
							|  |  |  |             n -= 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # representation of prolog code that automatically supports proper indentation | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PrologObject(object): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PrologClause(PrologObject): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, head, body): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object, object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.head = head | 
					
						
							|  |  |  |         self.body = body | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def pp(self, out, offset): | 
					
						
							|  |  |  |         out.indent_to(offset) | 
					
						
							|  |  |  |         out.write(self.head) | 
					
						
							|  |  |  |         out.writeln(" :-") | 
					
						
							|  |  |  |         self.body.pp(out, offset+8) | 
					
						
							|  |  |  |         out.writeln(".") | 
					
						
							|  |  |  |         out.writeln() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PrologLiteral(PrologObject): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, lit): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.literal = lit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def pp(self, out, offset): | 
					
						
							|  |  |  |         out.indent_to(offset) | 
					
						
							|  |  |  |         out.write(self.literal) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PrologIF(PrologObject): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, cond, left, right): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object, object, object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.cond = cond | 
					
						
							|  |  |  |         self.left = left | 
					
						
							|  |  |  |         self.right = right | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def pp(self, out, offset): | 
					
						
							|  |  |  |         out.indent_to(offset) | 
					
						
							|  |  |  |         out.write("("), | 
					
						
							|  |  |  |         indent = offset+1 | 
					
						
							|  |  |  |         self.cond.pp(out, indent) | 
					
						
							|  |  |  |         out.writeln() | 
					
						
							|  |  |  |         out.indent_to(indent) | 
					
						
							|  |  |  |         out.write("-> ") | 
					
						
							|  |  |  |         self.left.pp(out, indent+3) | 
					
						
							|  |  |  |         out.writeln() | 
					
						
							|  |  |  |         out.indent_to(indent) | 
					
						
							|  |  |  |         out.write(";  ") | 
					
						
							|  |  |  |         self.right.pp(out, indent+3) | 
					
						
							|  |  |  |         out.write(")") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ENUM_CLASSES = None | 
					
						
							|  |  |  | ENUM_CLASSES_AVOID = ('ScriptMode','ViewSelStatus','ExecStatus', | 
					
						
							|  |  |  |                       'ActorProperty','SpaceStatus') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def enum_classes(): | 
					
						
							|  |  |  |     global ENUM_CLASSES | 
					
						
							|  |  |  |     if ENUM_CLASSES is None: | 
					
						
							|  |  |  |         filename = "gecode-enums-%s.py" % gecode_version() | 
					
						
							|  |  |  |         import imp | 
					
						
							|  |  |  |         ENUM_CLASSES = imp.load_source( | 
					
						
							|  |  |  |             "gecode_enums", | 
					
						
							|  |  |  |             filename).ENUM_CLASSES | 
					
						
							|  |  |  |         ENUM_CLASSES = tuple(x for x in ENUM_CLASSES if x.TYPE not in ENUM_CLASSES_AVOID) | 
					
						
							|  |  |  |     return ENUM_CLASSES | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumImpl(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							|  |  |  |         self._generate_atoms() | 
					
						
							|  |  |  |         self._generate_from_term() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _generate_atoms(self): | 
					
						
							|  |  |  |         for x in self.ENUM: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print("static YAP_Term gecode_%s;" % x) | 
					
						
							|  |  |  |         print() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _generate_from_term(self): | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         t2 = self.TYPE | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print("static %s gecode_%s_from_term(YAP_Term X)" % (self.TYPE,t2)) | 
					
						
							|  |  |  |         print("{") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         for x in self.ENUM: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print("  if (X==gecode_%s) return %s;" % (x,x)) | 
					
						
							|  |  |  |         print('  cerr << "this should never happen" << endl; exit(1);') | 
					
						
							|  |  |  |         print("}") | 
					
						
							|  |  |  |         print() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _generate_from_term_forward_decl(self): | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         t2 =  self.TYPE | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print("static %s gecode_%s_from_term(YAP_Term);" % (self.TYPE,t2)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumImplGenerator(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         # generate_space_function_forward() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         for c in enum_classes(): | 
					
						
							|  |  |  |             class C(c,YAPEnumImpl): pass | 
					
						
							|  |  |  |             o = C() | 
					
						
							|  |  |  |             o.generate() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumForwardGenerator(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         # gsenerate_space_function_forward() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         for c in enum_classes(): | 
					
						
							|  |  |  |             class C(c,YAPEnumImpl): pass | 
					
						
							|  |  |  |             o = C() | 
					
						
							|  |  |  |             o._generate_from_term_forward_decl() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumInit(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							|  |  |  |         for x in self.ENUM: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print('{ YAP_Atom X= YAP_LookupAtom("%s");' % x) | 
					
						
							|  |  |  |             print('  gecode_%s = YAP_MkAtomTerm(X);' % x) | 
					
						
							|  |  |  |             print('  YAP_AtomGetHold(X); }') | 
					
						
							|  |  |  |         print() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumInitGenerator(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							|  |  |  |         for c in enum_classes(): | 
					
						
							|  |  |  |             class C(c,YAPEnumInit): pass | 
					
						
							|  |  |  |             o = C() | 
					
						
							|  |  |  |             o.generate() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumProlog(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate(self): | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |         t = self.TYPE | 
					
						
							|  |  |  |         if t.startswith("std::function") and t.endswith(")>"): | 
					
						
							|  |  |  |             t = "std_function" | 
					
						
							|  |  |  |         elif t.endswith(">"): | 
					
						
							|  |  |  |             sp = t.split("<") | 
					
						
							|  |  |  |             if len(sp) > 1: | 
					
						
							|  |  |  |                 t = sp[1].rstrip(">") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         for x in self.ENUM: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print("is_%s_('%s')." % (t, x)) | 
					
						
							| 
									
										
										
										
											2018-06-18 10:25:17 +01:00
										 |  |  |         print("\n") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         for x in self.ENUM: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |             print("is_%s_('%s','%s')." % (t, x, x)) | 
					
						
							| 
									
										
										
										
											2018-06-18 10:25:17 +01:00
										 |  |  |         print("\n") | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print("is_%s(X,Y) :- nonvar(X), is_%s_(X,Y)." % (t,t)) | 
					
						
							|  |  |  |         print("is_%s(X) :- is_%s_(X,_)." % (t,t)) | 
					
						
							| 
									
										
										
										
											2018-06-18 10:25:17 +01:00
										 |  |  |         print("\n") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class YAPEnumPrologGenerator(object): | 
					
						
							| 
									
										
										
										
											2016-12-10 03:09:59 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     def generate(self): | 
					
						
							|  |  |  |         for c in enum_classes(): | 
					
						
							|  |  |  |             class C(c,YAPEnumProlog): pass | 
					
						
							|  |  |  |             o = C() | 
					
						
							|  |  |  |             o.generate() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CCDescriptor(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, argtypes, api): | 
					
						
							| 
									
										
										
										
											2017-06-19 19:02:36 +01:00
										 |  |  |         # type: (object, object, object) -> object | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.argtypes = argtypes | 
					
						
							|  |  |  |         self.api = api | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def generate_impl(self): | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print("static YAP_Bool gecode_constraint_%s(void)" % self.api) | 
					
						
							|  |  |  |         print("{") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         i = 1 | 
					
						
							|  |  |  |         args = [] | 
					
						
							|  |  |  |         has_space = False | 
					
						
							|  |  |  |         for t in self.argtypes: | 
					
						
							|  |  |  |             v = "X%d" % i | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |             v2 = v | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             a = "YAP_ARG%d" % i | 
					
						
							|  |  |  |             if t=="Space": | 
					
						
							|  |  |  |                 v = "*space" | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |                 print("  GenericSpace* space = gecode_Space_from_term(%s);" % (a)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |                 has_space = True | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |                 v2 = v | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2017-07-24 18:14:57 +01:00
										 |  |  |                 t2 = t | 
					
						
							| 
									
										
										
										
											2017-11-08 10:19:10 +01:00
										 |  |  |                 if t.startswith("std::function") and t.endswith(")>"): | 
					
						
							|  |  |  |                     sp = t.split("&") | 
					
						
							|  |  |  |                     if len(sp) > 1: | 
					
						
							|  |  |  |                         t2 = "std_function" | 
					
						
							|  |  |  |                         v2 = v #sp[0]+"("+v+")" | 
					
						
							|  |  |  |                 elif t.endswith(">"): | 
					
						
							|  |  |  |                     sp = t.split("<") | 
					
						
							|  |  |  |                     if len(sp) > 1: | 
					
						
							|  |  |  |                         t2 = sp[1].rstrip(">") | 
					
						
							|  |  |  |                         v2 = v #sp[0]+"("+v+")" | 
					
						
							|  |  |  |                 extra = "" | 
					
						
							|  |  |  |                 if t in ("IntVar","BoolVar","SetVar","FloatVar","IntVarArgs","BoolVarArgs","SetVarArgs","FloatVarArgs","std_function"): | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |                     extra = "space," | 
					
						
							|  |  |  |                     if has_space == False: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |                         print("  GenericSpace* space = gecode_Space_from_term(%s);" % a) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |                         has_space = True | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |                 print("  %s %s = gecode_%s_from_term(%s%s);" % (t,v,t2,extra,a)) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |             args.append(v) | 
					
						
							|  |  |  |             i += 1 | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print("  %s(%s);" % (self.name, ",".join(args))) | 
					
						
							|  |  |  |         print("  return TRUE;") | 
					
						
							|  |  |  |         print("}") | 
					
						
							|  |  |  |         print() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def generate_init(self): | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         print('YAP_UserCPredicate("gecode_constraint_%s", gecode_constraint_%s, %d);' \ | 
					
						
							|  |  |  |             % (self.api, self.api, len(self.argtypes))) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-18 14:47:29 +00:00
										 |  |  | GECODE_VERSION = "6.1.1" | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def gecode_version(): | 
					
						
							| 
									
										
										
										
											2017-11-17 23:56:51 +00:00
										 |  |  |     #import pdb; pdb.set_trace() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     global GECODE_VERSION | 
					
						
							|  |  |  |     if GECODE_VERSION is not None: | 
					
						
							|  |  |  |         return GECODE_VERSION | 
					
						
							|  |  |  |     from distutils.ccompiler import new_compiler | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         from distutils.ccompiler import customize_compiler | 
					
						
							|  |  |  |     except: | 
					
						
							|  |  |  |         from distutils.sysconfig import customize_compiler | 
					
						
							|  |  |  |     import os | 
					
						
							|  |  |  |     cxx = new_compiler() | 
					
						
							|  |  |  |     customize_compiler(cxx) | 
					
						
							|  |  |  |     pid = os.getpid() | 
					
						
							|  |  |  |     file_hh = "_gecode_version_%d.hh" % pid | 
					
						
							|  |  |  |     file_txt = "_gecode_version_%d.txt" % pid | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |     f = open(file_hh,"w") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     f.write("""#include "gecode/support/config.hpp"
 | 
					
						
							|  |  |  | @@GECODE_VERSION""")
 | 
					
						
							|  |  |  |     f.close() | 
					
						
							|  |  |  |     cxx.preprocess(file_hh,output_file=file_txt) | 
					
						
							| 
									
										
										
										
											2015-10-18 23:44:32 +01:00
										 |  |  |     if True: | 
					
						
							|  |  |  |         f = open(file_txt) | 
					
						
							|  |  |  |         version = "" | 
					
						
							|  |  |  |         for line in f: | 
					
						
							|  |  |  |             if line.startswith("@@"): | 
					
						
							|  |  |  |                 version = line[3:-2] | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         f.close() | 
					
						
							|  |  |  |         os.remove(file_hh) | 
					
						
							|  |  |  |         os.remove(file_txt) | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2018-03-12 15:11:59 +00:00
										 |  |  |         version = "6.0.0" | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     GECODE_VERSION = version | 
					
						
							|  |  |  |     return version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def generate_files(): | 
					
						
							| 
									
										
										
										
											2017-11-17 23:56:51 +00:00
										 |  |  |     DIR = "../" + gecode_version() | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     import os, os.path | 
					
						
							|  |  |  |     DIR = os.path.abspath(DIR) | 
					
						
							|  |  |  |     if not os.path.exists(DIR): | 
					
						
							|  |  |  |         os.mkdir(DIR) | 
					
						
							| 
									
										
										
										
											2017-11-17 23:56:51 +00:00
										 |  |  |     filename = "gecode-prototypes-" + gecode_version() + ".hh" | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |     import sys | 
					
						
							|  |  |  |     stdout = sys.stdout | 
					
						
							|  |  |  |     try: | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         sys.stdout = open(os.path.join(DIR,"gecode-version.txt"),"w") | 
					
						
							|  |  |  |         print(gecode_version()) | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         sys.stdout.close() | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         sys.stdout = open(os.path.join(DIR,"gecode_yap_auto_generated.yap"),"w") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         prolog_print_notice() | 
					
						
							|  |  |  |         YAPEnumPrologGenerator().generate() | 
					
						
							|  |  |  |         YAPConstraintPrologGenerator(filename).generate() | 
					
						
							|  |  |  |         sys.stdout.close() | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         sys.stdout = open(os.path.join(DIR,"gecode_yap_cc_impl_auto_generated.icc"),"w") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         cc_print_notice() | 
					
						
							|  |  |  |         YAPEnumImplGenerator().generate() | 
					
						
							|  |  |  |         YAPConstraintCCGenerator(filename).generate_impl() | 
					
						
							|  |  |  |         sys.stdout.close() | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         sys.stdout = open(os.path.join(DIR,"gecode_yap_cc_init_auto_generated.icc"),"w") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         cc_print_notice() | 
					
						
							|  |  |  |         YAPEnumInitGenerator().generate() | 
					
						
							|  |  |  |         YAPConstraintCCGenerator(filename).generate_init() | 
					
						
							|  |  |  |         sys.stdout.close() | 
					
						
							| 
									
										
										
										
											2017-11-11 21:50:55 +00:00
										 |  |  |         sys.stdout = open(os.path.join(DIR,"gecode_yap_cc_forward_auto_generated.icc"),"w") | 
					
						
							| 
									
										
										
										
											2013-11-03 14:12:38 +00:00
										 |  |  |         cc_print_notice() | 
					
						
							|  |  |  |         YAPEnumForwardGenerator().generate() | 
					
						
							|  |  |  |         sys.stdout.close() | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         sys.stdout = stdout | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     generate_files() |