warsaw@nlm.nih.gov (Barry A. Warsaw) (05/16/91)
Has anyone out there tried to write some RPC code using Sun C++ 2.0? I've found quite a few problems with some of the <rpc/*.h> header files that come in the C++ distribution. In particular, there are bogosities in <rpc/pmap_clnt.h>, <rpc/clnt.h>, and <rpc/svc.h>. The first two I've been able to patch, but I haven't been able to get a corrected <rpc/svc.h> file to work yet. Note that I'm new to RPC (but not C++) so I could be doing some things wrong. Also, even though C++ comes with a version of rpcgen, I've found that I cannot use the generated files directly, except for the generated <protocol>.h file. And the C++ rpcgen takes entirely different switches than "regular" rpcgen. It would be nice if rpcgen(++) where just a superset of rpcgen! Does anybody else have similar experiences? Also, does anybody know if these files have been fixed for 2.1, and if Sun C++ 2.1 FCS is shipping? Thanks very much. -Barry NAME: Barry A. Warsaw USMAIL: National Library of Medicine TELE: (301) 496-1936 Lister Hill National Center for INET: warsaw@nlm.nih.gov Biomedical Communications UUCP: uunet!nlm.nih.gov!warsaw Information Technology Branch 8600 Rockville Pike Bldg. 38A, Rm. 7s722 Bethesda, Md. 20894
bergquis@nms.gdc.portal.com (Brett Bergquist) (05/20/91)
I to have been trying to get Sun C++ 2.0 to work with RPC. I found the same faults in the header files that you did. I took a different approach by compiling the _svc.c, _xdr.c, and _clnt.c using the C compiler. Somewhere in the manual for C++, it mentioned that you probably want to compile these files with the C compile so that you do not get a lot of warnings. I've noticed that rpcgen++ is not the same as the version that came with with system. In fact, its a revision or so behind. I also found that it has a major bug in that if I retype a basic type, it generates the wrong code. For example: typedef long Mtime; produces typedef u_long *Mtime; For me this was a show stopper. In order to get C++ compatible headers, I fetched a copy of version 3.9 of rpcgen from the net, and modified it myself to support C++. In fact this worked better, because I also modified it to support my error handling mechanisms, coding standards, etc. I've seen a copy of C++ 2.1 and rpcgen++ has been updated to the latest version, without TLI I believe. My Sun salesman said that it has officially been released and is shipping, but I am still waiting for our systems administrator to get it. -- Brett M. Bergquist, Principal Engineer | "Remind me" ... "to write an General DataComm, Inc., | "article on the compulsive reading Middlebury, CT 06762 | of news." - Stranger in a Strange Land Email: bergquis@nms.gdc.portal.com or ...!portal!gdc!nms!bergquis
warsaw@nlm.nih.gov (Barry A. Warsaw) (05/23/91)
>>>>> "Brett" == Brett Bergquist <bergquis@nms.gdc.portal.com> writes: Brett> I to have been trying to get Sun C++ 2.0 to work with RPC. Brett> I found the same faults in the header files that you did. Brett> I took a different approach by compiling the _svc.c, Brett> _xdr.c, and _clnt.c using the C compiler. Somewhere in the Brett> manual for C++, it mentioned that you probably want to Brett> compile these files with the C compile so that you do not Brett> get a lot of warnings. Well, after a bit of hacking (at patching the Sun C++ 2.0 header files), I've finally got a couple of classes implementing RPC. I took the files generated by rpcgen (not rpcgen++) as a template and wrapped it all up into a `server' and `client' class. One limitation is that I only implemented tcp since that's what I'm using, but there's nothing to prevent you from extending/modifying the class internals. Below are the public interfaces to the two classes. You'll still need rpcgen to create the _xdr.c file and the <protocol>.h file if you don't do it by hand (which really is almost as easy, except there's more to maintain when the protocol changes). If you're interested in more details send email. `server' usage requires you to derive your own class from server and make member functions of this class take two char*& arguments. These member functions will be the callback associated with RPC procedure numbers. The two char*& arguments can be cast to pointers to the appropriate data type within the callback function. Allocation and de-allocation of memory, serializing and deserializing XDR streams, etc, are all handled internally, as well as registration with portmapper and other bookkeeping. `client' usage more straightforward. You just give it a server hostname to connect to and then remotely execute procedures. You give rexec the size of the receiving data type (out) so that it is zeroed out before the deserialized data is placed there. -Barry NAME: Barry A. Warsaw USMAIL: National Library of Medicine TELE: (301) 496-1936 Lister Hill National Center for INET: warsaw@nlm.nih.gov Biomedical Communications UUCP: uunet!nlm.nih.gov!warsaw Information Technology Branch 8600 Rockville Pike Bldg. 38A, Rm. 7s722 Bethesda, Md. 20894 ==================== server class ==================== typedef void (server::*rpc_handler)( char*& in, char*& out ); class server { public: // it is an error to instantiate more than 1 instance of the // server class. Note that there is no destructor for this // class since (due to the nature of the run method), it will // never be destroyed until the entire process dies. server( u_long prognum, u_long versnum ); // register an RPC procedure with the appropriate callback // function which can handle this procedure call. void regproc( u_long procedure, rpc_handler callback, xdrproc_t in_converter, int indata_size, xdrproc_t out_converter, int outdata_size ); // once all callbacks are registered, the application can call // the run method below to start up the RPC server. it is an // error for run() to ever return so if a callback is used to // terminate the server, it should call exit() inline void run( void ); // returns non-zero if the application was started up by the // inet daemon inline int inetd_p( void ); protected: // this function actually does the dispatching to the member // function that can handle the remote procedure call. it is // not directly called by the RPC level code. virtual void dispatch( svc_req* request, SVCXPRT* transp ); }; ==================== client class interface ==================== static timeval _DEFAULT_CLIENT_TIMEOUT = { 25, 0 }; class client { public: inline client( string server, u_long prognum, u_long versnum, timeval timeout = _DEFAULT_CLIENT_TIMEOUT ); inline ~client( void ); // execute remote procedure on server, returns zero on // success, non-zero on failure. inline int rexec( u_long procedure, xdrproc_t in_converter, char*& in, xdrproc_t out_converter, char*& out, int sizeof_outdata ); // get and set the timeout value inline void timeout( timeval& new_timeout ); inline timeval timeout( void ); };