473,406 Members | 2,273 Online
Bytes | Software Development & Data Engineering Community
Post Job

Home Posts Topics Members FAQ

Join Bytes to post your question to a community of 473,406 software developers and data experts.

Passing values from C# from/to a C++ dll

Hi, I have this in C++ and I like to call it from c# to get the value
but I fail. it will be good if you can give me some information. I
tried it in VB.net it works but I use almost the same way as VB in C#
but it doens't work.

c++: (csp2.dll)
NoMangle long DLL_IMPORT_EXPORT csp2TimeStamp2Str(unsigned char
*Stamp, char *value, long nMaxLength);


VB.net: (this works correctly)

Declare Function csp2TimeStamp2Str Lib "csp2.dll" (ByRef Stamp As
Byte, ByVal value As String, ByVal nMaxLength As Integer) As Integer

Dim nRC As Integer
Dim arrbyteBarcode(99) As Byte '100 elements
Dim nBytesRead As Integer
Dim bstrTmp As New VB6.FixedLengthString(50)

nBytesRead = csp2GetPacket(arrbyteBarcode(0), i, 100)

nRC = csp2TimeStamp2Str(arrbyteBarcode(nBytesRead - 4), bstrTmp.Value,
Len(bstrTmp.Value))
TextBox1.text= VB.Left(bstrTmp.Value, 20)


C#: (this doesn't work :( )

[System.Runtime.InteropServices.DllImport("csp2.DLL ")] static extern
int csp2TimeStamp2Str(byte value, string Stamp, int nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
FixedLengthString bstrTmp = new FixedLengthString(50);

nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
bstrTmp=" ";

nRC = csp2TimeStamp2Str(arrbyteBarcode[nBytesRead-4],
bstrTmp.Value, bstrTmp.Value.Length);
TextBox1.text= bstrTmp.Value.toString();

Oct 3 '07 #1
11 7157
"Bob Yang" <bo******@gmail.comwrote in message
news:11*********************@d55g2000hsg.googlegro ups.com...
Hi, I have this in C++ and I like to call it from c# to get the value
but I fail. it will be good if you can give me some information. I
tried it in VB.net it works but I use almost the same way as VB in C#
but it doens't work.

c++: (csp2.dll)
NoMangle long DLL_IMPORT_EXPORT csp2TimeStamp2Str(unsigned char
*Stamp, char *value, long nMaxLength);


VB.net: (this works correctly)

Declare Function csp2TimeStamp2Str Lib "csp2.dll" (ByRef Stamp As
Byte, ByVal value As String, ByVal nMaxLength As Integer) As Integer

Dim nRC As Integer
Dim arrbyteBarcode(99) As Byte '100 elements
Dim nBytesRead As Integer
Dim bstrTmp As New VB6.FixedLengthString(50)

nBytesRead = csp2GetPacket(arrbyteBarcode(0), i, 100)

nRC = csp2TimeStamp2Str(arrbyteBarcode(nBytesRead - 4), bstrTmp.Value,
Len(bstrTmp.Value))
TextBox1.text= VB.Left(bstrTmp.Value, 20)


C#: (this doesn't work :( )

[System.Runtime.InteropServices.DllImport("csp2.DLL ")] static extern
int csp2TimeStamp2Str(byte value, string Stamp, int nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
FixedLengthString bstrTmp = new FixedLengthString(50);

nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
bstrTmp=" ";

nRC = csp2TimeStamp2Str(arrbyteBarcode[nBytesRead-4],
bstrTmp.Value, bstrTmp.Value.Length);
TextBox1.text= bstrTmp.Value.toString();


..... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();

First you have to pass a byte[] as first parameter to the function, you are
passing the first byte of the array by value. Note that the VB code is
flawed too, you should pass a byte[] not a refrence to a byte, this code
will fail on 64 bit Windows!
Second [1]get rid of the VB6 dependency and use a StringBuilder to pass a
fixed string buffer. Not sure why you are passing a " " char when calling
this function though.
[2] and [3] are used to get a byte array segment out of the original array.

Willy.

Oct 3 '07 #2
.... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();
[snip]
[2] and [3] are used to get a byte array segment out of the original
array.
No, it doesn't. ArraySegment<T>.Array is the entire array, not a subset.
Oct 3 '07 #3
Thank you! However, it doens't really work. sb.ToString() turns a
space " "

I capture some screens and code here: http://docs.google.com/Doc?id=dd5djd97_44d5nz2p

another question, this is not really related, how does it handle
access the value to "sb" wiht out "ref" or "out"? does c# handle this
by itself? I just wonder how to use it in the pure c# without c++
without global variables nor ref or out.
"Bob Yang" <bobya...@gmail.comwrote in message

news:11*********************@d55g2000hsg.googlegro ups.com...


Hi, I have this in C++ and I like to call it from c# to get the value
but I fail. it will be good if you can give me some information. I
tried it in VB.net it works but I use almost the same way as VB in C#
but it doens't work.
c++: (csp2.dll)
NoMangle long DLL_IMPORT_EXPORT csp2TimeStamp2Str(unsigned char
*Stamp, char *value, long nMaxLength);
VB.net: (this works correctly)
Declare Function csp2TimeStamp2Str Lib "csp2.dll" (ByRef Stamp As
Byte, ByVal value As String, ByVal nMaxLength As Integer) As Integer
Dim nRC As Integer
Dim arrbyteBarcode(99) As Byte '100 elements
Dim nBytesRead As Integer
Dim bstrTmp As New VB6.FixedLengthString(50)
nBytesRead = csp2GetPacket(arrbyteBarcode(0), i, 100)
nRC = csp2TimeStamp2Str(arrbyteBarcode(nBytesRead - 4), bstrTmp.Value,
Len(bstrTmp.Value))
TextBox1.text= VB.Left(bstrTmp.Value, 20)
C#: (this doesn't work :( )
[System.Runtime.InteropServices.DllImport("csp2.DLL ")] static extern
int csp2TimeStamp2Str(byte value, string Stamp, int nMaxLength);
int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
FixedLengthString bstrTmp = new FixedLengthString(50);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
bstrTmp=" ";
nRC = csp2TimeStamp2Str(arrbyteBarcode[nBytesRead-4],
bstrTmp.Value, bstrTmp.Value.Length);
TextBox1.text= bstrTmp.Value.toString();

.... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();

First you have to pass a byte[] as first parameter to the function, you are
passing the first byte of the array by value. Note that the VB code is
flawed too, you should pass a byte[] not a refrence to a byte, this code
will fail on 64 bit Windows!
Second [1]get rid of the VB6 dependency and use a StringBuilder to pass a
fixed string buffer. Not sure why you are passing a " " char when calling
this function though.
[2] and [3] are used to get a byte array segment out of the original array.

Willy.- -

- -

Oct 3 '07 #4
thank you. you are right so I change to this but I still not able to
get the value for "sb". any recommadation? thank you!

byte[] bb2 = new byte[100];
nBytesRead2 = nBytesRead - 4;

for (i = 0; i < 4; i++)
{
bb2[i] = arrbyteBarcode[nBytesRead2];
nBytesRead2++;
}
nRC = csp2TimeStamp2Str(bb2, sb, sb.Length);

.... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int nMaxLength);
int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();

[snip]
[2] and [3] are used to get a byte array segment out of the original
array.

No, it doesn't. ArraySegment<T>.Array is the entire array, not a subset.

Oct 3 '07 #5
I changed something and start to read value now. even it is not what I
want yet I think I may pass the wrong bb2.. I will try to test more.
and thank you to all of you! once I am done I will post the final
codes and share with everyone.

by the way, if someone can tell me how does c# assign a value to a
parameter without "ref" or "out" it will be great. thank you!
thank you. you are right so I change to this but I still not able to
get the value for "sb". any recommadation? thank you!

byte[] bb2 = new byte[100];
nBytesRead2 = nBytesRead - 4;

for (i = 0; i < 4; i++)
{
bb2[i] = arrbyteBarcode[nBytesRead2];
nBytesRead2++;
}

nRC = csp2TimeStamp2Str(bb2, sb, sb.Length);
.... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int nMaxLength);
int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();
[snip]
[2] and [3] are used to get a byte array segment out of the original
array.
No, it doesn't. ArraySegment<T>.Array is the entire array, not a subset.- -

- -

Oct 3 '07 #6
"Ben Voigt [C++ MVP]" <rb*@nospam.nospamwrote in message
news:%2***************@TK2MSFTNGP04.phx.gbl...
>
>.... csp2TimeStamp2Str(byte[] value, StringBuilder Stamp, int
nMaxLength);

int nRC, nBytesRead;
byte[] arrbyteBarcode= new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
ArraySegment<byteas = new ArraySegment<byte>(arrbyteBarcode,
nBytesRead-4, 4); // [2]
nRC = csp2TimeStamp2Str(as.Array, sb, sb.Length); // [3]
TextBox1.text= sb.toString();

[snip]
>[2] and [3] are used to get a byte array segment out of the original
array.

No, it doesn't. ArraySegment<T>.Array is the entire array, not a subset.

Very true, ArraySegment is of little use in general and especially here.
Important point is that a byte[] must be passed as an argument, like this...
....
nBytesRead = csp2GetPacket(arrbyteBarcode[0], i, 100);
byte[] ba = new byte[4];
Array.Copy(arrbyteBarcode, nBytesRead - 4, ba, 0, 4);
nRC = csp2TimeStamp2Str(ba, sb, sb.Length);
...

The same applies to the csp2GetPacket function, which is wrong too.

Willy.

Oct 3 '07 #7
thank you to Willy and Ben! yes, the key point is using "unsigned char
*Stamp, char *value" in c# to the right type.

1. by the way... how to make a method able to assign the value to the
parameter without using "ref" or "out"? thank you! I was surprice, it
can assign "arrbyteBarcode" and "sb" values wihtout using those
keyword ref and out!!

2. moreover, how come "char *value" =StringBuilder but "unsigned char
*Stamp" is NOT = to StringBuilder and must use byte array?


for everyone's information. here is the finally code in c#: (please
see previous post for the c++ and VB.net part)

[System.Runtime.InteropServices.DllImport("csp2.DLL ")]
static extern int csp2TimeStamp2Str(byte[] value, StringBuilder
Stamp, int nMaxLength);
int nRC;
int nBytesRead, nBytesRead2;
byte[] arrbyteBarcode = new byte[100];
byte[] bb2 = new byte[100];

StringBuilder sb = new StringBuilder(" ", 100);

nBytesRead = csp2GetPacket(arrbyteBarcode, 1, 100); //get
the packages's byte data into arrbyteBarcode; return the array size to
nBytesRead

nBytesRead2 = nBytesRead - 4; // find the starting point
for the last 4 byte

for (i = 0; i < 4; i++) // assing last
4 bytes data to the new byte array
{
bb2[i] =
arrbyteBarcode[nBytesRead2];
nBytesRead2++;
}

////get timestamp

nRC = csp2TimeStamp2Str(bb2, sb,
100);
richTextBox1.Text = richTextBox1.Text
+ "Time: " + sb.ToString() + "\n";

Oct 4 '07 #8
See inline
Willy.

"Bob Yang" <bo******@gmail.comwrote in message
news:11*********************@w3g2000hsg.googlegrou ps.com...
thank you to Willy and Ben! yes, the key point is using "unsigned char
*Stamp, char *value" in c# to the right type.

1. by the way... how to make a method able to assign the value to the
parameter without using "ref" or "out"? thank you! I was surprice, it
can assign "arrbyteBarcode" and "sb" values wihtout using those
keyword ref and out!!
When you pass a *reference type* like byte[] (or any other array type), the
interop layer pins the array instance and passes a pointer to the first
element in the array to the callee.
It's important that you pass an array when the C function is expecting a
pointer (to an array), you should not pass a *reference* to an array
element.
// OK
[DllImport...] ..... Test(byte[] ar, int size);
byte[] ba =...
Test(ba, ...);

// NOK
[DllImport...] ... Test(ref byte, int size);
byte[] ba = ...;
Test(ref ba[0], ...);

The latter may work on 32 bit windows, but fails on 64 bit windows due to an
optimization in the interop layer of the 64 bit CLR

2. moreover, how come "char *value" =StringBuilder but "unsigned char
*Stamp" is NOT = to StringBuilder and must use byte array?

The underlying buffer of StringBuilder is a (UNICODE) char array, the
interop layer will apply the necessary conversions as defined by the
marshaling attributes (MarshalAs) applied to the parameter.

>

for everyone's information. here is the finally code in c#: (please
see previous post for the c++ and VB.net part)

[System.Runtime.InteropServices.DllImport("csp2.DLL ")]
static extern int csp2TimeStamp2Str(byte[] value, StringBuilder
Stamp, int nMaxLength);
int nRC;
int nBytesRead, nBytesRead2;
byte[] arrbyteBarcode = new byte[100];
byte[] bb2 = new byte[100];

StringBuilder sb = new StringBuilder(" ", 100);

nBytesRead = csp2GetPacket(arrbyteBarcode, 1, 100); //get
the packages's byte data into arrbyteBarcode; return the array size to
nBytesRead

