Salome HOME
PR: mergefrom_BSEC_br1_14Mar04
[modules/kernel.git] / src / SALOMEGUI / QAD_PyInterp.cxx
1 //  SALOME SALOMEGUI : implementation of desktop and GUI kernel
2 //
3 //  Copyright (C) 2003  OPEN CASCADE, EADS/CCR, LIP6, CEA/DEN,
4 //  CEDRAT, EDF R&D, LEG, PRINCIPIA R&D, BUREAU VERITAS 
5 // 
6 //  This library is free software; you can redistribute it and/or 
7 //  modify it under the terms of the GNU Lesser General Public 
8 //  License as published by the Free Software Foundation; either 
9 //  version 2.1 of the License. 
10 // 
11 //  This library is distributed in the hope that it will be useful, 
12 //  but WITHOUT ANY WARRANTY; without even the implied warranty of 
13 //  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU 
14 //  Lesser General Public License for more details. 
15 // 
16 //  You should have received a copy of the GNU Lesser General Public 
17 //  License along with this library; if not, write to the Free Software 
18 //  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA 
19 // 
20 //  See http://www.opencascade.org/SALOME/ or email : webmaster.salome@opencascade.org 
21 //
22 //
23 //
24 //  File   : QAD_PyInterp.cxx
25 //  Author : Nicolas REJNERI
26 //  Module : SALOME
27 //  $Header$
28
29 #include "QAD_PyInterp.h"
30 #include "utilities.h"
31 using namespace std;
32
33 /*!
34  * constructor : multi Python interpreter, one per SALOME study.
35  * calls initialize method defined in base class, which calls virtual methods
36  * initstate & initcontext redefined here.
37  */
38 QAD_PyInterp::QAD_PyInterp(): PyInterp_base()
39 {
40   initialize();
41 }
42
43 QAD_PyInterp::~QAD_PyInterp()
44 {
45 }
46  
47 /*!
48  * EDF-CCAR
49  * When SALOME uses multi Python interpreter feature,
50  * Every study has its own interpreter and thread state (_tstate = Py_NewInterpreter())
51  * This is fine because every study has its own modules (sys.modules) stdout and stderr
52  * BUT some Python modules must be imported only once. In multi interpreter context Python
53  * modules (*.py) are imported several times.
54  * The pyqt module must be imported only once because it registers classes in a C module.
55  * It's quite the same with omniorb modules (internals and generated with omniidl)
56  * This problem is handled with "shared modules" defined in salome_shared_modules.py
57  * These "shared modules" are imported only once and only copied in all the other interpreters
58  * BUT it's not the only problem. Every interpreter has its own __builtin__ module. That's fine
59  * but if we have copied some modules and imported others problems may arise with operations that
60  * are not allowed in restricted execution environment. So we must impose that all interpreters
61  * have identical __builtin__ module.
62  * That's all, for the moment ...
63  */
64
65 void QAD_PyInterp::initState()
66 {
67   MESSAGE("QAD_PyInterp::initState");
68   salomeAcquireLock();           //acquire python global lock (one for all interpreters)
69   _tstate = Py_NewInterpreter(); //create an interpreter and save current state
70   SCRUTE(_tstate);
71   SCRUTE(PyInterp_base::_argc);
72   SCRUTE(PyInterp_base::_argv[0]);
73   PySys_SetArgv(PyInterp_base::_argc,PyInterp_base::_argv);    // initialize sys.argv
74
75   if(builtinmodule == NULL)return;
76   /*
77    * If builtinmodule has been initialized all the sub interpreters
78    * will have the same __builtin__ module
79    */
80   PyObject *m=PyImport_GetModuleDict();
81   PyDict_SetItemString(m, "__builtin__", builtinmodule);
82   SCRUTE(builtinmodule->ob_refcnt);                            // builtinmodule reference counter
83   _tstate->interp->builtins = PyModule_GetDict(builtinmodule);
84   Py_INCREF(_tstate->interp->builtins);
85 }
86
87 void QAD_PyInterp::initContext()
88 {
89   MESSAGE("QAD_PyInterp::initContext");
90   PyObject *m;
91   m=PyImport_AddModule("__main__");  // interpreter main module (module context)
92   if(m == NULL)
93     {
94       MESSAGE("problem...");
95       PyErr_Print();
96       ASSERT(0);
97       salomeReleaseLock(); 
98       return;
99     }  
100   _g = PyModule_GetDict(m);          // get interpreter dictionnary context
101   SCRUTE(_g);
102
103   if(builtinmodule)
104     {
105       PyDict_SetItemString(_g, "__builtins__", builtinmodule); // assign singleton __builtin__ module
106     }
107 // Debut modif CCAR
108   /*
109    * Import special module to change the import mechanism
110    */
111   m =PyImport_ImportModule("import_hook");
112   if(m == NULL){
113       MESSAGE("initContext: problem with import_hook import");
114       PyErr_Print();
115       PyErr_Clear();
116       ASSERT(0);
117   }
118   /*
119    * Call init_shared_modules to initialize the shared import mechanism for modules 
120    * that must not be imported twice
121    */
122   if(m != NULL){
123       m= PyObject_CallMethod(m,
124                          "init_shared_modules","O",salome_shared_modules_module);
125       if (m == NULL){
126           MESSAGE("initContext: problem with init_shared_modules call");
127           PyErr_Print();
128           PyErr_Clear();
129           ASSERT(0);
130       }
131   }
132 // Fin   modif CCAR
133
134 }