From ce0a8d7beffccbaeb6b603a96b7729278c4c9e75 Mon Sep 17 00:00:00 2001 From: Sean Dague Date: Mon, 8 Sep 2008 20:34:45 +0000 Subject: changes to Test directory structure per opensim-dev conversation --- .../Shared/CodeTools/Tests/CSCodeGeneratorTest.cs | 1605 ++++++++++++++++++++ .../Shared/CodeTools/Tests/CompilerTest.cs | 153 ++ 2 files changed, 1758 insertions(+) create mode 100644 OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CSCodeGeneratorTest.cs create mode 100644 OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CompilerTest.cs (limited to 'OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests') diff --git a/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CSCodeGeneratorTest.cs b/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CSCodeGeneratorTest.cs new file mode 100644 index 0000000..fefcada --- /dev/null +++ b/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CSCodeGeneratorTest.cs @@ -0,0 +1,1605 @@ +/* + * Copyright (c) Contributors, http://opensimulator.org/ + * See CONTRIBUTORS.TXT for a full list of copyright holders. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * * Neither the name of the OpenSim Project nor the + * names of its contributors may be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +using System.Collections.Generic; +using System.Text.RegularExpressions; +using NUnit.Framework; +using OpenSim.Region.ScriptEngine.Shared.CodeTools; + +namespace OpenSim.Region.ScriptEngine.Shared.CodeTools.Tests +{ + /// + /// Tests the LSL compiler, both the code generation and transformation. + /// Each test has some LSL code as input and C# code as expected output. + /// The generated C# code is compared against the expected C# code. + /// + [TestFixture] + public class CSCodeGeneratorTest + { + [Test] + public void TestDefaultState() + { + string input = @"default +{ + state_entry() + { + } +} +"; + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestCustomState() + { + string input = @"default +{ + state_entry() + { + } +} + +state another_state +{ + no_sensor() + { + } +} +"; + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n }" + + "\n public void another_state_event_no_sensor()" + + "\n {" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestEventWithArguments() + { + string input = @"default +{ + at_rot_target(integer tnum, rotation targetrot, rotation ourrot) + { + } +} +"; + string expected = + "\n public void default_event_at_rot_target(LSL_Types.LSLInteger tnum, LSL_Types.Quaternion targetrot, LSL_Types.Quaternion ourrot)" + + "\n {" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIntegerDeclaration() + { + string input = @"default +{ + touch_start(integer num_detected) + { + integer x; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = 0;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestAssignments() + { + string input = @"default +{ + touch_start(integer num_detected) + { + string y; + integer x = 14; + y = ""Hello""; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLString y = \"\";" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(14);" + + "\n y = new LSL_Types.LSLString(\"Hello\");" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestAdditionSubtractionOperator() + { + string input = @"default +{ + touch_start(integer num_detected) + { + integer y = -3; + integer x = 14 + 6; + y = 12 +45+20+x + 23 + 1 + x + y; + y = 12 + -45 + - 20 + x + 23 + -1 + x + y; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)\n" + + " {\n" + + " LSL_Types.LSLInteger y = -new LSL_Types.LSLInteger(3);\n" + + " LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(14) + new LSL_Types.LSLInteger(6);\n" + + " y = new LSL_Types.LSLInteger(12) + new LSL_Types.LSLInteger(45) + new LSL_Types.LSLInteger(20) + x + new LSL_Types.LSLInteger(23) + new LSL_Types.LSLInteger(1) + x + y;\n" + + " y = new LSL_Types.LSLInteger(12) + -new LSL_Types.LSLInteger(45) + -new LSL_Types.LSLInteger(20) + x + new LSL_Types.LSLInteger(23) + -new LSL_Types.LSLInteger(1) + x + y;\n" + + " }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestStrings() + { + string input = @"default +{ + touch_start(integer num_detected) + { + llOwnerSay(""Testing, 1, 2, 3""); + llSay(0, ""I can hear you!""); + some_custom_function(1, 2, 3 +x, 4, ""five"", ""arguments""); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n llOwnerSay(new LSL_Types.LSLString(\"Testing, 1, 2, 3\"));" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"I can hear you!\"));" + + "\n some_custom_function(new LSL_Types.LSLInteger(1), new LSL_Types.LSLInteger(2), new LSL_Types.LSLInteger(3) + x, new LSL_Types.LSLInteger(4), new LSL_Types.LSLString(\"five\"), new LSL_Types.LSLString(\"arguments\"));" + + "\n }" + + "\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestBinaryExpression() + { + string input = @"default +{ + touch_start(integer num_detected) + { + integer y; + integer x = 14 + 6; + y = 12 - 3; + y = 12 * 3; + y = 12 / 3; + y = 12 | 3; + y = 12 & 3; + y = 12 % 3; + y = 12 + 45 - 20 * x / 23 | 1 & x + y; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger y = 0;" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(14) + new LSL_Types.LSLInteger(6);" + + "\n y = new LSL_Types.LSLInteger(12) - new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) * new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) / new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) | new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) & new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) % new LSL_Types.LSLInteger(3);" + + "\n y = new LSL_Types.LSLInteger(12) + new LSL_Types.LSLInteger(45) - new LSL_Types.LSLInteger(20) * x / new LSL_Types.LSLInteger(23) | new LSL_Types.LSLInteger(1) & x + y;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestFloatConstants() + { + string input = @"default +{ + touch_start(integer num_detected) + { + float y = 1.1; + y = 1.123E3; + y = 1.123e3; + y = 1.123E+3; + y = 1.123e+3; + y = 1.123E-3; + y = 1.123e-3; + y = .4; + y = -1.123E3; + y = -1.123e3; + y = -1.123E+3; + y = -1.123e+3; + y = -1.123E-3; + y = -1.123e-3; + y = -.4; + y = 12.3 + -1.45E3 - 1.20e-2; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLFloat y = new LSL_Types.LSLFloat(1.1);" + + "\n y = new LSL_Types.LSLFloat(1.123E3);" + + "\n y = new LSL_Types.LSLFloat(1.123e3);" + + "\n y = new LSL_Types.LSLFloat(1.123E+3);" + + "\n y = new LSL_Types.LSLFloat(1.123e+3);" + + "\n y = new LSL_Types.LSLFloat(1.123E-3);" + + "\n y = new LSL_Types.LSLFloat(1.123e-3);" + + "\n y = new LSL_Types.LSLFloat(.4);" + + "\n y = -new LSL_Types.LSLFloat(1.123E3);" + + "\n y = -new LSL_Types.LSLFloat(1.123e3);" + + "\n y = -new LSL_Types.LSLFloat(1.123E+3);" + + "\n y = -new LSL_Types.LSLFloat(1.123e+3);" + + "\n y = -new LSL_Types.LSLFloat(1.123E-3);" + + "\n y = -new LSL_Types.LSLFloat(1.123e-3);" + + "\n y = -new LSL_Types.LSLFloat(.4);" + + "\n y = new LSL_Types.LSLFloat(12.3) + -new LSL_Types.LSLFloat(1.45E3) - new LSL_Types.LSLFloat(1.20e-2);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestComments() + { + string input = @"// this test tests comments +default +{ + touch_start(integer num_detected) // this should be stripped + { + // fill in code here... + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestStringsWithEscapedQuotesAndComments() + { + string input = @"// this test tests strings, with escaped quotes and comments in strings +default +{ + touch_start(integer num_detected) + { + string s1 = ""this is a string.""; + string s2 = ""this is a string ""+""with an escaped \"" inside it.""; + s1 = s2+"" and this ""+""is a string with // comments.""; + + string onemore = ""[\^@]""; + + string multiline = ""Good evening Sir, + my name is Steve. + I come from a rough area. + I used to be addicted to crack + but now I am off it and trying to stay clean. + That is why I am selling magazine subscriptions.""; // http://www.imdb.com/title/tt0151804/quotes + } +} +"; + + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLString s1 = new LSL_Types.LSLString(\"this is a string.\");" + + "\n LSL_Types.LSLString s2 = new LSL_Types.LSLString(\"this is a string \") + new LSL_Types.LSLString(\"with an escaped \\\" inside it.\");" + + "\n s1 = s2 + new LSL_Types.LSLString(\" and this \") + new LSL_Types.LSLString(\"is a string with // comments.\");" + + "\n LSL_Types.LSLString onemore = new LSL_Types.LSLString(\"[\\^@]\");" + + "\n LSL_Types.LSLString multiline = new LSL_Types.LSLString(\"Good evening Sir,\\n my name is Steve.\\n I come from a rough area.\\n I used to be addicted to crack\\n but now I am off it and trying to stay clean.\\n That is why I am selling magazine subscriptions.\");" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestGlobalDefinedFunctions() + { + string input = @"// this test tests custom defined functions + +string onefunc() +{ + return ""Hi from onefunc()!""; +} + +twofunc(string s) +{ + llSay(1000, s); +} + +default +{ + touch_start(integer num_detected) + { + llSay(2000, onefunc()); + twofunc(); + } +} +"; + string expected = + "\n LSL_Types.LSLString onefunc()" + + "\n {" + + "\n return new LSL_Types.LSLString(\"Hi from onefunc()!\");" + + "\n }" + + "\n void twofunc(LSL_Types.LSLString s)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(1000), s);" + + "\n }" + + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(2000), onefunc());" + + "\n twofunc();" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestGlobalDeclaredVariables() + { + string input = @"// this test tests custom defined functions and global variables + +string globalString; +integer globalInt = 14; +integer anotherGlobal = 20 * globalInt; + +string onefunc() +{ + globalString = "" ...and the global!""; + return ""Hi "" + + ""from "" + + ""onefunc()!"" + globalString; +} + +twofunc(string s) +{ + llSay(1000, s); +} + +default +{ + touch_start(integer num_detected) + { + llSay(2000, onefunc()); + twofunc(); + } +} +"; + string expected = + "\n LSL_Types.LSLString globalString = \"\";" + + "\n LSL_Types.LSLInteger globalInt = new LSL_Types.LSLInteger(14);" + + "\n LSL_Types.LSLInteger anotherGlobal = new LSL_Types.LSLInteger(20) * globalInt;" + + "\n LSL_Types.LSLString onefunc()" + + "\n {" + + "\n globalString = new LSL_Types.LSLString(\" ...and the global!\");" + + "\n return new LSL_Types.LSLString(\"Hi \") + new LSL_Types.LSLString(\"from \") + new LSL_Types.LSLString(\"onefunc()!\") + globalString;" + + "\n }" + + "\n void twofunc(LSL_Types.LSLString s)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(1000), s);" + + "\n }" + + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(2000), onefunc());" + + "\n twofunc();" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestMoreAssignments() + { + string input = @"// this test tests +=, -=, *=, /=, %= + +string globalString; +integer globalInt = 14; + +string onefunc(string addition) +{ + globalInt -= 2; + + globalString += addition; + return ""Hi "" + + ""from "" + + ""onefunc()! "" + globalString; +} + +default +{ + touch_start(integer num_detected) + { + llSay(2000, onefunc()); + + integer x = 2; + x *= 3; + x /= 14 + -2; + x %= 10; + } +} +"; + string expected = + "\n LSL_Types.LSLString globalString = \"\";" + + "\n LSL_Types.LSLInteger globalInt = new LSL_Types.LSLInteger(14);" + + "\n LSL_Types.LSLString onefunc(LSL_Types.LSLString addition)" + + "\n {" + + "\n globalInt -= new LSL_Types.LSLInteger(2);" + + "\n globalString += addition;" + + "\n return new LSL_Types.LSLString(\"Hi \") + new LSL_Types.LSLString(\"from \") + new LSL_Types.LSLString(\"onefunc()! \") + globalString;" + + "\n }" + + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(2000), onefunc());" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(2);" + + "\n x *= new LSL_Types.LSLInteger(3);" + + "\n x /= new LSL_Types.LSLInteger(14) + -new LSL_Types.LSLInteger(2);" + + "\n x %= new LSL_Types.LSLInteger(10);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestVectorConstantNotation() + { + string input = @"default +{ + touch_start(integer num_detected) + { + vector y = <1.2, llGetMeAFloat(), 4.4>; + rotation x = <0.1, 0.1, one + 2, 0.9>; + + y = <0.1, 0.1, 1.1 - three - two+eight*8>; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.Vector3 y = new LSL_Types.Vector3(new LSL_Types.LSLFloat(1.2), llGetMeAFloat(), new LSL_Types.LSLFloat(4.4));" + + "\n LSL_Types.Quaternion x = new LSL_Types.Quaternion(new LSL_Types.LSLFloat(0.1), new LSL_Types.LSLFloat(0.1), one + new LSL_Types.LSLInteger(2), new LSL_Types.LSLFloat(0.9));" + + "\n y = new LSL_Types.Vector3(new LSL_Types.LSLFloat(0.1), new LSL_Types.LSLFloat(0.1), new LSL_Types.LSLFloat(1.1) - three - two + eight * new LSL_Types.LSLInteger(8));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestVectorMemberAccess() + { + string input = @"default +{ + touch_start(integer num_detected) + { + vector y = <1.2, llGetMeAFloat(), 4.4>; + x = y.x + 1.1; + y.x = 1.1; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.Vector3 y = new LSL_Types.Vector3(new LSL_Types.LSLFloat(1.2), llGetMeAFloat(), new LSL_Types.LSLFloat(4.4));" + + "\n x = y.x + new LSL_Types.LSLFloat(1.1);" + + "\n y.x = new LSL_Types.LSLFloat(1.1);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestExpressionInParentheses() + { + string input = @"default +{ + touch_start(integer num_detected) + { + integer y = -3; + integer x = 14 + 6; + y = 12 +45+20+x + (23 + 1) + x + y; + y = (12 + -45 + -20 + x + 23 )+ -1 + x + y; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger y = -new LSL_Types.LSLInteger(3);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(14) + new LSL_Types.LSLInteger(6);" + + "\n y = new LSL_Types.LSLInteger(12) + new LSL_Types.LSLInteger(45) + new LSL_Types.LSLInteger(20) + x + (new LSL_Types.LSLInteger(23) + new LSL_Types.LSLInteger(1)) + x + y;" + + "\n y = (new LSL_Types.LSLInteger(12) + -new LSL_Types.LSLInteger(45) + -new LSL_Types.LSLInteger(20) + x + new LSL_Types.LSLInteger(23)) + -new LSL_Types.LSLInteger(1) + x + y;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIncrementDecrementOperator() + { + string input = @"// here we'll test the ++ and -- operators + +default +{ + touch_start(integer num_detected) + { + integer y = -3; + integer x = 14 + 6; + y = 12 +45+20+x++ + (23 + 1) + ++x + -- y; + y = (12 + -45 + -20 + x-- + 23 )+ -1 + x -- + ++y; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger y = -new LSL_Types.LSLInteger(3);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(14) + new LSL_Types.LSLInteger(6);" + + "\n y = new LSL_Types.LSLInteger(12) + new LSL_Types.LSLInteger(45) + new LSL_Types.LSLInteger(20) + x++ + (new LSL_Types.LSLInteger(23) + new LSL_Types.LSLInteger(1)) + ++x + --y;" + + "\n y = (new LSL_Types.LSLInteger(12) + -new LSL_Types.LSLInteger(45) + -new LSL_Types.LSLInteger(20) + x-- + new LSL_Types.LSLInteger(23)) + -new LSL_Types.LSLInteger(1) + x-- + ++y;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestLists() + { + string input = @"// testing lists + +default +{ + touch_start(integer num_detected) + { + list l = []; + list m = [1, two, ""three"", <4.0, 4.0, 4.0>, 5 + 5]; + llCallSomeFunc(1, llAnotherFunc(), [1, 2, 3]); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.list l = new LSL_Types.list();" + + "\n LSL_Types.list m = new LSL_Types.list(new LSL_Types.LSLInteger(1), two, new LSL_Types.LSLString(\"three\"), new LSL_Types.Vector3(new LSL_Types.LSLFloat(4.0), new LSL_Types.LSLFloat(4.0), new LSL_Types.LSLFloat(4.0)), new LSL_Types.LSLInteger(5) + new LSL_Types.LSLInteger(5));" + + "\n llCallSomeFunc(new LSL_Types.LSLInteger(1), llAnotherFunc(), new LSL_Types.list(new LSL_Types.LSLInteger(1), new LSL_Types.LSLInteger(2), new LSL_Types.LSLInteger(3)));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIfStatement() + { + string input = @"// let's test if statements + +default +{ + touch_start(integer num_detected) + { + integer x = 1; + + if (x) llSay(0, ""Hello""); + if (1) + { + llSay(0, ""Hi""); + integer r = 3; + return; + } + + if (f(x)) llSay(0, ""f(x) is true""); + else llSay(0, ""f(x) is false""); + + if (x + y) llSay(0, ""x + y is true""); + else if (y - x) llSay(0, ""y - x is true""); + else llSay(0, ""Who needs x and y anyway?""); + + if (x * y) llSay(0, ""x * y is true""); + else if (y / x) + { + llSay(0, ""uh-oh, y / x is true, exiting""); + return; + } + else llSay(0, ""Who needs x and y anyway?""); + + // and now for my last trick + if (x % y) llSay(0, ""x is true""); + else if (y & x) llSay(0, ""y is true""); + else if (z | x) llSay(0, ""z is true""); + else if (a * (b + x)) llSay(0, ""a is true""); + else if (b) llSay(0, ""b is true""); + else if (v) llSay(0, ""v is true""); + else llSay(0, ""Everything is lies!""); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n if (x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Hello\"));" + + "\n if (new LSL_Types.LSLInteger(1))" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Hi\"));" + + "\n LSL_Types.LSLInteger r = new LSL_Types.LSLInteger(3);" + + "\n return ;" + + "\n }" + + "\n if (f(x))" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"f(x) is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"f(x) is false\"));" + + "\n if (x + y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x + y is true\"));" + + "\n else" + + "\n if (y - x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"y - x is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Who needs x and y anyway?\"));" + + "\n if (x * y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x * y is true\"));" + + "\n else" + + "\n if (y / x)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"uh-oh, y / x is true, exiting\"));" + + "\n return ;" + + "\n }" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Who needs x and y anyway?\"));" + + "\n if (x % y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is true\"));" + + "\n else" + + "\n if (y & x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"y is true\"));" + + "\n else" + + "\n if (z | x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"z is true\"));" + + "\n else" + + "\n if (a * (b + x))" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"a is true\"));" + + "\n else" + + "\n if (b)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"b is true\"));" + + "\n else" + + "\n if (v)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"v is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Everything is lies!\"));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIfElseStatement() + { + string input = @"// let's test complex logical expressions + +default +{ + touch_start(integer num_detected) + { + integer x = 1; + integer y = 0; + + if (x && y) llSay(0, ""Hello""); + if (x || y) + { + llSay(0, ""Hi""); + integer r = 3; + return; + } + + if (x && y || z) llSay(0, ""x is true""); + else llSay(0, ""x is false""); + + if (x == y) llSay(0, ""x is true""); + else if (y < x) llSay(0, ""y is true""); + else llSay(0, ""Who needs x and y anyway?""); + + if (x > y) llSay(0, ""x is true""); + else if (y <= x) + { + llSay(0, ""uh-oh, y is true, exiting""); + return; + } + else llSay(0, ""Who needs x and y anyway?""); + + // and now for my last trick + if (x >= y) llSay(0, ""x is true""); + else if (y != x) llSay(0, ""y is true""); + else if (!z) llSay(0, ""z is true""); + else if (!(a && b)) llSay(0, ""a is true""); + else if (b) llSay(0, ""b is true""); + else if (v) llSay(0, ""v is true""); + else llSay(0, ""Everything is lies!""); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n LSL_Types.LSLInteger y = new LSL_Types.LSLInteger(0);" + + "\n if (x && y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Hello\"));" + + "\n if (x || y)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Hi\"));" + + "\n LSL_Types.LSLInteger r = new LSL_Types.LSLInteger(3);" + + "\n return ;" + + "\n }" + + "\n if (x && y || z)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is false\"));" + + "\n if (x == y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is true\"));" + + "\n else" + + "\n if (y < x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"y is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Who needs x and y anyway?\"));" + + "\n if (x > y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is true\"));" + + "\n else" + + "\n if (y <= x)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"uh-oh, y is true, exiting\"));" + + "\n return ;" + + "\n }" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Who needs x and y anyway?\"));" + + "\n if (x >= y)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"x is true\"));" + + "\n else" + + "\n if (y != x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"y is true\"));" + + "\n else" + + "\n if (!z)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"z is true\"));" + + "\n else" + + "\n if (!(a && b))" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"a is true\"));" + + "\n else" + + "\n if (b)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"b is true\"));" + + "\n else" + + "\n if (v)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"v is true\"));" + + "\n else" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Everything is lies!\"));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestWhileLoop() + { + string input = @"// let's test while loops + +default +{ + touch_start(integer num_detected) + { + integer x = 1; + integer y = 0; + + while (x) llSay(0, ""To infinity, and beyond!""); + while (0 || (x && 0)) + { + llSay(0, ""Never say never.""); + return; + } + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n LSL_Types.LSLInteger y = new LSL_Types.LSLInteger(0);" + + "\n while (x)" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"To infinity, and beyond!\"));" + + "\n while (new LSL_Types.LSLInteger(0) || (x && new LSL_Types.LSLInteger(0)))" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Never say never.\"));" + + "\n return ;" + + "\n }" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestDoWhileLoop() + { + string input = @"// let's test do-while loops + +default +{ + touch_start(integer num_detected) + { + integer x = 1; + integer y = 0; + + do llSay(0, ""And we're doing...""); + while (x); + + do + { + llSay(0, ""I like it here. I wish we could stay here forever.""); + y--; + } while (y); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n LSL_Types.LSLInteger y = new LSL_Types.LSLInteger(0);" + + "\n do" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"And we're doing...\"));" + + "\n while (x);" + + "\n do" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"I like it here. I wish we could stay here forever.\"));" + + "\n y--;" + + "\n }" + + "\n while (y);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestForLoop() + { + string input = @"// let's test for loops + +default +{ + touch_start(integer num_detected) + { + integer x = 1; + integer y = 0; + + for (x = 10; x >= 0; x--) + { + llOwnerSay(""Launch in T minus "" + x); + IncreaseRocketPower(); + } + + for (x = 0, y = 6; y > 0 && x != y; x++, y--) llOwnerSay(""Hi "" + x + "", "" + y); + for (x = 0, y = 6; ! y; x++,y--) llOwnerSay(""Hi "" + x + "", "" + y); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n LSL_Types.LSLInteger y = new LSL_Types.LSLInteger(0);" + + "\n for (x = new LSL_Types.LSLInteger(10); x >= new LSL_Types.LSLInteger(0); x--)" + + "\n {" + + "\n llOwnerSay(new LSL_Types.LSLString(\"Launch in T minus \") + x);" + + "\n IncreaseRocketPower();" + + "\n }" + + "\n for (x = new LSL_Types.LSLInteger(0), y = new LSL_Types.LSLInteger(6); y > new LSL_Types.LSLInteger(0) && x != y; x++, y--)" + + "\n llOwnerSay(new LSL_Types.LSLString(\"Hi \") + x + new LSL_Types.LSLString(\", \") + y);" + + "\n for (x = new LSL_Types.LSLInteger(0), y = new LSL_Types.LSLInteger(6); !y; x++, y--)" + + "\n llOwnerSay(new LSL_Types.LSLString(\"Hi \") + x + new LSL_Types.LSLString(\", \") + y);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestFloatsWithTrailingDecimal() + { + string input = @"// a curious feature of LSL that allows floats to be defined with a trailing dot + +default +{ + touch_start(integer num_detected) + { + float y = 1.; + y = 1.E3; + y = 1.e3; + y = 1.E+3; + y = 1.e+3; + y = 1.E-3; + y = 1.e-3; + y = -1.E3; + y = -1.e3; + y = -1.E+3; + y = -1.e+3; + y = -1.E-3; + y = -1.e-3; + y = 12. + -1.E3 - 1.e-2; + vector v = <0.,0.,0.>; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLFloat y = new LSL_Types.LSLFloat(1.0);" + + "\n y = new LSL_Types.LSLFloat(1.0E3);" + + "\n y = new LSL_Types.LSLFloat(1.0e3);" + + "\n y = new LSL_Types.LSLFloat(1.0E+3);" + + "\n y = new LSL_Types.LSLFloat(1.0e+3);" + + "\n y = new LSL_Types.LSLFloat(1.0E-3);" + + "\n y = new LSL_Types.LSLFloat(1.0e-3);" + + "\n y = -new LSL_Types.LSLFloat(1.0E3);" + + "\n y = -new LSL_Types.LSLFloat(1.0e3);" + + "\n y = -new LSL_Types.LSLFloat(1.0E+3);" + + "\n y = -new LSL_Types.LSLFloat(1.0e+3);" + + "\n y = -new LSL_Types.LSLFloat(1.0E-3);" + + "\n y = -new LSL_Types.LSLFloat(1.0e-3);" + + "\n y = new LSL_Types.LSLFloat(12.0) + -new LSL_Types.LSLFloat(1.0E3) - new LSL_Types.LSLFloat(1.0e-2);" + + "\n LSL_Types.Vector3 v = new LSL_Types.Vector3(new LSL_Types.LSLFloat(0.0), new LSL_Types.LSLFloat(0.0), new LSL_Types.LSLFloat(0.0));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestUnaryAndBinaryOperators() + { + string input = @"// let's test a few more operators + +default +{ + touch_start(integer num_detected) + { + integer x = 2; + integer y = 1; + integer z = x ^ y; + x = ~ z; + x = ~(y && z); + y = x >> z; + z = y << x; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(2);" + + "\n LSL_Types.LSLInteger y = new LSL_Types.LSLInteger(1);" + + "\n LSL_Types.LSLInteger z = x ^ y;" + + "\n x = ~z;" + + "\n x = ~(y && z);" + + "\n y = x >> z;" + + "\n z = y << x;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestTypecasts() + { + string input = @"// let's test typecasts + +default +{ + touch_start(integer num_detected) + { + string s = """"; + integer x = 1; + + s = (string) x++; + s = (string) x; + s = (string) <0., 0., 0.>; + s = (string) <1., 1., 1., 1.>; + s = (integer) ""1""; + s = (string) llSomethingThatReturnsInteger(); + s = (string) 134; + s = (string) (x ^ y | (z && l)) + (string) (x + y - 13); + llOwnerSay(""s is: "" + s); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLString s = new LSL_Types.LSLString(\"\");" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(1);" + + "\n s = (LSL_Types.LSLString) (x++);" + + "\n s = (LSL_Types.LSLString) (x);" + + "\n s = (LSL_Types.LSLString) (new LSL_Types.Vector3(new LSL_Types.LSLFloat(0.0), new LSL_Types.LSLFloat(0.0), new LSL_Types.LSLFloat(0.0)));" + + "\n s = (LSL_Types.LSLString) (new LSL_Types.Quaternion(new LSL_Types.LSLFloat(1.0), new LSL_Types.LSLFloat(1.0), new LSL_Types.LSLFloat(1.0), new LSL_Types.LSLFloat(1.0)));" + + "\n s = (LSL_Types.LSLInteger) (new LSL_Types.LSLString(\"1\"));" + + "\n s = (LSL_Types.LSLString) (llSomethingThatReturnsInteger());" + + "\n s = (LSL_Types.LSLString) (new LSL_Types.LSLInteger(134));" + + "\n s = (LSL_Types.LSLString) (x ^ y | (z && l)) + (LSL_Types.LSLString) (x + y - new LSL_Types.LSLInteger(13));" + + "\n llOwnerSay(new LSL_Types.LSLString(\"s is: \") + s);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestStates() + { + string input = @"// let's test states + +default +{ + touch_start(integer num_detected) + { + llSay(0, ""Going to state 'statetwo'""); + state statetwo; + } +} + +state statetwo +{ + state_entry() + { + llSay(0, ""Going to the default state""); + state default; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Going to state 'statetwo'\"));" + + "\n state(\"statetwo\");" + + "\n }" + + "\n public void statetwo_event_state_entry()" + + "\n {" + + "\n llSay(new LSL_Types.LSLInteger(0), new LSL_Types.LSLString(\"Going to the default state\"));" + + "\n state(\"default\");" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestHexIntegerConstants() + { + string input = @"// let's test hex integers + +default +{ + touch_start(integer num_detected) + { + integer x = 0x23; + integer x = 0x2f34B; + integer x = 0x2F34b; + integer x = 0x2F34B; + integer x = 0x2f34b; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(0x23);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(0x2f34B);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(0x2F34b);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(0x2F34B);" + + "\n LSL_Types.LSLInteger x = new LSL_Types.LSLInteger(0x2f34b);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestJumps() + { + string input = @"// let's test jumps + +default +{ + touch_start(integer num_detected) + { + jump here; + llOwnerSay(""Uh oh, the jump didn't work""); + @here; + llOwnerSay(""After the jump""); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n goto here;" + + "\n llOwnerSay(new LSL_Types.LSLString(\"Uh oh, the jump didn't work\"));" + + "\n here:" + + "\n llOwnerSay(new LSL_Types.LSLString(\"After the jump\"));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestImplicitVariableInitialization() + { + string input = @"// let's test implicitly initializing variables + +default +{ + touch_start(integer num_detected) + { + integer i; integer j = 14; + float f; float g = 14.0; + string s; string t = ""Hi there""; + list l; list m = [1, 2, 3]; + vector v; vector w = <1.0, 0.1, 0.5>; + rotation r; rotation u = <0.8, 0.7, 0.6, llSomeFunc()>; + key k; key n = ""ping""; + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger i = 0;" + + "\n LSL_Types.LSLInteger j = new LSL_Types.LSLInteger(14);" + + "\n LSL_Types.LSLFloat f = 0.0;" + + "\n LSL_Types.LSLFloat g = new LSL_Types.LSLFloat(14.0);" + + "\n LSL_Types.LSLString s = \"\";" + + "\n LSL_Types.LSLString t = new LSL_Types.LSLString(\"Hi there\");" + + "\n LSL_Types.list l = new LSL_Types.list();" + + "\n LSL_Types.list m = new LSL_Types.list(new LSL_Types.LSLInteger(1), new LSL_Types.LSLInteger(2), new LSL_Types.LSLInteger(3));" + + "\n LSL_Types.Vector3 v = new LSL_Types.Vector3(0.0, 0.0, 0.0);" + + "\n LSL_Types.Vector3 w = new LSL_Types.Vector3(new LSL_Types.LSLFloat(1.0), new LSL_Types.LSLFloat(0.1), new LSL_Types.LSLFloat(0.5));" + + "\n LSL_Types.Quaternion r = new LSL_Types.Quaternion(0.0, 0.0, 0.0, 0.0);" + + "\n LSL_Types.Quaternion u = new LSL_Types.Quaternion(new LSL_Types.LSLFloat(0.8), new LSL_Types.LSLFloat(0.7), new LSL_Types.LSLFloat(0.6), llSomeFunc());" + + "\n LSL_Types.LSLString k = \"\";" + + "\n LSL_Types.LSLString n = new LSL_Types.LSLString(\"ping\");" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestMultipleEqualsExpression() + { + string input = @"// let's test x = y = 5 type expressions + +default +{ + touch_start(integer num_detected) + { + integer x; + integer y; + x = y = 5; + x += y -= 5; + llOwnerSay(""x is: "" + (string) x + "", y is: "" + (string) y); + } +} +"; + string expected = + "\n public void default_event_touch_start(LSL_Types.LSLInteger num_detected)" + + "\n {" + + "\n LSL_Types.LSLInteger x = 0;" + + "\n LSL_Types.LSLInteger y = 0;" + + "\n x = y = new LSL_Types.LSLInteger(5);" + + "\n x += y -= new LSL_Types.LSLInteger(5);" + + "\n llOwnerSay(new LSL_Types.LSLString(\"x is: \") + (LSL_Types.LSLString) (x) + new LSL_Types.LSLString(\", y is: \") + (LSL_Types.LSLString) (y));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestUnaryExpressionLastInVectorConstant() + { + string input = @"// let's test unary expressions some more + +default +{ + state_entry() + { + vector v = ; + } +} +"; + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n LSL_Types.Vector3 v = new LSL_Types.Vector3(x, y, -new LSL_Types.LSLFloat(0.5));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestVectorMemberPlusEquals() + { + string input = @"// let's test unary expressions some more + +default +{ + state_entry() + { + vector v = llGetPos(); + v.z += 4; + v.z -= 4; + v.z *= 4; + v.z /= 4; + v.z %= 4; + } +} +"; + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n LSL_Types.Vector3 v = llGetPos();" + + "\n v.z += new LSL_Types.LSLInteger(4);" + + "\n v.z -= new LSL_Types.LSLInteger(4);" + + "\n v.z *= new LSL_Types.LSLInteger(4);" + + "\n v.z /= new LSL_Types.LSLInteger(4);" + + "\n v.z %= new LSL_Types.LSLInteger(4);" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestWhileLoopWithNoBody() + { + string input = @"default +{ + state_entry() + { + while (1<0); + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n while (new LSL_Types.LSLInteger(1) < new LSL_Types.LSLInteger(0))" + + "\n ;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestDoWhileLoopWithNoBody() + { + string input = @"default +{ + state_entry() + { + do; + while (1<0); + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n do" + + "\n ;" + + "\n while (new LSL_Types.LSLInteger(1) < new LSL_Types.LSLInteger(0));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIfWithNoBody() + { + string input = @"default +{ + state_entry() + { + if (1<0); + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n if (new LSL_Types.LSLInteger(1) < new LSL_Types.LSLInteger(0))" + + "\n ;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestIfElseWithNoBody() + { + string input = @"default +{ + state_entry() + { + if (1<0); + else; + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n if (new LSL_Types.LSLInteger(1) < new LSL_Types.LSLInteger(0))" + + "\n ;" + + "\n else" + + "\n ;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestForLoopWithNoBody() + { + string input = @"default +{ + state_entry() + { + for (x = 4; 1<0; x += 2); + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n for (x = new LSL_Types.LSLInteger(4); new LSL_Types.LSLInteger(1) < new LSL_Types.LSLInteger(0); x += new LSL_Types.LSLInteger(2))" + + "\n ;" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestAssignmentInIfWhileDoWhile() + { + string input = @"default +{ + state_entry() + { + integer x; + + while (x = 14) llOwnerSay(""x is: "" + (string) x); + + if (x = 24) llOwnerSay(""x is: "" + (string) x); + + do + llOwnerSay(""x is: "" + (string) x); + while (x = 44); + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n LSL_Types.LSLInteger x = 0;" + + "\n while (x = new LSL_Types.LSLInteger(14))" + + "\n llOwnerSay(new LSL_Types.LSLString(\"x is: \") + (LSL_Types.LSLString) (x));" + + "\n if (x = new LSL_Types.LSLInteger(24))" + + "\n llOwnerSay(new LSL_Types.LSLString(\"x is: \") + (LSL_Types.LSLString) (x));" + + "\n do" + + "\n llOwnerSay(new LSL_Types.LSLString(\"x is: \") + (LSL_Types.LSLString) (x));" + + "\n while (x = new LSL_Types.LSLInteger(44));" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + public void TestLSLListHack() + { + string input = @"default +{ + state_entry() + { + list l = [""hello""]; + l = (l=[]) + l + ""world""; + } +}"; + + string expected = + "\n public void default_event_state_entry()" + + "\n {" + + "\n LSL_Types.list l = new LSL_Types.list(new LSL_Types.LSLString(\"hello\"));" + + "\n l = (l = new LSL_Types.list()) + l + new LSL_Types.LSLString(\"world\");" + + "\n }\n"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = cg.Convert(input); + Assert.AreEqual(expected, output); + } + + [Test] + [ExpectedException(typeof(Tools.CSToolsException))] + public void TestSyntaxError() + { + string input = @"default +{ + state_entry() + { + integer y + } +} +"; + try + { + CSCodeGenerator cg = new CSCodeGenerator(); + cg.Convert(input); + } + catch (Tools.CSToolsException e) + { + // The syntax error is on line 6, char 5 (expected ';', found + // '}'). + Regex r = new Regex("Line ([0-9]+), char ([0-9]+)"); + Match m = r.Match(e.Message); + Assert.AreEqual("6", m.Groups[1].Value); + Assert.AreEqual("5", m.Groups[2].Value); + + throw; + } + } + + [Test] + [ExpectedException(typeof(Tools.CSToolsException))] + public void TestSyntaxErrorDeclaringVariableInForLoop() + { + string input = @"default +{ + state_entry() + { + for (integer x = 0; x < 10; x++) llOwnerSay(""x is: "" + (string) x); + } +} +"; + try + { + CSCodeGenerator cg = new CSCodeGenerator(); + cg.Convert(input); + } + catch (Tools.CSToolsException e) + { + // The syntax error is on line 5, char 14 (Syntax error) + Regex r = new Regex("Line ([0-9]+), char ([0-9]+)"); + Match m = r.Match(e.Message); + Assert.AreEqual("5", m.Groups[1].Value); + Assert.AreEqual("14", m.Groups[2].Value); + + throw; + } + } + } +} diff --git a/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CompilerTest.cs b/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CompilerTest.cs new file mode 100644 index 0000000..7725d8d --- /dev/null +++ b/OpenSim/Region/ScriptEngine/Shared/CodeTools/Tests/CompilerTest.cs @@ -0,0 +1,153 @@ +/* + * Copyright (c) Contributors, http://opensimulator.org/ + * See CONTRIBUTORS.TXT for a full list of copyright holders. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * * Neither the name of the OpenSim Project nor the + * names of its contributors may be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +using System.IO; +using System.CodeDom.Compiler; +using System.Collections.Generic; +using Microsoft.CSharp; +using NUnit.Framework; +using OpenSim.Region.ScriptEngine.Shared.CodeTools; + +namespace OpenSim.Region.ScriptEngine.Shared.CodeTools.Tests +{ + /// + /// Tests the LSL compiler. Among other things, test that error messages + /// generated by the C# compiler can be mapped to prper lines/columns in + /// the LSL source. + /// + [TestFixture] + public class CompilerTest + { + private string m_testDir; + private CSharpCodeProvider m_CSCodeProvider; + private CompilerParameters m_compilerParameters; + private CompilerResults m_compilerResults; + + /// + /// Creates a temporary directory where build artifacts are stored. + /// + [TestFixtureSetUp] + public void Init() + { + m_testDir = Path.Combine(Path.GetTempPath(), "opensim_compilerTest_" + Path.GetRandomFileName()); + + if (!Directory.Exists(m_testDir)) + { + // Create the temporary directory for housing build artifacts. + Directory.CreateDirectory(m_testDir); + } + + // Create a CSCodeProvider and CompilerParameters. + m_CSCodeProvider = new CSharpCodeProvider(); + m_compilerParameters = new CompilerParameters(); + + string rootPath = Path.Combine(Path.GetDirectoryName(System.AppDomain.CurrentDomain.BaseDirectory), "bin"); + m_compilerParameters.ReferencedAssemblies.Add(Path.Combine(rootPath, "OpenSim.Region.ScriptEngine.Shared.dll")); + m_compilerParameters.ReferencedAssemblies.Add(Path.Combine(rootPath, "OpenSim.Region.ScriptEngine.Shared.Api.Runtime.dll")); + m_compilerParameters.GenerateExecutable = false; + } + + /// + /// Removes the temporary build directory and any build artifacts + /// inside it. + /// + [TestFixtureTearDown] + public void CleanUp() + { + if (Directory.Exists(m_testDir)) + { + // Blow away the temporary directory with artifacts. + Directory.Delete(m_testDir, true); + } + } + + /// + /// Test the C# compiler error message can be mapped to the correct + /// line/column in the LSL source when an undeclared variable is used. + /// + //[Test] + public void TestUseUndeclaredVariable() + { + m_compilerParameters.OutputAssembly = Path.Combine(m_testDir, Path.GetRandomFileName() + ".dll"); + + string input = @"default +{ + state_entry() + { + integer y = x + 3; + } +}"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = "using OpenSim.Region.ScriptEngine.Shared; using System.Collections.Generic;\n" + + "namespace SecondLife { " + + "public class Script : OpenSim.Region.ScriptEngine.Shared.ScriptBase.ScriptBaseClass {\n" + + "public Script() { } " + + cg.Convert(input) + + "} }\n"; + Dictionary, KeyValuePair> positionMap = cg.PositionMap; + + m_compilerResults = m_CSCodeProvider.CompileAssemblyFromSource(m_compilerParameters, output); + + Assert.AreEqual(new KeyValuePair(5, 21), + positionMap[new KeyValuePair(m_compilerResults.Errors[0].Line, m_compilerResults.Errors[0].Column)]); + } + + /// + /// Test that a string can be cast to string and another string + /// concatenated. + /// + //[Test] + public void TestCastAndConcatString() + { + m_compilerParameters.OutputAssembly = Path.Combine(m_testDir, Path.GetRandomFileName() + ".dll"); + + string input = @"string s = "" a string""; + +default +{ + state_entry() + { + key gAvatarKey = llDetectedKey(0); + string tmp = (string) gAvatarKey + s; + llSay(0, tmp); + } +}"; + + CSCodeGenerator cg = new CSCodeGenerator(); + string output = "using OpenSim.Region.ScriptEngine.Shared; using System.Collections.Generic;\n" + + "namespace SecondLife { " + + "public class Script : OpenSim.Region.ScriptEngine.Shared.ScriptBase.ScriptBaseClass {\n" + + "public Script() { } " + + cg.Convert(input) + + "} }\n"; + m_compilerResults = m_CSCodeProvider.CompileAssemblyFromSource(m_compilerParameters, output); + + Assert.AreEqual(0, m_compilerResults.Errors.Count); + } + } +} -- cgit v1.1