A test of creating custom classes for control over RPC. Creates an LED class with member functions callable over serial.

Dependencies:   mbed

Committer:
JimmyTheHack
Date:
Wed Sep 29 01:32:32 2010 +0000
Revision:
2:fe4c1d5a97fa
Parent:
1:ddf1739fcbb8
Child:
3:d70703d85ff9
still testing autodocumentation

Who changed what in which revision?

UserRevisionLine numberNew contents of line
JimmyTheHack 0:b0ab03e03fdc 1 /* Test of custom RPC classes. An introduction to the RPC environment.
JimmyTheHack 0:b0ab03e03fdc 2
JimmyTheHack 0:b0ab03e03fdc 3 Based heavily off of code from iva2k
JimmyTheHack 0:b0ab03e03fdc 4 http://mbed.org/users/iva2k/programs/pub_iva2k_ethrpc/gpdz3x
JimmyTheHack 0:b0ab03e03fdc 5 in turn based off of servo library
JimmyTheHack 0:b0ab03e03fdc 6 http://mbed.org/projects/cookbook/svn/Servo/trunk/Servo.h
JimmyTheHack 0:b0ab03e03fdc 7 http://mbed.org/projects/cookbook/svn/Servo/trunk/Servo.cpp
JimmyTheHack 0:b0ab03e03fdc 8
JimmyTheHack 0:b0ab03e03fdc 9 This code features heavy copy-pasting, so I take little credit for the content. My goal was not to create original code, but to put together a basic introduction
JimmyTheHack 0:b0ab03e03fdc 10 to help myself and others understand how to implement custom classes in RPC.
JimmyTheHack 0:b0ab03e03fdc 11
JimmyTheHack 0:b0ab03e03fdc 12 This code creates a custom RPC class called LED. The LED class contains two member functions, toggle and blink.
JimmyTheHack 0:b0ab03e03fdc 13 Normally when compiling a program the original names of the objects are lost. In order to be able to call these functions by name after compiling, we must save the name in a string.
JimmyTheHack 0:b0ab03e03fdc 14 The name of our object and necessary arguments for the initialization function are saved using the get_rpc_class() function. The name of each member function and their arguments must also be
JimmyTheHack 0:b0ab03e03fdc 15 registered using get_rpc_methods().
JimmyTheHack 0:b0ab03e03fdc 16
JimmyTheHack 0:b0ab03e03fdc 17
JimmyTheHack 0:b0ab03e03fdc 18
JimmyTheHack 0:b0ab03e03fdc 19
JimmyTheHack 0:b0ab03e03fdc 20 From: http://mbed.org/cookbook/Interfacing-Using-RPC
JimmyTheHack 0:b0ab03e03fdc 21
JimmyTheHack 0:b0ab03e03fdc 22 RPC Commands are in the format: "/<Object name>/<Method name> <Arguments separated by spaces>
JimmyTheHack 0:b0ab03e03fdc 23 If you send just "/" mbed will return a list of objects that can be used
JimmyTheHack 0:b0ab03e03fdc 24 If you send "/<object name>/" then mbed will return the methods which can be used on this object.
JimmyTheHack 0:b0ab03e03fdc 25
JimmyTheHack 0:b0ab03e03fdc 26 I haven't finished digging through this yet, but I think most of the documentation for the RPC functions is in the files:
JimmyTheHack 0:b0ab03e03fdc 27 http://mbed.org/projects/libraries/svn/mbed/trunk/Base.h
JimmyTheHack 0:b0ab03e03fdc 28 http://mbed.org/projects/libraries/svn/mbed/trunk/rpc.h
JimmyTheHack 0:b0ab03e03fdc 29
JimmyTheHack 0:b0ab03e03fdc 30
JimmyTheHack 0:b0ab03e03fdc 31 */
JimmyTheHack 0:b0ab03e03fdc 32
JimmyTheHack 0:b0ab03e03fdc 33 #include "mbed.h"
JimmyTheHack 0:b0ab03e03fdc 34 #include "rpc.h"
JimmyTheHack 0:b0ab03e03fdc 35
JimmyTheHack 0:b0ab03e03fdc 36
JimmyTheHack 2:fe4c1d5a97fa 37 /**Class: LED
JimmyTheHack 2:fe4c1d5a97fa 38 *
JimmyTheHack 2:fe4c1d5a97fa 39 *attached to an LED pin, contains simple blink functionality over serial
JimmyTheHack 2:fe4c1d5a97fa 40 *
JimmyTheHack 2:fe4c1d5a97fa 41 */
JimmyTheHack 0:b0ab03e03fdc 42
JimmyTheHack 0:b0ab03e03fdc 43 class LED :public Base { //make sure to define the class with inheritance from the Base RPC class.
JimmyTheHack 0:b0ab03e03fdc 44 public:
JimmyTheHack 0:b0ab03e03fdc 45 //constructor//
JimmyTheHack 0:b0ab03e03fdc 46 LED(PinName mypin, const char *name=NULL);
JimmyTheHack 0:b0ab03e03fdc 47 //LED class functions//
JimmyTheHack 0:b0ab03e03fdc 48 void blink(int n);
JimmyTheHack 0:b0ab03e03fdc 49 int toggle();
JimmyTheHack 0:b0ab03e03fdc 50 DigitalOut LEDpin;
JimmyTheHack 0:b0ab03e03fdc 51 int state;
JimmyTheHack 0:b0ab03e03fdc 52
JimmyTheHack 0:b0ab03e03fdc 53 #ifdef MBED_RPC //this code will not compile unless we have included the rpc.h file. So this class can also be used without RPC.
JimmyTheHack 0:b0ab03e03fdc 54 virtual const struct rpc_method *get_rpc_methods();
JimmyTheHack 0:b0ab03e03fdc 55 static struct rpc_class *get_rpc_class();
JimmyTheHack 0:b0ab03e03fdc 56 #endif // MBED_RPC
JimmyTheHack 0:b0ab03e03fdc 57 };
JimmyTheHack 0:b0ab03e03fdc 58
JimmyTheHack 0:b0ab03e03fdc 59
JimmyTheHack 1:ddf1739fcbb8 60 /**Initialization Function**/
JimmyTheHack 1:ddf1739fcbb8 61 LED::LED(PinName mypin, const char *name) : Base(name), LEDpin(mypin) { //initialize pin
JimmyTheHack 0:b0ab03e03fdc 62 LEDpin.write(0);
JimmyTheHack 0:b0ab03e03fdc 63 state=false; //set LED to off
JimmyTheHack 0:b0ab03e03fdc 64 }
JimmyTheHack 1:ddf1739fcbb8 65 /**switch the state of the LED**/
JimmyTheHack 1:ddf1739fcbb8 66 int LED::toggle() {
JimmyTheHack 0:b0ab03e03fdc 67 if (state==0) {
JimmyTheHack 0:b0ab03e03fdc 68 state=1;
JimmyTheHack 0:b0ab03e03fdc 69 } else {
JimmyTheHack 0:b0ab03e03fdc 70 state=0;
JimmyTheHack 0:b0ab03e03fdc 71 }
JimmyTheHack 0:b0ab03e03fdc 72 LEDpin=state;
JimmyTheHack 0:b0ab03e03fdc 73 return state; //print the current state of the LED
JimmyTheHack 0:b0ab03e03fdc 74 }
JimmyTheHack 0:b0ab03e03fdc 75
JimmyTheHack 1:ddf1739fcbb8 76 /**blink the LED n times**/
JimmyTheHack 1:ddf1739fcbb8 77 void LED::blink(int n=1) {
JimmyTheHack 0:b0ab03e03fdc 78 do { //blink at least once
JimmyTheHack 0:b0ab03e03fdc 79 toggle(); //toggle LED state
JimmyTheHack 0:b0ab03e03fdc 80 wait(.2);
JimmyTheHack 0:b0ab03e03fdc 81 toggle(); //return LED to original state
JimmyTheHack 0:b0ab03e03fdc 82 wait(.2);
JimmyTheHack 0:b0ab03e03fdc 83 n--;
JimmyTheHack 0:b0ab03e03fdc 84 } while (n>=1);
JimmyTheHack 0:b0ab03e03fdc 85 }
JimmyTheHack 0:b0ab03e03fdc 86
JimmyTheHack 0:b0ab03e03fdc 87
JimmyTheHack 0:b0ab03e03fdc 88 #ifdef MBED_RPC
JimmyTheHack 1:ddf1739fcbb8 89 /**Create a list of the available methods which can be called for this class**/
JimmyTheHack 1:ddf1739fcbb8 90 const rpc_method *LED::get_rpc_methods() {
JimmyTheHack 0:b0ab03e03fdc 91 static const rpc_method rpc_methods[] = {
JimmyTheHack 0:b0ab03e03fdc 92 { "toggle", rpc_method_caller<int, LED, &LED::toggle> }, //first specify the name string. The arguments to rpc_method_caller appear to be <outputs, RPC class, inputs, reference to function>.
JimmyTheHack 0:b0ab03e03fdc 93 //In this case, we have one output and no inputs so they are skipped. We must only specify the class of the object the method belongs to and the address of the method.
JimmyTheHack 0:b0ab03e03fdc 94
JimmyTheHack 0:b0ab03e03fdc 95 { "blink", rpc_method_caller<LED, int, &LED::blink> }, //this method has no outputs so they are skipped, but we must specify the input to be of type int.
JimmyTheHack 0:b0ab03e03fdc 96 RPC_METHOD_SUPER(Base),
JimmyTheHack 0:b0ab03e03fdc 97 };
JimmyTheHack 0:b0ab03e03fdc 98 return rpc_methods;
JimmyTheHack 0:b0ab03e03fdc 99 }
JimmyTheHack 1:ddf1739fcbb8 100 /**Register the class itself as an RPC-callable class**/
JimmyTheHack 1:ddf1739fcbb8 101 rpc_class *LED::get_rpc_class() {
JimmyTheHack 0:b0ab03e03fdc 102 static const rpc_function funcs[] = {
JimmyTheHack 0:b0ab03e03fdc 103 { "new", rpc_function_caller<const char*, PinName, const char*, &Base::construct<LED,PinName,const char*> > }, //still don't fully understand the arguments in this line. I suppose the first argument may just be an output echo of the name.
JimmyTheHack 0:b0ab03e03fdc 104 RPC_METHOD_END
JimmyTheHack 0:b0ab03e03fdc 105 };
JimmyTheHack 0:b0ab03e03fdc 106 static rpc_class c = { "LED", funcs, NULL };
JimmyTheHack 0:b0ab03e03fdc 107 return &c;
JimmyTheHack 0:b0ab03e03fdc 108 }
JimmyTheHack 0:b0ab03e03fdc 109 #endif // MBED_RPC
JimmyTheHack 0:b0ab03e03fdc 110
JimmyTheHack 0:b0ab03e03fdc 111
JimmyTheHack 0:b0ab03e03fdc 112 /* end LED class definition */
JimmyTheHack 0:b0ab03e03fdc 113
JimmyTheHack 0:b0ab03e03fdc 114
JimmyTheHack 0:b0ab03e03fdc 115
JimmyTheHack 0:b0ab03e03fdc 116
JimmyTheHack 0:b0ab03e03fdc 117
JimmyTheHack 0:b0ab03e03fdc 118 Serial pc(USBTX, USBRX); //set up serial communication
JimmyTheHack 1:ddf1739fcbb8 119 /**Wait for RPC commands and then call the interpreter**/
JimmyTheHack 1:ddf1739fcbb8 120 int main() {
JimmyTheHack 0:b0ab03e03fdc 121 // specify which classes we would like to be able to call over LED
JimmyTheHack 0:b0ab03e03fdc 122 Base::add_rpc_class<Timer>(); //a class included in the core mbed RPC library
JimmyTheHack 0:b0ab03e03fdc 123 Base::add_rpc_class<LED>(); //my own custom LED class
JimmyTheHack 0:b0ab03e03fdc 124
JimmyTheHack 0:b0ab03e03fdc 125 // receive commands, and send back the responses
JimmyTheHack 0:b0ab03e03fdc 126 char buf[256], outbuf[256];
JimmyTheHack 0:b0ab03e03fdc 127 while (1) {
JimmyTheHack 0:b0ab03e03fdc 128 pc.gets(buf, 256); //grab serial commands
JimmyTheHack 0:b0ab03e03fdc 129 rpc(buf, outbuf); //interpret the RPC commands
JimmyTheHack 0:b0ab03e03fdc 130 //this function only requires an input string and output string, so we can obtain RPC commands via a different interface than serial if desired.
JimmyTheHack 0:b0ab03e03fdc 131 pc.printf("%s\n", outbuf); //print back over serial
JimmyTheHack 0:b0ab03e03fdc 132 }
JimmyTheHack 0:b0ab03e03fdc 133 }
JimmyTheHack 0:b0ab03e03fdc 134
JimmyTheHack 0:b0ab03e03fdc 135
JimmyTheHack 0:b0ab03e03fdc 136