public static int Init([MarshalAs(UnmanagedType.LPWStr)] string /*IntPtr*/ initStr, IntPtr caps, ref IntPtr obj) { string text = $"!!! Init !!!\tobj:{obj}\tCaps:{caps}\tInitStr:{initStr/*Marshal.PtrToStringBSTR(initStr)*/ ?? "null"}\t"; WriteToLog(text); try { UnMemory <int> .SaveInMemArr(new[] { 12, 1, 0 }, ref caps); var capsR = UnMemory <int> .ReadInMemArr(caps, 3); text = $"!!! Init internal Caps !!!\tsize:{capsR[0]}\tVolatileKeySlotCount:{capsR[1]}\tNonvolatileKeySlotCount:{capsR[2]}\t"; //WriteToLog(text); var res = InitializeReader().Result; // выделение памяти под obj int pInt = 0; UnMemory <int> .SaveInMem(pInt, ref obj); _obj = obj; if (_keys == null) { _keys = new Dictionary <Tuple <int, int>, Tuple <int, bool, byte[]> >(); } else { _keys.Clear(); } return(res); } catch (Exception e) { WriteToLog($"Init ERROR !!! reader:{_reader} card: {_card}\r\n {e}"); return((int)ErrorCodes.E_CARDREADER_NOT_INIT); } }
public static int Anticollision(IntPtr obj, IntPtr serialNumberBuf, int bufSize, IntPtr serialNumberSize) { string text = $"!!! Anticollision !!!\tobj:{obj}\tSerialNumberBuf:{serialNumberBuf}\tBufSize:{bufSize}\tSerialNumberSize:{serialNumberSize}\t"; WriteToLog(text); try { var uid = _card?.GetUid().Result; if (uid != null) { byte[] truncUid = new byte[Math.Min(bufSize, uid.Length)]; Array.Copy(uid, truncUid, truncUid.Length); UnMemory <int> .SaveInMem(truncUid.Length, ref serialNumberSize); UnMemory <byte> .SaveInMemArr(uid, ref serialNumberBuf); //Marshal.StructureToPtr(memory_object, SerialNumberBuf, true); var uidWrited = UnMemory <byte> .ReadInMemArr(serialNumberBuf, truncUid.Length); var serialNumberSizeWrited = UnMemory <int> .ReadInMem(serialNumberSize); text = $"!!! Anticollision over !!!\tobj:{obj}\tSerialNumberBuf:{serialNumberBuf}\tBufSize:{bufSize}\tuid:{BitConverter.ToString(uid)}\tuid_w:{BitConverter.ToString(uidWrited ?? new byte[] {})}\tSerialNumberSize:{serialNumberSize}\tSerialNumberSize_writed:{serialNumberSizeWrited}\t"; WriteToLog(text); return((int)ErrorCodes.E_SUCCESS); } WriteToLog($"Anticollision ERROR !!! _card:{_card == null}\r\n uid:{uid == null}"); return((int)ErrorCodes.E_CARDREADER_NOT_INIT); } catch (Exception e) { WriteToLog($"Anticollision ERROR !!! {text}\r\n {e}"); return((int)ErrorCodes.E_GENERIC); } }
// ОПЕРАЦИЯ ПО КАРТЕ /// <summary> /// ОПЕРАЦИЯ ПО КАРТЕ /// </summary> /// <param name="type">тип операции</param> /// <param name="terminal">терминал (азс)</param> /// <param name="transact">номер транзакции</param> /// <param name="opItem">параметры операции</param> /// <returns></returns> public static ServioCardInfo CardOperationExecute(CardOperationType type, int terminal, long transact, CardOperationItem opItem) { ServioCardInfo res = new ServioCardInfo { ErrorCore = -1, CardNumber = "", IssuerID = -1 }; IntPtr obj = new IntPtr(); IntPtr serialNumberPtr = Marshal.AllocHGlobal(sizeof(SerialnumberT)); IntPtr cardImagePtr = Marshal.AllocHGlobal(sizeof(Mf1S70T)); IntPtr operationPtr = CardOperation_Create(); try { if ((res.ErrorCore = InitInternal(_config, ref obj)) != (int)ErrorCodes.ESuccess) { if (res.ErrorCore == (int)ErrorCodes.ECancel) { return(res); } throw new CardShellException("Error Init No" + res, res.ErrorCore); } UnMemory <byte> .SaveInMemArr(new byte[Mf1S70T.Size], ref cardImagePtr); if ((res.ErrorCore = ReadCard(obj, serialNumberPtr, cardImagePtr, false)) != (int)ErrorCodes.ESuccess) { if (res.ErrorCore == (int)ErrorCodes.ECancel) { return(res); } throw new CardShellException("Error ReadCard No" + res, res.ErrorCore); } var operation = UnMemory <CardOperation> .ReadInMem(operationPtr); SerialnumberT serialNumber = UnMemory <SerialnumberT> .ReadInMem(serialNumberPtr); operation.Size = sizeof(CardOperation); operation.OpCode = (byte)OperationSelector(type); operation.POS = terminal; // Номер АЗС или терминала operation.OpTime = DateTime.Now.ToOADate(); // Реальное время с вашего терминала operation.TransactID = transact; // Тут желательно ваш код код транзакции указать operation.CardType = (int)CardType.CtPayment; // Предположим что тип карты неизвестен operation.SerialNumber = serialNumber; // Указать серийный номер, считанный с метки operation.IssuerID = -1; // Указать -1, программа сама переопределит. Если указать другой код - программа будет использовать его operation.CardNumber = ""; // Не указывать !!! operation.AddPrefixZeros = 0; // Нужна доп. настройка. Аналогичная нашей опции в обработчиках operation.ItemCount = 1; // При аутентификации позиции не заполнять. IntPtr pItem1 = Marshal.AllocHGlobal(sizeof(CardOperationItem)); UnMemory <CardOperationItem> .SaveInMem(opItem, ref pItem1); IntPtr ppItem1 = Marshal.AllocHGlobal(sizeof(IntPtr)); UnMemory <IntPtr> .SaveInMem(pItem1, ref ppItem1); operation.Items = ppItem1; UnMemory <CardOperation> .SaveInMem(operation, ref operationPtr); if ((res.ErrorCore = OperationSelector(type, obj, serialNumberPtr, cardImagePtr, operationPtr)) != (int)ErrorCodes.ESuccess) { throw new CardShellException("Error Operation No" + res, res.ErrorCore); } // флаг указывает на то что образ карты был изменен в процессе операции // и нужно записать его обратно. // Нужно для пополнения карты при просмотре информации operation = UnMemory <CardOperation> .ReadInMem(operationPtr); if (operation.CardImageChanged != 0) { // при ошибке можно попробовать повторить операцию несколько раз 3-10... // если не проходит то это патовая ситуация и для её дальнейшего // разрешения нужно сохранить SerialNumber, CardImage и CardInfo // В теории, сохранённый SerialNumber и CardImage можно попробовать // записать обратно даже при перезапуске ПО, но до следующего // обслуживания по карте. Чтобы решить ситуацию можно сохранить CardInfo // чтобы в офисе уменьшили или увеличили баланс карты на соотв. значение. if ((res.ErrorCore = WriteCard(obj, serialNumberPtr, cardImagePtr)) != (int)ErrorCodes.ESuccess) { throw new CardShellException("Error WriteCard No" + res, res.ErrorCore); } } operation = (CardOperation)Marshal.PtrToStructure(operationPtr, typeof(CardOperation)); res.CardInfo = operation.CheckImage; return(res); } finally { //CardOperation_Free(operationPtr); UnMemory.FreeIntPtr(serialNumberPtr); UnMemory.FreeIntPtr(cardImagePtr); //MarshalHelper.UnMemory.FreeIntPtr(operationPtr); } }
// АВТОРИЗАЦИЯ public static ServioCardInfo Authorize() { ServioCardInfo res = new ServioCardInfo { ErrorCore = -1, CardNumber = "", IssuerID = -1 }; IntPtr obj = new IntPtr(); IntPtr serialNumberPtr = Marshal.AllocHGlobal(sizeof(SerialnumberT)); IntPtr cardImagePtr = Marshal.AllocHGlobal(sizeof(Mf1S70T)); IntPtr operationPtr = CardOperation_Create(); try { if ((res.ErrorCore = InitInternal(_config, ref obj)) != (int)ErrorCodes.ESuccess) { if (res.ErrorCore == (int)ErrorCodes.ECancel) { return(res); } throw new CardShellException("Error Init No" + res, res.ErrorCore); } if ((res.ErrorCore = ReadCard(obj, serialNumberPtr, cardImagePtr, true)) != (int)ErrorCodes.ESuccess) { if (res.ErrorCore == (int)ErrorCodes.ECancel) { return(res); } throw new CardShellException("Error ReadCard No" + res, res.ErrorCore); } var operation = UnMemory <CardOperation> .ReadInMem(operationPtr); SerialnumberT serialNumber = UnMemory <SerialnumberT> .ReadInMem(serialNumberPtr); operation.Size = sizeof(CardOperation); operation.POS = 1; // Номер АЗС или терминала operation.OpTime = DateTime.Now.ToOADate(); // Реальное время с вашего терминала operation.TransactID = 0; // Тут желательно ваш код код транзакции указать operation.OpCode = (byte)OpCode.OpAuthorization; operation.IsPostpay = 0; // как есть - если постоплата - указать true operation.CardType = (int)CardType.Unknown; // Предположим что тип карты неизвестен operation.SerialNumber = serialNumber; // Указать серийный номер, считанный с метки operation.IssuerID = -1; // Указать -1, программа сама переопределит. Если указать другой код - программа будет использовать его operation.CardNumber = ""; // Не указывать !!! operation.AddPrefixZeros = 0; // Нужна доп. настройка. Аналогичная нашей опции в обработчиках operation.WoCard = 0; // operation.PINChecked = 0; // Если указать false то драйвер запросит форму ввода пароля через callback в случае, если на карту поставили PIN-код operation.ItemCount = 0; // Позиции не нужны UnMemory <CardOperation> .SaveInMem(operation, ref operationPtr); if ((res.ErrorCore = Auth(obj, serialNumberPtr, cardImagePtr, operationPtr)) != (int)ErrorCodes.ESuccess) { if (res.ErrorCore == (int)ErrorCodes.ECancel) { return(res); } throw new CardShellException("Error Auth No" + res, res.ErrorCore); } operation = UnMemory <CardOperation> .ReadInMem(operationPtr); res.IssuerID = operation.IssuerID; res.CardNumber = operation.CardNumber.PadLeft(20, '0'); return(res); } catch (CardShellException ex) { return(new ServioCardInfo { ErrorCore = ex.ErrorCode }); } catch { return(new ServioCardInfo { ErrorCore = (int)ErrorCodes.EGeneric, }); } finally { UnMemory.FreeIntPtr(serialNumberPtr); UnMemory.FreeIntPtr(cardImagePtr); //MarshalHelper.UnMemory.FreeIntPtr(operationPtr); CardOperation_Free(operationPtr); } }