Merged 15170:15635 from trunk (no conflicts or even merges)
[blender.git] / source / gameengine / GameLogic / SCA_RandomSensor.cpp
1 /**
2  * Generate random pulses
3  *
4  * $Id$
5  *
6  * ***** BEGIN GPL LICENSE BLOCK *****
7  *
8  * This program is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU General Public License
10  * as published by the Free Software Foundation; either version 2
11  * of the License, or (at your option) any later version.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program; if not, write to the Free Software Foundation,
20  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
21  *
22  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
23  * All rights reserved.
24  *
25  * The Original Code is: all of this file.
26  *
27  * Contributor(s): none yet.
28  *
29  * ***** END GPL LICENSE BLOCK *****
30  */
31
32 #include "SCA_RandomSensor.h"
33 #include "SCA_EventManager.h"
34 #include "SCA_RandomEventManager.h"
35 #include "SCA_LogicManager.h"
36 #include "ConstExpr.h"
37 #include <iostream>
38
39 #ifdef HAVE_CONFIG_H
40 #include <config.h>
41 #endif
42
43 /* ------------------------------------------------------------------------- */
44 /* Native functions                                                          */
45 /* ------------------------------------------------------------------------- */
46
47 SCA_RandomSensor::SCA_RandomSensor(SCA_EventManager* eventmgr, 
48                                  SCA_IObject* gameobj, 
49                                  int startseed,
50                                  PyTypeObject* T)
51     : SCA_ISensor(gameobj,eventmgr, T)
52 {
53         // m_basegenerator is never deleted => memory leak
54         m_basegenerator = new SCA_RandomNumberGenerator(startseed);
55         Init();
56 }
57
58
59
60 SCA_RandomSensor::~SCA_RandomSensor() 
61 {
62     /* Nothing to be done here. */
63 }
64
65 void SCA_RandomSensor::Init()
66 {
67     m_iteration  = 0;
68         m_interval = 0;
69         m_lastdraw   = false;
70     m_currentDraw = m_basegenerator->Draw();
71 }
72
73
74 CValue* SCA_RandomSensor::GetReplica()
75 {
76         CValue* replica = new SCA_RandomSensor(*this);
77         // replication copies m_basegenerator pointer => share same generator
78         // this will copy properties and so on...
79         CValue::AddDataToReplica(replica);
80
81         return replica;
82 }
83
84
85
86 bool SCA_RandomSensor::IsPositiveTrigger()
87
88         return (m_invert !=m_lastdraw);
89 }
90
91
92 bool SCA_RandomSensor::Evaluate(CValue* event)
93 {
94     /* Random generator is the generator from Line 25 of Table 1 in          */
95     /* [KNUTH 1981, The Art of Computer Programming Vol. 2                   */
96     /* (2nd Ed.), pp102]                                                     */
97     /* It's a very simple max. length sequence generator. We can             */
98     /* draw 32 bool values before having to generate the next                */
99     /* sequence value. There are some theorems that will tell you            */
100     /* this is a reasonable way of generating bools. Check Knuth.            */
101     /* Furthermore, we only draw each <delay>-eth frame.                     */
102
103         bool evaluateResult = false;
104
105         if (++m_interval > m_pulse_frequency) {
106             bool drawResult = false;
107                 m_interval = 0;
108                 if (m_iteration > 31) {
109                         m_currentDraw = m_basegenerator->Draw();
110                         drawResult = (m_currentDraw & 0x1) == 0;
111                         m_iteration = 1;
112                 } else {
113                         drawResult = ((m_currentDraw >> m_iteration) & 0x1) == 0;
114                         m_iteration++;
115                 }
116                 evaluateResult = drawResult != m_lastdraw;
117                 m_lastdraw = drawResult;
118         }
119     
120     /* now pass this result to some controller */
121         return evaluateResult;
122 }
123
124 /* ------------------------------------------------------------------------- */
125 /* Python functions                                                          */
126 /* ------------------------------------------------------------------------- */
127
128 /* Integration hooks ------------------------------------------------------- */
129 PyTypeObject SCA_RandomSensor::Type = {
130         PyObject_HEAD_INIT(&PyType_Type)
131         0,
132         "SCA_RandomSensor",
133         sizeof(SCA_RandomSensor),
134         0,
135         PyDestructor,
136         0,
137         __getattr,
138         __setattr,
139         0, //&MyPyCompare,
140         __repr,
141         0, //&cvalue_as_number,
142         0,
143         0,
144         0,
145         0
146 };
147
148 PyParentObject SCA_RandomSensor::Parents[] = {
149         &SCA_RandomSensor::Type,
150         &SCA_ISensor::Type,
151         &SCA_ILogicBrick::Type,
152         &CValue::Type,
153         NULL
154 };
155
156 PyMethodDef SCA_RandomSensor::Methods[] = {
157         {"setSeed",     (PyCFunction) SCA_RandomSensor::sPySetSeed, METH_VARARGS, SetSeed_doc},
158         {"getSeed",     (PyCFunction) SCA_RandomSensor::sPyGetSeed, METH_VARARGS, GetSeed_doc},
159         {"getLastDraw", (PyCFunction) SCA_RandomSensor::sPyGetLastDraw, METH_VARARGS, GetLastDraw_doc},
160         {NULL,NULL} //Sentinel
161 };
162
163 PyObject* SCA_RandomSensor::_getattr(const STR_String& attr) {
164         _getattr_up(SCA_ISensor);
165 }
166
167 /* 1. setSeed                                                            */
168 char SCA_RandomSensor::SetSeed_doc[] = 
169 "setSeed(seed)\n"
170 "\t- seed: integer\n"
171 "\tSet the initial seed of the generator. Equal seeds produce\n"
172 "\tequal series. If the seed is 0, the generator will produce\n"
173 "\tthe same value on every call.\n";
174 PyObject* SCA_RandomSensor::PySetSeed(PyObject* self, PyObject* args, PyObject* kwds) {
175         long seedArg;
176         if(!PyArg_ParseTuple(args, "i", &seedArg)) {
177                 return NULL;
178         }
179         
180         m_basegenerator->SetSeed(seedArg);
181         
182         Py_Return;
183 }
184
185 /* 2. getSeed                                                            */
186 char SCA_RandomSensor::GetSeed_doc[] = 
187 "getSeed()\n"
188 "\tReturns the initial seed of the generator. Equal seeds produce\n"
189 "\tequal series.\n";
190 PyObject* SCA_RandomSensor::PyGetSeed(PyObject* self, PyObject* args, PyObject* kwds) {
191         return PyInt_FromLong(m_basegenerator->GetSeed());
192 }
193
194 /* 3. getLastDraw                                                            */
195 char SCA_RandomSensor::GetLastDraw_doc[] = 
196 "getLastDraw()\n"
197 "\tReturn the last value that was drawn.\n";
198 PyObject* SCA_RandomSensor::PyGetLastDraw(PyObject* self, PyObject* args, PyObject* kwds) {
199         return PyInt_FromLong(m_lastdraw);
200 }
201
202 /* eof */