Add new comment

Native (Delphi) callbacks in .NET (C#) COM assembly


I’ve posted the English version of this article in codeproject.com


نشرت النسخة الانكليزية لهذا المقال في موقع codeproject.com

مقدمة

لنفترض لدينا الحالة التالية:

-    تريد كتابة إجراء في بيئة الدوت نت و تريد أن تجعله متاحاً للغات البرمجة الأصلية (native) مثل دلفي.
-    هذا الاجراء يأخذ تابع منادى (callback) كأحد بارامتراته/التابع في هذه الحالة سيكون مكتوباً بلغة أصلية كدلفي/.
-    تريد استدعاء تابع الدوت نت هذا من برنامج دلفي.

قبل البدء

بداية ما هو التابع المنادى (callback function) وبالمناسبة هذه هي ترجمتي لأني لم أجد ترجمة عربية أخرى معتمدة لهذا المصطلح الانكليزي. حسب الويكيبديا التابع المنادى هو عبارة عن مرجعية لقطعة من النص المصدري يمرر كمتحول إلى نص مصدري آخر. وهذه التقنية شائعة جداً في لغات البرمجة الأصلية مثل سي بلس بلس و دلفي. أما في دوت نت فالتقنية الأقرب لهذا المفهوم هي تقنية التفويض  delegate. إن التعامل مع هذا النوع من التوابع قد يكون سهلاً عندما نبقى ضمن نطاق اللغات الأصلية مثل سي و دلفي وعندما يبقى استخدامها وتبادلها ضمن النطاق نفسه، لكن كيف يمكن تمرير مرجعية لنص مصدري موجود في الذاكرة الحقيقية إلى نص مصدري آخر موجود في الذاكرة المدارة و هي هنا بيئة الدوت نت؟ هذا ما سأحاول تبيانه في هذا المقال.

أمرٌ آخر سيواجهنا في السيناريو السابق و هو كيف نجعل تابع دوت نت متاحاً لاستخدامه في لغات البرمجة الأصلية، من المعروف أن هذا الأمر يمكن تحقيقه عن طريق جعل مكتبة الدوت نت .net assembly متاحة لعناصر الكوم COM-visible وهنا لابد من الآخذ بعين الاعتبار الخطوات التالية:

•    تسجيل مكتبة الدوت نت كعنصر كوم COM.
•    إضافة المكتبة إلى ذاكرة المكتبات العامة GAC  (Global Assembly Cache) في حال أردنا استدعاء المكتبة من مسارات مختلفة.
•    يجب إعطاء المكتبة اسماً فريداً (قوياً) في حال أردنا إضافتها إلى ال GAC.

قسم الدوت نت

سأستخدم فيجوال ستديو 2010 و سي شارب لبناء هذا القسم.
في فيجوال ستوديو قم بإنشاء مشروع مكتبة صنف جديد و سمّها  CSDemoLibrary.
من ضمن خصائص المشروع وفي مربع معلومات المكتبة assembly information قم بتفعيل خيار COM-Visible.

Native (Delphi) callbacks in .NET (C#) COM assembly

نحن بحاجة إلى إعطاء المكتبة اسماً فريداً إذا كنا سنضيفها إلى الـ GAC لاستخدامها من مسارات مختلفة –يمكنك تجاهل هذه الخطوة إذا كنت ستحفظ هذه المكتبة في نفس المسار لتطبيق دلفي- ولإعطاء المكتبة هذا الاسم الفريد:
في خصائص المشروع ومن لسان التبويب signing اختر(Sign The Assembly)   ثم من القائمة المنسدلة اختر جديد و ليكن اسم الملف CSDemoLibrary.

Native (Delphi) callbacks in .NET (C#) COM assembly

ولأننا نريد تمرير مؤشر من برنامج دلفي إلى هذه المكتبة فهناك خطوة إضافية لابد منها وهي (السماح بالشيفرة غير الآمنة) unsafe code وذلك بتفعيل الخيار (Allow unsafe Code) في لسان التبويب Build مزيد من المعلومات عن المؤشرات في الدوت نت في مقالي السابق

Native (Delphi) callbacks in .NET (C#) COM assembly

الآن أصبحنا جاهزين لكتابة الكود، احذف الصنف الافتراضي في المشروع و أضف صنف جديد و سمّه MyClass.
من أجل التبسيط سوف نعتمد في هذا المثال تابع منادى بسيط يأخذ بارامترين فقط.. في دلفي سيكون شكل هذا التابع كما يلي:

procedure callback(intParam: Integer; strParam: pChar); stdcall;

المكافئ لهذا التابع في الدوت نت سيكون تفويض delegate يأخذ الشكل التالي:

public delegate void NativeCallback(Int32 intParam, [MarshalAs(UnmanagedType.LPWStr)] string strParam);

لقد استخدمت وسم [MarshalAs(UnmanagedType.LPWStr)] للبارمتر النصي لتحويله إلى يونيكود عند تبادله مع اللغات الأصلية native ويمكنك إغفال هذا الوسم في حال أردت التعامل مع الـ Ansistring لكن عندها يجب تغيير نوع البارمتر النصي في تابع الدلفي ليصبح PAnsichar.
الآن نستطيع كتابة تابع الدوت نت، مع مراعاة أننا يجب أن نجعل الصنف MyClass متاحاً لـ COM وذلك بوسمه بالوسم  ComVisible(true)  مع نص guid جديد وبما أنه سيكون فريداً سيشكل ما يشبه البصمة الخاصة بعنصر الكوم هذا. الأمر الآخر المهم هنا ولكون هذا التابع سيتعامل مع المؤشرات فيجب وسمه بالوسم unsafe (راجع مقالي السابق) وفي مايلي النص الكامل للصنف MyClass:

using System;
using System.Runtime.InteropServices;
using System.Threading;

namespace CSDemoLibrary
{
    public delegate void NativeCallback(Int32 intParam, [MarshalAs(UnmanagedType.LPWStr)] string strParam);
   
    [ComVisible(true),
    GuidAttribute("3A65D04A-3F2F-4CB3-B65A-8D402B8C64CE")]
    public class MyClass
    {
       
        public unsafe int Process(
            int intValue,
            Int32 callbackPointer,
            int intParam,
            string strParam)
        {
            IntPtr ptr = new IntPtr(callbackPointer);
            NativeCallback callbackMethod = (NativeCallback)Marshal.GetDelegateForFunctionPointer(ptr, typeof(NativeCallback));
           
            callbackMethod(intParam, strParam);
            Thread.Sleep(1000);
           
            callbackMethod(25, "المرحلة الأولى");
            Thread.Sleep(1000);
            callbackMethod(50, "المرحلة الثانية");
            Thread.Sleep(1000);
            callbackMethod(75, "المرحلة الثالثة");
            Thread.Sleep(1000);
            callbackMethod(100, "المرحلة الرابعة");

            return intValue * 10;
        }
    }
}

في النص السابق: التابع Process يحاكي عملية تستغرق فترة طويلة من الزمن (مثل جلب كمية كبيرة من البيانات من الانترنت)  مع إعلام التطبيق الطالب بشكل مستمر بحالة العملية أو بمقدار التقدم. في مثالنا هذا ومن أجل التبسيط فقط سنقوم بتوقيف التنفيذ لمدة ثانية واحدة في كل مرة وذلك لمحاكاة عملية تستغرق وقتاً طويلاً و تعيد في نهايتها رقم كنتيجة للعملية. في الكود السابق لدينا callbackPointer هو مؤشر للتابع المنادى، ولدينا (intParam, strParam) هي بارامترات ذلك التابع، هذين البارامترين يمكن استخدامهما كقيم ابتدائية وفي معظم السيناريوهات لن يكون هناك حاجة لقيم ابتدائية ولكني ابقيت عليهم هنا لشرح كيفية تمرير بارامترات التوابع من البيئة الأصلية إلى تابع الدوت نت.
GetDelegateForFunctionPointer هو المفتاح في هذه المسألة كلها حيث يقوم هذا التابع بتحويل مؤشر لتابع أصلي إلى التفويض delegate المكافئ له وهناك مزيد من العلومات عن هذا التابع يمكن أن تجدها هنا

تسجيل مكتبة الدوت نت

لتجنب اشكالية المسارات في الدوس سأستخدم هنا (Visual Studio Command Prompt) وهناك اختصار لهذه الاداة تجده في المسار التالي:

(start menu-> All programs-> Microsoft Visual Studio 2010-> Visual Studio Tools)

باستخدام هذه الاداة يمكنك تسجيل مكتبة الدوت نت كعنصر كوم باتباع الخطوات التالية:

شغل Visual Studio Command Prompt /as administrator/ وانتقل إلى المجلد الذي يحتوي المكتبة CSDemoLibrary.dll باستخدام تعليمات الدوس.
إذا كنت ستستخدم CSDemoLibrary.dll من نفس مسار برنامج دلفي فقط فلا داعي لاضافة المكتبة الى GAC فيما عدا ذلك يجب إضافة المكتبة الى الـ GAC ولتحقيق هذا سنستخدم الاداة gacutil مع البارامتر –i كمايلي:

gacutil -i CSDemoLibrary.dll

لتسجيل المكتبة كعنصر COM سنستخدم أداة regasm كما يلي:

regasm CSDemoLibrary.dll

ملاحظة:
لإزالة المكتبة من الـ GAC نستخدم الأمر التالي:

gacutil -u CSDemoLibrary

لإلغاء تسجيل المكتبة نستخدم الامر:

regasm -u CSDemoLibrary.dll

قسم الدلفي:

سأستخدم في هذا القسم دلفي 2010.
في دلفي 2010 انشئ مشروع تطبيق VCL جديد   
أضف (TLabel, TButton, TGauge) إلى النموذج الرئيسي

Native (Delphi) callbacks in .NET (C#) COM assembly

في الكود خلف النموذج الرئيسي أضف ComObj إلى قسم uses 
الآن لنقوم بانشاء تابع بسيط في دلفي سيكون هو التابع المنادى الذي سنرسله (بالأحرى نرسل مرجعيته) إلى مكتبة الدوت نت لتقوم بدورها باستدعاءه لإظهار وتحديث حالة العملية للمستخدم.

procedure callback(intParam: Integer; strParam: pChar); stdcall;    
Begin
Form2.Gauge1.Progress := intParam;    
Form2.lbMessage.Caption := strParam;
Application.ProcessMessages;
End;

سأستخدم في هذه المشروع تقنية الاسناد المتأخر (late binding) في انشاء عنصر الكوم، لذلك سنغير كود الضغط على الـ btnProcess ليصبح كمايلي:

procedure TForm2.btnProcessClick(Sender: TObject);
var
  oleObject: OleVariant;
begin
  try
    oleObject := CreateOleObject('CSDemoLibrary.MyClass');
    ShowMessage('النتيجة= ' + IntToStr(oleObject.Process(10, LongInt(@callback), 0, 'القيمة الابتدائية')));
  except on E: Exception do
    ShowMessage('COM Error: ' + #13 + #10 + e.Message);
  end;
end;

وبهذا يصبح لدينا تطبيق دلفي يستدعي عملية طويلة في الدوت نت التي بدورها تقوم بتنبيه تطبيق دلفي حول حالة العملية و مقدار التقدم الحاصل باستخدام التابع المنادى المكتوب في دلفي.

Native (Delphi) callbacks in .NET (C#) COM assembly

مراجع

http://en.wikipedia.org/wiki/Callback_%28computer_programming%29     
http://edn.embarcadero.com/article/32754
http://msdn.microsoft.com/en-us/library/system.runtime.interopservices.marshal.getdelegateforfunctionpointer%28v=vs.85%29.aspx
Articles Categories: