xrubysupport.cpp
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13  ***************************************************************************/
14 //---------------------------------------------------------------------------
15 #include "xrubysupport.h"
16 #include "measure.h"
17 #include <QFile>
18 #include <QDataStream>
19 #include <math.h>
20 
21 
22 #define XRUBYSUPPORT_RB ":/script/xrubysupport.rb" //in the qrc.
23 
24 XRuby::XRuby(const char *name, bool runtime, const shared_ptr<XMeasure> &measure)
25 : XAliasListNode<XRubyThread>(name, runtime),
26 m_measure(measure),
27 m_thread(shared_from_this(), &XRuby::execute) {
28  iterate_commit([=](Transaction &tr){
29  m_lsnChildCreated = tr[ *this].onChildCreated().connectWeakly(shared_from_this(),
30  &XRuby::onChildCreated, XListener::FLAG_MAIN_THREAD_CALL);
31  });
32 }
33 XRuby::~XRuby() {
34 }
35 
36 Ruby::Value
37 XRuby::rnode_create(const shared_ptr<XNode> &node) {
38  if(auto vnode = dynamic_pointer_cast<XValueNodeBase>(node)) {
39  return m_rubyClassValueNode->rubyObject(vnode);
40  }
41  else if(auto lnode = dynamic_pointer_cast<XListNodeBase>(node)) {
42  return m_rubyClassListNode->rubyObject(lnode);
43  }
44  else {
45  return m_rubyClassNode->rubyObject(node);
46  }
47 }
48 
49 Ruby::Value
50 XRuby::rnode_child(const shared_ptr<XNode> &node, Ruby::Value var) {
51  shared_ptr<XNode> child;
52 
53  if(Ruby::isConvertible<long>(var)) {
54  long idx = Ruby::convert<long>(var);
55  Snapshot shot( *node);
56  if(shot.size()) {
57  if ((idx >= 0) && (idx < (int)shot.size()))
58  child = shot.list()->at(idx);
59  }
60  if(! child ) {
61  throw (std::string)formatString("No such node idx:%ld on %s\n",
62  idx, node->getName().c_str());
63  }
64  }
65  else if(Ruby::isConvertible<const char*>(var)) {
66  const char *name = Ruby::convert<const char*>(var);
67  child = node->getChild(name);
68  if( !child ) {
69  throw (std::string)formatString("No such node name:%s on %s\n",
70  name, node->getName().c_str());
71  }
72  }
73  else
74  throw (std::string)formatString("Ill format to find node on %s\n", node->getName().c_str());
75  return rnode_create(child);
76 }
77 Ruby::Value
78 XRuby::rlistnode_create_child(const shared_ptr<XNode> &node, Ruby::Value rbtype, Ruby::Value rbname) {
79  const char *type = Ruby::convert<const char*>(rbtype);
80  const char *name = Ruby::convert<const char*>(rbname);
81 
82  shared_ptr<XNode> child;
83  Snapshot shot( *node);
84  if( shot[ *node].isRuntime() ) {
85  throw (std::string)formatString("Node %s is run-time node!\n", node->getName().c_str());
86  }
87  auto lnode = dynamic_pointer_cast<XListNodeBase>(node);
88  if( !lnode) {
89  throw (std::string)formatString("Error on %s : Not ListNode. Could not make child"
90  " name = %s, type = %s\n", node->getName().c_str(), name, type);
91  }
92  if(strlen(name))
93  child = node->getChild(name);
94 
95  if( !child) {
96  if(lnode->isThreadSafeDuringCreationByTypename()) {
97  child = lnode->createByTypename(type, name);
98  }
99  else {
100  shared_ptr<Payload::tCreateChild> x(new Payload::tCreateChild);
101  x->lnode = lnode;
102  x->type = type;
103  x->name = name;
104  Snapshot shot( *this);
105  shot.talk(shot[ *this].onChildCreated(), x);
106  XScopedLock<XCondition> lock(x->cond);
107  while(x->lnode) {
108  x->cond.wait();
109  }
110  child = x->child;
111  x->child.reset();
112  }
113  }
114  if( !child) {
115  throw (std::string)formatString(
116  "Error on %s : Could not make child"
117  " name = %s, type = %s\n", node->getName().c_str(), name, type);
118  }
119  return rnode_create(child);
120 }
121 void
122 XRuby::onChildCreated(const Snapshot &, const shared_ptr<Payload::tCreateChild> &x) {
123  x->child = x->lnode->createByTypename(x->type, x->name);
124  x->lnode.reset();
125  XScopedLock<XCondition> lock(x->cond);
126  x->cond.signal();
127 }
128 Ruby::Value
129 XRuby::rlistnode_release_child(const shared_ptr<XNode> &node, Ruby::Value rbchild) {
130  auto lnode = dynamic_pointer_cast<XListNodeBase>(node);
131  if( !lnode) {
132  throw (std::string)formatString("Error on %s : Not ListNode.", node->getName().c_str());
133  }
134  Snapshot shot( *lnode);
135  if( !shot[ *lnode].isUIEnabled()) {
136  throw (std::string)formatString("Node %s is read-only!\n", node->getName().c_str());
137  }
138  shared_ptr<XNode> child(m_rubyClassNode->unwrap(rbchild));
139  lnode->release(child);
140  return Ruby::Nil;
141 }
142 
143 Ruby::Value
144 XRuby::rnode_name(const shared_ptr<XNode> &node) {
145  return Ruby::convertToRuby((std::string)node->getName());
146 }
147 Ruby::Value
148 XRuby::rnode_count(const shared_ptr<XNode> &node) {
149  Snapshot shot( *node);
150  return Ruby::convertToRuby(shot.size());
151 }
152 Ruby::Value
153 XRuby::rnode_touch(const shared_ptr<XNode> &node) {
154  auto tnode = dynamic_pointer_cast<XTouchableNode>(node);
155  if( !node)
156  throw (std::string)formatString("Type mismatch on node %s\n", node->getName().c_str());
157  dbgPrint(QString("Ruby, Node %1, touching."));
158  tnode->iterate_commit([=](Transaction &tr){
159  if(tr[ *tnode].isUIEnabled() )
160  tr[ *tnode].touch();
161  else
162  throw (std::string)formatString("Node %s is read-only!\n", node->getName().c_str());
163  });
164  return Ruby::Nil;
165 }
166 Ruby::Value
167 XRuby::rvaluenode_set(const shared_ptr<XNode> &node, Ruby::Value var) {
168  Snapshot shot( *node);
169  auto vnode = dynamic_pointer_cast<XValueNodeBase>(node);
170  assert(vnode);
171  dbgPrint(QString("Ruby, Node %1, setting new value.").arg(node->getName()) );
172  if( !shot[ *node].isUIEnabled()) {
173  throw (std::string)formatString("Node %s is read-only!\n", node->getName().c_str());
174  }
175  XRuby::strOnNode(vnode, var);
176  dbgPrint(QString("Ruby, Node %1, new value: %2.").arg(node->getName()).arg(shot[ *vnode].to_str()) );
177  return XRuby::getValueOfNode(vnode);
178 }
179 Ruby::Value
180 XRuby::rvaluenode_load(const shared_ptr<XNode> &node, Ruby::Value var) {
181  Snapshot shot( *node);
182  auto vnode = dynamic_pointer_cast<XValueNodeBase>(node);
183  assert(vnode);
184  dbgPrint(QString("Ruby, Node %1, loading new value.").arg(node->getName()) );
185  if( shot[ *node].isRuntime()) {
186  throw (std::string)formatString("Node %s is run-time node!\n", node->getName().c_str());
187  }
188  XRuby::strOnNode(vnode, var);
189  dbgPrint(QString("Ruby, Node %1, new value: %2.").arg(node->getName()).arg(shot[ *vnode].to_str()) );
190  return XRuby::getValueOfNode(vnode);
191 }
192 Ruby::Value
193 XRuby::rvaluenode_get(const shared_ptr<XNode> &node) {
194  auto vnode = dynamic_pointer_cast<XValueNodeBase>(node);
195  assert(vnode);
196  return XRuby::getValueOfNode(vnode);
197 }
198 Ruby::Value
199 XRuby::rvaluenode_to_str(const shared_ptr<XNode> &node) {
200  auto vnode = dynamic_pointer_cast<XValueNodeBase>(node);
201  assert(vnode);
202  return Ruby::convertToRuby((std::string)( **vnode)->to_str());
203 }
204 void
205 XRuby::strOnNode(const shared_ptr<XValueNodeBase> &node, Ruby::Value value) {
206  auto dnode = dynamic_pointer_cast<XDoubleNode>(node);
207  auto inode = dynamic_pointer_cast<XIntNode>(node);
208  auto uinode = dynamic_pointer_cast<XUIntNode>(node);
209  auto lnode = dynamic_pointer_cast<XLongNode>(node);
210  auto ulnode = dynamic_pointer_cast<XULongNode>(node);
211  auto bnode = dynamic_pointer_cast<XBoolNode>(node);
212 
213  if(Ruby::isConvertible<long>(value)) {
214  long integer = Ruby::convert<long>(value);
215  if(uinode || ulnode) {
216  if(integer < 0) {
217  throw (std::string)formatString("Negative FIXNUM on %s\n", node->getName().c_str());
218  }
219  }
220  if(inode) { trans( *inode) = integer; return; }
221  if(lnode) { trans( *lnode) = integer; return; }
222  if(uinode) { trans( *uinode) = integer; return; }
223  if(ulnode) { trans( *ulnode) = integer; return; }
224  double dbl = integer;
225  if(dnode) { trans( *dnode) = dbl; return;}
226  throw (std::string)formatString("FIXNUM is not appropreate on %s\n", node->getName().c_str());
227  }
228  else if(Ruby::isConvertible<double>(value)) {
229  double dbl = Ruby::convert<double>(value);
230  if(dnode) { trans( *dnode) = dbl; return;}
231  throw (std::string)formatString("FLOAT is not appropreate on %s\n", node->getName().c_str());
232  }
233  else if(Ruby::isConvertible<const char*>(value)) {
234  try {
235  trans( *node).str(Ruby::convert<const char*>(value));
236  }
237  catch (XKameError &e) {
238  throw (std::string)formatString("Validation error %s on %s\n"
239  , (const char*)e.msg().c_str(), node->getName().c_str());
240  }
241  }
242  else if(Ruby::isConvertible<bool>(value)) {
243  bool v = Ruby::convert<bool>(value);
244  if(bnode) { trans( *bnode) = v; return;}
245  throw (std::string)formatString("TRUE is not appropreate on %s\n", node->getName().c_str());
246  }
247  else
248  throw (std::string)formatString("UNKNOWN TYPE is not appropreate on %s\n", node->getName().c_str());
249 }
250 Ruby::Value
251 XRuby::getValueOfNode(const shared_ptr<XValueNodeBase> &node) {
252  auto dnode = dynamic_pointer_cast<XDoubleNode>(node);
253  auto inode = dynamic_pointer_cast<XIntNode>(node);
254  auto uinode = dynamic_pointer_cast<XUIntNode>(node);
255  auto lnode = dynamic_pointer_cast<XLongNode>(node);
256  auto ulnode = dynamic_pointer_cast<XULongNode>(node);
257  auto bnode = dynamic_pointer_cast<XBoolNode>(node);
258  auto snode = dynamic_pointer_cast<XStringNode>(node);
259  Snapshot shot( *node);
260  if(dnode) {return Ruby::convertToRuby((double)shot[ *dnode]);}
261  if(inode) {return Ruby::convertToRuby((int)shot[ *inode]);}
262  if(uinode) {return Ruby::convertToRuby((unsigned int)shot[ *uinode]);}
263  if(lnode) {return Ruby::convertToRuby((long)shot[ *lnode]);}
264  if(ulnode) {return Ruby::convertToRuby((unsigned long)shot[ *ulnode]);}
265  if(bnode) {return Ruby::convertToRuby((bool)shot[ *bnode]);}
266  if(snode) {return Ruby::convertToRuby((const std::string&)shot[ *snode]);}
267  return Ruby::Nil;
268 }
269 
270 shared_ptr<XRubyThread>
271 XRuby::findRubyThread(const shared_ptr<XNode> &, Ruby::Value threadid)
272 {
273  long id = Ruby::convert<long>(threadid);
274  shared_ptr<XRubyThread> rubythread;
275  Snapshot shot(*this);
276  if(shot.size()) {
277  for(XNode::const_iterator it = shot.list()->begin(); it != shot.list()->end(); ++it) {
278  auto th = dynamic_pointer_cast<XRubyThread>( *it);
279  assert(th);
280  if(id == shot[ *th->threadID()])
281  rubythread = th;
282  }
283  }
284  return rubythread;
285 }
286 Ruby::Value
287 XRuby::my_rbdefout(const shared_ptr<XNode> &node, Ruby::Value str, Ruby::Value threadid) {
288  shared_ptr<XString> sstr(new XString(Ruby::convert<const char*>(str)));
289  shared_ptr<XRubyThread> rubythread(findRubyThread(node, threadid));
290  if(rubythread) {
291  Snapshot shot( *rubythread);
292  shot.talk(shot[ *rubythread].onMessageOut(), sstr);
293  dbgPrint(QString("Ruby [%1]; %2").arg(shot[ *rubythread->filename()].to_str()).arg( *sstr));
294  }
295  else {
296  dbgPrint(QString("Ruby [global]; %1").arg(*sstr));
297  }
298  return Ruby::Nil;
299 }
300 Ruby::Value
301 XRuby::my_rbdefin(const shared_ptr<XNode> &node, Ruby::Value threadid) {
302  shared_ptr<XRubyThread> rubythread(findRubyThread(node, threadid));
303  if(rubythread) {
304  XString line = rubythread->gets();
305  if(line.length())
306  return Ruby::convertToRuby((std::string)line);
307  return Ruby::Nil;
308  }
309  else {
310  throw XString("UNKNOWN Ruby thread\n");
311  }
312 }
313 Ruby::Value
314 XRuby::is_main_terminated(const shared_ptr<XNode> &) {
315  return Ruby::convertToRuby(m_thread.isTerminated());
316 }
317 
318 void *
319 XRuby::execute(const atomic<bool> &) {
320 
321  m_ruby.reset(new Ruby("KAME"));
322  shared_ptr<XRuby> xruby = dynamic_pointer_cast<XRuby>(shared_from_this());
323  m_rubyClassNode.reset(new Ruby::Class<XRuby, XNode>(xruby, "XNode"));
324  m_rubyClassNode->defineMethod<&XRuby::rnode_name>("name");
325  m_rubyClassNode->defineMethod<&XRuby::rnode_touch>("touch");
326  m_rubyClassNode->defineMethod1<&XRuby::rnode_child>("child");
327  m_rubyClassNode->defineMethod<&XRuby::rnode_count>("count");
328 
329  m_rubyClassValueNode.reset(new Ruby::Class<XRuby, XNode>(xruby, "XValueNode",
330  m_rubyClassNode->rubyClassObject()));
331  m_rubyClassValueNode->defineMethod1<&XRuby::rvaluenode_set>("internal_set");
332  m_rubyClassValueNode->defineMethod1<&XRuby::rvaluenode_load>("internal_load");
333  m_rubyClassValueNode->defineMethod<&XRuby::rvaluenode_get>("internal_get");
334 
335  m_rubyClassListNode.reset(new Ruby::Class<XRuby, XNode>(xruby, "XListNode",
336  m_rubyClassNode->rubyClassObject()));
337  m_rubyClassListNode->defineMethod2<&XRuby::rlistnode_create_child>("internal_create");
338  m_rubyClassListNode->defineMethod1<&XRuby::rlistnode_release_child>("release");
339 
340  {
341  shared_ptr<XMeasure> measure = m_measure.lock();
342  assert(measure);
343  XString name = measure->getName();
344  name[0] = toupper(name[0]);
345  Ruby::Value rbRootNode = rnode_create(measure);
346  m_ruby->defineGlobalConst(name.c_str(), rbRootNode);
347  m_ruby->defineGlobalConst("RootNode", rbRootNode);
348  }
349  {
350  Ruby::Value rbRubyThreads = rnode_create(shared_from_this());
351  m_ruby->defineGlobalConst("XRubyThreads", rbRubyThreads);
352  m_rubyClassNode->defineSingletonMethod2<&XRuby::my_rbdefout>(
353  rbRubyThreads, "my_rbdefout");
354  m_rubyClassNode->defineSingletonMethod1<&XRuby::my_rbdefin>(
355  rbRubyThreads, "my_rbdefin");
356  m_rubyClassNode->defineSingletonMethod<&XRuby::is_main_terminated>(
357  rbRubyThreads, "is_main_terminated");
358  }
359 
360  {
361  QFile scriptfile(XRUBYSUPPORT_RB);
362  if( !scriptfile.open(QIODevice::ReadOnly | QIODevice::Text)) {
363  gErrPrint("No KAME ruby support file installed.");
364  return NULL;
365  }
366  fprintf(stderr, "Loading ruby scripting monitor.\n");
367  char data[65536];
368  QDataStream( &scriptfile).readRawData(data, sizeof(data));
369 
370  int state = m_ruby->evalProtect(data);
371  if(state) {
372  fprintf(stderr, "Ruby, exception(s) occurred.\n");
373  m_ruby->printErrorInfo();
374  }
375  }
376 
377  fprintf(stderr, "ruby fin");
378  m_ruby.reset();
379  fprintf(stderr, "ished\n");
380  return NULL;
381 }

Generated for KAME4 by  doxygen 1.8.3