-- Raspberry Pi LPC1114 I/O Processor Expansion Board SPI Agent Firmware -- loopback test program -- Copyright (C)2013-2018, Philip Munts, President, Munts AM Corp. -- -- 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. -- -- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 COPYRIGHT HOLDER OR 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. WITH System; USE System; WITH Ada.Command_Line; USE Ada.Command_Line; WITH Ada.Real_Time; USE Ada.Real_Time; WITH Ada.Strings; USE Ada.Strings; WITH Ada.Strings.Fixed; USE Ada.Strings.Fixed; WITH Ada.Strings.Unbounded; USE Ada.Strings.Unbounded; WITH Ada.Text_IO; USE Ada.Text_IO; WITH Interfaces; USE Interfaces; WITH lpc11xx; USE lpc11xx; WITH spi_agent.commands; USE spi_agent.commands; WITH spi_agent.exceptions; USE spi_agent.exceptions; WITH spi_agent.messages; USE spi_agent.messages; WITH spi_agent.pins; USE spi_agent.pins; WITH AWS.Client; WITH AWS.Response; PROCEDURE spi_agent_http_client IS servername : Unbounded_String; iterations : Integer := 5000; error : Integer; cmd : SPIAGENT_COMMAND_MSG_t; resp : SPIAGENT_RESPONSE_MSG_t; starttime : Time; stoptime : Time; elapsed : Duration; rate : Duration; PACKAGE int_io IS NEW Integer_IO(Integer); USE int_io; PACKAGE duration_io IS NEW Fixed_IO(Duration); USE duration_io; PROCEDURE command (cmd : IN SPIAGENT_COMMAND_MSG_t; resp : OUT SPIAGENT_RESPONSE_MSG_t; error : OUT Integer) IS httprequest : Unbounded_String; httpresponse : String(1 .. 80); c1, c2, c3, c4, c5 : Natural; BEGIN resp.command := cmd.command; resp.pin := cmd.pin; resp.data := 0; resp.error := 0; -- Build the request string httprequest := "http://" & servername & ":8081/SPIAGENT?cmd=" & Trim(To_Unbounded_String(Unsigned_32'IMAGE(cmd.command)), Left) & "," & Trim(To_Unbounded_String(Unsigned_32'IMAGE(cmd.pin)), Left) & "," & Trim(To_Unbounded_String(Unsigned_32'IMAGE(cmd.data)), Left); -- Send the request string and get the response string httpresponse := Head(AWS.Response.Message_Body(AWS.Client.Get(To_String(httprequest))), httpresponse'LAST, ASCII.NUL); -- Parse the response string c1 := Index(httpresponse, ",", 1); IF c1 = 0 THEN RAISE DATA_ERROR WITH "Invalid response from HTTP server"; END IF; c2 := Index(httpresponse, ",", c1+1); IF c2 = 0 THEN RAISE DATA_ERROR WITH "Invalid response from HTTP server"; END IF; c3 := Index(httpresponse, ",", c2+1); IF c3 = 0 THEN RAISE DATA_ERROR WITH "Invalid response from HTTP server"; END IF; c4 := Index(httpresponse, ",", c3+1); IF c4 = 0 THEN RAISE DATA_ERROR WITH "Invalid response from HTTP server"; END IF; c5 := Index(httpresponse, ";", c4+1); IF c5 = 0 THEN RAISE DATA_ERROR WITH "Invalid response from HTTP server"; END IF; -- Convert the response string fields to variables error := Integer'VALUE(httpresponse(1 .. c1 - 1)); resp.command := Unsigned_32'VALUE(httpresponse(c1 + 1 .. c2 - 1)); resp.pin := Unsigned_32'VALUE(httpresponse(c2 + 1 .. c3 - 1)); resp.data := Unsigned_32'VALUE(httpresponse(c3 + 1 .. c4 - 1)); resp.error := Unsigned_32'VALUE(httpresponse(c4 + 1 .. c5 - 1)); END command; BEGIN New_Line; Put_Line("Raspberry Pi LPC1114 I/O Processor Expansion Board SPI Agent Firmware Test"); New_Line; -- Analyze command line parameters CASE Argument_Count IS WHEN 0 => servername := To_Unbounded_String("localhost"); WHEN 1 => servername := To_Unbounded_String(Argument(1)); WHEN 2 => servername := To_Unbounded_String(Argument(1)); iterations := Integer'VALUE(Argument(2)); WHEN OTHERS => Put_Line("Usage: " & Command_Name & " [hostname]"); RETURN; END CASE; -- Issue a NOP command cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_NOP); cmd.pin := 0; cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; Put("Response: "); Put("command:"); Put(Integer(resp.command), 0, 10); Put(" pin:"); Put(Integer(resp.pin), 0, 10); Put(" data:"); Put(Integer(resp.data), 0, 10); Put(" error:"); Put(Integer(resp.error), 0, 10); New_Line(1); -- Issue a LOOPBACK command cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_LOOPBACK); cmd.pin := 2; cmd.data := 3; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; Put("Response: "); Put("command:"); Put(Integer(resp.command), 0, 10); Put(" pin:"); Put(Integer(resp.pin), 0, 10); Put(" data:"); Put(Integer(resp.data), 0, 10); Put(" error:"); Put(Integer(resp.error), 0, 10); New_Line(1); -- Issue an illegal GET_GPIO command cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_GET_GPIO); cmd.pin := 99; cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; Put("Response: "); Put("command:"); Put(Integer(resp.command), 0, 10); Put(" pin:"); Put(Integer(resp.pin), 0, 10); Put(" data:"); Put(Integer(resp.data), 0, 10); Put(" error:"); Put(Integer(resp.error), 0, 10); New_Line(1); -- Issue an illegal command cmd.command := 99; cmd.pin := 0; cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; Put("Response: "); Put("command:"); Put(Integer(resp.command), 0, 10); Put(" pin:"); Put(Integer(resp.pin), 0, 10); Put(" data:"); Put(Integer(resp.data), 0, 10); Put(" error:"); Put(Integer(resp.error), 0, 10); New_Line(2); -- Query the LPC1114 firmware version cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_NOP); cmd.pin := 0; cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; IF resp.error /= 0 THEN RAISE SpiAgentError WITH "SPI Agent Firmware returned error:" & Unsigned_32'IMAGE(resp.error); END IF; Put("The LPC1114 firmware version is "); Put(Integer(resp.data), 1, 10); New_Line(1); -- Query the LPC1114 device ID cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_GET_SFR); cmd.pin := 16#400483F4#; cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; IF resp.error /= 0 THEN RAISE SpiAgentError WITH "SPI Agent Firmware returned error:" & Unsigned_32'IMAGE(resp.error); END IF; Put("The LPC1114 device ID is "); Put(Integer(resp.data), 1, 16); New_Line(1); -- Query the LED state cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_GET_GPIO); cmd.pin := pin_id_t'POS(LPC1114_LED); cmd.data := 0; command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; IF resp.error /= 0 THEN RAISE SpiAgentError WITH "SPI Agent Firmware returned error:" & Unsigned_32'IMAGE(resp.error); END IF; Put("The expansion board LED is "); CASE Integer(resp.data) IS WHEN 0 => Put("OFF"); WHEN 1 => Put("ON"); WHEN OTHERS => RAISE SpiAgentError WITH "Invalid value for GPIO state"; END CASE; New_Line(2); -- Perform loopback test Put("Starting" & Integer'IMAGE(iterations) & " SPI agent loopback test transactions..."); New_Line(2); starttime := Clock; FOR i IN 1 .. iterations LOOP cmd.command := SPIAGENT_COMMAND_t'POS(SPIAGENT_CMD_LOOPBACK); cmd.pin := Unsigned_32(i*17); cmd.data := Unsigned_32(i*19); command(cmd, resp, error); IF error /= 0 THEN RAISE SpiAgentError WITH "command() failed, error:" & Integer'IMAGE(error); END IF; IF (resp.command /= cmd.command) OR (resp.pin /= cmd.pin) OR (resp.data /= cmd.data) OR (resp.error /= 0) THEN Put("Iteration:"); Put(i, 0, 10); Put(" Response: "); Put("command:"); Put(Integer(resp.command), 0, 10); Put(" pin:"); Put(Integer(resp.pin), 0, 10); Put(" data:"); Put(Integer(resp.data), 0, 10); Put(" error:"); Put(Integer(resp.error), 0, 10); New_Line(1); END IF; END LOOP; stoptime := Clock; -- Display statistics elapsed := To_Duration(stoptime - starttime); rate := Duration(iterations) / elapsed; Put("Performed" & Integer'IMAGE(iterations) & " loopback tests in "); Put(elapsed, 0, 1); Put_Line(" seconds"); Put(" "); Put(rate, 0, 1); Put_Line(" iterations per second"); Put(" "); Put(1000000.0 / rate, 0, 1); Put_Line(" microseconds per iteration"); END spi_agent_http_client;