nBytesRead2 = nBytesRead - 4; // find the starting point
for the last 4 byte

for (i = 0; i < 4; i++) // assing last
4 bytes data to the new byte array
{
bb2[i] =
arrbyteBarcode[nBytesRead2];
nBytesRead2++;
}

////get timestamp

nRC = csp2TimeStamp2Str(bb2, sb,
100);
richTextBox1.Text = richTextBox1.Text
+ "Time: " + sb.ToString() + "\n";

Oct 4 '07 #9
thank you! great information.

1. in DllImport situation, C++ is able to change value for a private
variable (I think it is because c++ will locate the variable's address
and change the value there?). what's about in the pure c# without
using any dll? for example, like the code below. is it possible to
change the passing parameters value without using "out" or "ref"?
thank you!
class OutReturnExample
{
static void Method(out int i, out string s1, out string s2)
{
i = 44;
s1 = "I've been returned";
s2 = null;
}
static void Main()
{
int value;
string str1, str2;
Method(out value, out str1, out str2);
// value is now 44
// str1 is now "I've been returned"
// str2 is (still) null;
}
}

2. if I have a c++ as (unsigned char[] ca1, char[] ca2), how can I
call it correctly in c#? is it equal to (byte[] ca1, byte[] ca2)?
Moreover, ca1 and ca2 are only passing IN the the value and c++ cannot
change the value because it is not a pointer in c++?

thank you!


See inline
Willy.

"Bob Yang" <bobya...@gmail.comwrote in message

news:11*********************@w3g2000hsg.googlegrou ps.com...
thank you to Willy and Ben! yes, the key point is using "unsigned char
*Stamp, char *value" in c# to the right type.
1. by the way... how to make a method able to assign the value to the
parameter without using "ref" or "out"? thank you! I was surprice, it
can assign "arrbyteBarcode" and "sb" values wihtout using those
keyword ref and out!!

When you pass a *reference type* like byte[] (or any other array type), the
interop layer pins the array instance and passes a pointer to the first
element in the array to the callee.
It's important that you pass an array when the C function is expecting a
pointer (to an array), you should not pass a *reference* to an array
element.
// OK
[DllImport...] ..... Test(byte[] ar, int size);
byte[] ba =...
Test(ba, ...);

// NOK
[DllImport...] ... Test(ref byte, int size);
byte[] ba = ...;
Test(ref ba[0], ...);

The latter may work on 32 bit windows, but fails on 64 bit windows due to an
optimization in the interop layer of the 64 bit CLR
2. moreover, how come "char *value" =StringBuilder but "unsigned char
*Stamp" is NOT = to StringBuilder and must use byte array?

The underlying buffer of StringBuilder is a (UNICODE) char array, the
interop layer will apply the necessary conversions as defined by the
marshaling attributes (MarshalAs) applied to the parameter.


for everyone's information. here is the finally code in c#: (please
see previous post for the c++ and VB.net part)
[System.Runtime.InteropServices.DllImport("csp2.DLL ")]
static extern int csp2TimeStamp2Str(byte[] value, StringBuilder
Stamp, int nMaxLength);
int nRC;
int nBytesRead, nBytesRead2;
byte[] arrbyteBarcode = new byte[100];
byte[] bb2 = new byte[100];
StringBuilder sb = new StringBuilder(" ", 100);
nBytesRead = csp2GetPacket(arrbyteBarcode, 1, 100); //get
the packages's byte data into arrbyteBarcode; return the array size to
nBytesRead
nBytesRead2 = nBytesRead - 4; // find the starting point
for the last 4 byte
for (i = 0; i < 4; i++) // assing last
4 bytes data to the new byte array
{
bb2[i] =
arrbyteBarcode[nBytesRead2];
nBytesRead2++;
}
////get timestamp
nRC = csp2TimeStamp2Str(bb2, sb,
100);
richTextBox1.Text = richTextBox1.Text
+ "Time: " + sb.ToString() + "\n";- -

- -

Oct 4 '07 #10

"Bob Yang" <bo******@gmail.comwrote in message
news:11**********************@57g2000hsv.googlegro ups.com...
thank you! great information.

2. if I have a c++ as (unsigned char[] ca1, char[] ca2), how can I
call it correctly in c#? is it equal to (byte[] ca1, byte[] ca2)?
More precisely ca2 would be sbyte[], but byte[] should also work.
Moreover, ca1 and ca2 are only passing IN the the value and c++ cannot
change the value because it is not a pointer in c++?
The DLL cannot change the caller's variable to a different array. It can
however change the content of the array passed in the call, which is shared
with the caller.
Oct 8 '07 #11
Please read the following article:
http://blog.rednael.com/2008/08/29/M...DLLsInNET.aspx

It's an in depth article about marshalling between native code and managed code. It shows which types are interoperable, how to import a DLL, how to pass strings, how to pass structures and how to de-reference pointers.

Code examples in C# are included.
Sep 10 '08 #12

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

1
by: Paul | last post by:
Hmmm, didn't seem to work. I have set session.use_cookies = 1 and session.use_trans_sid = 1 in my php.ini file. Index.php contains:...
12
by: Kevin Lyons | last post by:
Hello, I am trying to get my select options (courses) passed correctly from the following URL: http://www.dslextreme.com/users/kevinlyons/selectBoxes.html I am having difficulty getting the...
3
by: domeceo | last post by:
can anyone tell me why I cannot pass values in a setTimeout function whenever I use this function it says "menu is undefined" after th alert. function imgOff(menu, num) { if (document.images) {...
2
by: Gordon Hudson | last post by:
Hello Here is what I am trying to do: Make a hyperlink on a data entry form with its URL built from the values for fields in that entry on the database. So, I would go to record 65785 click...
2
by: craigkenisston | last post by:
Hi, I created an array of objects like this : object Values = {myObject.myprop, otherobject.otherprop, thirdobject.xprop}; Then I pass it to a method. and I get the values filled in that...
1
by: olduncleamos | last post by:
Hello all. With a background in ASP, I am finding the work required for passing values between pages mystifying. For various obvious reasons, I have eliminated using cookies and session to store...
11
by: John Pass | last post by:
Hi, In the attached example, I do understand that the references are not changed if an array is passed by Val. What I do not understand is the result of line 99 (If one can find this by line...
1
by: vncntj | last post by:
I have a C#.NET that simply passes 6 values to a Stored Procedure. But I'm trying to get the (Default.aspx.cs page) to handle passing the values to the sp. The goal is to pass the values and see...
3
by: ishwarbg | last post by:
Hi Everyone, I have a .Net Application, through which I am invoking a function from a legacy DLL developed in C++. My structure in C# contains some data of type double which I need to pass to to...
5
by: jmartmem | last post by:
Greetings, I have built an Update Record Form in an ASP page. This form contains a number of fields, such as text boxes and menus, to name a few. Upon clicking the 'submit' button, I want the...
1
by: Sonnysonu | last post by:
This is the data of csv file 1 2 3 1 2 3 1 2 3 1 2 3 2 3 2 3 3 the lengths should be different i have to store the data by column-wise with in the specific length. suppose the i have to...
0
by: Hystou | last post by:
There are some requirements for setting up RAID: 1. The motherboard and BIOS support RAID configuration. 2. The motherboard has 2 or more available SATA protocol SSD/HDD slots (including MSATA, M.2...
0
marktang
by: marktang | last post by:
ONU (Optical Network Unit) is one of the key components for providing high-speed Internet services. Its primary function is to act as an endpoint device located at the user's premises. However,...
0
by: Hystou | last post by:
Most computers default to English, but sometimes we require a different language, especially when relocating. Forgot to request a specific language before your computer shipped? No problem! You can...
0
jinu1996
by: jinu1996 | last post by:
In today's digital age, having a compelling online presence is paramount for businesses aiming to thrive in a competitive landscape. At the heart of this digital strategy lies an intricately woven...
0
by: Hystou | last post by:
Overview: Windows 11 and 10 have less user interface control over operating system update behaviour than previous versions of Windows. In Windows 11 and 10, there is no way to turn off the Windows...
0
tracyyun
by: tracyyun | last post by:
Dear forum friends, With the development of smart home technology, a variety of wireless communication protocols have appeared on the market, such as Zigbee, Z-Wave, Wi-Fi, Bluetooth, etc. Each...
0
agi2029
by: agi2029 | last post by:
Let's talk about the concept of autonomous AI software engineers and no-code agents. These AIs are designed to manage the entire lifecycle of a software development project—planning, coding, testing,...
0
isladogs
by: isladogs | last post by:
The next Access Europe User Group meeting will be on Wednesday 1 May 2024 starting at 18:00 UK time (6PM UTC+1) and finishing by 19:30 (7.30PM). In this session, we are pleased to welcome a new...

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.