The message structures passed to the io_read, io_write, and io_openfd handlers contain a member called xtype. From struct _io_read:
struct _io_read {
    ...
    uint32_t            xtype;
    ...
}
Basically, the xtype contains extended information that can be used to adjust the behavior of a standard I/O function. This information includes a type and optionally some flags. To isolate the type from the flags, AND the xtype member with _IO_XTYPE_MASK:
if ((msg->i.xtype & _IO_XTYPE_MASK) == _IO_XTYPE_OFFSET) {
   ...
}
Most resource managers care about only a few types:
For example:
struct myread_offset {
    struct _io_read        read;
    struct _xtype_offset   offset;
}   
  Some resource managers can be sure that their clients will never call pread*() or pwrite*(). (For example, a resource manager that's controlling a robot arm probably wouldn't care.) In this case, you can treat this type of message as an error.
struct myreadcond {
    struct _io_read        read;
    struct _xtype_readcond cond;
}   
  As with _IO_XTYPE_OFFSET, if your resource manager isn't prepared to handle readcond(), you can treat this type of message as an error.
The following types are used for special purposes, so your resource manager probably doesn't need to handle them:
The xtype member may also include some flags. Your resource manager might be interested in the following:
The other defined flags are used for special purposes, so your resource manager probably doesn't need to handle them: