blob: 7e4e0c2f68bf4c319c0749fd7ed650fe8fd6e2f3 [file] [log] [blame]
/*
Copyright (c) 2012 250bpm s.r.o.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom
the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
IN THE SOFTWARE.
*/
#include "../pattern.h"
#include "../transport.h"
#include "sock.h"
#include "../utils/err.h"
#include "../utils/cont.h"
#define SP_SOCK_EVENT_IN 1
#define SP_SOCK_EVENT_OUT 2
void sp_sockbase_init (struct sp_sockbase *self,
const struct sp_sockbase_vfptr *vfptr, int fd)
{
self->vfptr = vfptr;
self->zombie = 0;
sp_cp_init (&self->cp);
sp_cond_init (&self->cond);
self->fd = fd;
}
void sp_sock_zombify (struct sp_sock *self)
{
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
sp_cp_lock (&sockbase->cp);
sockbase->zombie = 1;
sp_cond_post (&sockbase->cond);
sp_cp_unlock (&sockbase->cp);
}
void sp_sock_term (struct sp_sock *self)
{
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
/* Terminate the derived class. */
sockbase->vfptr->term (sockbase);
/* Terminate the sp_sockbase itself. */
sp_cond_term (&sockbase->cond);
sp_cp_term (&sockbase->cp);
}
void sp_sockbase_unblock_recv (struct sp_sockbase *self)
{
sp_cond_post (&self->cond);
}
void sp_sockbase_unblock_send (struct sp_sockbase *self)
{
sp_cond_post (&self->cond);
}
struct sp_cp *sp_sockbase_getcp (struct sp_sockbase *self)
{
return &self->cp;
}
struct sp_cp *sp_sock_getcp (struct sp_sock *self)
{
return &((struct sp_sockbase*) self)->cp;
}
int sp_sock_setopt (struct sp_sock *self, int level, int option,
const void *optval, size_t optvallen)
{
int rc;
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
sp_cp_lock (&sockbase->cp);
/* If sp_term() was already called, return ETERM. */
if (sp_slow (sockbase->zombie)) {
sp_cp_unlock (&sockbase->cp);
return -ETERM;
}
/* Generic socket-level options. */
if (level == SP_SOL_SOCKET) {
sp_cp_unlock (&sockbase->cp);
return -ENOPROTOOPT;
}
/* Pattern-specific socket options. */
if (level > SP_SOL_SOCKET) {
rc = sockbase->vfptr->setopt (sockbase, level, option,
optval, optvallen);
sp_cp_unlock (&sockbase->cp);
return rc;
}
/* Transport-specific options. */
if (level < SP_SOL_SOCKET) {
sp_cp_unlock (&sockbase->cp);
return -ENOPROTOOPT;
}
sp_assert (0);
}
int sp_sock_getopt (struct sp_sock *self, int level, int option,
void *optval, size_t *optvallen)
{
int rc;
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
sp_cp_lock (&sockbase->cp);
/* If sp_term() was already called, return ETERM. */
if (sp_slow (sockbase->zombie)) {
sp_cp_unlock (&sockbase->cp);
return -ETERM;
}
/* Generic socket-level options. */
if (level == SP_SOL_SOCKET) {
sp_cp_unlock (&sockbase->cp);
return -ENOPROTOOPT;
}
/* Pattern-specific socket options. */
if (level > SP_SOL_SOCKET) {
rc = sockbase->vfptr->getopt (sockbase, level, option,
optval, optvallen);
sp_cp_unlock (&sockbase->cp);
return rc;
}
/* Transport-specific options. */
if (level < SP_SOL_SOCKET) {
sp_cp_unlock (&sockbase->cp);
return -ENOPROTOOPT;
}
sp_assert (0);
}
int sp_sock_send (struct sp_sock *self, const void *buf, size_t len, int flags)
{
int rc;
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
sp_cp_lock (&sockbase->cp);
while (1) {
/* If sp_term() was already called, return ETERM. */
if (sp_slow (sockbase->zombie)) {
sp_cp_unlock (&sockbase->cp);
return -ETERM;
}
/* Try to send the message in a non-blocking way. */
rc = sockbase->vfptr->send (sockbase, buf, len);
if (sp_fast (rc == 0)) {
sp_cp_unlock (&sockbase->cp);
return 0;
}
/* Any unexpected error is forwarded to the caller. */
if (sp_slow (rc != -EAGAIN)) {
sp_cp_unlock (&sockbase->cp);
return rc;
}
/* If the message cannot be sent at the moment and the send call
is non-blocking, return immediately. */
if (sp_fast (flags & SP_DONTWAIT)) {
sp_cp_unlock (&sockbase->cp);
return -EAGAIN;
}
/* With blocking send, wait while there are new pipes available
for sending. */
rc = sp_cond_wait (&sockbase->cond, &sockbase->cp.sync, -1);
errnum_assert (rc == 0, rc);
}
}
int sp_sock_recv (struct sp_sock *self, void *buf, size_t *len, int flags)
{
int rc;
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
sp_cp_lock (&sockbase->cp);
while (1) {
/* If sp_term() was already called, return ETERM. */
if (sp_slow (sockbase->zombie)) {
sp_cp_unlock (&sockbase->cp);
return -ETERM;
}
/* Try to receive the message in a non-blocking way. */
rc = sockbase->vfptr->recv (sockbase, buf, len);
if (sp_fast (rc == 0)) {
sp_cp_unlock (&sockbase->cp);
return 0;
}
/* Any unexpected error is forwarded to the caller. */
if (sp_slow (rc != -EAGAIN)) {
sp_cp_unlock (&sockbase->cp);
return rc;
}
/* If the message cannot be received at the moment and the recv call
is non-blocking, return immediately. */
if (sp_fast (flags & SP_DONTWAIT)) {
sp_cp_unlock (&sockbase->cp);
return -EAGAIN;
}
/* With blocking recv, wait while there are new pipes available
for receiving. */
rc = sp_cond_wait (&sockbase->cond, &sockbase->cp.sync, -1);
errnum_assert (rc == 0, rc);
}
}
int sp_sock_fd (struct sp_sock *self)
{
struct sp_sockbase *sockbase;
sockbase = (struct sp_sockbase*) self;
return sockbase->fd;
}
int sp_sock_add (struct sp_sock *self, struct sp_pipe *pipe)
{
struct sp_sockbase *sockbase;
/* Forward the call to the specific socket type. */
sockbase = (struct sp_sockbase*) self;
return sockbase->vfptr->add (sockbase, pipe);
}
void sp_sock_rm (struct sp_sock *self, struct sp_pipe *pipe)
{
struct sp_sockbase *sockbase;
/* Forward the call to the specific socket type. */
sockbase = (struct sp_sockbase*) self;
sockbase->vfptr->rm (sockbase, pipe);
}
void sp_sock_in (struct sp_sock *self, struct sp_pipe *pipe)
{
int rc;
struct sp_sockbase *sockbase;
/* Forward the call to the specific socket type. */
sockbase = (struct sp_sockbase*) self;
rc = sockbase->vfptr->in (sockbase, pipe);
errnum_assert (rc >= 0, -rc);
if (rc == 1)
sp_cond_post (&sockbase->cond);
}
void sp_sock_out (struct sp_sock *self, struct sp_pipe *pipe)
{
int rc;
struct sp_sockbase *sockbase;
/* Forward the call to the specific socket type. */
sockbase = (struct sp_sockbase*) self;
rc = sockbase->vfptr->out (sockbase, pipe);
errnum_assert (rc >= 0, -rc);
if (rc == 1)
sp_cond_post (&sockbase->cond);
}