Advertisement
Guest User

theano_python

a guest
Dec 29th, 2016
475
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Python 42.92 KB | None | 0 0
  1. 00180   {
  2. 00181       CLazyLinker *self;
  3. 00182
  4. 00183       self = (CLazyLinker *)type->tp_alloc(type, 0);
  5. 00184       if (self != NULL) {
  6. 00185         self->nodes = NULL;
  7. 00186         self->thunks = NULL;
  8. 00187         self->pre_call_clear = NULL;
  9. 00188
  10. 00189         self->allow_gc = 1;
  11. 00190         self->n_applies = 0;
  12. 00191         self->n_vars = 0;
  13. 00192         self->var_computed = NULL;
  14. 00193         self->var_computed_cells = NULL;
  15. 00194         self->var_value_cells = NULL;
  16. 00195         self->dependencies = NULL;
  17. 00196         self->n_dependencies = NULL;
  18. 00197
  19. 00198         self->n_output_vars = 0;
  20. 00199         self->output_vars = NULL;
  21. 00200
  22. 00201         self->is_lazy = NULL;
  23. 00202
  24. 00203         self->var_owner = NULL;
  25. 00204         self->var_has_owner = NULL;
  26. 00205
  27. 00206         self->node_n_inputs = NULL;
  28. 00207         self->node_n_outputs = NULL;
  29. 00208         self->node_inputs = NULL;
  30. 00209         self->node_outputs = NULL;
  31. 00210         self->node_inputs_outputs_base = NULL;
  32. 00211         self->node_prereqs = NULL;
  33. 00212         self->node_n_prereqs = NULL;
  34. 00213
  35. 00214         self->update_storage = NULL;
  36. 00215         self->n_updates = 0;
  37. 00216
  38. 00217         self->thunk_cptr_data = NULL;
  39. 00218         self->thunk_cptr_fn = NULL;
  40. 00219         self->call_times = NULL;
  41. 00220         self->call_counts = NULL;
  42. 00221         self->do_timing = 0;
  43. 00222
  44. 00223         self->need_update_inputs = 0;
  45. 00224         self->position_of_error = -1;
  46. 00225       }
  47. 00226       return (PyObject *)self;
  48. 00227   }
  49. 00228
  50. 00229   static int
  51. 00230   CLazyLinker_init(CLazyLinker *self, PyObject *args, PyObject *kwds)
  52. 00231   {
  53. 00232       static char *kwlist[] = {
  54. 00233         (char*)"nodes",
  55. 00234         (char*)"thunks",
  56. 00235         (char*)"pre_call_clear",
  57. 00236         (char*)"allow_gc",
  58. 00237         (char*)"call_counts",
  59. 00238         (char*)"call_times",
  60. 00239         (char*)"compute_map_list",
  61. 00240         (char*)"storage_map_list",
  62. 00241         (char*)"base_input_output_list",
  63. 00242         (char*)"node_n_inputs",
  64. 00243         (char*)"node_n_outputs",
  65. 00244         (char*)"node_input_offset",
  66. 00245         (char*)"node_output_offset",
  67. 00246         (char*)"var_owner",
  68. 00247         (char*)"is_lazy_list",
  69. 00248         (char*)"output_vars",
  70. 00249         (char*)"node_prereqs",
  71. 00250         (char*)"node_output_size",
  72. 00251         (char*)"update_storage",
  73. 00252         (char*)"dependencies",
  74. 00253         NULL};
  75. 00254
  76. 00255       PyObject *compute_map_list=NULL,
  77. 00256                *storage_map_list=NULL,
  78. 00257                *base_input_output_list=NULL,
  79. 00258                *node_n_inputs=NULL,
  80. 00259                *node_n_outputs=NULL,
  81. 00260                *node_input_offset=NULL,
  82. 00261                *node_output_offset=NULL,
  83. 00262                *var_owner=NULL,
  84. 00263                *is_lazy=NULL,
  85. 00264                *output_vars=NULL,
  86. 00265                *node_prereqs=NULL,
  87. 00266                *node_output_size=NULL,
  88. 00267                *update_storage=NULL,
  89. 00268                *dependencies=NULL;
  90. 00269
  91. 00270       assert(!self->nodes);
  92. 00271       if (! PyArg_ParseTupleAndKeywords(args, kwds, "OOOiOOOOOOOOOOOOOOOO", kwlist,
  93. 00272                                         &self->nodes,
  94. 00273                                         &self->thunks,
  95. 00274                                         &self->pre_call_clear,
  96. 00275                                         &self->allow_gc,
  97. 00276                                         &self->call_counts,
  98. 00277                                         &self->call_times,
  99. 00278                                         &compute_map_list,
  100. 00279                                         &storage_map_list,
  101. 00280                                         &base_input_output_list,
  102. 00281                                         &node_n_inputs,
  103. 00282                                         &node_n_outputs,
  104. 00283                                         &node_input_offset,
  105. 00284                                         &node_output_offset,
  106. 00285                                         &var_owner,
  107. 00286                                         &is_lazy,
  108. 00287                                         &output_vars,
  109. 00288                                         &node_prereqs,
  110. 00289                                         &node_output_size,
  111. 00290                                         &update_storage,
  112. 00291                                         &dependencies
  113. 00292                                         ))
  114. 00293           return -1;
  115. 00294       Py_INCREF(self->nodes);
  116. 00295       Py_INCREF(self->thunks);
  117. 00296       Py_INCREF(self->pre_call_clear);
  118. 00297       Py_INCREF(self->call_counts);
  119. 00298       Py_INCREF(self->call_times);
  120. 00299
  121. 00300       Py_ssize_t n_applies = PyList_Size(self->nodes);
  122. 00301
  123. 00302       self->n_applies = n_applies;
  124. 00303       self->n_vars = PyList_Size(var_owner);
  125. 00304
  126. 00305       if (PyList_Size(self->thunks) != n_applies) return -1;
  127. 00306       if (PyList_Size(self->call_counts) != n_applies) return -1;
  128. 00307       if (PyList_Size(self->call_times) != n_applies) return -1;
  129. 00308
  130. 00309       // allocated and initialize thunk_cptr_data and thunk_cptr_fn
  131. 00310       if (n_applies)
  132. 00311         {
  133. 00312           self->thunk_cptr_data = (void**)calloc(n_applies, sizeof(void*));
  134. 00313           self->thunk_cptr_fn = (void**)calloc(n_applies, sizeof(void*));
  135. 00314           self->is_lazy = (int*)calloc(n_applies, sizeof(int));
  136. 00315           self->node_prereqs = (Py_ssize_t**)calloc(n_applies, sizeof(Py_ssize_t*));
  137. 00316           self->node_n_prereqs = (Py_ssize_t*)calloc(n_applies, sizeof(Py_ssize_t));
  138. 00317           assert(self->node_prereqs);
  139. 00318           assert(self->node_n_prereqs);
  140. 00319           assert(self->is_lazy);
  141. 00320           assert(self->thunk_cptr_fn);
  142. 00321           assert(self->thunk_cptr_data);
  143. 00322
  144. 00323           for (int i = 0; i < n_applies; ++i)
  145. 00324             {
  146. 00325               PyObject * thunk = PyList_GetItem(self->thunks, i);
  147. 00326               //thunk is borrowed
  148. 00327               if (PyObject_HasAttrString(thunk, "cthunk"))
  149. 00328                 {
  150. 00329                   PyObject * cthunk = PyObject_GetAttrString(thunk, "cthunk");
  151. 00330                   //new reference
  152. 00331                   assert (cthunk && PyCObject_Check(cthunk));
  153. 00332                   self->thunk_cptr_fn[i] = PyCObject_AsVoidPtr(cthunk);
  154. 00333                   self->thunk_cptr_data[i] = PyCObject_GetDesc(cthunk);
  155. 00334                   Py_DECREF(cthunk);
  156. 00335                   // cthunk is kept alive by membership in self->thunks
  157. 00336                 }
  158. 00337
  159. 00338               PyObject * el_i = PyList_GetItem(is_lazy, i);
  160. 00339               self->is_lazy[i] = PyNumber_AsSsize_t(el_i, NULL);
  161. 00340
  162. 00341               /* now get the prereqs */
  163. 00342               el_i = PyList_GetItem(node_prereqs, i);
  164. 00343               assert (PyList_Check(el_i));
  165. 00344               self->node_n_prereqs[i] = PyList_Size(el_i);
  166. 00345               if (self->node_n_prereqs[i])
  167. 00346                 {
  168. 00347                   self->node_prereqs[i] = (Py_ssize_t*)malloc(
  169. 00348                                 PyList_Size(el_i)*sizeof(Py_ssize_t));
  170. 00349                   for (int j = 0; j < PyList_Size(el_i); ++j)
  171. 00350                     {
  172. 00351                       PyObject * el_ij = PyList_GetItem(el_i, j);
  173. 00352                       Py_ssize_t N = PyNumber_AsSsize_t(el_ij, PyExc_IndexError);
  174. 00353                       if (PyErr_Occurred())
  175. 00354                         return -1;
  176. 00355                       // N < n. variables
  177. 00356                       assert(N < PyList_Size(var_owner));
  178. 00357                       self->node_prereqs[i][j] = N;
  179. 00358                     }
  180. 00359                 }
  181. 00360             }
  182. 00361         }
  183. 00362       if (PyList_Check(base_input_output_list))
  184. 00363         {
  185. 00364           Py_ssize_t n_inputs_outputs_base = PyList_Size(base_input_output_list);
  186. 00365           self->node_inputs_outputs_base = (Py_ssize_t*)calloc(n_inputs_outputs_base,sizeof(Py_ssize_t));
  187. 00366           assert(self->node_inputs_outputs_base);
  188. 00367           for (int i = 0; i < n_inputs_outputs_base; ++i)
  189. 00368             {
  190. 00369               PyObject *el_i = PyList_GetItem(base_input_output_list, i);
  191. 00370               Py_ssize_t idx = PyNumber_AsSsize_t(el_i, PyExc_IndexError);
  192. 00371               if (PyErr_Occurred()) return -1;
  193. 00372               self->node_inputs_outputs_base[i] = idx;
  194. 00373             }
  195. 00374           self->node_n_inputs = (Py_ssize_t*)calloc(n_applies,sizeof(Py_ssize_t));
  196. 00375           assert(self->node_n_inputs);
  197. 00376           self->node_n_outputs = (Py_ssize_t*)calloc(n_applies,sizeof(Py_ssize_t));
  198. 00377           assert(self->node_n_outputs);
  199. 00378           self->node_inputs = (Py_ssize_t**)calloc(n_applies,sizeof(Py_ssize_t*));
  200. 00379           assert(self->node_inputs);
  201. 00380           self->node_outputs = (Py_ssize_t**)calloc(n_applies,sizeof(Py_ssize_t*));
  202. 00381           assert(self->node_outputs);
  203. 00382           for (int i = 0; i < n_applies; ++i)
  204. 00383             {
  205. 00384               Py_ssize_t N;
  206. 00385               N = PyNumber_AsSsize_t(PyList_GetItem(node_n_inputs, i),PyExc_IndexError);
  207. 00386               if (PyErr_Occurred()) return -1;
  208. 00387               assert (N <= n_inputs_outputs_base);
  209. 00388               self->node_n_inputs[i] = N;
  210. 00389               N = PyNumber_AsSsize_t(PyList_GetItem(node_n_outputs, i),PyExc_IndexError);
  211. 00390               if (PyErr_Occurred()) return -1;
  212. 00391               assert (N <= n_inputs_outputs_base);
  213. 00392               self->node_n_outputs[i] = N;
  214. 00393               N = PyNumber_AsSsize_t(PyList_GetItem(node_input_offset, i),PyExc_IndexError);
  215. 00394               if (PyErr_Occurred()) return -1;
  216. 00395               assert (N <= n_inputs_outputs_base);
  217. 00396               self->node_inputs[i] = &self->node_inputs_outputs_base[N];
  218. 00397               N = PyNumber_AsSsize_t(PyList_GetItem(node_output_offset, i),PyExc_IndexError);
  219. 00398               if (PyErr_Occurred()) return -1;
  220. 00399               assert (N <= n_inputs_outputs_base);
  221. 00400               self->node_outputs[i] = &self->node_inputs_outputs_base[N];
  222. 00401             }
  223. 00402         }
  224. 00403       else
  225. 00404         {
  226. 00405           PyErr_SetString(PyExc_TypeError, "base_input_output_list must be list");
  227. 00406           return -1;
  228. 00407         }
  229. 00408
  230. 00409       // allocation for var_owner
  231. 00410       if (PyList_Check(var_owner))
  232. 00411         {
  233. 00412           self->var_owner = (Py_ssize_t*)calloc(self->n_vars,sizeof(Py_ssize_t));
  234. 00413           self->var_has_owner = (int*)calloc(self->n_vars,sizeof(int));
  235. 00414           self->var_computed = (int*)calloc(self->n_vars,sizeof(int));
  236. 00415           self->var_computed_cells = (PyObject**)calloc(self->n_vars,sizeof(PyObject*));
  237. 00416           self->var_value_cells = (PyObject**)calloc(self->n_vars,sizeof(PyObject*));
  238. 00417           for (int i = 0; i < self->n_vars; ++i)
  239. 00418             {
  240. 00419               PyObject * el_i = PyList_GetItem(var_owner, i);
  241. 00420               if (el_i == Py_None)
  242. 00421                 {
  243. 00422                   self->var_has_owner[i] = 0;
  244. 00423                 }
  245. 00424               else
  246. 00425                 {
  247. 00426                   Py_ssize_t N = PyNumber_AsSsize_t(el_i, PyExc_IndexError);
  248. 00427                   if (PyErr_Occurred()) return -1;
  249. 00428                   assert (N <= n_applies);
  250. 00429                   self->var_owner[i] = N;
  251. 00430                   self->var_has_owner[i] = 1;
  252. 00431                 }
  253. 00432               self->var_computed_cells[i] = PyList_GetItem(compute_map_list, i);
  254. 00433               Py_INCREF(self->var_computed_cells[i]);
  255. 00434               self->var_value_cells[i] = PyList_GetItem(storage_map_list, i);
  256. 00435               Py_INCREF(self->var_value_cells[i]);
  257. 00436             }
  258. 00437         }
  259. 00438       else
  260. 00439         {
  261. 00440           PyErr_SetString(PyExc_TypeError, "var_owner must be list");
  262. 00441           return -1;
  263. 00442         }
  264. 00443
  265. 00444       if (dependencies != Py_None)
  266. 00445         {
  267. 00446           self->dependencies = (Py_ssize_t**)calloc(self->n_vars, sizeof(Py_ssize_t *));
  268. 00447           self->n_dependencies = (Py_ssize_t*)calloc(self->n_vars, sizeof(Py_ssize_t));
  269. 00448           assert(self->dependencies);
  270. 00449           assert(self->n_dependencies);
  271. 00450
  272. 00451           for (int i = 0; i < self->n_vars; ++i)
  273. 00452             {
  274. 00453               PyObject *tmp = PyList_GetItem(dependencies, i);
  275. 00454               // refcounting - tmp is borrowed
  276. 00455               if (unpack_list_of_ssize_t(tmp, &self->dependencies[i], &self->n_dependencies[i],
  277. 00456                                          "dependencies"))
  278. 00457                 return -1;
  279. 00458             }
  280. 00459         }
  281. 00460
  282. 00461       if (unpack_list_of_ssize_t(output_vars, &self->output_vars, &self->n_output_vars,
  283. 00462                                  "output_vars"))
  284. 00463         return -1;
  285. 00464       for (int i = 0; i < self->n_output_vars; ++i)
  286. 00465         {
  287. 00466           assert(self->output_vars[i] < self->n_vars);
  288. 00467         }
  289. 00468       if (unpack_list_of_ssize_t(update_storage, &self->update_storage, &self->n_updates,
  290. 00469                                  "updates_storage"))
  291. 00470         return -1;
  292. 00471       return 0;
  293. 00472   }
  294. 00473   static void set_position_of_error(CLazyLinker * self, int owner_idx)
  295. 00474   {
  296. 00475     if (self->position_of_error == -1)
  297. 00476       {
  298. 00477         self->position_of_error = owner_idx;
  299. 00478       }
  300. 00479   }
  301. 00480   static PyObject * pycall(CLazyLinker * self, Py_ssize_t node_idx, int verbose)
  302. 00481   {
  303. 00482     // call thunk to see which inputs it wants
  304. 00483     PyObject * thunk = PyList_GetItem(self->thunks, node_idx);
  305. 00484     // refcounting - thunk is borrowed
  306. 00485     PyObject * rval = NULL;
  307. 00486     if (self->do_timing)
  308. 00487       {
  309. 00488         double t0 = pytime(NULL);
  310. 00489         if (verbose) fprintf(stderr, "calling via Python (node %i)\n", (int)node_idx);
  311. 00490         rval = PyObject_CallObject(thunk, NULL);
  312. 00491         if (rval)
  313. 00492           {
  314. 00493             double t1 = pytime(NULL);
  315. 00494             double ti = PyFloat_AsDouble(
  316. 00495                            PyList_GetItem(self->call_times, node_idx));
  317. 00496             PyList_SetItem(self->call_times, node_idx,
  318. 00497                            PyFloat_FromDouble(t1 - t0 + ti));
  319. 00498             PyObject * count = PyList_GetItem(self->call_counts, node_idx);
  320. 00499             long icount = PyInt_AsLong(count);
  321. 00500             PyList_SetItem(self->call_counts, node_idx,
  322. 00501                            PyInt_FromLong(icount + 1));
  323. 00502         }
  324. 00503       }
  325. 00504     else
  326. 00505       {
  327. 00506         if (verbose)
  328. 00507           {
  329. 00508             fprintf(stderr, "calling via Python (node %i)\n", (int)node_idx);
  330. 00509           }
  331. 00510         rval = PyObject_CallObject(thunk, NULL);
  332. 00511       }
  333. 00512     return rval;
  334. 00513   }
  335. 00514   static int c_call(CLazyLinker * self, Py_ssize_t node_idx, int verbose)
  336. 00515   {
  337. 00516     void * ptr_addr = self->thunk_cptr_fn[node_idx];
  338. 00517     int (*fn)(void*) = (int (*)(void*))(ptr_addr);
  339. 00518     if (verbose) fprintf(stderr, "calling non-lazy shortcut (node %i)\n", (int)node_idx);
  340. 00519     int err = 0;
  341. 00520     if (self->do_timing)
  342. 00521       {
  343. 00522         double t0 = pytime(NULL);
  344. 00523         err = fn(self->thunk_cptr_data[node_idx]);
  345. 00524         double t1 = pytime(NULL);
  346. 00525         double ti = PyFloat_AsDouble(PyList_GetItem(self->call_times, node_idx));
  347. 00526         PyList_SetItem(self->call_times, node_idx, PyFloat_FromDouble(t1 - t0 + ti));
  348. 00527         PyObject * count = PyList_GetItem(self->call_counts, node_idx);
  349. 00528         long icount = PyInt_AsLong(count);
  350. 00529         PyList_SetItem(self->call_counts, node_idx, PyInt_FromLong(icount+1));
  351. 00530       }
  352. 00531     else
  353. 00532       {
  354. 00533         err = fn(self->thunk_cptr_data[node_idx]);
  355. 00534       }
  356. 00535
  357. 00536     if (err)
  358. 00537       {
  359. 00538         // cast the argument to a PyList (as described near line 226 of cc.py)
  360. 00539         PyObject * __ERROR = ((PyObject**)self->thunk_cptr_data[node_idx])[0];
  361. 00540         assert (PyList_Check(__ERROR));
  362. 00541         assert (PyList_Size(__ERROR) == 3);
  363. 00542         PyObject * err_type = PyList_GetItem(__ERROR, 0); //stolen ref
  364. 00543         PyObject * err_msg = PyList_GetItem(__ERROR, 1); //stolen ref
  365. 00544         PyObject * err_trace = PyList_GetItem(__ERROR, 2); //stolen ref
  366. 00545         PyList_SET_ITEM(__ERROR, 0, Py_None); Py_INCREF(Py_None); //clobbers old ref
  367. 00546         PyList_SET_ITEM(__ERROR, 1, Py_None); Py_INCREF(Py_None); //clobbers old ref
  368. 00547         PyList_SET_ITEM(__ERROR, 2, Py_None); Py_INCREF(Py_None); //clobbers old ref
  369. 00548
  370. 00549         assert(!PyErr_Occurred()); // because CLinker hid the exception in __ERROR aka data
  371. 00550         PyErr_Restore(err_type, err_msg, err_trace); //steals refs to args
  372. 00551       }
  373. 00552     if (err) set_position_of_error(self, node_idx);
  374. 00553     return err;
  375. 00554   }
  376. 00555   static
  377. 00556   int lazy_rec_eval(CLazyLinker * self, Py_ssize_t var_idx, PyObject*one, PyObject*zero)
  378. 00557   {
  379. 00558     PyObject *rval = NULL;
  380. 00559     int verbose = 0;
  381. 00560     int err = 0;
  382. 00561
  383. 00562     if (verbose) fprintf(stderr, "lazy_rec computing %i\n", (int)var_idx);
  384. 00563
  385. 00564     if (self->var_computed[var_idx] || !self->var_has_owner[var_idx])
  386. 00565       return 0;
  387. 00566
  388. 00567     Py_ssize_t owner_idx = self->var_owner[var_idx];
  389. 00568
  390. 00569     // STEP 1: compute the pre-requirements of the node
  391. 00570     // Includes input nodes for non-lazy ops.
  392. 00571     for (int i = 0; i < self->node_n_prereqs[owner_idx]; ++i)
  393. 00572       {
  394. 00573         Py_ssize_t prereq_idx = self->node_prereqs[owner_idx][i];
  395. 00574         if (!self->var_computed[prereq_idx])
  396. 00575           {
  397. 00576             err = lazy_rec_eval(self, prereq_idx, one, zero);
  398. 00577             if (err) return err;
  399. 00578           }
  400. 00579         assert (self->var_computed[prereq_idx]);
  401. 00580       }
  402. 00581
  403. 00582     // STEP 2: compute the node itself
  404. 00583     if (self->is_lazy[owner_idx])
  405. 00584       {
  406. 00585         // update the compute_map cells corresponding to the inputs of this thunk
  407. 00586         for (int i = 0; i < self->node_n_inputs[owner_idx]; ++i)
  408. 00587           {
  409. 00588             int in_idx = self->node_inputs[owner_idx][i];
  410. 00589             if (self->var_computed[in_idx])
  411. 00590               {
  412. 00591                 Py_INCREF(one);
  413. 00592                 err = PyList_SetItem(self->var_computed_cells[in_idx], 0, one);
  414. 00593               }
  415. 00594             else
  416. 00595               {
  417. 00596                 Py_INCREF(zero);
  418. 00597                 err = PyList_SetItem(self->var_computed_cells[in_idx], 0, zero);
  419. 00598               }
  420. 00599             if (err) goto fail;
  421. 00600           }
  422. 00601
  423. 00602         rval = pycall(self, owner_idx, verbose);
  424. 00603         // refcounting - rval is new ref
  425. 00604         //TODO: to prevent infinite loops
  426. 00605         // - consider check that a thunk does not ask for an input that is already computed
  427. 00606         if (rval == NULL)
  428. 00607           {
  429. 00608             assert (PyErr_Occurred());
  430. 00609             err = 1;
  431. 00610             goto fail;
  432. 00611           }
  433. 00612
  434. 00613         //update the computed-ness of any output cells
  435. 00614         for (int i = 0; i < self->node_n_outputs[owner_idx]; ++i)
  436. 00615           {
  437. 00616             int out_idx = self->node_outputs[owner_idx][i];
  438. 00617             PyObject * el_i = PyList_GetItem(self->var_computed_cells[out_idx], 0);
  439. 00618             Py_ssize_t N = PyNumber_AsSsize_t(el_i, PyExc_IndexError);
  440. 00619             if (PyErr_Occurred())
  441. 00620               {
  442. 00621                 err = -1;
  443. 00622                 goto pyfail;
  444. 00623               }
  445. 00624             assert (N==0 || N==1);
  446. 00625             self->var_computed[out_idx] = N;
  447. 00626           }
  448. 00627         if (!self->var_computed[var_idx])
  449. 00628           {
  450. 00629             /*
  451. 00630              * If self is not computed after the call, this means that some
  452. 00631              * inputs are needed.  Compute the ones on the returned list
  453. 00632              * and try to compute the current node again (with recursive call).
  454. 00633              * This allows a node to request more nodes more than once before
  455. 00634              * finally yielding a result.
  456. 00635              */
  457. 00636             if (!PyList_Check(rval))
  458. 00637               {
  459. 00638                 //TODO: More helpful error to help find *which node* made this
  460. 00639                 // bad thunk
  461. 00640                 PyErr_SetString(PyExc_TypeError,
  462. 00641                                 "lazy thunk should return a list");
  463. 00642                 err = 1;
  464. 00643                 goto pyfail;
  465. 00644               }
  466. 00645
  467. 00646             if (!PyList_Size(rval))
  468. 00647               {
  469. 00648                 PyErr_SetString(PyExc_ValueError,
  470. 00649                                 "lazy thunk returned empty list without computing output");
  471. 00650                 err = 1;
  472. 00651                 goto pyfail;
  473. 00652               }
  474. 00653
  475. 00654             for (int i = 0; i < PyList_Size(rval); ++i)
  476. 00655               {
  477. 00656                 PyObject * el_i = PyList_GetItem(rval, i);
  478. 00657                 Py_ssize_t N = PyNumber_AsSsize_t(el_i, PyExc_IndexError);
  479. 00658                 if (PyErr_Occurred())
  480. 00659                   {
  481. 00660                     err = 1;
  482. 00661                     goto pyfail;
  483. 00662                   }
  484. 00663                 assert (N <= self->node_n_inputs[owner_idx]);
  485. 00664                 Py_ssize_t input_idx = self->node_inputs[owner_idx][N];
  486. 00665                 err = lazy_rec_eval(self, input_idx, one, zero);
  487. 00666                 if (err) goto pyfail;
  488. 00667               }
  489. 00668
  490. 00669             Py_DECREF(rval);
  491. 00670             /*
  492. 00671              * We intentionally skip all the end-of-function processing
  493. 00672              * (mark outputs, GC) as it will be performed by the call
  494. 00673              * that actually manages to compute the result.
  495. 00674              */
  496. 00675             return lazy_rec_eval(self, var_idx, one, zero);
  497. 00676           }
  498. 00677
  499. 00678         Py_DECREF(rval);
  500. 00679       }
  501. 00680     else //owner is not a lazy op. Ensure all intputs are evaluated.
  502. 00681       {
  503. 00682         // loop over inputs to owner
  504. 00683         // call lazy_rec_eval on each one that is not computed.
  505. 00684         // if there's an error, pass it up the stack
  506. 00685         for (int i = 0; i < self->node_n_inputs[owner_idx]; ++i)
  507. 00686           {
  508. 00687             Py_ssize_t input_idx = self->node_inputs[owner_idx][i];
  509. 00688             if (!self->var_computed[input_idx])
  510. 00689               {
  511. 00690                 err = lazy_rec_eval(self, input_idx, one, zero);
  512. 00691                 if (err) return err;
  513. 00692               }
  514. 00693             assert (self->var_computed[input_idx]);
  515. 00694           }
  516. 00695
  517. 00696         // call the thunk for this owner.
  518. 00697         if (self->thunk_cptr_fn[owner_idx])
  519. 00698           {
  520. 00699             err = c_call(self, owner_idx, verbose);
  521. 00700             if (err) goto fail;
  522. 00701           }
  523. 00702         else
  524. 00703           {
  525. 00704             rval = pycall(self, owner_idx, verbose);
  526. 00705             //rval is new ref
  527. 00706             if (rval) //pycall returned normally (no exception)
  528. 00707               {
  529. 00708                 if (rval == Py_None)
  530. 00709                   {
  531. 00710                     Py_DECREF(rval); //ignore a return of None
  532. 00711                   }
  533. 00712                 else if (PyList_Check(rval))
  534. 00713                   {
  535. 00714                     PyErr_SetString(PyExc_TypeError,
  536. 00715                                     "non-lazy thunk should return None, not list");
  537. 00716                     err = 1;
  538. 00717                     goto pyfail;
  539. 00718                   }
  540. 00719                 else // don't know what it returned, but it wasn't right.
  541. 00720                   {
  542. 00721                     PyErr_SetObject(PyExc_TypeError, rval);
  543. 00722                     err = 1;
  544. 00723                     // We don't release rval since we put it in the error above
  545. 00724                     goto fail;
  546. 00725                   }
  547. 00726               }
  548. 00727             else // pycall returned NULL (internal error)
  549. 00728               {
  550. 00729                 err = 1;
  551. 00730                 goto fail;
  552. 00731               }
  553. 00732           }
  554. 00733       }
  555. 00734
  556. 00735     // loop over all outputs and mark them as computed
  557. 00736     for (int i = 0; i < self->node_n_outputs[owner_idx]; ++i)
  558. 00737       {
  559. 00738         self->var_computed[self->node_outputs[owner_idx][i]] = 1;
  560. 00739       }
  561. 00740
  562. 00741     // Free vars that are not needed anymore
  563. 00742     if (self->allow_gc)
  564. 00743       {
  565. 00744         for (int i = 0; i < self->node_n_inputs[owner_idx]; ++i)
  566. 00745           {
  567. 00746             int cleanup = 1;
  568. 00747             Py_ssize_t i_idx = self->node_inputs[owner_idx][i];
  569. 00748             if (!self->var_has_owner[i_idx])
  570. 00749               continue;
  571. 00750
  572. 00751             for (int j = 0; j < self->n_output_vars; ++j)
  573. 00752               {
  574. 00753                 if (i_idx == self->output_vars[j])
  575. 00754                   {
  576. 00755                     cleanup = 0;
  577. 00756                     break;
  578. 00757                   }
  579. 00758               }
  580. 00759             if (!cleanup) continue;
  581. 00760
  582. 00761             for (int j = 0; j < self->n_dependencies[i_idx]; ++j)
  583. 00762               {
  584. 00763                 if (!self->var_computed[self->dependencies[i_idx][j]])
  585. 00764                   {
  586. 00765                     cleanup = 0;
  587. 00766                     break;
  588. 00767                   }
  589. 00768               }
  590. 00769             if (!cleanup) continue;
  591. 00770
  592. 00771             Py_INCREF(Py_None);
  593. 00772             err = PyList_SetItem(self->var_value_cells[i_idx], 0, Py_None);
  594. 00773   //See the Stack gc implementation for why we change it to 2 and not 0.
  595. 00774             self->var_computed[i_idx] = 2;
  596. 00775             if (err) goto fail;
  597. 00776           }
  598. 00777       }
  599. 00778
  600. 00779     return 0;
  601. 00780    pyfail:
  602. 00781     Py_DECREF(rval);
  603. 00782    fail:
  604. 00783     set_position_of_error(self, owner_idx);
  605. 00784     return err;
  606. 00785   }
  607. 00786
  608. 00787   static PyObject *
  609. 00788   CLazyLinker_call(PyObject *_self, PyObject *args, PyObject *kwds)
  610. 00789   {
  611. 00790     CLazyLinker * self = (CLazyLinker*)_self;
  612. 00791     static char *kwlist[] = {
  613. 00792       (char *)"time_thunks",
  614. 00793       (char *)"n_calls",
  615. 00794       (char *)"output_subset",
  616. 00795       NULL};
  617. 00796     int n_calls=1;
  618. 00797     PyObject *output_subset_ptr = NULL;
  619. 00798     if (! PyArg_ParseTupleAndKeywords(args, kwds, "|iiO", kwlist,
  620. 00799                                       &self->do_timing,
  621. 00800                                       &n_calls,
  622. 00801                                       &output_subset_ptr))
  623. 00802       return NULL;
  624. 00803
  625. 00804     int err = 0;
  626. 00805     // parse an output_subset list
  627. 00806     // it is stored as a bool list of length n_output_vars: calculate a var or not
  628. 00807     char *output_subset = NULL;
  629. 00808     int output_subset_size = -1;
  630. 00809     if (output_subset_ptr != NULL)
  631. 00810       {
  632. 00811         if (! PyList_Check(output_subset_ptr))
  633. 00812           {
  634. 00813             err = 1;
  635. 00814             PyErr_SetString(PyExc_RuntimeError, "Output_subset is not a list");
  636. 00815           }
  637. 00816         else
  638. 00817           {
  639. 00818             output_subset_size = PyList_Size(output_subset_ptr);
  640. 00819             output_subset = (char*)calloc(self->n_output_vars, sizeof(char));
  641. 00820             for (int it = 0; it < output_subset_size; ++it)
  642. 00821               {
  643. 00822                 PyObject *elem = PyList_GetItem(output_subset_ptr, it);
  644. 00823                 if (! PyInt_Check(elem))
  645. 00824                   {
  646. 00825                     err = 1;
  647. 00826                     PyErr_SetString(PyExc_RuntimeError, "Some elements of output_subset list are not int");
  648. 00827                   }
  649. 00828                 output_subset[PyInt_AsLong(elem)] = 1;
  650. 00829               }
  651. 00830           }
  652. 00831       }
  653. 00832
  654. 00833     self->position_of_error = -1;
  655. 00834     // create constants used to fill the var_compute_cells
  656. 00835     PyObject * one = PyInt_FromLong(1);
  657. 00836     PyObject * zero = PyInt_FromLong(0);
  658. 00837
  659. 00838     // pre-allocate our return value
  660. 00839     Py_INCREF(Py_None);
  661. 00840     PyObject * rval = Py_None;
  662. 00841     //clear storage of pre_call_clear elements
  663. 00842     for (int call_i = 0; call_i < n_calls && (!err); ++call_i)
  664. 00843       {
  665. 00844         Py_ssize_t n_pre_call_clear = PyList_Size(self->pre_call_clear);
  666. 00845         assert(PyList_Check(self->pre_call_clear));
  667. 00846         for (int i = 0; i < n_pre_call_clear; ++i)
  668. 00847           {
  669. 00848             PyObject * el_i = PyList_GetItem(self->pre_call_clear, i);
  670. 00849             Py_INCREF(Py_None);
  671. 00850             PyList_SetItem(el_i, 0, Py_None);
  672. 00851           }
  673. 00852         //clear the computed flag out of all non-input vars
  674. 00853         for (int i = 0; i < self->n_vars; ++i)
  675. 00854           {
  676. 00855             self->var_computed[i] = !self->var_has_owner[i];
  677. 00856             if (self->var_computed[i])
  678. 00857               {
  679. 00858                 Py_INCREF(one);
  680. 00859                 PyList_SetItem(self->var_computed_cells[i], 0, one);
  681. 00860               }
  682. 00861             else
  683. 00862               {
  684. 00863                 Py_INCREF(zero);
  685. 00864                 PyList_SetItem(self->var_computed_cells[i], 0, zero);
  686. 00865               }
  687. 00866           }
  688. 00867
  689. 00868         int first_updated = self->n_output_vars - self->n_updates;
  690. 00869         for (int i = 0; i < self->n_output_vars && (!err); ++i)
  691. 00870           {
  692. 00871             if (i >= first_updated || output_subset == NULL || output_subset[i] == 1)
  693. 00872               {
  694. 00873                 err = lazy_rec_eval(self, self->output_vars[i], one, zero);
  695. 00874               }
  696. 00875           }
  697. 00876
  698. 00877         if (!err)
  699. 00878           {
  700. 00879             // save references to outputs prior to updating storage containers
  701. 00880             assert (self->n_output_vars >= self->n_updates);
  702. 00881             Py_DECREF(rval);
  703. 00882             rval = PyList_New(self->n_output_vars);
  704. 00883             for (int i = 0; i < (self->n_output_vars); ++i)
  705. 00884               {
  706. 00885                 Py_ssize_t src = self->output_vars[i];
  707. 00886                 PyObject * item = PyList_GetItem(self->var_value_cells[src], 0);
  708. 00887                 if ((output_subset == NULL || output_subset[i]) &&
  709. 00888                     self->var_computed[src] != 1)
  710. 00889                   {
  711. 00890                     err = 1;
  712. 00891                     PyErr_Format(PyExc_AssertionError,
  713. 00892                                  "The compute map of output %d should contain "
  714. 00893                                  "1 at the end of execution, not %d.",
  715. 00894                                  i, self->var_computed[src]);
  716. 00895                     break;
  717. 00896                   }
  718. 00897                 Py_INCREF(item);
  719. 00898                 PyList_SetItem(rval, i, item);
  720. 00899               }
  721. 00900           }
  722. 00901
  723. 00902         if (!err)
  724. 00903           {
  725. 00904             // Update the inputs that have an update rule
  726. 00905             for (int i = 0; i < self->n_updates; ++i)
  727. 00906               {
  728. 00907                 PyObject* tmp = PyList_GetItem(rval, self->n_output_vars - self->n_updates + i);
  729. 00908                 Py_INCREF(tmp);
  730. 00909                 Py_ssize_t dst = self->update_storage[i];
  731. 00910                 PyList_SetItem(self->var_value_cells[dst], 0, tmp);
  732. 00911               }
  733. 00912           }
  734. 00913       }
  735. 00914
  736. 00915     /*
  737. 00916       Clear everything that is left and not an output. This is needed
  738. 00917       for lazy evaluation since the current GC algo is too conservative
  739. 00918       with lazy graphs.
  740. 00919     */
  741. 00920     if (self->allow_gc && !err)
  742. 00921       {
  743. 00922         for (Py_ssize_t i = 0; i < self->n_vars; ++i)
  744. 00923           {
  745. 00924             int do_cleanup = 1;
  746. 00925             if (!self->var_has_owner[i] || !self->var_computed[i])
  747. 00926               continue;
  748. 00927             for (int j = 0; j < self->n_output_vars; ++j)
  749. 00928               {
  750. 00929                 if (i == self->output_vars[j])
  751. 00930                   {
  752. 00931                     do_cleanup = 0;
  753. 00932                     break;
  754. 00933                   }
  755. 00934               }
  756. 00935             if (!do_cleanup)
  757. 00936               continue;
  758. 00937             Py_INCREF(Py_None);
  759. 00938             PyList_SetItem(self->var_value_cells[i], 0, Py_None);
  760. 00939           }
  761. 00940       }
  762. 00941     if (output_subset != NULL)
  763. 00942       free(output_subset);
  764. 00943
  765. 00944     Py_DECREF(one);
  766. 00945     Py_DECREF(zero);
  767. 00946     if (err)
  768. 00947       {
  769. 00948         Py_DECREF(rval);
  770. 00949         return NULL;
  771. 00950       }
  772. 00951     return rval;
  773. 00952   }
  774. 00953
  775. 00954   #if 0
  776. 00955   static PyMethodDef CLazyLinker_methods[] = {
  777. 00956       {
  778. 00957         //"name", (PyCFunction)CLazyLinker_accept, METH_VARARGS, "Return the name, combining the first and last name"
  779. 00958       },
  780. 00959       {NULL}  /* Sentinel */
  781. 00960   };
  782. 00961   #endif
  783. 00962
  784. 00963
  785. 00964   static PyObject *
  786. 00965   CLazyLinker_get_allow_gc(CLazyLinker *self, void *closure)
  787. 00966   {
  788. 00967       return PyBool_FromLong(self->allow_gc);
  789. 00968   }
  790. 00969
  791. 00970   static int
  792. 00971   CLazyLinker_set_allow_gc(CLazyLinker *self, PyObject *value, void *closure)
  793. 00972   {
  794. 00973     if(!PyBool_Check(value))
  795. 00974       return -1;
  796. 00975
  797. 00976     if (value == Py_True)
  798. 00977       self->allow_gc = true;
  799. 00978     else
  800. 00979       self->allow_gc = false;
  801. 00980     return 0;
  802. 00981   }
  803. 00982
  804. 00983   static PyGetSetDef CLazyLinker_getset[] = {
  805. 00984     {(char*)"allow_gc",
  806. 00985      (getter)CLazyLinker_get_allow_gc,
  807. 00986      (setter)CLazyLinker_set_allow_gc,
  808. 00987      (char*)"do this function support allow_gc",
  809. 00988      NULL},
  810. 00989     {NULL, NULL, NULL, NULL}  /* Sentinel */
  811. 00990   };
  812. 00991   static PyMemberDef CLazyLinker_members[] = {
  813. 00992       {(char*)"nodes", T_OBJECT_EX, offsetof(CLazyLinker, nodes), 0,
  814. 00993        (char*)"list of nodes"},
  815. 00994       {(char*)"thunks", T_OBJECT_EX, offsetof(CLazyLinker, thunks), 0,
  816. 00995        (char*)"list of thunks in program"},
  817. 00996       {(char*)"call_counts", T_OBJECT_EX, offsetof(CLazyLinker, call_counts), 0,
  818. 00997        (char*)"number of calls of each thunk"},
  819. 00998       {(char*)"call_times", T_OBJECT_EX, offsetof(CLazyLinker, call_times), 0,
  820. 00999        (char*)"total runtime in each thunk"},
  821. 01000       {(char*)"position_of_error", T_INT, offsetof(CLazyLinker, position_of_error), 0,
  822. 01001        (char*)"position of failed thunk"},
  823. 01002       {(char*)"time_thunks", T_INT, offsetof(CLazyLinker, do_timing), 0,
  824. 01003        (char*)"bool: nonzero means call will time thunks"},
  825. 01004       {(char*)"need_update_inputs", T_INT, offsetof(CLazyLinker, need_update_inputs), 0,
  826. 01005        (char*)"bool: nonzero means Function.__call__ must implement update mechanism"},
  827. 01006       {NULL}  /* Sentinel */
  828. 01007   };
  829. 01008
  830. 01009   static PyTypeObject lazylinker_ext_CLazyLinkerType = {
  831. 01010   #if defined(NPY_PY3K)
  832. 01011       PyVarObject_HEAD_INIT(NULL, 0)
  833. 01012   #else
  834. 01013       PyObject_HEAD_INIT(NULL)
  835. 01014       0,                         /*ob_size*/
  836. 01015   #endif
  837. 01016       "lazylinker_ext.CLazyLinker",             /*tp_name*/
  838. 01017       sizeof(CLazyLinker), /*tp_basicsize*/
  839. 01018       0,                         /*tp_itemsize*/
  840. 01019       CLazyLinker_dealloc,       /*tp_dealloc*/
  841. 01020       0,                         /*tp_print*/
  842. 01021       0,                         /*tp_getattr*/
  843. 01022       0,                         /*tp_setattr*/
  844. 01023       0,                         /*tp_compare*/
  845. 01024       0,                         /*tp_repr*/
  846. 01025       0,                         /*tp_as_number*/
  847. 01026       0,                         /*tp_as_sequence*/
  848. 01027       0,                         /*tp_as_mapping*/
  849. 01028       0,                         /*tp_hash */
  850. 01029       CLazyLinker_call,          /*tp_call*/
  851. 01030       0,                         /*tp_str*/
  852. 01031       0,                         /*tp_getattro*/
  853. 01032       0,                         /*tp_setattro*/
  854. 01033       0,                         /*tp_as_buffer*/
  855. 01034       Py_TPFLAGS_DEFAULT|Py_TPFLAGS_BASETYPE,        /*tp_flags*/
  856. 01035       "CLazyLinker object",      /* tp_doc */
  857. 01036       0,                         /* tp_traverse */
  858. 01037       0,                         /* tp_clear */
  859. 01038       0,                         /* tp_richcompare */
  860. 01039       0,                         /* tp_weaklistoffset */
  861. 01040       0,                         /* tp_iter */
  862. 01041       0,                         /* tp_iternext */
  863. 01042       0,//CLazyLinker_methods,       /* tp_methods */
  864. 01043       CLazyLinker_members,       /* tp_members */
  865. 01044       CLazyLinker_getset,        /* tp_getset */
  866. 01045       0,                         /* tp_base */
  867. 01046       0,                         /* tp_dict */
  868. 01047       0,                         /* tp_descr_get */
  869. 01048       0,                         /* tp_descr_set */
  870. 01049       0,                         /* tp_dictoffset */
  871. 01050       (initproc)CLazyLinker_init,/* tp_init */
  872. 01051       0,                         /* tp_alloc */
  873. 01052       CLazyLinker_new,           /* tp_new */
  874. 01053   };
  875. 01054
  876. 01055   static PyObject * get_version(PyObject *dummy, PyObject *args)
  877. 01056   {
  878. 01057     PyObject *result = PyFloat_FromDouble(0.211);
  879. 01058     return result;
  880. 01059   }
  881. 01060
  882. 01061   static PyMethodDef lazylinker_ext_methods[] = {
  883. 01062     {"get_version",  get_version, METH_VARARGS, "Get extension version."},
  884. 01063     {NULL, NULL, 0, NULL}        /* Sentinel */
  885. 01064   };
  886. 01065
  887. 01066   #if defined(NPY_PY3K)
  888. 01067   static struct PyModuleDef moduledef = {
  889. 01068           PyModuleDef_HEAD_INIT,
  890. 01069           "lazylinker_ext",
  891. 01070           NULL,
  892. 01071           -1,
  893. 01072           lazylinker_ext_methods,
  894. 01073           NULL,
  895. 01074           NULL,
  896. 01075           NULL,
  897. 01076           NULL
  898. 01077   };
  899. 01078   #endif
  900. 01079   #if defined(NPY_PY3K)
  901. 01080   #define RETVAL m
  902. 01081   PyMODINIT_FUNC
  903. 01082   PyInit_lazylinker_ext(void) {
  904. 01083   #else
  905. 01084   #define RETVAL
  906. 01085   PyMODINIT_FUNC
  907. 01086   initlazylinker_ext(void)
  908. 01087   {
  909. 01088   #endif
  910. 01089       PyObject* m;
  911. 01090
  912. 01091       lazylinker_ext_CLazyLinkerType.tp_new = PyType_GenericNew;
  913. 01092       if (PyType_Ready(&lazylinker_ext_CLazyLinkerType) < 0)
  914. 01093           return RETVAL;
  915. 01094   #if defined(NPY_PY3K)
  916. 01095       m = PyModule_Create(&moduledef);
  917. 01096   #else
  918. 01097       m = Py_InitModule3("lazylinker_ext", lazylinker_ext_methods,
  919. 01098                          "Example module that creates an extension type.");
  920. 01099   #endif
  921. 01100       Py_INCREF(&lazylinker_ext_CLazyLinkerType);
  922. 01101       PyModule_AddObject(m, "CLazyLinker", (PyObject *)&lazylinker_ext_CLazyLinkerType);
  923. 01102
  924. 01103       return RETVAL;
  925. 01104   }
  926. 01105
  927. ===============================
  928. Problem occurred during compilation with the command line below:
  929. "c:\Rtools\mingw_32\bin\g++.exe" -shared -g -DNPY_NO_DEPRECATED_API=NPY_1_7_API_VERSION -m64 -DMS_WIN64 -I"C:\Program Files\Anaconda2\lib\site-packages\numpy\core\include" -I"C:\Program Files\Anaconda2\include" -I"C:\Users\silverstone\Theano\theano\gof" -L"C:\Program Files\Anaconda2\libs" -L"C:\Program Files\Anaconda2" -o C:\Users\silverstone\AppData\Local\Theano\compiledir_Windows-7-6.1.7601-SP1-Intel64_Family_6_Model_58_Stepping_9_GenuineIntel-2.7.13-64\lazylinker_ext\lazylinker_ext.pyd C:\Users\silverstone\AppData\Local\Theano\compiledir_Windows-7-6.1.7601-SP1-Intel64_Family_6_Model_58_Stepping_9_GenuineIntel-2.7.13-64\lazylinker_ext\mod.cpp -lpython27
  930. C:\Users\silverstone\AppData\Local\Theano\compiledir_Windows-7-6.1.7601-SP1-Intel64_Family_6_Model_58_Stepping_9_GenuineIntel-2.7.13-64\lazylinker_ext\mod.cpp:1:0: sorry, unimplemented: 64-bit mode not compiled in
  931.  #include <Python.h>
  932.  ^
  933.  
  934. ---------------------------------------------------------------------------
  935. Exception                                 Traceback (most recent call last)
  936. <ipython-input-1-3397704bd624> in <module>()
  937. ----> 1 import theano
  938.  
  939. C:\Users\silverstone\Theano\theano\__init__.pyc in <module>()
  940.      64     object2, utils)
  941.      65
  942. ---> 66 from theano.compile import (
  943.      67     SymbolicInput, In,
  944.      68     SymbolicOutput, Out,
  945.  
  946. C:\Users\silverstone\Theano\theano\compile\__init__.py in <module>()
  947.       8         SpecifyShape, specify_shape, register_specify_shape_c_code)
  948.       9
  949. ---> 10 from theano.compile.function_module import *
  950.      11
  951.      12 from theano.compile.mode import *
  952.  
  953. C:\Users\silverstone\Theano\theano\compile\function_module.py in <module>()
  954.      19 from theano.compat import izip
  955.      20 from theano.gof import graph
  956. ---> 21 import theano.compile.mode
  957.      22 from theano.compile.io import (
  958.      23     In, SymbolicInput, SymbolicOutput)
  959.  
  960. C:\Users\silverstone\Theano\theano\compile\mode.py in <module>()
  961.       8 import theano
  962.       9 from theano import gof
  963. ---> 10 import theano.gof.vm
  964.      11 from theano.configparser import config
  965.      12 from theano.compile.ops import _output_guard
  966.  
  967. C:\Users\silverstone\Theano\theano\gof\vm.py in <module>()
  968.     657
  969.     658 try:
  970. --> 659     from . import lazylinker_c
  971.     660
  972.     661     class CVM(lazylinker_c.CLazyLinker, VM):
  973.  
  974. C:\Users\silverstone\Theano\theano\gof\lazylinker_c.py in <module>()
  975.     125             args = cmodule.GCC_compiler.compile_args()
  976.     126             cmodule.GCC_compiler.compile_str(dirname, code, location=loc,
  977. --> 127                                              preargs=args)
  978.     128             # Save version into the __init__.py file.
  979.     129             init_py = os.path.join(loc, '__init__.py')
  980.  
  981. C:\Users\silverstone\Theano\theano\gof\cmodule.pyc in compile_str(module_name, src_code, location, include_dirs, lib_dirs, libs, preargs, py_module, hide_symbols)
  982.    2307             # difficult to read.
  983.    2308             raise Exception('Compilation failed (return status=%s): %s' %
  984. -> 2309                             (status, compile_stderr.replace('\n', '. ')))
  985.    2310         elif config.cmodule.compilation_warning and compile_stderr:
  986.    2311             # Print errors just below the command line.
  987.  
  988. Exception: Compilation failed (return status=1): C:\Users\silverstone\AppData\Local\Theano\compiledir_Windows-7-6.1.7601-SP1-Intel64_Family_6_Model_58_Stepping_9_GenuineIntel-2.7.13-64\lazylinker_ext\mod.cpp:1:0: sorry, unimplemented: 64-bit mo.  ^include <Python.h>
  989.  
  990. In [2]:
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement