TinyJS on mbed. TinyJS is very simple JavaScript engine.

Dependencies:   mbed

TinyJS on mbed

what's this ?

TinyJS is an extremely simple (under than 2000 lines) JavaScript interpreter engine.
I ported on mbed. but it restrict any features.
TinyJS project is https://github.com/gfwilliams/tiny-js

TinyJSは2000行以下で書かれた非常に小さいJavaScriptインタプリタエンジンです。
これをmbedに移植してみました。(ただし、いろいろ制限があります)
本家はこちら。 https://github.com/gfwilliams/tiny-js

how to use

You must use on serial terminal application by mbed serial USB.
baud is 57600bps
USBシリアルとして、ターミナルソフトを接続するとコンソールが表示されます。
ボーレートは57600bpsになってます。

functions

functions for mbed.
mbed用関数

  • mbed.DigitalIn(pinName, mode)
  • mbed.DigitalOut(pinName, val)
  • mbed.AnalogIn(pinName)
  • mbed.AnalogOut(pinName, val)
  • mbed.InterruptIn(pinName, edge, mode, callback)
  • mbed.TimerStart()
  • mbed.TimerStop()
  • mbed.TimerReset()
  • mbed.TimerRead()
  • mbed.Timeout(callback, t)
  • mbed.wait(s)
  • mbed.memfree()

sample JavaScript codes

DigitalOut

mbed.DigitalOut('LED1', 1);
mbed.DigitalOut('LED2', 0);
mbed.DigitalOut('LED3', 1);
mbed.DigitalOut('LED4', 0);

LED1 = On, LED2=Off, LED3=On, ED4=Off
LED1 = 点灯、LED2=消灯、LED3=点灯、LED4=消灯

DigitalIn

print(mbed.DigitalIn('p5', 'PullUp'));

p5 is pull up, read, and print on console.
p5をPullUpして読みプリントする。

AnalogOut

mbed.AnalogOut('p18', 0.8);

p18 is analog output, value is 0.8.
p18を 値0.8でアナログ出力する。

AnalogIn

print(mbed.AnalogIn('p20'));

p20 is read analog voltage, and print on console.
p20をアナログ入力しプリントする。

InterruptIn

var led1 = 0;
mbed.InterruptIn('p5', 'fall', 'PullUp', function() {led1 = !led1; mbed.DigitalOut('LED1', led1);});

Interrupt on p5, and ON/OFF does LED1.
p5で割り込んでLED1をON/OFFする。

Timeout and wait sample code

mbed.Timeout(function() {mbed.DigitalOut('LED1', 1);mbed.wait(3);mbed.DigitalOut('LED1', 0);}, 4);

LED1=on when wait for 4 seconds. and LED1=off for 3 seconds later.
LED1を4秒待って点灯して3秒後に消灯する。

memfree

print(mbed.memfree());

This prints the number of bytes of the remainder memory on mbed where TinyJS is usable.
これはTinyJSが使えるmbed上での残りメモリのバイト数をプリントアウトする。

LED Blinker by Timeout

blinker = function() {var led = 0; mbed.Timeout(function() {led = !led; mbed.DigitalOut('LED1', led);blinker();}, 0.5);};
blinker();

LED Blinker by Timeout.
Timeoutを使ったLチカ。

restrictions

  • There is very little available memory. (Less than 9kbytes on LPC1768)
  • Registration of InterruptIn is 4 limit.
  • The loop to 8,192 times.
  • The built-in functions (general JavaScript functions) that TinyJS prepares for for securing of memory is not included.

more, more, more ....

制限事項

  • 利用できるメモリは非常に少ない。(LPC1768で9kbytes以下)
  • InterruptInで登録できる割り込みは4つまで。4つを超えると1つめから順番に削除される。
  • ループは8192回まで。
  • メモリ確保のためTinyJSが用意している組み込み関数(一般的なJavaScript関数)は含まれない。

他、多数....

sample movies

http://www.youtube.com/watch?v=ARp0DK70JGM
http://www.youtube.com/watch?v=UOZQ4eEC4xA

