You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
353 lines
12 KiB
353 lines
12 KiB
// Copyright (c) 2004-2008 MySQL AB, 2008-2009 Sun Microsystems, Inc.
|
|
//
|
|
// This program is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License version 2 as published by
|
|
// the Free Software Foundation
|
|
//
|
|
// There are special exceptions to the terms and conditions of the GPL
|
|
// as it is applied to this software. View the full text of the
|
|
// exception in file EXCEPTIONS in the directory of this software
|
|
// distribution.
|
|
//
|
|
// This program 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 General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
using System.IO;
|
|
using MySql.Data.MySqlClient;
|
|
using System.Diagnostics;
|
|
|
|
|
|
namespace MySql.Data.Common
|
|
{
|
|
#if !PocketPC
|
|
|
|
/// <summary>
|
|
/// Helper class to encapsulate shared memory functionality
|
|
/// Also cares of proper cleanup of file mapping object and cew
|
|
/// </summary>
|
|
internal class SharedMemory : IDisposable
|
|
{
|
|
private const uint FILE_MAP_WRITE = 0x0002;
|
|
|
|
IntPtr fileMapping;
|
|
IntPtr view;
|
|
|
|
public SharedMemory(string name, IntPtr size)
|
|
{
|
|
fileMapping = NativeMethods.OpenFileMapping(FILE_MAP_WRITE, false,
|
|
name);
|
|
if (fileMapping == IntPtr.Zero)
|
|
{
|
|
throw new MySqlException("Cannot open file mapping " + name);
|
|
}
|
|
view = NativeMethods.MapViewOfFile(fileMapping, FILE_MAP_WRITE, 0, 0, size);
|
|
}
|
|
|
|
public IntPtr View
|
|
{
|
|
get { return view; }
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
{
|
|
if (disposing)
|
|
{
|
|
if (view != IntPtr.Zero)
|
|
{
|
|
NativeMethods.UnmapViewOfFile(view);
|
|
view = IntPtr.Zero;
|
|
}
|
|
if (fileMapping != IntPtr.Zero)
|
|
{
|
|
// Free the handle
|
|
NativeMethods.CloseHandle(fileMapping);
|
|
fileMapping = IntPtr.Zero;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
/// <summary>
|
|
/// Summary description for SharedMemoryStream.
|
|
/// </summary>
|
|
internal class SharedMemoryStream : Stream
|
|
{
|
|
private string memoryName;
|
|
private EventWaitHandle serverRead;
|
|
private EventWaitHandle serverWrote;
|
|
private EventWaitHandle clientRead;
|
|
private EventWaitHandle clientWrote;
|
|
private EventWaitHandle connectionClosed;
|
|
private SharedMemory data;
|
|
private int bytesLeft;
|
|
private int position;
|
|
private int connectNumber;
|
|
|
|
private const int BUFFERLENGTH = 16004;
|
|
|
|
private int readTimeout = System.Threading.Timeout.Infinite;
|
|
private int writeTimeout = System.Threading.Timeout.Infinite;
|
|
|
|
public SharedMemoryStream(string memName)
|
|
{
|
|
memoryName = memName;
|
|
}
|
|
|
|
public void Open(uint timeOut)
|
|
{
|
|
if (connectionClosed != null)
|
|
{
|
|
Debug.Assert(false, "Connection is already open");
|
|
}
|
|
GetConnectNumber(timeOut);
|
|
SetupEvents();
|
|
}
|
|
|
|
public override void Close()
|
|
{
|
|
if (connectionClosed != null)
|
|
{
|
|
bool isClosed = connectionClosed.WaitOne(0);
|
|
if (!isClosed)
|
|
{
|
|
connectionClosed.Set();
|
|
connectionClosed.Close();
|
|
}
|
|
connectionClosed = null;
|
|
EventWaitHandle[] handles =
|
|
{serverRead, serverWrote, clientRead, clientWrote};
|
|
|
|
for(int i=0; i< handles.Length; i++)
|
|
{
|
|
if(handles[i] != null)
|
|
handles[i].Close();
|
|
}
|
|
if (data != null)
|
|
{
|
|
data.Dispose();
|
|
data = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
private void GetConnectNumber(uint timeOut)
|
|
{
|
|
EventWaitHandle connectRequest;
|
|
try
|
|
{
|
|
connectRequest =
|
|
EventWaitHandle.OpenExisting(memoryName + "_CONNECT_REQUEST");
|
|
|
|
}
|
|
catch (Exception)
|
|
{
|
|
// If server runs as service, its shared memory is global
|
|
// And if connector runs in user session, it needs to prefix
|
|
// shared memory name with "Global\"
|
|
string prefixedMemoryName = @"Global\" + memoryName;
|
|
connectRequest =
|
|
EventWaitHandle.OpenExisting(prefixedMemoryName + "_CONNECT_REQUEST");
|
|
memoryName = prefixedMemoryName;
|
|
}
|
|
EventWaitHandle connectAnswer =
|
|
EventWaitHandle.OpenExisting(memoryName + "_CONNECT_ANSWER");
|
|
using (SharedMemory connectData =
|
|
new SharedMemory(memoryName + "_CONNECT_DATA", (IntPtr)4))
|
|
{
|
|
// now start the connection
|
|
if (!connectRequest.Set())
|
|
throw new MySqlException("Failed to open shared memory connection");
|
|
if (!connectAnswer.WaitOne((int)(timeOut * 1000), false))
|
|
throw new MySqlException("Timeout during connection");
|
|
connectNumber = Marshal.ReadInt32(connectData.View);
|
|
}
|
|
}
|
|
|
|
|
|
private void SetupEvents()
|
|
{
|
|
string prefix = memoryName + "_" + connectNumber;
|
|
data = new SharedMemory(prefix + "_DATA", (IntPtr)BUFFERLENGTH);
|
|
serverWrote = EventWaitHandle.OpenExisting(prefix + "_SERVER_WROTE");
|
|
serverRead = EventWaitHandle.OpenExisting(prefix + "_SERVER_READ");
|
|
clientWrote = EventWaitHandle.OpenExisting(prefix + "_CLIENT_WROTE");
|
|
clientRead = EventWaitHandle.OpenExisting(prefix + "_CLIENT_READ");
|
|
connectionClosed = EventWaitHandle.OpenExisting(prefix + "_CONNECTION_CLOSED");
|
|
|
|
// tell the server we are ready
|
|
serverRead.Set();
|
|
}
|
|
|
|
#region Properties
|
|
public override bool CanRead
|
|
{
|
|
get { return true; }
|
|
}
|
|
|
|
public override bool CanSeek
|
|
{
|
|
get { return false; }
|
|
}
|
|
|
|
public override bool CanWrite
|
|
{
|
|
get { return true; }
|
|
}
|
|
|
|
public override long Length
|
|
{
|
|
get { throw new NotSupportedException("SharedMemoryStream does not support seeking - length"); }
|
|
}
|
|
|
|
public override long Position
|
|
{
|
|
get { throw new NotSupportedException("SharedMemoryStream does not support seeking - position"); }
|
|
set { }
|
|
}
|
|
|
|
#endregion
|
|
|
|
public override void Flush()
|
|
{
|
|
// No need to flush anything to disk ,as our shared memory is backed
|
|
// by the page file
|
|
}
|
|
|
|
public override int Read(byte[] buffer, int offset, int count)
|
|
{
|
|
int timeLeft = readTimeout;
|
|
WaitHandle[] waitHandles = { serverWrote, connectionClosed };
|
|
LowResolutionStopwatch stopwatch = new LowResolutionStopwatch();
|
|
while (bytesLeft == 0)
|
|
{
|
|
stopwatch.Start();
|
|
int index = WaitHandle.WaitAny(waitHandles, timeLeft);
|
|
stopwatch.Stop();
|
|
if (index == WaitHandle.WaitTimeout)
|
|
throw new TimeoutException("Timeout when reading from shared memory");
|
|
|
|
if (waitHandles[index] == connectionClosed)
|
|
throw new MySqlException("Connection to server lost",true, null);
|
|
|
|
if (readTimeout != System.Threading.Timeout.Infinite)
|
|
{
|
|
timeLeft = readTimeout - (int)stopwatch.ElapsedMilliseconds;
|
|
if (timeLeft < 0)
|
|
throw new TimeoutException("Timeout when reading from shared memory");
|
|
}
|
|
|
|
bytesLeft = Marshal.ReadInt32(data.View);
|
|
position = 4;
|
|
}
|
|
|
|
int len = Math.Min(count, bytesLeft);
|
|
long baseMem = data.View.ToInt64() + position;
|
|
|
|
for (int i = 0; i < len; i++, position++)
|
|
buffer[offset + i] = Marshal.ReadByte((IntPtr)(baseMem + i));
|
|
|
|
bytesLeft -= len;
|
|
if (bytesLeft == 0)
|
|
clientRead.Set();
|
|
|
|
return len;
|
|
}
|
|
|
|
public override long Seek(long offset, SeekOrigin origin)
|
|
{
|
|
throw new NotSupportedException("SharedMemoryStream does not support seeking");
|
|
}
|
|
|
|
public override void Write(byte[] buffer, int offset, int count)
|
|
{
|
|
int leftToDo = count;
|
|
int buffPos = offset;
|
|
WaitHandle[] waitHandles = { serverRead, connectionClosed };
|
|
LowResolutionStopwatch stopwatch = new LowResolutionStopwatch();
|
|
int timeLeft = writeTimeout;
|
|
|
|
while (leftToDo > 0)
|
|
{
|
|
stopwatch.Start();
|
|
int index = WaitHandle.WaitAny(waitHandles, timeLeft);
|
|
stopwatch.Stop();
|
|
|
|
if (waitHandles[index] == connectionClosed)
|
|
throw new MySqlException("Connection to server lost",true, null);
|
|
|
|
if (index == WaitHandle.WaitTimeout)
|
|
throw new TimeoutException("Timeout when reading from shared memory");
|
|
|
|
if (writeTimeout != System.Threading.Timeout.Infinite)
|
|
{
|
|
timeLeft = writeTimeout - (int)stopwatch.ElapsedMilliseconds;
|
|
if (timeLeft < 0)
|
|
throw new TimeoutException("Timeout when writing to shared memory");
|
|
}
|
|
int bytesToDo = Math.Min(leftToDo, BUFFERLENGTH);
|
|
long baseMem = data.View.ToInt64() + 4;
|
|
Marshal.WriteInt32(data.View, bytesToDo);
|
|
Marshal.Copy(buffer, buffPos, (IntPtr)baseMem, bytesToDo);
|
|
buffPos += bytesToDo;
|
|
leftToDo -= bytesToDo;
|
|
if (!clientWrote.Set())
|
|
throw new MySqlException("Writing to shared memory failed");
|
|
}
|
|
}
|
|
|
|
public override void SetLength(long value)
|
|
{
|
|
throw new NotSupportedException("SharedMemoryStream does not support seeking");
|
|
}
|
|
|
|
public override bool CanTimeout
|
|
{
|
|
get
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
public override int ReadTimeout
|
|
{
|
|
get
|
|
{
|
|
return readTimeout;
|
|
}
|
|
set
|
|
{
|
|
readTimeout = value;
|
|
}
|
|
}
|
|
|
|
public override int WriteTimeout
|
|
{
|
|
get
|
|
{
|
|
return writeTimeout;
|
|
}
|
|
set
|
|
{
|
|
writeTimeout = value;
|
|
}
|
|
}
|
|
|
|
}
|
|
#endif
|
|
}
|
|
|