Home | History | Annotate | Line # | Download | only in libcc1
rpc.hh revision 1.6
      1  1.1  mrg /* RPC call and callback templates
      2  1.6  mrg    Copyright (C) 2014-2020 Free Software Foundation, Inc.
      3  1.1  mrg 
      4  1.1  mrg This file is part of GCC.
      5  1.1  mrg 
      6  1.1  mrg GCC is free software; you can redistribute it and/or modify it under
      7  1.1  mrg the terms of the GNU General Public License as published by the Free
      8  1.1  mrg Software Foundation; either version 3, or (at your option) any later
      9  1.1  mrg version.
     10  1.1  mrg 
     11  1.1  mrg GCC is distributed in the hope that it will be useful, but WITHOUT ANY
     12  1.1  mrg WARRANTY; without even the implied warranty of MERCHANTABILITY or
     13  1.1  mrg FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
     14  1.1  mrg for more details.
     15  1.1  mrg 
     16  1.1  mrg You should have received a copy of the GNU General Public License
     17  1.1  mrg along with GCC; see the file COPYING3.  If not see
     18  1.1  mrg <http://www.gnu.org/licenses/>.  */
     19  1.1  mrg 
     20  1.1  mrg #ifndef CC1_PLUGIN_RPC_HH
     21  1.1  mrg #define CC1_PLUGIN_RPC_HH
     22  1.1  mrg 
     23  1.1  mrg #include "status.hh"
     24  1.1  mrg #include "connection.hh"
     25  1.1  mrg 
     26  1.1  mrg namespace cc1_plugin
     27  1.1  mrg {
     28  1.1  mrg   // The plugin API may contain some "const" method parameters.
     29  1.1  mrg   // However, when unmarshalling we cannot unmarshall into a const
     30  1.1  mrg   // object; and furthermore we want to be able to deallocate pointers
     31  1.1  mrg   // when finished with them.  This wrapper class lets us properly
     32  1.1  mrg   // remove the "const" and handle deallocation from pointer types.
     33  1.1  mrg 
     34  1.1  mrg   template<typename T>
     35  1.1  mrg   class argument_wrapper
     36  1.1  mrg   {
     37  1.1  mrg   public:
     38  1.1  mrg 
     39  1.1  mrg     argument_wrapper () { }
     40  1.1  mrg     ~argument_wrapper () { }
     41  1.1  mrg 
     42  1.1  mrg     operator T () const { return m_object; }
     43  1.1  mrg 
     44  1.1  mrg     status unmarshall (connection *conn)
     45  1.1  mrg     {
     46  1.1  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
     47  1.1  mrg     }
     48  1.1  mrg 
     49  1.1  mrg   private:
     50  1.1  mrg 
     51  1.1  mrg     T m_object;
     52  1.1  mrg 
     53  1.1  mrg     // No copying or assignment allowed.
     54  1.1  mrg     argument_wrapper (const argument_wrapper &);
     55  1.1  mrg     argument_wrapper &operator= (const argument_wrapper &);
     56  1.1  mrg   };
     57  1.1  mrg 
     58  1.1  mrg   // Specialization for any kind of pointer.  This is declared but not
     59  1.1  mrg   // defined to avoid bugs if a new pointer type is introduced into
     60  1.1  mrg   // the API.  Instead you will just get a compilation error.
     61  1.1  mrg   template<typename T>
     62  1.1  mrg   class argument_wrapper<const T *>;
     63  1.1  mrg 
     64  1.1  mrg   // Specialization for string types.
     65  1.1  mrg   template<>
     66  1.1  mrg   class argument_wrapper<const char *>
     67  1.1  mrg   {
     68  1.1  mrg   public:
     69  1.1  mrg     argument_wrapper () : m_object (NULL) { }
     70  1.1  mrg     ~argument_wrapper ()
     71  1.1  mrg     {
     72  1.1  mrg       delete[] m_object;
     73  1.1  mrg     }
     74  1.1  mrg 
     75  1.1  mrg     operator const char * () const
     76  1.1  mrg     {
     77  1.1  mrg       return m_object;
     78  1.1  mrg     }
     79  1.1  mrg 
     80  1.1  mrg     status unmarshall (connection *conn)
     81  1.1  mrg     {
     82  1.1  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
     83  1.1  mrg     }
     84  1.1  mrg 
     85  1.1  mrg   private:
     86  1.1  mrg 
     87  1.1  mrg     char *m_object;
     88  1.1  mrg 
     89  1.1  mrg     // No copying or assignment allowed.
     90  1.1  mrg     argument_wrapper (const argument_wrapper &);
     91  1.1  mrg     argument_wrapper &operator= (const argument_wrapper &);
     92  1.1  mrg   };
     93  1.1  mrg 
     94  1.1  mrg   // Specialization for gcc_type_array.
     95  1.1  mrg   template<>
     96  1.1  mrg   class argument_wrapper<const gcc_type_array *>
     97  1.1  mrg   {
     98  1.1  mrg   public:
     99  1.1  mrg     argument_wrapper () : m_object (NULL) { }
    100  1.1  mrg     ~argument_wrapper ()
    101  1.1  mrg     {
    102  1.1  mrg       // It would be nicer if gcc_type_array could have a destructor.
    103  1.1  mrg       // But, it is in code shared with gdb and cannot.
    104  1.1  mrg       if (m_object != NULL)
    105  1.1  mrg 	delete[] m_object->elements;
    106  1.1  mrg       delete m_object;
    107  1.1  mrg     }
    108  1.1  mrg 
    109  1.1  mrg     operator const gcc_type_array * () const
    110  1.1  mrg     {
    111  1.1  mrg       return m_object;
    112  1.1  mrg     }
    113  1.1  mrg 
    114  1.1  mrg     status unmarshall (connection *conn)
    115  1.1  mrg     {
    116  1.1  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
    117  1.1  mrg     }
    118  1.1  mrg 
    119  1.1  mrg   private:
    120  1.1  mrg 
    121  1.1  mrg     gcc_type_array *m_object;
    122  1.1  mrg 
    123  1.1  mrg     // No copying or assignment allowed.
    124  1.1  mrg     argument_wrapper (const argument_wrapper &);
    125  1.1  mrg     argument_wrapper &operator= (const argument_wrapper &);
    126  1.1  mrg   };
    127  1.1  mrg 
    128  1.3  mrg #ifdef GCC_CP_INTERFACE_H
    129  1.3  mrg   // Specialization for gcc_vbase_array.
    130  1.3  mrg   template<>
    131  1.3  mrg   class argument_wrapper<const gcc_vbase_array *>
    132  1.3  mrg   {
    133  1.3  mrg   public:
    134  1.3  mrg     argument_wrapper () : m_object (NULL) { }
    135  1.3  mrg     ~argument_wrapper ()
    136  1.3  mrg     {
    137  1.3  mrg       // It would be nicer if gcc_type_array could have a destructor.
    138  1.3  mrg       // But, it is in code shared with gdb and cannot.
    139  1.3  mrg       if (m_object != NULL)
    140  1.3  mrg 	{
    141  1.3  mrg 	  delete[] m_object->flags;
    142  1.3  mrg 	  delete[] m_object->elements;
    143  1.3  mrg 	}
    144  1.3  mrg       delete m_object;
    145  1.3  mrg     }
    146  1.3  mrg 
    147  1.3  mrg     operator const gcc_vbase_array * () const
    148  1.3  mrg     {
    149  1.3  mrg       return m_object;
    150  1.3  mrg     }
    151  1.3  mrg 
    152  1.3  mrg     status unmarshall (connection *conn)
    153  1.3  mrg     {
    154  1.3  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
    155  1.3  mrg     }
    156  1.3  mrg 
    157  1.3  mrg   private:
    158  1.3  mrg 
    159  1.3  mrg     gcc_vbase_array *m_object;
    160  1.3  mrg 
    161  1.3  mrg     // No copying or assignment allowed.
    162  1.3  mrg     argument_wrapper (const argument_wrapper &);
    163  1.3  mrg     argument_wrapper &operator= (const argument_wrapper &);
    164  1.3  mrg   };
    165  1.3  mrg 
    166  1.3  mrg   // Specialization for gcc_cp_template_args.
    167  1.3  mrg   template<>
    168  1.3  mrg   class argument_wrapper<const gcc_cp_template_args *>
    169  1.3  mrg   {
    170  1.3  mrg   public:
    171  1.3  mrg     argument_wrapper () : m_object (NULL) { }
    172  1.3  mrg     ~argument_wrapper ()
    173  1.3  mrg     {
    174  1.3  mrg       // It would be nicer if gcc_type_array could have a destructor.
    175  1.3  mrg       // But, it is in code shared with gdb and cannot.
    176  1.3  mrg       if (m_object != NULL)
    177  1.3  mrg 	{
    178  1.3  mrg 	  delete[] m_object->elements;
    179  1.3  mrg 	  delete[] m_object->kinds;
    180  1.3  mrg 	}
    181  1.3  mrg       delete m_object;
    182  1.3  mrg     }
    183  1.3  mrg 
    184  1.3  mrg     operator const gcc_cp_template_args * () const
    185  1.3  mrg     {
    186  1.3  mrg       return m_object;
    187  1.3  mrg     }
    188  1.3  mrg 
    189  1.3  mrg     status unmarshall (connection *conn)
    190  1.3  mrg     {
    191  1.3  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
    192  1.3  mrg     }
    193  1.3  mrg 
    194  1.3  mrg   private:
    195  1.3  mrg 
    196  1.3  mrg     gcc_cp_template_args *m_object;
    197  1.3  mrg 
    198  1.3  mrg     // No copying or assignment allowed.
    199  1.3  mrg     argument_wrapper (const argument_wrapper &);
    200  1.3  mrg     argument_wrapper &operator= (const argument_wrapper &);
    201  1.3  mrg   };
    202  1.3  mrg 
    203  1.3  mrg   // Specialization for gcc_cp_function_args.
    204  1.3  mrg   template<>
    205  1.3  mrg   class argument_wrapper<const gcc_cp_function_args *>
    206  1.3  mrg   {
    207  1.3  mrg   public:
    208  1.3  mrg     argument_wrapper () : m_object (NULL) { }
    209  1.3  mrg     ~argument_wrapper ()
    210  1.3  mrg     {
    211  1.3  mrg       // It would be nicer if gcc_type_array could have a destructor.
    212  1.3  mrg       // But, it is in code shared with gdb and cannot.
    213  1.3  mrg       if (m_object != NULL)
    214  1.3  mrg 	{
    215  1.3  mrg 	  delete[] m_object->elements;
    216  1.3  mrg 	}
    217  1.3  mrg       delete m_object;
    218  1.3  mrg     }
    219  1.3  mrg 
    220  1.3  mrg     operator const gcc_cp_function_args * () const
    221  1.3  mrg     {
    222  1.3  mrg       return m_object;
    223  1.3  mrg     }
    224  1.3  mrg 
    225  1.3  mrg     status unmarshall (connection *conn)
    226  1.3  mrg     {
    227  1.3  mrg       return ::cc1_plugin::unmarshall (conn, &m_object);
    228  1.3  mrg     }
    229  1.3  mrg 
    230  1.3  mrg   private:
    231  1.3  mrg 
    232  1.3  mrg     gcc_cp_function_args *m_object;
    233  1.3  mrg 
    234  1.3  mrg     // No copying or assignment allowed.
    235  1.3  mrg     argument_wrapper (const argument_wrapper &);
    236  1.3  mrg     argument_wrapper &operator= (const argument_wrapper &);
    237  1.3  mrg   };
    238  1.3  mrg #endif /* GCC_CP_INTERFACE_H */
    239  1.3  mrg 
    240  1.1  mrg   // There are two kinds of template functions here: "call" and
    241  1.1  mrg   // "callback".  They are each repeated multiple times to handle
    242  1.1  mrg   // different numbers of arguments.  (This would be improved with
    243  1.1  mrg   // C++11, though applying a call is still tricky until C++14 can be
    244  1.1  mrg   // used.)
    245  1.1  mrg 
    246  1.1  mrg   // The "call" template is used for making a remote procedure call.
    247  1.1  mrg   // It starts a query ('Q') packet, marshalls its arguments, waits
    248  1.1  mrg   // for a result, and finally reads and returns the result via an
    249  1.1  mrg   // "out" parameter.
    250  1.1  mrg 
    251  1.1  mrg   // The "callback" template is used when receiving a remote procedure
    252  1.1  mrg   // call.  This template function is suitable for use with the
    253  1.1  mrg   // "callbacks" and "connection" classes.  It decodes incoming
    254  1.1  mrg   // arguments, passes them to the wrapped function, and finally
    255  1.1  mrg   // marshalls a reply packet.
    256  1.1  mrg 
    257  1.1  mrg   template<typename R>
    258  1.1  mrg   status
    259  1.1  mrg   call (connection *conn, const char *method, R *result)
    260  1.1  mrg   {
    261  1.1  mrg     if (!conn->send ('Q'))
    262  1.1  mrg       return FAIL;
    263  1.1  mrg     if (!marshall (conn, method))
    264  1.1  mrg       return FAIL;
    265  1.1  mrg     if (!marshall (conn, 0))
    266  1.1  mrg       return FAIL;
    267  1.1  mrg     if (!conn->wait_for_result ())
    268  1.1  mrg       return FAIL;
    269  1.1  mrg     if (!unmarshall (conn, result))
    270  1.1  mrg       return FAIL;
    271  1.1  mrg     return OK;
    272  1.1  mrg   }
    273  1.1  mrg 
    274  1.1  mrg   template<typename R, R (*func) (connection *)>
    275  1.1  mrg   status
    276  1.1  mrg   callback (connection *conn)
    277  1.1  mrg   {
    278  1.1  mrg     R result;
    279  1.1  mrg 
    280  1.1  mrg     if (!unmarshall_check (conn, 0))
    281  1.1  mrg       return FAIL;
    282  1.1  mrg     result = func (conn);
    283  1.1  mrg     if (!conn->send ('R'))
    284  1.1  mrg       return FAIL;
    285  1.1  mrg     return marshall (conn, result);
    286  1.1  mrg   }
    287  1.1  mrg 
    288  1.1  mrg   template<typename R, typename A>
    289  1.1  mrg   status
    290  1.1  mrg   call (connection *conn, const char *method, R *result, A arg)
    291  1.1  mrg   {
    292  1.1  mrg     if (!conn->send ('Q'))
    293  1.1  mrg       return FAIL;
    294  1.1  mrg     if (!marshall (conn, method))
    295  1.1  mrg       return FAIL;
    296  1.1  mrg     if (!marshall (conn, 1))
    297  1.1  mrg       return FAIL;
    298  1.1  mrg     if (!marshall (conn, arg))
    299  1.1  mrg       return FAIL;
    300  1.1  mrg     if (!conn->wait_for_result ())
    301  1.1  mrg       return FAIL;
    302  1.1  mrg     if (!unmarshall (conn, result))
    303  1.1  mrg       return FAIL;
    304  1.1  mrg     return OK;
    305  1.1  mrg   }
    306  1.1  mrg 
    307  1.1  mrg   template<typename R, typename A, R (*func) (connection *, A)>
    308  1.1  mrg   status
    309  1.1  mrg   callback (connection *conn)
    310  1.1  mrg   {
    311  1.1  mrg     argument_wrapper<A> arg;
    312  1.1  mrg     R result;
    313  1.1  mrg 
    314  1.1  mrg     if (!unmarshall_check (conn, 1))
    315  1.1  mrg       return FAIL;
    316  1.1  mrg     if (!arg.unmarshall (conn))
    317  1.1  mrg       return FAIL;
    318  1.1  mrg     result = func (conn, arg);
    319  1.1  mrg     if (!conn->send ('R'))
    320  1.1  mrg       return FAIL;
    321  1.1  mrg     return marshall (conn, result);
    322  1.1  mrg   }
    323  1.1  mrg 
    324  1.1  mrg   template<typename R, typename A1, typename A2>
    325  1.1  mrg   status
    326  1.1  mrg   call (connection *conn, const char *method, R *result, A1 arg1, A2 arg2)
    327  1.1  mrg   {
    328  1.1  mrg     if (!conn->send ('Q'))
    329  1.1  mrg       return FAIL;
    330  1.1  mrg     if (!marshall (conn, method))
    331  1.1  mrg       return FAIL;
    332  1.1  mrg     if (!marshall (conn, 2))
    333  1.1  mrg       return FAIL;
    334  1.1  mrg     if (!marshall (conn, arg1))
    335  1.1  mrg       return FAIL;
    336  1.1  mrg     if (!marshall (conn, arg2))
    337  1.1  mrg       return FAIL;
    338  1.1  mrg     if (!conn->wait_for_result ())
    339  1.1  mrg       return FAIL;
    340  1.1  mrg     if (!unmarshall (conn, result))
    341  1.1  mrg       return FAIL;
    342  1.1  mrg     return OK;
    343  1.1  mrg   }
    344  1.1  mrg 
    345  1.1  mrg   template<typename R, typename A1, typename A2, R (*func) (connection *,
    346  1.1  mrg 							    A1, A2)>
    347  1.1  mrg   status
    348  1.1  mrg   callback (connection *conn)
    349  1.1  mrg   {
    350  1.1  mrg     argument_wrapper<A1> arg1;
    351  1.1  mrg     argument_wrapper<A2> arg2;
    352  1.1  mrg     R result;
    353  1.1  mrg 
    354  1.1  mrg     if (!unmarshall_check (conn, 2))
    355  1.1  mrg       return FAIL;
    356  1.1  mrg     if (!arg1.unmarshall (conn))
    357  1.1  mrg       return FAIL;
    358  1.1  mrg     if (!arg2.unmarshall (conn))
    359  1.1  mrg       return FAIL;
    360  1.1  mrg     result = func (conn, arg1, arg2);
    361  1.1  mrg     if (!conn->send ('R'))
    362  1.1  mrg       return FAIL;
    363  1.1  mrg     return marshall (conn, result);
    364  1.1  mrg   }
    365  1.1  mrg 
    366  1.1  mrg   template<typename R, typename A1, typename A2, typename A3>
    367  1.1  mrg   status
    368  1.1  mrg   call (connection *conn, const char *method, R *result, A1 arg1, A2 arg2,
    369  1.1  mrg 	A3 arg3)
    370  1.1  mrg   {
    371  1.1  mrg     if (!conn->send ('Q'))
    372  1.1  mrg       return FAIL;
    373  1.1  mrg     if (!marshall (conn, method))
    374  1.1  mrg       return FAIL;
    375  1.1  mrg     if (!marshall (conn, 3))
    376  1.1  mrg       return FAIL;
    377  1.1  mrg     if (!marshall (conn, arg1))
    378  1.1  mrg       return FAIL;
    379  1.1  mrg     if (!marshall (conn, arg2))
    380  1.1  mrg       return FAIL;
    381  1.1  mrg     if (!marshall (conn, arg3))
    382  1.1  mrg       return FAIL;
    383  1.1  mrg     if (!conn->wait_for_result ())
    384  1.1  mrg       return FAIL;
    385  1.1  mrg     if (!unmarshall (conn, result))
    386  1.1  mrg       return FAIL;
    387  1.1  mrg     return OK;
    388  1.1  mrg   }
    389  1.1  mrg 
    390  1.1  mrg   template<typename R, typename A1, typename A2, typename A3,
    391  1.1  mrg 	   R (*func) (connection *, A1, A2, A3)>
    392  1.1  mrg   status
    393  1.1  mrg   callback (connection *conn)
    394  1.1  mrg   {
    395  1.1  mrg     argument_wrapper<A1> arg1;
    396  1.1  mrg     argument_wrapper<A2> arg2;
    397  1.1  mrg     argument_wrapper<A3> arg3;
    398  1.1  mrg     R result;
    399  1.1  mrg 
    400  1.1  mrg     if (!unmarshall_check (conn, 3))
    401  1.1  mrg       return FAIL;
    402  1.1  mrg     if (!arg1.unmarshall (conn))
    403  1.1  mrg       return FAIL;
    404  1.1  mrg     if (!arg2.unmarshall (conn))
    405  1.1  mrg       return FAIL;
    406  1.1  mrg     if (!arg3.unmarshall (conn))
    407  1.1  mrg       return FAIL;
    408  1.1  mrg     result = func (conn, arg1, arg2, arg3);
    409  1.1  mrg     if (!conn->send ('R'))
    410  1.1  mrg       return FAIL;
    411  1.1  mrg     return marshall (conn, result);
    412  1.1  mrg   }
    413  1.1  mrg 
    414  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4>
    415  1.1  mrg   status
    416  1.1  mrg   call (connection *conn, const char *method, R *result, A1 arg1, A2 arg2,
    417  1.1  mrg 	A3 arg3, A4 arg4)
    418  1.1  mrg   {
    419  1.1  mrg     if (!conn->send ('Q'))
    420  1.1  mrg       return FAIL;
    421  1.1  mrg     if (!marshall (conn, method))
    422  1.1  mrg       return FAIL;
    423  1.1  mrg     if (!marshall (conn, 4))
    424  1.1  mrg       return FAIL;
    425  1.1  mrg     if (!marshall (conn, arg1))
    426  1.1  mrg       return FAIL;
    427  1.1  mrg     if (!marshall (conn, arg2))
    428  1.1  mrg       return FAIL;
    429  1.1  mrg     if (!marshall (conn, arg3))
    430  1.1  mrg       return FAIL;
    431  1.1  mrg     if (!marshall (conn, arg4))
    432  1.1  mrg       return FAIL;
    433  1.1  mrg     if (!conn->wait_for_result ())
    434  1.1  mrg       return FAIL;
    435  1.1  mrg     if (!unmarshall (conn, result))
    436  1.1  mrg       return FAIL;
    437  1.1  mrg     return OK;
    438  1.1  mrg   }
    439  1.1  mrg 
    440  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4,
    441  1.1  mrg 	   R (*func) (connection *, A1, A2, A3, A4)>
    442  1.1  mrg   status
    443  1.1  mrg   callback (connection *conn)
    444  1.1  mrg   {
    445  1.1  mrg     argument_wrapper<A1> arg1;
    446  1.1  mrg     argument_wrapper<A2> arg2;
    447  1.1  mrg     argument_wrapper<A3> arg3;
    448  1.1  mrg     argument_wrapper<A4> arg4;
    449  1.1  mrg     R result;
    450  1.1  mrg 
    451  1.1  mrg     if (!unmarshall_check (conn, 4))
    452  1.1  mrg       return FAIL;
    453  1.1  mrg     if (!arg1.unmarshall (conn))
    454  1.1  mrg       return FAIL;
    455  1.1  mrg     if (!arg2.unmarshall (conn))
    456  1.1  mrg       return FAIL;
    457  1.1  mrg     if (!arg3.unmarshall (conn))
    458  1.1  mrg       return FAIL;
    459  1.1  mrg     if (!arg4.unmarshall (conn))
    460  1.1  mrg       return FAIL;
    461  1.1  mrg     result = func (conn, arg1, arg2, arg3, arg4);
    462  1.1  mrg     if (!conn->send ('R'))
    463  1.1  mrg       return FAIL;
    464  1.1  mrg     return marshall (conn, result);
    465  1.1  mrg   }
    466  1.1  mrg 
    467  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4,
    468  1.1  mrg 	   typename A5>
    469  1.1  mrg   status
    470  1.1  mrg   call (connection *conn, const char *method, R *result, A1 arg1, A2 arg2,
    471  1.1  mrg 	A3 arg3, A4 arg4, A5 arg5)
    472  1.1  mrg   {
    473  1.1  mrg     if (!conn->send ('Q'))
    474  1.1  mrg       return FAIL;
    475  1.1  mrg     if (!marshall (conn, method))
    476  1.1  mrg       return FAIL;
    477  1.1  mrg     if (!marshall (conn, 5))
    478  1.1  mrg       return FAIL;
    479  1.1  mrg     if (!marshall (conn, arg1))
    480  1.1  mrg       return FAIL;
    481  1.1  mrg     if (!marshall (conn, arg2))
    482  1.1  mrg       return FAIL;
    483  1.1  mrg     if (!marshall (conn, arg3))
    484  1.1  mrg       return FAIL;
    485  1.1  mrg     if (!marshall (conn, arg4))
    486  1.1  mrg       return FAIL;
    487  1.1  mrg     if (!marshall (conn, arg5))
    488  1.1  mrg       return FAIL;
    489  1.1  mrg     if (!conn->wait_for_result ())
    490  1.1  mrg       return FAIL;
    491  1.1  mrg     if (!unmarshall (conn, result))
    492  1.1  mrg       return FAIL;
    493  1.1  mrg     return OK;
    494  1.1  mrg   }
    495  1.1  mrg 
    496  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4,
    497  1.1  mrg 	   typename A5, R (*func) (connection *, A1, A2, A3, A4, A5)>
    498  1.1  mrg   status
    499  1.1  mrg   callback (connection *conn)
    500  1.1  mrg   {
    501  1.1  mrg     argument_wrapper<A1> arg1;
    502  1.1  mrg     argument_wrapper<A2> arg2;
    503  1.1  mrg     argument_wrapper<A3> arg3;
    504  1.1  mrg     argument_wrapper<A4> arg4;
    505  1.1  mrg     argument_wrapper<A5> arg5;
    506  1.1  mrg     R result;
    507  1.1  mrg 
    508  1.1  mrg     if (!unmarshall_check (conn, 5))
    509  1.1  mrg       return FAIL;
    510  1.1  mrg     if (!arg1.unmarshall (conn))
    511  1.1  mrg       return FAIL;
    512  1.1  mrg     if (!arg2.unmarshall (conn))
    513  1.1  mrg       return FAIL;
    514  1.1  mrg     if (!arg3.unmarshall (conn))
    515  1.1  mrg       return FAIL;
    516  1.1  mrg     if (!arg4.unmarshall (conn))
    517  1.1  mrg       return FAIL;
    518  1.1  mrg     if (!arg5.unmarshall (conn))
    519  1.1  mrg       return FAIL;
    520  1.1  mrg     result = func (conn, arg1, arg2, arg3, arg4, arg5);
    521  1.1  mrg     if (!conn->send ('R'))
    522  1.1  mrg       return FAIL;
    523  1.1  mrg     return marshall (conn, result);
    524  1.1  mrg   }
    525  1.1  mrg 
    526  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4,
    527  1.1  mrg 	   typename A5, typename A6, typename A7>
    528  1.1  mrg   status
    529  1.1  mrg   call (connection *conn, const char *method, R *result, A1 arg1, A2 arg2,
    530  1.1  mrg 	A3 arg3, A4 arg4, A5 arg5, A6 arg6, A7 arg7)
    531  1.1  mrg   {
    532  1.1  mrg     if (!conn->send ('Q'))
    533  1.1  mrg       return FAIL;
    534  1.1  mrg     if (!marshall (conn, method))
    535  1.1  mrg       return FAIL;
    536  1.1  mrg     if (!marshall (conn, 7))
    537  1.1  mrg       return FAIL;
    538  1.1  mrg     if (!marshall (conn, arg1))
    539  1.1  mrg       return FAIL;
    540  1.1  mrg     if (!marshall (conn, arg2))
    541  1.1  mrg       return FAIL;
    542  1.1  mrg     if (!marshall (conn, arg3))
    543  1.1  mrg       return FAIL;
    544  1.1  mrg     if (!marshall (conn, arg4))
    545  1.1  mrg       return FAIL;
    546  1.1  mrg     if (!marshall (conn, arg5))
    547  1.1  mrg       return FAIL;
    548  1.1  mrg     if (!marshall (conn, arg6))
    549  1.1  mrg       return FAIL;
    550  1.1  mrg     if (!marshall (conn, arg7))
    551  1.1  mrg       return FAIL;
    552  1.1  mrg     if (!conn->wait_for_result ())
    553  1.1  mrg       return FAIL;
    554  1.1  mrg     if (!unmarshall (conn, result))
    555  1.1  mrg       return FAIL;
    556  1.1  mrg     return OK;
    557  1.1  mrg   }
    558  1.1  mrg 
    559  1.1  mrg   template<typename R, typename A1, typename A2, typename A3, typename A4,
    560  1.1  mrg 	   typename A5, typename A6, typename A7,
    561  1.1  mrg 	   R (*func) (connection *, A1, A2, A3, A4, A5, A6, A7)>
    562  1.1  mrg   status
    563  1.1  mrg   callback (connection *conn)
    564  1.1  mrg   {
    565  1.1  mrg     argument_wrapper<A1> arg1;
    566  1.1  mrg     argument_wrapper<A2> arg2;
    567  1.1  mrg     argument_wrapper<A3> arg3;
    568  1.1  mrg     argument_wrapper<A4> arg4;
    569  1.1  mrg     argument_wrapper<A5> arg5;
    570  1.1  mrg     argument_wrapper<A6> arg6;
    571  1.1  mrg     argument_wrapper<A7> arg7;
    572  1.1  mrg     R result;
    573  1.1  mrg 
    574  1.1  mrg     if (!unmarshall_check (conn, 7))
    575  1.1  mrg       return FAIL;
    576  1.1  mrg     if (!arg1.unmarshall (conn))
    577  1.1  mrg       return FAIL;
    578  1.1  mrg     if (!arg2.unmarshall (conn))
    579  1.1  mrg       return FAIL;
    580  1.1  mrg     if (!arg3.unmarshall (conn))
    581  1.1  mrg       return FAIL;
    582  1.1  mrg     if (!arg4.unmarshall (conn))
    583  1.1  mrg       return FAIL;
    584  1.1  mrg     if (!arg5.unmarshall (conn))
    585  1.1  mrg       return FAIL;
    586  1.1  mrg     if (!arg6.unmarshall (conn))
    587  1.1  mrg       return FAIL;
    588  1.1  mrg     if (!arg7.unmarshall (conn))
    589  1.1  mrg       return FAIL;
    590  1.1  mrg     result = func (conn, arg1, arg2, arg3, arg4, arg5, arg6, arg7);
    591  1.1  mrg     if (!conn->send ('R'))
    592  1.1  mrg       return FAIL;
    593  1.1  mrg     return marshall (conn, result);
    594  1.1  mrg   }
    595  1.1  mrg };
    596  1.1  mrg 
    597  1.1  mrg #endif // CC1_PLUGIN_RPC_HH
    598