mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-06 23:45:35 +00:00
7b6023d9cf
Adding cudaipc{src,sink} element for CUDA IPC support. Implementation note: * For the communication between end points, Win32 named-pipe and unix domain socket will be used on Windows and Linux respectively. * cudaipcsink behaves as a server, and all GPU resources will be owned by the server process and exported for other processes, then cudaipcsrc (client) will import each exported handle. * User can select IPC mode via "ipc-mode" property of cudaipcsink. There are two IPC mode, one is "legacy" which uses legacy CUDA IPC method and the other is "mmap" which uses CUDA virtual memory API with OS's resource handle sharing method such as DuplicateHandle() on Windows. The "mmap" mode might be better than "legacy" in terms of stability since it relies on OS's resource management but it would consume more GPU memory than "legacy" mode. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4510>
554 lines
14 KiB
C++
554 lines
14 KiB
C++
/* GStreamer
|
|
* Copyright (C) 2023 Seungha Yang <seungha@centricular.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "gstcudaipc.h"
|
|
#include <gst/cuda/gstcuda-private.h>
|
|
#include <string.h>
|
|
#include <cctype>
|
|
#include <locale>
|
|
#include <codecvt>
|
|
#include <algorithm>
|
|
#include <mutex>
|
|
|
|
#ifdef G_OS_WIN32
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#define GST_CUDA_IPC_MAGIC_NUMBER 0xC0DA10C0
|
|
|
|
constexpr guint GST_CUDA_IPC_PKT_HAVE_DATA_PAYLOAD_MIN_SIZE =
|
|
sizeof (GstClockTime) + sizeof (CUipcMemHandle) +
|
|
sizeof (GstCudaIpcMemLayout) + sizeof (guint8);
|
|
constexpr guint GST_CUDA_IPC_PKT_RELEASE_DATA_PAYLOAD_SIZE =
|
|
sizeof (CUipcMemHandle);
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_identify (std::vector < guint8 > &buf,
|
|
GstCudaIpcPacketHeader & header)
|
|
{
|
|
g_return_val_if_fail (buf.size () >= GST_CUDA_IPC_PKT_HEADER_SIZE, false);
|
|
|
|
memcpy (&header, &buf[0], GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
if (header.magic != GST_CUDA_IPC_MAGIC_NUMBER)
|
|
return false;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE + header.payload_size);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_build_config (std::vector < guint8 > &buf,
|
|
GstCudaPid pid, gboolean use_mmap, GstCaps * caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
guint8 *ptr;
|
|
gchar *caps_str = nullptr;
|
|
guint caps_size = 0;
|
|
|
|
g_return_val_if_fail (GST_IS_CAPS (caps), false);
|
|
|
|
caps_str = gst_caps_serialize (caps, GST_SERIALIZE_FLAG_NONE);
|
|
if (!caps_str)
|
|
return false;
|
|
|
|
caps_size = strlen (caps_str) + 1;
|
|
|
|
header.type = GstCudaIpcPktType::CONFIG;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = sizeof (GstCudaPid) + sizeof (gboolean) + caps_size;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE + header.payload_size);
|
|
|
|
ptr = &buf[0];
|
|
|
|
memcpy (ptr, &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
*((GstCudaPid *) ptr) = pid;
|
|
ptr += sizeof (GstCudaPid);
|
|
|
|
*((gboolean *) ptr) = use_mmap;
|
|
ptr += sizeof (gboolean);
|
|
|
|
strcpy ((char *) ptr, caps_str);
|
|
g_free (caps_str);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_parse_config (std::vector < guint8 > &buf, GstCudaPid * pid,
|
|
gboolean * use_mmap, GstCaps ** caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
const guint8 *ptr;
|
|
std::string str;
|
|
|
|
g_return_val_if_fail (buf.size () > GST_CUDA_IPC_PKT_HEADER_SIZE, false);
|
|
g_return_val_if_fail (caps, false);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (&header, ptr, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
if (header.type != GstCudaIpcPktType::CONFIG ||
|
|
header.magic != GST_CUDA_IPC_MAGIC_NUMBER ||
|
|
header.payload_size <= sizeof (GstCudaPid) + sizeof (gboolean)) {
|
|
return false;
|
|
}
|
|
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (pid, ptr, sizeof (GstCudaPid));
|
|
ptr += sizeof (GstCudaPid);
|
|
|
|
memcpy (use_mmap, ptr, sizeof (gboolean));
|
|
ptr += sizeof (gboolean);
|
|
|
|
*caps = gst_caps_from_string ((const gchar *) ptr);
|
|
if (*caps == nullptr)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_need_data (std::vector < guint8 > &buf)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
|
|
header.type = GstCudaIpcPktType::NEED_DATA;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = 0;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
memcpy (&buf[0], &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
bool
|
|
gst_cuda_ipc_pkt_build_have_data (std::vector < guint8 > &buf, GstClockTime pts,
|
|
const GstVideoInfo & info, const CUipcMemHandle & handle, GstCaps * caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
GstCudaIpcMemLayout layout;
|
|
guint8 *ptr;
|
|
gchar *caps_str = nullptr;
|
|
guint caps_size = 1;
|
|
|
|
if (caps) {
|
|
caps_str = gst_caps_serialize (caps, GST_SERIALIZE_FLAG_NONE);
|
|
if (!caps_str)
|
|
return false;
|
|
|
|
caps_size += strlen (caps_str) + 1;
|
|
}
|
|
|
|
header.type = GstCudaIpcPktType::HAVE_DATA;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = sizeof (GstClockTime) + sizeof (CUipcMemHandle) +
|
|
sizeof (GstCudaIpcMemLayout) + caps_size;
|
|
|
|
layout.size = layout.max_size = info.size;
|
|
layout.pitch = info.stride[0];
|
|
for (guint i = 0; i < 4; i++)
|
|
layout.offset[i] = info.offset[i];
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE + header.payload_size);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (ptr, &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (ptr, &pts, sizeof (GstClockTime));
|
|
ptr += sizeof (GstClockTime);
|
|
|
|
memcpy (ptr, &layout, sizeof (GstCudaIpcMemLayout));
|
|
ptr += sizeof (GstCudaIpcMemLayout);
|
|
|
|
memcpy (ptr, &handle, sizeof (CUipcMemHandle));
|
|
ptr += sizeof (CUipcMemHandle);
|
|
|
|
if (caps) {
|
|
*ptr = 1;
|
|
ptr++;
|
|
|
|
strcpy ((char *) ptr, caps_str);
|
|
} else {
|
|
*ptr = 0;
|
|
}
|
|
|
|
g_free (caps_str);
|
|
|
|
return true;
|
|
}
|
|
/* *INDENT-ON* */
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_parse_have_data (const std::vector < guint8 > &buf,
|
|
GstClockTime & pts, GstCudaIpcMemLayout & layout, CUipcMemHandle & handle,
|
|
GstCaps ** caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
const guint8 *ptr;
|
|
std::string str;
|
|
|
|
g_return_val_if_fail (buf.size () >=
|
|
GST_CUDA_IPC_PKT_HEADER_SIZE +
|
|
GST_CUDA_IPC_PKT_HAVE_DATA_PAYLOAD_MIN_SIZE, false);
|
|
g_return_val_if_fail (caps, false);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (&header, ptr, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
if (header.type != GstCudaIpcPktType::HAVE_DATA ||
|
|
header.magic != GST_CUDA_IPC_MAGIC_NUMBER ||
|
|
header.payload_size < GST_CUDA_IPC_PKT_HAVE_DATA_PAYLOAD_MIN_SIZE) {
|
|
return false;
|
|
}
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (&pts, ptr, sizeof (GstClockTime));
|
|
ptr += sizeof (GstClockTime);
|
|
|
|
memcpy (&layout, ptr, sizeof (GstCudaIpcMemLayout));
|
|
ptr += sizeof (GstCudaIpcMemLayout);
|
|
|
|
memcpy (&handle, ptr, sizeof (CUipcMemHandle));
|
|
ptr += sizeof (CUipcMemHandle);
|
|
|
|
if (*ptr) {
|
|
ptr++;
|
|
|
|
*caps = gst_caps_from_string ((const gchar *) ptr);
|
|
if (*caps == nullptr)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
bool
|
|
gst_cuda_ipc_pkt_build_have_mmap_data (std::vector<guint8> & buf,
|
|
GstClockTime pts, const GstVideoInfo & info, guint32 max_size,
|
|
GstCudaSharableHandle handle, GstCaps * caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
GstCudaIpcMemLayout layout;
|
|
guint8 *ptr;
|
|
gchar *caps_str = nullptr;
|
|
guint caps_size = 1;
|
|
|
|
if (caps) {
|
|
caps_str = gst_caps_serialize (caps, GST_SERIALIZE_FLAG_NONE);
|
|
if (!caps_str)
|
|
return false;
|
|
|
|
caps_size = strlen (caps_str) + 1;
|
|
}
|
|
|
|
header.type = GstCudaIpcPktType::HAVE_MMAP_DATA;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = sizeof (GstClockTime) + sizeof (GstCudaIpcMemLayout) +
|
|
sizeof (GstCudaSharableHandle) + caps_size;
|
|
|
|
layout.size = info.size;
|
|
layout.max_size = max_size;
|
|
layout.pitch = info.stride[0];
|
|
for (guint i = 0; i < 4; i++)
|
|
layout.offset[i] = info.offset[i];
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE + header.payload_size);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (ptr, &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (ptr, &pts, sizeof (GstClockTime));
|
|
ptr += sizeof (GstClockTime);
|
|
|
|
memcpy (ptr, &layout, sizeof (GstCudaIpcMemLayout));
|
|
ptr += sizeof (GstCudaIpcMemLayout);
|
|
|
|
*((GstCudaSharableHandle *) ptr) = handle;
|
|
ptr += sizeof (GstCudaSharableHandle);
|
|
|
|
if (caps) {
|
|
*ptr = 1;
|
|
ptr++;
|
|
|
|
strcpy ((char *) ptr, caps_str);
|
|
} else {
|
|
*ptr = 0;
|
|
}
|
|
|
|
g_free (caps_str);
|
|
|
|
return true;
|
|
}
|
|
/* *INDENT-ON* */
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_parse_have_mmap_data (const std::vector < guint8 > &buf,
|
|
GstClockTime & pts, GstCudaIpcMemLayout & layout,
|
|
GstCudaSharableHandle * handle, GstCaps ** caps)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
const guint8 *ptr;
|
|
std::string str;
|
|
|
|
g_return_val_if_fail (buf.size () >
|
|
GST_CUDA_IPC_PKT_HEADER_SIZE +
|
|
sizeof (GstClockTime) + sizeof (GstCudaIpcMemLayout) +
|
|
sizeof (GstCudaSharableHandle), false);
|
|
g_return_val_if_fail (caps, false);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (&header, ptr, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
if (header.type != GstCudaIpcPktType::HAVE_MMAP_DATA ||
|
|
header.magic != GST_CUDA_IPC_MAGIC_NUMBER ||
|
|
header.payload_size <=
|
|
sizeof (GstClockTime) + sizeof (GstCudaIpcMemLayout) +
|
|
sizeof (GstCudaSharableHandle)) {
|
|
return false;
|
|
}
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (&pts, ptr, sizeof (GstClockTime));
|
|
ptr += sizeof (GstClockTime);
|
|
|
|
memcpy (&layout, ptr, sizeof (GstCudaIpcMemLayout));
|
|
ptr += sizeof (GstCudaIpcMemLayout);
|
|
|
|
*handle = *((GstCudaSharableHandle *) ptr);
|
|
ptr += sizeof (GstCudaSharableHandle);
|
|
|
|
if (*ptr) {
|
|
ptr++;
|
|
|
|
*caps = gst_caps_from_string ((const gchar *) ptr);
|
|
if (*caps == nullptr)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_read_done (std::vector < guint8 > &buf)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
|
|
header.type = GstCudaIpcPktType::READ_DONE;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = 0;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
memcpy (&buf[0], &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_release_data (std::vector < guint8 > &buf,
|
|
const CUipcMemHandle & handle)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
guint8 *ptr;
|
|
|
|
header.type = GstCudaIpcPktType::RELEASE_DATA;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = sizeof (CUipcMemHandle);
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE +
|
|
GST_CUDA_IPC_PKT_RELEASE_DATA_PAYLOAD_SIZE);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (ptr, &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
memcpy (ptr, &handle, GST_CUDA_IPC_PKT_RELEASE_DATA_PAYLOAD_SIZE);
|
|
}
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_parse_release_data (std::vector < guint8 > &buf,
|
|
CUipcMemHandle & handle)
|
|
{
|
|
g_return_val_if_fail (buf.size () >=
|
|
GST_CUDA_IPC_PKT_HEADER_SIZE + GST_CUDA_IPC_PKT_RELEASE_DATA_PAYLOAD_SIZE,
|
|
false);
|
|
|
|
memcpy (&handle, &buf[0] + GST_CUDA_IPC_PKT_HEADER_SIZE,
|
|
GST_CUDA_IPC_PKT_RELEASE_DATA_PAYLOAD_SIZE);
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_release_mmap_data (std::vector < guint8 > &buf,
|
|
GstCudaSharableHandle handle)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
guint8 *ptr;
|
|
|
|
header.type = GstCudaIpcPktType::RELEASE_MMAP_DATA;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = sizeof (GstCudaSharableHandle);
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE + header.payload_size);
|
|
|
|
ptr = &buf[0];
|
|
memcpy (ptr, &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
*((GstCudaSharableHandle *) ptr) = handle;
|
|
}
|
|
|
|
bool
|
|
gst_cuda_ipc_pkt_parse_release_mmap_data (std::vector < guint8 > &buf,
|
|
GstCudaSharableHandle * handle)
|
|
{
|
|
guint8 *ptr;
|
|
|
|
g_return_val_if_fail (buf.size () >=
|
|
GST_CUDA_IPC_PKT_HEADER_SIZE + sizeof (GstCudaSharableHandle), false);
|
|
|
|
ptr = &buf[0];
|
|
ptr += GST_CUDA_IPC_PKT_HEADER_SIZE;
|
|
|
|
*handle = *((GstCudaSharableHandle *) ptr);
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_eos (std::vector < guint8 > &buf)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
|
|
header.type = GstCudaIpcPktType::EOS;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = 0;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
memcpy (&buf[0], &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
}
|
|
|
|
void
|
|
gst_cuda_ipc_pkt_build_fin (std::vector < guint8 > &buf)
|
|
{
|
|
GstCudaIpcPacketHeader header;
|
|
|
|
header.type = GstCudaIpcPktType::FIN;
|
|
header.magic = GST_CUDA_IPC_MAGIC_NUMBER;
|
|
header.payload_size = 0;
|
|
|
|
buf.resize (GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
|
|
memcpy (&buf[0], &header, GST_CUDA_IPC_PKT_HEADER_SIZE);
|
|
}
|
|
|
|
std::string
|
|
gst_cuda_ipc_mem_handle_to_string (const CUipcMemHandle & handle)
|
|
{
|
|
std::string dump (68, '\0');
|
|
guint val[16];
|
|
for (guint i = 0; i < 16; i++) {
|
|
val[i] = *((guint *) (handle.reserved + i * 4));
|
|
}
|
|
|
|
sprintf (&dump[0], "%x%x%x%x-%x%x%x%x-%x%x%x%x-%x%x%x%x", val[0],
|
|
val[1], val[2], val[3], val[4], val[5], val[6], val[7], val[8],
|
|
val[9], val[10], val[11], val[12], val[13], val[14], val[15]);
|
|
|
|
return dump;
|
|
}
|
|
|
|
bool
|
|
gst_cuda_ipc_clock_is_system (GstClock * clock)
|
|
{
|
|
GstClockType clock_type = GST_CLOCK_TYPE_MONOTONIC;
|
|
GstClock *mclock;
|
|
|
|
if (G_OBJECT_TYPE (clock) != GST_TYPE_SYSTEM_CLOCK)
|
|
return false;
|
|
|
|
g_object_get (clock, "clock-type", &clock_type, nullptr);
|
|
if (clock_type != GST_CLOCK_TYPE_MONOTONIC)
|
|
return false;
|
|
|
|
mclock = gst_clock_get_master (clock);
|
|
if (!mclock)
|
|
return true;
|
|
|
|
gst_object_unref (mclock);
|
|
return false;
|
|
}
|
|
|
|
#ifdef G_OS_WIN32
|
|
/* *INDENT-OFF* */
|
|
static inline void rtrim(std::string &s) {
|
|
s.erase (std::find_if (s.rbegin(), s.rend(),
|
|
[](unsigned char ch) {
|
|
return !std::isspace (ch);
|
|
}).base (), s.end ());
|
|
}
|
|
/* *INDENT-ON* */
|
|
|
|
std::string
|
|
gst_cuda_ipc_win32_error_to_string (guint err)
|
|
{
|
|
wchar_t buffer[1024];
|
|
|
|
if (!FormatMessageW (FORMAT_MESSAGE_IGNORE_INSERTS |
|
|
FORMAT_MESSAGE_FROM_SYSTEM, nullptr, err, 0, buffer, 1024, nullptr)) {
|
|
return std::string ("");
|
|
}
|
|
|
|
std::wstring_convert < std::codecvt_utf8 < wchar_t >, wchar_t >converter;
|
|
std::string ret = converter.to_bytes (buffer);
|
|
rtrim (ret);
|
|
|
|
return ret;
|
|
}
|
|
#else
|
|
std::string
|
|
gst_cuda_ipc_win32_error_to_string (guint err)
|
|
{
|
|
return std::string ("");
|
|
}
|
|
#endif /* G_OS_WIN32 */
|
|
|
|
bool
|
|
gst_cuda_ipc_handle_is_equal (const CUipcMemHandle & handle,
|
|
const CUipcMemHandle & other)
|
|
{
|
|
if (memcmp (&handle, &other, sizeof (CUipcMemHandle)) == 0)
|
|
return true;
|
|
|
|
return false;
|
|
}
|