-
Notifications
You must be signed in to change notification settings - Fork 749
Python to CLR string marshaling LRU cache. #538
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Closed
Closed
Changes from all commits
Commits
Show all changes
19 commits
Select commit
Hold shift + click to select a range
9de6f7b
Runtime/Shutdown loop stores old caches and static variables. It's pr…
e9e5c60
Added all tests finalizer routine.
64acfcb
Bug: Py_Initialize/Py_Finalize calls during alive PythonEngine. Fixed.
7898833
Init/Shutdown state variable (IsFinalizing) fix.
ed29200
TestPythonEngineProperties fixes.
fc549e3
Managed to python string marshaling cache added.
06f4faa
Net40 compatibility fix.
5136c85
Build warnings fix.
507edc9
Update TestRuntime.cs
741070b
Python string to CLR string marshaling cache added.
88c9bc8
Merge branch 'master' into string-marshaling-cache
den-run-ai ffceeed
Merge branch 'master' into string-marshaling-cache
filmor f7433f6
Merge branch 'master' into string-marshaling-cache
filmor b200bcb
Merge branch 'master' into string-marshaling-cache
filmor 0df879e
Merge branch 'master' into string-marshaling-cache
filmor 382c4d6
Merge branch 'master' into string-marshaling-cache
filmor 1865e21
Merge branch 'master' into string-marshaling-cache
filmor 46f2498
Merge branch 'master' into string-marshaling-cache
filmor 49d0d89
Merge branch 'master' into string-marshaling-cache
filmor File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,5 @@ | ||
using System; | ||
using System.Diagnostics; | ||
using NUnit.Framework; | ||
using Python.Runtime; | ||
|
||
|
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
using NUnit.Framework; | ||
using Python.Runtime; | ||
|
||
namespace Python.EmbeddingTest | ||
{ | ||
[SetUpFixture] | ||
public class TestsSuite | ||
{ | ||
[OneTimeTearDown] | ||
public void FinalCleanup() | ||
{ | ||
if (PythonEngine.IsInitialized) | ||
{ | ||
PythonEngine.Shutdown(); | ||
} | ||
} | ||
} | ||
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -18,7 +18,7 @@ public object MarshalNativeToManaged(IntPtr pNativeData) | |
|
||
public abstract IntPtr MarshalManagedToNative(object managedObj); | ||
|
||
public void CleanUpNativeData(IntPtr pNativeData) | ||
public virtual void CleanUpNativeData(IntPtr pNativeData) | ||
{ | ||
Marshal.FreeHGlobal(pNativeData); | ||
} | ||
|
@@ -44,7 +44,12 @@ internal class UcsMarshaler : MarshalerBase | |
private static readonly MarshalerBase Instance = new UcsMarshaler(); | ||
private static readonly Encoding PyEncoding = Runtime.PyEncoding; | ||
|
||
public override IntPtr MarshalManagedToNative(object managedObj) | ||
private const int MaxStringLength = 100; | ||
private const int MaxItemSize = 4 * (MaxStringLength + 1); | ||
private static readonly EncodedStringsFifoDictionary EncodedStringsDictionary = | ||
new EncodedStringsFifoDictionary(10000, MaxItemSize); | ||
|
||
public override unsafe IntPtr MarshalManagedToNative(object managedObj) | ||
{ | ||
var s = managedObj as string; | ||
|
||
|
@@ -53,16 +58,36 @@ public override IntPtr MarshalManagedToNative(object managedObj) | |
return IntPtr.Zero; | ||
} | ||
|
||
byte[] bStr = PyEncoding.GetBytes(s + "\0"); | ||
IntPtr mem = Marshal.AllocHGlobal(bStr.Length); | ||
try | ||
IntPtr mem; | ||
int stringBytesCount; | ||
if (s.Length <= MaxStringLength) | ||
{ | ||
Marshal.Copy(bStr, 0, mem, bStr.Length); | ||
if (EncodedStringsDictionary.TryGetValue(s, out mem)) | ||
{ | ||
return mem; | ||
} | ||
|
||
stringBytesCount = PyEncoding.GetByteCount(s); | ||
mem = EncodedStringsDictionary.AddUnsafe(s); | ||
} | ||
catch (Exception) | ||
else | ||
{ | ||
Marshal.FreeHGlobal(mem); | ||
throw; | ||
stringBytesCount = PyEncoding.GetByteCount(s); | ||
mem = Marshal.AllocHGlobal(stringBytesCount + 4); | ||
} | ||
|
||
fixed (char* str = s) | ||
{ | ||
try | ||
{ | ||
PyEncoding.GetBytes(str, s.Length, (byte*)mem, stringBytesCount); | ||
} | ||
catch | ||
{ | ||
// Do nothing with this. Very strange problem. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. When does this happen? |
||
} | ||
|
||
*(int*)(mem + stringBytesCount) = 0; | ||
} | ||
|
||
return mem; | ||
|
@@ -106,6 +131,14 @@ public static int GetUnicodeByteLength(IntPtr p) | |
} | ||
} | ||
|
||
public override void CleanUpNativeData(IntPtr pNativeData) | ||
{ | ||
if (!EncodedStringsDictionary.IsKnownPtr(pNativeData)) | ||
{ | ||
base.CleanUpNativeData(pNativeData); | ||
} | ||
} | ||
|
||
/// <summary> | ||
/// Utility function for Marshaling Unicode on PY3 and AnsiStr on PY2. | ||
/// Use on functions whose Input signatures changed between PY2/PY3. | ||
|
@@ -118,11 +151,29 @@ public static int GetUnicodeByteLength(IntPtr p) | |
/// <remarks> | ||
/// You MUST deallocate the IntPtr of the Return when done with it. | ||
/// </remarks> | ||
public static IntPtr Py3UnicodePy2StringtoPtr(string s) | ||
public unsafe static IntPtr Py3UnicodePy2StringtoPtr(string s) | ||
{ | ||
return Runtime.IsPython3 | ||
? Instance.MarshalManagedToNative(s) | ||
: Marshal.StringToHGlobalAnsi(s); | ||
if (Runtime.IsPython3) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This means that your optimisation will only work at all for Python 3, is that necessary? |
||
{ | ||
int stringBytesCount = PyEncoding.GetByteCount(s); | ||
IntPtr mem = Marshal.AllocHGlobal(stringBytesCount + 4); | ||
fixed (char* str = s) | ||
{ | ||
try | ||
{ | ||
PyEncoding.GetBytes(str, s.Length, (byte*)mem, stringBytesCount); | ||
} | ||
catch | ||
{ | ||
// Do nothing with this. Very strange problem. | ||
} | ||
|
||
*(int*)(mem + stringBytesCount) = 0; | ||
} | ||
return mem; | ||
} | ||
|
||
return Marshal.StringToHGlobalAnsi(s); | ||
} | ||
|
||
/// <summary> | ||
|
@@ -208,7 +259,12 @@ internal class Utf8Marshaler : MarshalerBase | |
private static readonly MarshalerBase Instance = new Utf8Marshaler(); | ||
private static readonly Encoding PyEncoding = Encoding.UTF8; | ||
|
||
public override IntPtr MarshalManagedToNative(object managedObj) | ||
private const int MaxStringLength = 100; | ||
|
||
private static readonly EncodedStringsFifoDictionary EncodedStringsDictionary = | ||
new EncodedStringsFifoDictionary(10000, 4 * (MaxStringLength + 1)); | ||
|
||
public override unsafe IntPtr MarshalManagedToNative(object managedObj) | ||
{ | ||
var s = managedObj as string; | ||
|
||
|
@@ -217,21 +273,49 @@ public override IntPtr MarshalManagedToNative(object managedObj) | |
return IntPtr.Zero; | ||
} | ||
|
||
byte[] bStr = PyEncoding.GetBytes(s + "\0"); | ||
IntPtr mem = Marshal.AllocHGlobal(bStr.Length); | ||
try | ||
IntPtr mem; | ||
int stringBytesCount; | ||
if (s.Length <= MaxStringLength) | ||
{ | ||
Marshal.Copy(bStr, 0, mem, bStr.Length); | ||
if (EncodedStringsDictionary.TryGetValue(s, out mem)) | ||
{ | ||
return mem; | ||
} | ||
|
||
stringBytesCount = PyEncoding.GetByteCount(s); | ||
mem = EncodedStringsDictionary.AddUnsafe(s); | ||
} | ||
catch (Exception) | ||
else | ||
{ | ||
Marshal.FreeHGlobal(mem); | ||
throw; | ||
stringBytesCount = PyEncoding.GetByteCount(s); | ||
mem = Marshal.AllocHGlobal(stringBytesCount + 1); | ||
} | ||
|
||
fixed (char* str = s) | ||
{ | ||
try | ||
{ | ||
PyEncoding.GetBytes(str, s.Length, (byte*)mem, stringBytesCount); | ||
} | ||
catch | ||
{ | ||
// Do nothing with this. Very strange problem. | ||
} | ||
|
||
((byte*)mem)[stringBytesCount] = 0; | ||
} | ||
|
||
return mem; | ||
} | ||
|
||
public override void CleanUpNativeData(IntPtr pNativeData) | ||
{ | ||
if (!EncodedStringsDictionary.IsKnownPtr(pNativeData)) | ||
{ | ||
base.CleanUpNativeData(pNativeData); | ||
} | ||
} | ||
|
||
public static ICustomMarshaler GetInstance(string cookie) | ||
{ | ||
return Instance; | ||
|
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
using System; | ||
|
||
namespace Python.Runtime | ||
{ | ||
using System.Runtime.InteropServices; | ||
|
||
public class EncodedStringsFifoDictionary: IDisposable | ||
{ | ||
private readonly FifoDictionary<string, IntPtr> _innerDictionary; | ||
|
||
private readonly IntPtr _rawMemory; | ||
|
||
private readonly int _allocatedSize; | ||
|
||
public EncodedStringsFifoDictionary(int capacity, int maxItemSize) | ||
{ | ||
if (maxItemSize < 1) | ||
{ | ||
throw new ArgumentOutOfRangeException( | ||
nameof(maxItemSize), | ||
"Maximum item size should be non-zero positive."); | ||
} | ||
|
||
_innerDictionary = new FifoDictionary<string, IntPtr>(capacity); | ||
_allocatedSize = maxItemSize * capacity; | ||
_rawMemory = Marshal.AllocHGlobal(_allocatedSize); | ||
|
||
MaxItemSize = maxItemSize; | ||
} | ||
|
||
public int MaxItemSize { get; } | ||
|
||
public bool TryGetValue(string key, out IntPtr value) | ||
{ | ||
return _innerDictionary.TryGetValue(key, out value); | ||
} | ||
|
||
public IntPtr AddUnsafe(string key) | ||
{ | ||
int nextSlot = _innerDictionary.NextSlotToAdd; | ||
IntPtr ptr = _rawMemory + (MaxItemSize * nextSlot); | ||
_innerDictionary.AddUnsafe(key, ptr); | ||
return ptr; | ||
} | ||
|
||
public bool IsKnownPtr(IntPtr ptr) | ||
{ | ||
var uptr = (ulong)ptr; | ||
var umem = (ulong)_rawMemory; | ||
|
||
return uptr >= umem && uptr < umem + (ulong)_allocatedSize; | ||
} | ||
|
||
private void ReleaseUnmanagedResources() | ||
{ | ||
if (_rawMemory != IntPtr.Zero) | ||
{ | ||
Marshal.FreeHGlobal(_rawMemory); | ||
} | ||
} | ||
|
||
public void Dispose() | ||
{ | ||
ReleaseUnmanagedResources(); | ||
GC.SuppressFinalize(this); | ||
} | ||
|
||
~EncodedStringsFifoDictionary() | ||
{ | ||
ReleaseUnmanagedResources(); | ||
} | ||
} | ||
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Reflection; | ||
using System.Runtime.InteropServices; | ||
using System.Text; | ||
|
||
namespace Python.Runtime | ||
{ | ||
#if !NETSTANDARD | ||
/// <summary> | ||
/// This polyfill is thread unsafe. | ||
/// </summary> | ||
[CLSCompliant(false)] | ||
public static class EncodingGetStringPolyfill | ||
{ | ||
private static readonly MethodInfo PlatformGetStringMethodInfo = | ||
typeof(Encoding).GetMethod( | ||
"GetString", | ||
BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, null, | ||
new[] | ||
{ | ||
typeof(byte*), typeof(int) | ||
}, null); | ||
|
||
private static readonly byte[] StdDecodeBuffer = PlatformGetStringMethodInfo == null ? new byte[1024 * 1024] : null; | ||
|
||
private static Dictionary<Encoding, EncodingGetStringUnsafeDelegate> PlatformGetStringMethodsDelegatesCache = new Dictionary<Encoding, EncodingGetStringUnsafeDelegate>(); | ||
|
||
private unsafe delegate string EncodingGetStringUnsafeDelegate(byte* pstr, int size); | ||
|
||
public unsafe static string GetString(this Encoding encoding, byte* pstr, int size) | ||
{ | ||
if (PlatformGetStringMethodInfo != null) | ||
{ | ||
EncodingGetStringUnsafeDelegate getStringDelegate; | ||
if (!PlatformGetStringMethodsDelegatesCache.TryGetValue(encoding, out getStringDelegate)) | ||
{ | ||
getStringDelegate = | ||
(EncodingGetStringUnsafeDelegate)Delegate.CreateDelegate( | ||
typeof(EncodingGetStringUnsafeDelegate), encoding, PlatformGetStringMethodInfo); | ||
PlatformGetStringMethodsDelegatesCache.Add(encoding, getStringDelegate); | ||
} | ||
return getStringDelegate(pstr, size); | ||
} | ||
|
||
byte[] buffer = size <= StdDecodeBuffer.Length ? StdDecodeBuffer : new byte[size]; | ||
Marshal.Copy((IntPtr)pstr, buffer, 0, size); | ||
return encoding.GetString(buffer, 0, size); | ||
} | ||
} | ||
#endif | ||
|
||
} |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why
+4
, wouldn't one additional byte be enough?