2018-10-09 08:20:59 +00:00
|
|
|
#if MONO_SECURITY_ALIAS
|
|
|
|
extern alias MonoSecurity;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if MONO_SECURITY_ALIAS
|
|
|
|
using MX = MonoSecurity::Mono.Security.X509;
|
|
|
|
#else
|
|
|
|
using MX = Mono.Security.X509;
|
|
|
|
#endif
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
using System;
|
|
|
|
using System.Text;
|
|
|
|
using System.Runtime.InteropServices;
|
2018-10-09 08:20:59 +00:00
|
|
|
using System.Security.Cryptography;
|
|
|
|
using System.Security.Cryptography.X509Certificates;
|
|
|
|
using Microsoft.Win32.SafeHandles;
|
2016-08-03 10:59:49 +00:00
|
|
|
using XamMac.CoreFoundation;
|
|
|
|
|
2018-10-09 08:20:59 +00:00
|
|
|
namespace Mono.AppleTls
|
2016-08-03 10:59:49 +00:00
|
|
|
{
|
2019-04-12 14:10:50 +00:00
|
|
|
class X509CertificateImplApple : X509Certificate2ImplUnix
|
2016-08-03 10:59:49 +00:00
|
|
|
{
|
|
|
|
IntPtr handle;
|
|
|
|
X509CertificateImpl fallback;
|
|
|
|
|
|
|
|
public X509CertificateImplApple (IntPtr handle, bool owns)
|
|
|
|
{
|
|
|
|
this.handle = handle;
|
|
|
|
if (!owns)
|
|
|
|
CFHelpers.CFRetain (handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override bool IsValid {
|
|
|
|
get { return handle != IntPtr.Zero; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override IntPtr Handle {
|
|
|
|
get { return handle; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override IntPtr GetNativeAppleCertificate ()
|
|
|
|
{
|
|
|
|
ThrowIfContextInvalid ();
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override X509CertificateImpl Clone ()
|
|
|
|
{
|
|
|
|
ThrowIfContextInvalid ();
|
|
|
|
return new X509CertificateImplApple (handle, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CFHelpers.SecurityLibrary)]
|
|
|
|
extern static IntPtr SecCertificateCopySubjectSummary (IntPtr cert);
|
|
|
|
|
|
|
|
[DllImport (CFHelpers.SecurityLibrary)]
|
|
|
|
extern static IntPtr SecCertificateCopyData (IntPtr cert);
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
protected override byte[] GetRawCertData ()
|
|
|
|
{
|
|
|
|
ThrowIfContextInvalid ();
|
|
|
|
var data = SecCertificateCopyData (handle);
|
|
|
|
if (data == IntPtr.Zero)
|
|
|
|
throw new ArgumentException ("Not a valid certificate");
|
|
|
|
|
|
|
|
try {
|
|
|
|
return CFHelpers.FetchDataBuffer (data);
|
|
|
|
} finally {
|
|
|
|
CFHelpers.CFRelease (data);
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string GetSubjectSummary ()
|
|
|
|
{
|
|
|
|
ThrowIfContextInvalid ();
|
|
|
|
IntPtr cfstr = SecCertificateCopySubjectSummary (handle);
|
|
|
|
string ret = CFHelpers.FetchString (cfstr);
|
|
|
|
CFHelpers.CFRelease (cfstr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override bool Equals (X509CertificateImpl other, out bool result)
|
|
|
|
{
|
|
|
|
var otherAppleImpl = other as X509CertificateImplApple;
|
|
|
|
if (otherAppleImpl != null && otherAppleImpl.handle == handle) {
|
|
|
|
result = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MustFallback ()
|
|
|
|
{
|
|
|
|
ThrowIfContextInvalid ();
|
|
|
|
if (fallback != null)
|
|
|
|
return;
|
2018-10-09 08:20:59 +00:00
|
|
|
var mxCert = new MX.X509Certificate (RawData);
|
|
|
|
fallback = new X509Certificate2ImplMono (mxCert);
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
#region X509Certificate2Impl implementation
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
/*
|
|
|
|
* The AppleTls backend does not support X509Certificate2 yet, so we can safely throw
|
|
|
|
* PlatformNotSupportedException here.
|
|
|
|
*/
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override bool HasPrivateKey => throw new PlatformNotSupportedException ();
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override AsymmetricAlgorithm PrivateKey {
|
|
|
|
get => throw new PlatformNotSupportedException ();
|
|
|
|
set => throw new PlatformNotSupportedException ();
|
|
|
|
}
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override RSA GetRSAPrivateKey ()
|
|
|
|
{
|
|
|
|
throw new PlatformNotSupportedException ();
|
|
|
|
}
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override DSA GetDSAPrivateKey ()
|
|
|
|
{
|
|
|
|
throw new PlatformNotSupportedException ();
|
|
|
|
}
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override PublicKey PublicKey => throw new PlatformNotSupportedException ();
|
2018-10-09 08:20:59 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
internal override X509CertificateImplCollection IntermediateCertificates => throw new PlatformNotSupportedException ();
|
2016-08-03 10:59:49 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
internal override X509Certificate2Impl FallbackImpl => throw new PlatformNotSupportedException ();
|
2018-10-09 08:20:59 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
public override bool Verify (X509Certificate2 thisCertificate)
|
2016-08-03 10:59:49 +00:00
|
|
|
{
|
2019-04-12 14:10:50 +00:00
|
|
|
throw new PlatformNotSupportedException ();
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
#endregion
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
protected override void Dispose (bool disposing)
|
|
|
|
{
|
|
|
|
if (handle != IntPtr.Zero){
|
|
|
|
CFHelpers.CFRelease (handle);
|
|
|
|
handle = IntPtr.Zero;
|
|
|
|
}
|
|
|
|
if (fallback != null) {
|
|
|
|
fallback.Dispose ();
|
|
|
|
fallback = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|