The second issue associated with handling combine messages is how to access the data area for subsequent message components.
For example, the writeblock() combine message format has an lseek() message first, followed by the write() message. This means that the data associated with the write() request is further in the received message buffer than would be the case for just a simple _IO_WRITE message:
This issue is easy to work around. There's a resource manager library function called resmgr_msgread() that knows how to get the data corresponding to the correct message component. Therefore, in the io_write handler, if you used resmgr_msgread() instead of MsgRead(), this would be transparent to you.
For reference, here's the source for resmgr_msgread():
int resmgr_msgread( resmgr_context_t *ctp, void *msg, int nbytes, int offset) { return MsgRead(ctp->rcvid, msg, nbytes, ctp->offset + offset); }
As you can see, resmgr_msgread() simply calls MsgRead() with the offset of the component message from the beginning of the combine message buffer. For completeness, there's also a resmgr_msgwrite() that works in an identical manner to MsgWrite(), except that it dereferences the passed ctp to obtain the rcvid.