To keep things clear, I've avoided talking about how you'd use message passing over a
network, even though this is a crucial part of QNX Neutrino's flexibility!
Everything you've learned so far applies to message passing over a network.
Earlier in this chapter, I showed you an example:
#include <fcntl.h>
#include <unistd.h>
int
main (void)
{
int fd;
fd = open ("/net/wintermute/home/rk/filename", O_WRONLY);
write (fd, "This is message passing\n", 24);
close (fd);
return (EXIT_SUCCESS);
}
At the time, I said that this was an example of using message passing
over a network.
The client creates a connection to a ND/PID/CHID (which just happens to be
on a different node), and the server performs a MsgReceive()
on its channel.
The client and server are identical in this case to the local, single-node case.
You could stop reading right here—there really isn't anything
tricky about message passing over a network.
But for those readers who are curious about the how of this, read on!
Now that we've seen some of the details of local message passing, we can
discuss in a little more depth how message passing over a network works.
While this discussion may seem complicated, it really boils down to two
phases: name resolution, and once that's been taken care of, simple message passing.
Here's a diagram that illustrates the steps we'll be talking about:
Figure 1. Message passing over a network. Notice that Qnet is divided into two sections.
In the diagram, our node is called magenta,
and, as implied by the example, the target node is called wintermute.
Let's analyze the interactions that occur when a client program
uses Qnet to access a server over a network:
- The client's open() function was told to open a filename
that happened to have /net in front of it.
(The name /net is the default name manifested by Qnet.)
This client has no idea who is responsible for that particular pathname,
so it connects to the process manager (step 1) in order to find out who actually
owns the resource.
This is done regardless of whether we're passing messages
over a network and happens automatically.
Since the native QNX Neutrino network manager, Qnet, owns
all pathnames that begin with /net, the process manager
returns information to the client telling it to ask Qnet
about the pathname.
- The client now sends a message to Qnet's resource manager thread,
hoping that Qnet will be able to handle the request.
However, Qnet on this node isn't responsible for providing the
ultimate service that the client wants, so it tells the client that it should
actually contact the process manager on node wintermute.
(The way this is done is via a redirect response, which gives
the client the ND/PID/CHID of a server that it should contact instead.)
This redirect response is also handled automatically by the client's library.
- The client now connects to the process manager on wintermute.
This involves sending an off-node message through Qnet's network-handler thread.
The Qnet process on the client's node gets the message and transports it over the
medium to the remote Qnet, which delivers it to the process
manager on wintermute.
The process manager there resolves the rest of the pathname (in our example, that
would be the /home/rk/filename part) and sends a
redirect message back.
This redirect message follows the reverse path (from the server's Qnet
over the medium to the Qnet on the client's node, and finally back to
the client).
This redirect message now contains the location of the server that the client
wanted to contact in the first place, that is, the ND/PID/CHID of the server that's
going to service the client's requests.
(In our example, the server was a filesystem.)
- The client now sends the request to that server. The path followed here
is identical to the path followed in step 3 above, except that the server is
contacted directly instead of going through the process manager.
Once steps 1 through 3 have been established, step 4 is the model for all future communications.
In our client example above, the open(), read(), and close()
messages all take path number 4.
Note that the client's open() is what triggered this sequence of events to happen
in the first place—but the actual open message flows as described (through
path number 4).
Note:
For the really interested reader: I've left out one step.
During step 2, when the client asks Qnet about wintermute,
Qnet needs to figure out who wintermute is.
This may result in Qnet performing one more network transaction to resolve
the nodename.
The diagram presented above is correct if we assume that Qnet already
knew about wintermute.
We'll come back to the messages used for the open(), read(),
and close() (and others) in the Resource Managers chapter.