mirror_corosync/lib/amf.c
Mark Haverkamp bf199836cf Move the closing of file descriptors from the destructor to
the finalize function.  This allows threads waiting on poll to
wake up and release handle references.

(Logical change 1.205)


git-svn-id: http://svn.fedorahosted.org/svn/corosync/trunk@662 fd59a12c-fef9-0310-b244-a6a79926bd2f
2005-06-15 21:04:28 +00:00

885 lines
25 KiB
C

/*
* Copyright (c) 2002-2003 MontaVista Software, Inc.
*
* All rights reserved.
*
* Author: Steven Dake (sdake@mvista.com)
*
* This software licensed under BSD license, the text of which follows:
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* - Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* - Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* - Neither the name of the MontaVista Software, Inc. nor the names of its
* contributors may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <pthread.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#include <sys/un.h>
#include "../include/ais_types.h"
#include "../include/ais_amf.h"
#include "../include/ais_msg.h"
#include "util.h"
struct res_overlay {
struct res_header header;
char data[4096];
};
/*
* Data structure for instance data
*/
struct amfInstance {
int response_fd;
int dispatch_fd;
SaAmfCallbacksT callbacks;
SaNameT compName;
int compRegistered;
int finalize;
pthread_mutex_t response_mutex;
pthread_mutex_t dispatch_mutex;
};
static void amfHandleInstanceDestructor (void *);
/*
* All instances in one database
*/
static struct saHandleDatabase amfHandleDatabase = {
.handleCount = 0,
.handles = 0,
.mutex = PTHREAD_MUTEX_INITIALIZER,
.handleInstanceDestructor = amfHandleInstanceDestructor
};
/*
* Versions supported
*/
static SaVersionT amfVersionsSupported[] = {
{ 'A', 1, 1 },
{ 'a', 1, 1 }
};
static struct saVersionDatabase amfVersionDatabase = {
sizeof (amfVersionsSupported) / sizeof (SaVersionT),
amfVersionsSupported
};
/*
* Implementation
*/
void amfHandleInstanceDestructor (void *instance)
{
}
SaErrorT
saAmfInitialize (
SaAmfHandleT *amfHandle,
const SaAmfCallbacksT *amfCallbacks,
const SaVersionT *version)
{
struct amfInstance *amfInstance;
SaErrorT error = SA_OK;
error = saVersionVerify (&amfVersionDatabase, version);
if (error != SA_OK) {
goto error_no_destroy;
}
error = saHandleCreate (&amfHandleDatabase, sizeof (struct amfInstance), amfHandle);
if (error != SA_OK) {
goto error_no_destroy;
}
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
goto error_destroy;
}
amfInstance->response_fd = -1;
amfInstance->dispatch_fd = -1;
error = saServiceConnectTwo (&amfInstance->response_fd,
&amfInstance->dispatch_fd, AMF_SERVICE);
if (error != SA_OK) {
goto error_put_destroy;
}
memcpy (&amfInstance->callbacks, amfCallbacks, sizeof (SaAmfCallbacksT));
pthread_mutex_init (&amfInstance->response_mutex, NULL);
pthread_mutex_init (&amfInstance->dispatch_mutex, NULL);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (SA_OK);
error_put_destroy:
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
error_destroy:
saHandleDestroy (&amfHandleDatabase, *amfHandle);
error_no_destroy:
return (error);
}
SaErrorT
saAmfSelectionObjectGet (
const SaAmfHandleT *amfHandle,
SaSelectionObjectT *selectionObject)
{
struct amfInstance *amfInstance;
SaErrorT error;
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
*selectionObject = amfInstance->dispatch_fd;
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (SA_OK);
}
SaErrorT
saAmfDispatch (
const SaAmfHandleT *amfHandle,
SaDispatchFlagsT dispatchFlags)
{
struct pollfd ufds;
int timeout = -1;
SaErrorT error;
int cont = 1; /* always continue do loop except when set to 0 */
int dispatch_avail;
struct amfInstance *amfInstance;
struct res_lib_amf_healthcheckcallback *res_lib_amf_healthcheckcallback;
struct res_lib_amf_readinessstatesetcallback *res_lib_amf_readinessstatesetcallback;
struct res_lib_amf_csisetcallback *res_lib_amf_csisetcallback;
struct res_lib_amf_csiremovecallback *res_lib_amf_csiremovecallback;
struct res_lib_amf_protectiongrouptrackcallback *res_lib_amf_protectiongrouptrackcallback;
SaAmfCallbacksT callbacks;
struct res_overlay dispatch_data;
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle,
(void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
/*
* Timeout instantly for SA_DISPATCH_ALL
*/
if (dispatchFlags == SA_DISPATCH_ALL) {
timeout = 0;
}
do {
/*
* Read data directly from socket
*/
ufds.fd = amfInstance->dispatch_fd;
ufds.events = POLLIN;
ufds.revents = 0;
error = saPollRetry (&ufds, 1, timeout);
if (error != SA_OK) {
goto error_nounlock;
}
pthread_mutex_lock (&amfInstance->dispatch_mutex);
error = saPollRetry (&ufds, 1, 0);
if (error != SA_OK) {
goto error_nounlock;
}
/*
* Handle has been finalized in another thread
*/
if (amfInstance->finalize == 1) {
error = SA_OK;
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
goto error_unlock;
}
dispatch_avail = ufds.revents & POLLIN;
if (dispatch_avail == 0 && dispatchFlags == SA_DISPATCH_ALL) {
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
break; /* exit do while cont is 1 loop */
} else
if (dispatch_avail == 0) {
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
continue; /* next poll */
}
if (ufds.revents & POLLIN) {
/*
* Queue empty, read response from socket
*/
error = saRecvRetry (amfInstance->dispatch_fd, &dispatch_data.header,
sizeof (struct res_header), MSG_WAITALL | MSG_NOSIGNAL);
if (error != SA_OK) {
goto error_unlock;
}
if (dispatch_data.header.size > sizeof (struct res_header)) {
error = saRecvRetry (amfInstance->dispatch_fd, &dispatch_data.data,
dispatch_data.header.size - sizeof (struct res_header),
MSG_WAITALL | MSG_NOSIGNAL);
if (error != SA_OK) {
goto error_unlock;
}
}
} else {
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
continue;
}
/*
* Make copy of callbacks, message data, unlock instance, and call callback
* A risk of this dispatch method is that the callback routines may
* operate at the same time that amfFinalize has been called in another thread.
*/
memcpy (&callbacks, &amfInstance->callbacks, sizeof (SaAmfCallbacksT));
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
/*
* Dispatch incoming response
*/
switch (dispatch_data.header.id) {
case MESSAGE_RES_AMF_HEALTHCHECKCALLBACK:
res_lib_amf_healthcheckcallback = (struct res_lib_amf_healthcheckcallback *)&dispatch_data;
callbacks.saAmfHealthcheckCallback (
res_lib_amf_healthcheckcallback->invocation,
&res_lib_amf_healthcheckcallback->compName,
res_lib_amf_healthcheckcallback->checkType);
break;
case MESSAGE_RES_AMF_READINESSSTATESETCALLBACK:
res_lib_amf_readinessstatesetcallback = (struct res_lib_amf_readinessstatesetcallback *)&dispatch_data;
callbacks.saAmfReadinessStateSetCallback (
res_lib_amf_readinessstatesetcallback->invocation,
&res_lib_amf_readinessstatesetcallback->compName,
res_lib_amf_readinessstatesetcallback->readinessState);
break;
case MESSAGE_RES_AMF_CSISETCALLBACK:
res_lib_amf_csisetcallback = (struct res_lib_amf_csisetcallback *)&dispatch_data;
callbacks.saAmfCSISetCallback (
res_lib_amf_csisetcallback->invocation,
&res_lib_amf_csisetcallback->compName,
&res_lib_amf_csisetcallback->csiName,
res_lib_amf_csisetcallback->csiFlags,
&res_lib_amf_csisetcallback->haState,
&res_lib_amf_csisetcallback->activeCompName,
res_lib_amf_csisetcallback->transitionDescriptor);
break;
case MESSAGE_RES_AMF_CSIREMOVECALLBACK:
res_lib_amf_csiremovecallback = (struct res_lib_amf_csiremovecallback *)&dispatch_data;
callbacks.saAmfCSIRemoveCallback (
res_lib_amf_csiremovecallback->invocation,
&res_lib_amf_csiremovecallback->compName,
&res_lib_amf_csiremovecallback->csiName,
&res_lib_amf_csiremovecallback->csiFlags);
break;
case MESSAGE_RES_AMF_PROTECTIONGROUPTRACKCALLBACK:
res_lib_amf_protectiongrouptrackcallback = (struct res_lib_amf_protectiongrouptrackcallback *)&dispatch_data;
memcpy (res_lib_amf_protectiongrouptrackcallback->notificationBufferAddress,
res_lib_amf_protectiongrouptrackcallback->notificationBuffer,
res_lib_amf_protectiongrouptrackcallback->numberOfItems * sizeof (SaAmfProtectionGroupNotificationT));
callbacks.saAmfProtectionGroupTrackCallback(
&res_lib_amf_protectiongrouptrackcallback->csiName,
res_lib_amf_protectiongrouptrackcallback->notificationBufferAddress,
res_lib_amf_protectiongrouptrackcallback->numberOfItems,
res_lib_amf_protectiongrouptrackcallback->numberOfMembers,
res_lib_amf_protectiongrouptrackcallback->error);
break;
default:
error = SA_ERR_LIBRARY;
goto error_nounlock;
break;
}
/*
* Determine if more messages should be processed
*/
switch (dispatchFlags) {
case SA_DISPATCH_ONE:
cont = 0;
break;
case SA_DISPATCH_ALL:
break;
case SA_DISPATCH_BLOCKING:
break;
}
} while (cont);
error_unlock:
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
error_nounlock:
return (error);
}
SaErrorT
saAmfFinalize (
const SaAmfHandleT *amfHandle)
{
struct amfInstance *amfInstance;
SaErrorT error;
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->dispatch_mutex);
pthread_mutex_lock (&amfInstance->response_mutex);
/*
* Another thread has already started finalizing
*/
if (amfInstance->finalize) {
pthread_mutex_unlock (&amfInstance->response_mutex);
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (SA_ERR_BAD_HANDLE);
}
amfInstance->finalize = 1;
pthread_mutex_unlock (&amfInstance->response_mutex);
pthread_mutex_unlock (&amfInstance->dispatch_mutex);
saHandleDestroy (&amfHandleDatabase, *amfHandle);
if (amfInstance->response_fd != -1) {
shutdown (amfInstance->response_fd, 0);
close (amfInstance->response_fd);
}
if (amfInstance->dispatch_fd != -1) {
shutdown (amfInstance->dispatch_fd, 0);
close (amfInstance->dispatch_fd);
}
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (error);
}
SaErrorT
saAmfComponentRegister (
const SaAmfHandleT *amfHandle,
const SaNameT *compName,
const SaNameT *proxyCompName)
{
struct amfInstance *amfInstance;
SaErrorT error;
struct req_lib_amf_componentregister req_lib_amf_componentregister;
struct res_lib_amf_componentregister res_lib_amf_componentregister;
req_lib_amf_componentregister.header.size = sizeof (struct req_lib_amf_componentregister);
req_lib_amf_componentregister.header.id = MESSAGE_REQ_AMF_COMPONENTREGISTER;
memcpy (&req_lib_amf_componentregister.compName, compName, sizeof (SaNameT));
if (proxyCompName) {
memcpy (&req_lib_amf_componentregister.proxyCompName, proxyCompName, sizeof (SaNameT));
} else {
memset (&req_lib_amf_componentregister.proxyCompName, 0, sizeof (SaNameT));
}
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->response_mutex);
error - saSendReceiveReply (amfInstance->response_fd, &req_lib_amf_componentregister,
sizeof (struct req_lib_amf_componentregister),
&res_lib_amf_componentregister,
sizeof (struct res_lib_amf_componentregister));
if (error != SA_OK) {
goto error_unlock;
}
if (res_lib_amf_componentregister.header.error == SA_OK) {
amfInstance->compRegistered = 1;
memcpy (&amfInstance->compName, compName, sizeof (SaNameT));
}
error = res_lib_amf_componentregister.header.error;
error_unlock:
pthread_mutex_unlock (&amfInstance->response_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (error);
}
SaErrorT
saAmfComponentUnregister (
const SaAmfHandleT *amfHandle,
const SaNameT *compName,
const SaNameT *proxyCompName)
{
struct req_lib_amf_componentunregister req_lib_amf_componentunregister;
struct res_lib_amf_componentunregister res_lib_amf_componentunregister;
struct amfInstance *amfInstance;
SaErrorT error;
req_lib_amf_componentunregister.header.size = sizeof (struct req_lib_amf_componentunregister);
req_lib_amf_componentunregister.header.id = MESSAGE_REQ_AMF_COMPONENTUNREGISTER;
memcpy (&req_lib_amf_componentunregister.compName, compName, sizeof (SaNameT));
if (proxyCompName) {
memcpy (&req_lib_amf_componentunregister.proxyCompName, proxyCompName, sizeof (SaNameT));
} else {
memset (&req_lib_amf_componentunregister.proxyCompName, 0, sizeof (SaNameT));
}
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->response_mutex);
error - saSendReceiveReply (amfInstance->response_fd,
&req_lib_amf_componentunregister,
sizeof (struct req_lib_amf_componentunregister),
&res_lib_amf_componentunregister,
sizeof (struct res_lib_amf_componentunregister));
if (error != SA_OK) {
goto error_unlock;
}
error = res_lib_amf_componentunregister.header.error;
if (error == SA_OK) {
amfInstance->compRegistered = 0;
}
error_unlock:
pthread_mutex_unlock (&amfInstance->response_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (error);
}
SaErrorT
saAmfCompNameGet (
const SaAmfHandleT *amfHandle,
SaNameT *compName)
{
struct amfInstance *amfInstance;
SaErrorT error;
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->response_mutex);
if (amfInstance->compRegistered == 0) {
pthread_mutex_unlock (&amfInstance->response_mutex);
return (SA_ERR_NOT_EXIST);
}
memcpy (compName, &amfInstance->compName, sizeof (SaNameT));
pthread_mutex_unlock (&amfInstance->response_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (SA_OK);
}
SaErrorT
saAmfReadinessStateGet (
const SaNameT *compName,
SaAmfReadinessStateT *readinessState)
{
int fd_response;
int fd_dispatch;
SaErrorT error;
struct req_amf_readinessstateget req_amf_readinessstateget;
struct res_lib_amf_readinessstateget res_lib_amf_readinessstateget;
error = saServiceConnectTwo (&fd_response, &fd_dispatch, AMF_SERVICE);
if (error != SA_OK) {
goto exit_noclose;
}
req_amf_readinessstateget.header.id = MESSAGE_RES_AMF_READINESSSTATEGET;
req_amf_readinessstateget.header.size = sizeof (struct req_amf_readinessstateget);
memcpy (&req_amf_readinessstateget.compName, compName, sizeof (SaNameT));
error - saSendReceiveReply (fd_response,
&req_amf_readinessstateget, sizeof (struct req_amf_readinessstateget),
&res_lib_amf_readinessstateget, sizeof (struct res_lib_amf_readinessstateget));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_readinessstateget.header.error;
if (error == SA_OK) {
memcpy (readinessState, &res_lib_amf_readinessstateget.readinessState,
sizeof (SaAmfReadinessStateT));
}
exit_close:
close (fd_response);
close (fd_dispatch);
exit_noclose:
return (error);
}
SaErrorT
saAmfStoppingComplete (
SaInvocationT invocation,
SaErrorT error)
{
struct req_amf_stoppingcomplete req_amf_stoppingcomplete;
struct res_lib_amf_stoppingcomplete res_lib_amf_stoppingcomplete;
int fd;
SaErrorT errorResult;
errorResult = saServiceConnect (&fd, MESSAGE_REQ_AMF_INIT);
if (errorResult != SA_OK) {
goto exit_noclose;
}
req_amf_stoppingcomplete.header.id = MESSAGE_REQ_AMF_STOPPINGCOMPLETE;
req_amf_stoppingcomplete.header.size = sizeof (struct req_amf_stoppingcomplete);
req_amf_stoppingcomplete.invocation = invocation;
req_amf_stoppingcomplete.error = error;
error = saSendReceiveReply (fd,
&req_amf_stoppingcomplete, sizeof (struct req_amf_stoppingcomplete),
&res_lib_amf_stoppingcomplete, sizeof (struct res_lib_amf_stoppingcomplete));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_stoppingcomplete.header.error;
// TODO executive needs to send reply of stopping complete
exit_close:
close (fd);
exit_noclose:
return (errorResult);
}
SaErrorT
saAmfHAStateGet (
const SaNameT *compName,
const SaNameT *csiName,
SaAmfHAStateT *haState) {
struct req_amf_hastateget req_amf_hastateget;
struct res_lib_amf_hastateget res_lib_amf_hastateget;
int fd;
SaErrorT error;
error = saServiceConnect (&fd, MESSAGE_REQ_AMF_INIT);
if (error != SA_OK) {
goto exit_noclose;
}
req_amf_hastateget.header.id = MESSAGE_REQ_AMF_HASTATEGET;
req_amf_hastateget.header.size = sizeof (struct req_amf_hastateget);
memcpy (&req_amf_hastateget.compName, compName, sizeof (SaNameT));
memcpy (&req_amf_hastateget.csiName, csiName, sizeof (SaNameT));
error = saSendReceiveReply (fd,
&req_amf_hastateget, sizeof (struct req_amf_hastateget),
&res_lib_amf_hastateget, sizeof (struct res_lib_amf_hastateget));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_hastateget.header.error;
if (error == SA_OK) {
memcpy (haState, &res_lib_amf_hastateget.haState, sizeof (SaAmfHAStateT));
}
exit_close:
close (fd);
exit_noclose:
return (error);
}
SaErrorT
saAmfProtectionGroupTrackStart (
const SaAmfHandleT *amfHandle,
const SaNameT *csiName,
SaUint8T trackFlags,
const SaAmfProtectionGroupNotificationT *notificationBuffer,
SaUint32T numberOfItems) {
struct amfInstance *amfInstance;
struct req_amf_protectiongrouptrackstart req_amf_protectiongrouptrackstart;
struct res_lib_amf_protectiongrouptrackstart res_lib_amf_protectiongrouptrackstart;
SaErrorT error;
req_amf_protectiongrouptrackstart.header.size = sizeof (struct req_amf_protectiongrouptrackstart);
req_amf_protectiongrouptrackstart.header.id = MESSAGE_REQ_AMF_PROTECTIONGROUPTRACKSTART;
memcpy (&req_amf_protectiongrouptrackstart.csiName, csiName, sizeof (SaNameT));
req_amf_protectiongrouptrackstart.trackFlags = trackFlags;
req_amf_protectiongrouptrackstart.notificationBufferAddress = (SaAmfProtectionGroupNotificationT *)notificationBuffer;
req_amf_protectiongrouptrackstart.numberOfItems = numberOfItems;
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->response_mutex);
error = saSendReceiveReply (amfInstance->response_fd,
&req_amf_protectiongrouptrackstart,
sizeof (struct req_amf_protectiongrouptrackstart),
&res_lib_amf_protectiongrouptrackstart,
sizeof (struct res_lib_amf_protectiongrouptrackstart));
if (error != SA_OK) {
goto error_unlock;
}
error = res_lib_amf_protectiongrouptrackstart.header.error;
error_unlock:
pthread_mutex_unlock (&amfInstance->response_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (error);
}
SaErrorT
saAmfProtectionGroupTrackStop (
const SaAmfHandleT *amfHandle,
const SaNameT *csiName) {
struct amfInstance *amfInstance;
struct req_amf_protectiongrouptrackstop req_amf_protectiongrouptrackstop;
struct res_lib_amf_protectiongrouptrackstop res_lib_amf_protectiongrouptrackstop;
SaErrorT error;
req_amf_protectiongrouptrackstop.header.size = sizeof (struct req_amf_protectiongrouptrackstop);
req_amf_protectiongrouptrackstop.header.id = MESSAGE_REQ_AMF_PROTECTIONGROUPTRACKSTOP;
memcpy (&req_amf_protectiongrouptrackstop.csiName, csiName, sizeof (SaNameT));
error = saHandleInstanceGet (&amfHandleDatabase, *amfHandle, (void *)&amfInstance);
if (error != SA_OK) {
return (error);
}
pthread_mutex_lock (&amfInstance->response_mutex);
error = saSendReceiveReply (amfInstance->response_fd,
&req_amf_protectiongrouptrackstop,
sizeof (struct req_amf_protectiongrouptrackstop),
&res_lib_amf_protectiongrouptrackstop,
sizeof (struct res_lib_amf_protectiongrouptrackstop));
if (error != SA_OK) {
goto error_unlock;
}
error = res_lib_amf_protectiongrouptrackstop.header.error;
error_unlock:
pthread_mutex_unlock (&amfInstance->response_mutex);
saHandleInstancePut (&amfHandleDatabase, *amfHandle);
return (error);
}
SaErrorT
saAmfErrorReport (
const SaNameT *reportingComponent,
const SaNameT *erroneousComponent,
SaTimeT errorDetectionTime,
const SaAmfErrorDescriptorT *errorDescriptor,
const SaAmfAdditionalDataT *additionalData) {
struct req_lib_amf_errorreport req_lib_amf_errorreport;
struct res_lib_amf_errorreport res_lib_amf_errorreport;
int fd;
SaErrorT error;
error = saServiceConnect (&fd, MESSAGE_REQ_AMF_INIT);
if (error != SA_OK) {
goto exit_noclose;
}
req_lib_amf_errorreport.header.id = MESSAGE_REQ_AMF_ERRORREPORT;
req_lib_amf_errorreport.header.size = sizeof (struct req_lib_amf_errorreport);
memcpy (&req_lib_amf_errorreport.reportingComponent, reportingComponent, sizeof (SaNameT));
memcpy (&req_lib_amf_errorreport.erroneousComponent, erroneousComponent, sizeof (SaNameT));
req_lib_amf_errorreport.errorDetectionTime = errorDetectionTime;
memcpy (&req_lib_amf_errorreport.errorDescriptor,
errorDescriptor, sizeof (SaAmfErrorDescriptorT));
/* TODO this is wrong, and needs some thinking
memcpy (&req_lib_amf_errorreport.additionalData,
additionalData, sizeof (SaAmfAdditionalDataT));
*/
error = saSendReceiveReply (fd,
&req_lib_amf_errorreport, sizeof (struct req_lib_amf_errorreport),
&res_lib_amf_errorreport, sizeof (struct res_lib_amf_errorreport));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_errorreport.header.error;
exit_close:
close (fd);
exit_noclose:
return (error);
}
SaErrorT
saAmfErrorCancelAll (
const SaNameT *compName) {
struct req_lib_amf_errorcancelall req_lib_amf_errorcancelall;
struct res_lib_amf_errorcancelall res_lib_amf_errorcancelall;
int fd;
SaErrorT error;
error = saServiceConnect (&fd, MESSAGE_REQ_AMF_INIT);
if (error != SA_OK) {
goto exit_noclose;
}
req_lib_amf_errorcancelall.header.id = MESSAGE_REQ_AMF_ERRORCANCELALL;
req_lib_amf_errorcancelall.header.size = sizeof (struct req_lib_amf_errorcancelall);
memcpy (&req_lib_amf_errorcancelall.compName, compName, sizeof (SaNameT));
error = saSendReceiveReply (fd,
&req_lib_amf_errorcancelall, sizeof (struct req_lib_amf_errorcancelall),
&res_lib_amf_errorcancelall, sizeof (struct res_lib_amf_errorcancelall));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_errorcancelall.header.error;
exit_close:
close (fd);
exit_noclose:
return (error);
}
SaErrorT
saAmfComponentCapabilityModelGet (
const SaNameT *compName,
SaAmfComponentCapabilityModelT *componentCapabilityModel)
{
int fd;
SaErrorT error;
struct req_amf_componentcapabilitymodelget req_amf_componentcapabilitymodelget;
struct res_lib_amf_componentcapabilitymodelget res_lib_amf_componentcapabilitymodelget;
error = saServiceConnect (&fd, MESSAGE_REQ_AMF_INIT);
if (error != SA_OK) {
goto exit_noclose;
}
req_amf_componentcapabilitymodelget.header.id = MESSAGE_REQ_AMF_COMPONENTCAPABILITYMODELGET;
req_amf_componentcapabilitymodelget.header.size = sizeof (struct req_amf_componentcapabilitymodelget);
memcpy (&req_amf_componentcapabilitymodelget.compName, compName, sizeof (SaNameT));
error = saSendReceiveReply (fd,
&req_amf_componentcapabilitymodelget,
sizeof (struct req_amf_componentcapabilitymodelget),
&res_lib_amf_componentcapabilitymodelget,
sizeof (struct res_lib_amf_componentcapabilitymodelget));
if (error != SA_OK) {
goto exit_close;
}
error = res_lib_amf_componentcapabilitymodelget.header.error;
if (error == SA_OK) {
memcpy (componentCapabilityModel,
&res_lib_amf_componentcapabilitymodelget.componentCapabilityModel,
sizeof (SaAmfComponentCapabilityModelT));
}
exit_close:
close (fd);
exit_noclose:
return (error);
}
SaErrorT
saAmfPendingOperationGet (
const SaNameT *compName,
SaAmfPendingOperationFlagsT *pendingOperationFlags) {
*pendingOperationFlags = 0;
return (SA_OK);
}
SaErrorT
saAmfResponse (
SaInvocationT invocation,
SaErrorT error)
{
struct req_amf_response req_amf_response;
struct res_lib_amf_response res_lib_amf_response;
int fd_response;
int fd_dispatch;
SaErrorT errorResult;
errorResult = saServiceConnectTwo (&fd_response, &fd_dispatch, AMF_SERVICE);
if (errorResult != SA_OK) {
goto exit_noclose;
}
req_amf_response.header.id = MESSAGE_REQ_AMF_RESPONSE;
req_amf_response.header.size = sizeof (struct req_amf_response);
req_amf_response.invocation = invocation;
req_amf_response.error = error;
errorResult = saSendReceiveReply (fd_response,
&req_amf_response,
sizeof (struct req_amf_response),
&res_lib_amf_response,
sizeof (struct res_lib_amf_response));
close (fd_response);
close (fd_dispatch);
if (errorResult == SA_OK) {
errorResult = res_lib_amf_response.header.error;
}
exit_noclose:
return (errorResult);
}