Committer:
ohneta
Date:
Mon Jan 20 00:07:35 2014 +0000
Revision:
8:819934a27c2d
Parent:
0:aae260bdcdd9
update InterruptIn, Timer, Timeout functons

Who changed what in which revision?

UserRevisionLine numberNew contents of line
ohneta 0:aae260bdcdd9 1 /*
ohneta 0:aae260bdcdd9 2 * TinyJS
ohneta 0:aae260bdcdd9 3 *
ohneta 0:aae260bdcdd9 4 * A single-file Javascript-alike engine
ohneta 0:aae260bdcdd9 5 *
ohneta 0:aae260bdcdd9 6 * - Useful language functions
ohneta 0:aae260bdcdd9 7 *
ohneta 0:aae260bdcdd9 8 * Authored By Gordon Williams <gw@pur3.co.uk>
ohneta 0:aae260bdcdd9 9 *
ohneta 0:aae260bdcdd9 10 * Copyright (C) 2009 Pur3 Ltd
ohneta 0:aae260bdcdd9 11 *
ohneta 0:aae260bdcdd9 12 * Permission is hereby granted, free of charge, to any person obtaining a copy of
ohneta 0:aae260bdcdd9 13 * this software and associated documentation files (the "Software"), to deal in
ohneta 0:aae260bdcdd9 14 * the Software without restriction, including without limitation the rights to
ohneta 0:aae260bdcdd9 15 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
ohneta 0:aae260bdcdd9 16 * of the Software, and to permit persons to whom the Software is furnished to do
ohneta 0:aae260bdcdd9 17 * so, subject to the following conditions:
ohneta 0:aae260bdcdd9 18
ohneta 0:aae260bdcdd9 19 * The above copyright notice and this permission notice shall be included in all
ohneta 0:aae260bdcdd9 20 * copies or substantial portions of the Software.
ohneta 0:aae260bdcdd9 21
ohneta 0:aae260bdcdd9 22 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
ohneta 0:aae260bdcdd9 23 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
ohneta 0:aae260bdcdd9 24 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
ohneta 0:aae260bdcdd9 25 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
ohneta 0:aae260bdcdd9 26 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
ohneta 0:aae260bdcdd9 27 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
ohneta 0:aae260bdcdd9 28 * SOFTWARE.
ohneta 0:aae260bdcdd9 29 */
ohneta 0:aae260bdcdd9 30
ohneta 0:aae260bdcdd9 31 #include "TinyJS_Functions.h"
ohneta 0:aae260bdcdd9 32 #include <math.h>
ohneta 0:aae260bdcdd9 33 #include <cstdlib>
ohneta 0:aae260bdcdd9 34 #include <sstream>
ohneta 0:aae260bdcdd9 35
ohneta 0:aae260bdcdd9 36 using namespace std;
ohneta 0:aae260bdcdd9 37 // ----------------------------------------------- Actual Functions
ohneta 0:aae260bdcdd9 38 void scTrace(CScriptVar *c, void *userdata) {
ohneta 0:aae260bdcdd9 39 CTinyJS *js = (CTinyJS*)userdata;
ohneta 0:aae260bdcdd9 40 js->root->trace();
ohneta 0:aae260bdcdd9 41 }
ohneta 0:aae260bdcdd9 42
ohneta 0:aae260bdcdd9 43 void scObjectDump(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 44 c->getParameter("this")->trace("> ");
ohneta 0:aae260bdcdd9 45 }
ohneta 0:aae260bdcdd9 46
ohneta 0:aae260bdcdd9 47 void scObjectClone(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 48 CScriptVar *obj = c->getParameter("this");
ohneta 0:aae260bdcdd9 49 c->getReturnVar()->copyValue(obj);
ohneta 0:aae260bdcdd9 50 }
ohneta 0:aae260bdcdd9 51
ohneta 0:aae260bdcdd9 52 void scMathRand(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 53 c->getReturnVar()->setDouble((double)rand()/RAND_MAX);
ohneta 0:aae260bdcdd9 54 }
ohneta 0:aae260bdcdd9 55
ohneta 0:aae260bdcdd9 56 void scMathRandInt(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 57 int min = c->getParameter("min")->getInt();
ohneta 0:aae260bdcdd9 58 int max = c->getParameter("max")->getInt();
ohneta 0:aae260bdcdd9 59 int val = min + (int)(rand()%(1+max-min));
ohneta 0:aae260bdcdd9 60 c->getReturnVar()->setInt(val);
ohneta 0:aae260bdcdd9 61 }
ohneta 0:aae260bdcdd9 62
ohneta 0:aae260bdcdd9 63 void scCharToInt(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 64 string str = c->getParameter("ch")->getString();;
ohneta 0:aae260bdcdd9 65 int val = 0;
ohneta 0:aae260bdcdd9 66 if (str.length()>0)
ohneta 0:aae260bdcdd9 67 val = (int)str.c_str()[0];
ohneta 0:aae260bdcdd9 68 c->getReturnVar()->setInt(val);
ohneta 0:aae260bdcdd9 69 }
ohneta 0:aae260bdcdd9 70
ohneta 0:aae260bdcdd9 71 void scStringIndexOf(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 72 string str = c->getParameter("this")->getString();
ohneta 0:aae260bdcdd9 73 string search = c->getParameter("search")->getString();
ohneta 0:aae260bdcdd9 74 size_t p = str.find(search);
ohneta 0:aae260bdcdd9 75 int val = (p==string::npos) ? -1 : p;
ohneta 0:aae260bdcdd9 76 c->getReturnVar()->setInt(val);
ohneta 0:aae260bdcdd9 77 }
ohneta 0:aae260bdcdd9 78
ohneta 0:aae260bdcdd9 79 void scStringSubstring(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 80 string str = c->getParameter("this")->getString();
ohneta 0:aae260bdcdd9 81 int lo = c->getParameter("lo")->getInt();
ohneta 0:aae260bdcdd9 82 int hi = c->getParameter("hi")->getInt();
ohneta 0:aae260bdcdd9 83
ohneta 0:aae260bdcdd9 84 int l = hi-lo;
ohneta 0:aae260bdcdd9 85 if (l>0 && lo>=0 && lo+l<=(int)str.length())
ohneta 0:aae260bdcdd9 86 c->getReturnVar()->setString(str.substr(lo, l));
ohneta 0:aae260bdcdd9 87 else
ohneta 0:aae260bdcdd9 88 c->getReturnVar()->setString("");
ohneta 0:aae260bdcdd9 89 }
ohneta 0:aae260bdcdd9 90
ohneta 0:aae260bdcdd9 91 void scStringCharAt(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 92 string str = c->getParameter("this")->getString();
ohneta 0:aae260bdcdd9 93 int p = c->getParameter("pos")->getInt();
ohneta 0:aae260bdcdd9 94 if (p>=0 && p<(int)str.length())
ohneta 0:aae260bdcdd9 95 c->getReturnVar()->setString(str.substr(p, 1));
ohneta 0:aae260bdcdd9 96 else
ohneta 0:aae260bdcdd9 97 c->getReturnVar()->setString("");
ohneta 0:aae260bdcdd9 98 }
ohneta 0:aae260bdcdd9 99
ohneta 0:aae260bdcdd9 100 void scStringCharCodeAt(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 101 string str = c->getParameter("this")->getString();
ohneta 0:aae260bdcdd9 102 int p = c->getParameter("pos")->getInt();
ohneta 0:aae260bdcdd9 103 if (p>=0 && p<(int)str.length())
ohneta 0:aae260bdcdd9 104 c->getReturnVar()->setInt(str.at(p));
ohneta 0:aae260bdcdd9 105 else
ohneta 0:aae260bdcdd9 106 c->getReturnVar()->setInt(0);
ohneta 0:aae260bdcdd9 107 }
ohneta 0:aae260bdcdd9 108
ohneta 0:aae260bdcdd9 109 void scStringSplit(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 110 string str = c->getParameter("this")->getString();
ohneta 0:aae260bdcdd9 111 string sep = c->getParameter("separator")->getString();
ohneta 0:aae260bdcdd9 112 CScriptVar *result = c->getReturnVar();
ohneta 0:aae260bdcdd9 113 result->setArray();
ohneta 0:aae260bdcdd9 114 int length = 0;
ohneta 0:aae260bdcdd9 115
ohneta 0:aae260bdcdd9 116 size_t pos = str.find(sep);
ohneta 0:aae260bdcdd9 117 while (pos != string::npos) {
ohneta 0:aae260bdcdd9 118 result->setArrayIndex(length++, new CScriptVar(str.substr(0,pos)));
ohneta 0:aae260bdcdd9 119 str = str.substr(pos+1);
ohneta 0:aae260bdcdd9 120 pos = str.find(sep);
ohneta 0:aae260bdcdd9 121 }
ohneta 0:aae260bdcdd9 122
ohneta 0:aae260bdcdd9 123 if (str.size()>0)
ohneta 0:aae260bdcdd9 124 result->setArrayIndex(length++, new CScriptVar(str));
ohneta 0:aae260bdcdd9 125 }
ohneta 0:aae260bdcdd9 126
ohneta 0:aae260bdcdd9 127 void scStringFromCharCode(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 128 char str[2];
ohneta 0:aae260bdcdd9 129 str[0] = c->getParameter("char")->getInt();
ohneta 0:aae260bdcdd9 130 str[1] = 0;
ohneta 0:aae260bdcdd9 131 c->getReturnVar()->setString(str);
ohneta 0:aae260bdcdd9 132 }
ohneta 0:aae260bdcdd9 133
ohneta 0:aae260bdcdd9 134 void scIntegerParseInt(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 135 string str = c->getParameter("str")->getString();
ohneta 0:aae260bdcdd9 136 int val = strtol(str.c_str(),0,0);
ohneta 0:aae260bdcdd9 137 c->getReturnVar()->setInt(val);
ohneta 0:aae260bdcdd9 138 }
ohneta 0:aae260bdcdd9 139
ohneta 0:aae260bdcdd9 140 void scIntegerValueOf(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 141 string str = c->getParameter("str")->getString();
ohneta 0:aae260bdcdd9 142
ohneta 0:aae260bdcdd9 143 int val = 0;
ohneta 0:aae260bdcdd9 144 if (str.length()==1)
ohneta 0:aae260bdcdd9 145 val = str[0];
ohneta 0:aae260bdcdd9 146 c->getReturnVar()->setInt(val);
ohneta 0:aae260bdcdd9 147 }
ohneta 0:aae260bdcdd9 148
ohneta 0:aae260bdcdd9 149 void scJSONStringify(CScriptVar *c, void *) {
ohneta 0:aae260bdcdd9 150 #ifndef MBED
ohneta 0:aae260bdcdd9 151 std::ostringstream result;
ohneta 0:aae260bdcdd9 152 c->getParameter("obj")->getJSON(result);
ohneta 0:aae260bdcdd9 153 c->getReturnVar()->setString(result.str());
ohneta 0:aae260bdcdd9 154 #else
ohneta 0:aae260bdcdd9 155 std::string result;
ohneta 0:aae260bdcdd9 156 c->getParameter("obj")->getJSON(result);
ohneta 0:aae260bdcdd9 157 c->getReturnVar()->setString(result);
ohneta 0:aae260bdcdd9 158 #endif
ohneta 0:aae260bdcdd9 159 }
ohneta 0:aae260bdcdd9 160
ohneta 0:aae260bdcdd9 161 void scExec(CScriptVar *c, void *data) {
ohneta 0:aae260bdcdd9 162 CTinyJS *tinyJS = (CTinyJS *)data;
ohneta 0:aae260bdcdd9 163 std::string str = c->getParameter("jsCode")->getString();
ohneta 0:aae260bdcdd9 164 tinyJS->execute(str);
ohneta 0:aae260bdcdd9 165 }
ohneta 0:aae260bdcdd9 166
ohneta 0:aae260bdcdd9 167 void scEval(CScriptVar *c, void *data) {
ohneta 0:aae260bdcdd9 168 CTinyJS *tinyJS = (CTinyJS *)data;
ohneta 0:aae260bdcdd9 169 std::string str = c->getParameter("jsCode")->getString();
ohneta 0:aae260bdcdd9 170 c->setReturnVar(tinyJS->evaluateComplex(str).var);
ohneta 0:aae260bdcdd9 171 }
ohneta 0:aae260bdcdd9 172
ohneta 0:aae260bdcdd9 173 void scArrayContains(CScriptVar *c, void *data) {
ohneta 0:aae260bdcdd9 174 CScriptVar *obj = c->getParameter("obj");
ohneta 0:aae260bdcdd9 175 CScriptVarLink *v = c->getParameter("this")->firstChild;
ohneta 0:aae260bdcdd9 176
ohneta 0:aae260bdcdd9 177 bool contains = false;
ohneta 0:aae260bdcdd9 178 while (v) {
ohneta 0:aae260bdcdd9 179 if (v->var->equals(obj)) {
ohneta 0:aae260bdcdd9 180 contains = true;
ohneta 0:aae260bdcdd9 181 break;
ohneta 0:aae260bdcdd9 182 }
ohneta 0:aae260bdcdd9 183 v = v->nextSibling;
ohneta 0:aae260bdcdd9 184 }
ohneta 0:aae260bdcdd9 185
ohneta 0:aae260bdcdd9 186 c->getReturnVar()->setInt(contains);
ohneta 0:aae260bdcdd9 187 }
ohneta 0:aae260bdcdd9 188
ohneta 0:aae260bdcdd9 189 void scArrayRemove(CScriptVar *c, void *data) {
ohneta 0:aae260bdcdd9 190 CScriptVar *obj = c->getParameter("obj");
ohneta 0:aae260bdcdd9 191 vector<int> removedIndices;
ohneta 0:aae260bdcdd9 192 CScriptVarLink *v;
ohneta 0:aae260bdcdd9 193 // remove
ohneta 0:aae260bdcdd9 194 v = c->getParameter("this")->firstChild;
ohneta 0:aae260bdcdd9 195 while (v) {
ohneta 0:aae260bdcdd9 196 if (v->var->equals(obj)) {
ohneta 0:aae260bdcdd9 197 removedIndices.push_back(v->getIntName());
ohneta 0:aae260bdcdd9 198 }
ohneta 0:aae260bdcdd9 199 v = v->nextSibling;
ohneta 0:aae260bdcdd9 200 }
ohneta 0:aae260bdcdd9 201 // renumber
ohneta 0:aae260bdcdd9 202 v = c->getParameter("this")->firstChild;
ohneta 0:aae260bdcdd9 203 while (v) {
ohneta 0:aae260bdcdd9 204 int n = v->getIntName();
ohneta 0:aae260bdcdd9 205 int newn = n;
ohneta 0:aae260bdcdd9 206 for (size_t i=0;i<removedIndices.size();i++)
ohneta 0:aae260bdcdd9 207 if (n>=removedIndices[i])
ohneta 0:aae260bdcdd9 208 newn--;
ohneta 0:aae260bdcdd9 209 if (newn!=n)
ohneta 0:aae260bdcdd9 210 v->setIntName(newn);
ohneta 0:aae260bdcdd9 211 v = v->nextSibling;
ohneta 0:aae260bdcdd9 212 }
ohneta 0:aae260bdcdd9 213 }
ohneta 0:aae260bdcdd9 214
ohneta 0:aae260bdcdd9 215 void scArrayJoin(CScriptVar *c, void *data) {
ohneta 0:aae260bdcdd9 216 string sep = c->getParameter("separator")->getString();
ohneta 0:aae260bdcdd9 217 CScriptVar *arr = c->getParameter("this");
ohneta 0:aae260bdcdd9 218
ohneta 0:aae260bdcdd9 219 ostringstream sstr;
ohneta 0:aae260bdcdd9 220 int l = arr->getArrayLength();
ohneta 0:aae260bdcdd9 221 for (int i=0;i<l;i++) {
ohneta 0:aae260bdcdd9 222 if (i>0) sstr << sep;
ohneta 0:aae260bdcdd9 223 sstr << arr->getArrayIndex(i)->getString();
ohneta 0:aae260bdcdd9 224 }
ohneta 0:aae260bdcdd9 225
ohneta 0:aae260bdcdd9 226 c->getReturnVar()->setString(sstr.str());
ohneta 0:aae260bdcdd9 227 }
ohneta 0:aae260bdcdd9 228
ohneta 0:aae260bdcdd9 229 // ----------------------------------------------- Register Functions
ohneta 0:aae260bdcdd9 230 void registerFunctions(CTinyJS *tinyJS) {
ohneta 0:aae260bdcdd9 231 tinyJS->addNative("function exec(jsCode)", scExec, tinyJS); // execute the given code
ohneta 0:aae260bdcdd9 232 tinyJS->addNative("function eval(jsCode)", scEval, tinyJS); // execute the given string (an expression) and return the result
ohneta 0:aae260bdcdd9 233 tinyJS->addNative("function trace()", scTrace, tinyJS);
ohneta 0:aae260bdcdd9 234 tinyJS->addNative("function Object.dump()", scObjectDump, 0);
ohneta 0:aae260bdcdd9 235 tinyJS->addNative("function Object.clone()", scObjectClone, 0);
ohneta 0:aae260bdcdd9 236 tinyJS->addNative("function Math.rand()", scMathRand, 0);
ohneta 0:aae260bdcdd9 237 tinyJS->addNative("function Math.randInt(min, max)", scMathRandInt, 0);
ohneta 0:aae260bdcdd9 238 tinyJS->addNative("function charToInt(ch)", scCharToInt, 0); // convert a character to an int - get its value
ohneta 0:aae260bdcdd9 239 tinyJS->addNative("function String.indexOf(search)", scStringIndexOf, 0); // find the position of a string in a string, -1 if not
ohneta 0:aae260bdcdd9 240 tinyJS->addNative("function String.substring(lo,hi)", scStringSubstring, 0);
ohneta 0:aae260bdcdd9 241 tinyJS->addNative("function String.charAt(pos)", scStringCharAt, 0);
ohneta 0:aae260bdcdd9 242 tinyJS->addNative("function String.charCodeAt(pos)", scStringCharCodeAt, 0);
ohneta 0:aae260bdcdd9 243 tinyJS->addNative("function String.fromCharCode(char)", scStringFromCharCode, 0);
ohneta 0:aae260bdcdd9 244 tinyJS->addNative("function String.split(separator)", scStringSplit, 0);
ohneta 0:aae260bdcdd9 245 tinyJS->addNative("function Integer.parseInt(str)", scIntegerParseInt, 0); // string to int
ohneta 0:aae260bdcdd9 246 tinyJS->addNative("function Integer.valueOf(str)", scIntegerValueOf, 0); // value of a single character
ohneta 0:aae260bdcdd9 247 tinyJS->addNative("function JSON.stringify(obj, replacer)", scJSONStringify, 0); // convert to JSON. replacer is ignored at the moment
ohneta 0:aae260bdcdd9 248 // JSON.parse is left out as you can (unsafely!) use eval instead
ohneta 0:aae260bdcdd9 249 tinyJS->addNative("function Array.contains(obj)", scArrayContains, 0);
ohneta 0:aae260bdcdd9 250 tinyJS->addNative("function Array.remove(obj)", scArrayRemove, 0);
ohneta 0:aae260bdcdd9 251 tinyJS->addNative("function Array.join(separator)", scArrayJoin, 0);
ohneta 0:aae260bdcdd9 252 }
ohneta 0:aae260bdcdd9 253
ohneta 0:aae260bdcdd9 254