pimgen: Add support for setProperty

Add code generation support for setProperty actions.

Signed-off-by: Brad Bishop <bradleyb@fuzziesquirrel.com>
Change-Id: Id367f6da075035824c8f2ae6302505ef625a629a
diff --git a/example/events.d/match1.yaml b/example/events.d/match1.yaml
index 87279d2..3a08305 100644
--- a/example/events.d/match1.yaml
+++ b/example/events.d/match1.yaml
@@ -11,6 +11,13 @@
             interface: org.freedesktop.DBus.Properties
             member: PropertiesChanged
       actions:
-          - name: noop
+          - name: setProperty
+            args:
+              interface: xyz.openbmc_project.Example.Iface1
+              property: ExampleProperty1
+              path: /test
+              value:
+                type: string
+                value: test
 
 # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
diff --git a/pimgen.py b/pimgen.py
index e5771d6..088590c 100755
--- a/pimgen.py
+++ b/pimgen.py
@@ -49,14 +49,20 @@
 
     def __init__(self, **kw):
         self.value = kw.pop('value')
+        self.cast = kw.pop('cast', None)
         super(Argument, self).__init__(**kw)
 
     def cppArg(self):
         '''Transform string types to c++ string constants.'''
-        if self.typeName == 'string':
-            return '"%s"' % self.value
 
-        return self.value
+        a = self.value
+        if self.typeName == 'string':
+            a = '"%s"' % a
+
+        if self.cast:
+            a = 'static_cast<%s>(%s)' % (self.cast, a)
+
+        return a
 
 
 class MethodCall(NamedElement, Renderer):
@@ -64,6 +70,7 @@
 
     def __init__(self, **kw):
         self.namespace = kw.pop('namespace', [])
+        self.template = kw.pop('template', '')
         self.pointer = kw.pop('pointer', False)
         self.args = \
             [Argument(**x) for x in kw.pop('args', [])]
@@ -72,7 +79,12 @@
     def bare_method(self):
         '''Provide the method name and encompassing
         namespace without any arguments.'''
-        return '::'.join(self.namespace + [self.name])
+
+        m = '::'.join(self.namespace + [self.name])
+        if self.template:
+            m += '<%s>' % self.template
+
+        return m
 
 
 class Filter(MethodCall):
@@ -111,6 +123,30 @@
         super(DestroyObject, self).__init__(**kw)
 
 
+class SetProperty(Action):
+    '''Render a setProperty action.'''
+
+    def __init__(self, **kw):
+        mapped = kw.pop('args')
+        member = Interface(mapped['interface']).namespace()
+        member = '&%s' % '::'.join(
+            member.split('::') + [NamedElement(
+                name=mapped['property']).camelCase])
+
+        memberType = Argument(**mapped['value']).cppTypeName
+
+        kw['template'] = Interface(mapped['interface']).namespace()
+        kw['args'] = [
+            {'value': mapped['path'], 'type':'string'},
+            {'value': mapped['interface'], 'type':'string'},
+            {'value': member, 'cast': '{0} ({1}::*)({0})'.format(
+                memberType,
+                Interface(mapped['interface']).namespace())},
+            mapped['value'],
+        ]
+        super(SetProperty, self).__init__(**kw)
+
+
 class NoopAction(Action):
     '''Render a noop action.'''
 
@@ -146,6 +182,7 @@
     action_map = {
         'noop': NoopAction,
         'destroyObject': DestroyObject,
+        'setProperty': SetProperty,
     }
 
     def __init__(self, **kw):