summaryrefslogtreecommitdiffabout
path: root/gammu/emb/common
Side-by-side diff
Diffstat (limited to 'gammu/emb/common') (more/less context) (ignore whitespace changes)
-rw-r--r--gammu/emb/common/common.pro179
-rw-r--r--gammu/emb/common/config.h15
-rw-r--r--gammu/emb/common/device/bluetoth/affix.c44
-rw-r--r--gammu/emb/common/device/bluetoth/affix.h3
-rw-r--r--gammu/emb/common/device/bluetoth/blue_w32.c216
-rw-r--r--gammu/emb/common/device/bluetoth/blue_w32.h188
-rw-r--r--gammu/emb/common/device/bluetoth/bluetoth.c85
-rw-r--r--gammu/emb/common/device/bluetoth/bluetoth.h15
-rw-r--r--gammu/emb/common/device/bluetoth/bluez.c207
-rw-r--r--gammu/emb/common/device/bluetoth/bluez.h3
-rw-r--r--gammu/emb/common/device/devfunc.c266
-rw-r--r--gammu/emb/common/device/devfunc.h38
-rw-r--r--gammu/emb/common/device/irda/irda.c187
-rw-r--r--gammu/emb/common/device/irda/irda.h22
-rw-r--r--gammu/emb/common/device/irda/irda_unx.h61
-rw-r--r--gammu/emb/common/device/irda/irda_w32.h35
-rw-r--r--gammu/emb/common/device/serial/ser_djg.c74
-rw-r--r--gammu/emb/common/device/serial/ser_djg.h15
-rw-r--r--gammu/emb/common/device/serial/ser_unx.c291
-rw-r--r--gammu/emb/common/device/serial/ser_unx.h22
-rw-r--r--gammu/emb/common/device/serial/ser_w32.c340
-rw-r--r--gammu/emb/common/device/serial/ser_w32.h19
-rw-r--r--gammu/emb/common/gammu.h23
-rw-r--r--gammu/emb/common/gsmcomon.c260
-rw-r--r--gammu/emb/common/gsmcomon.h92
-rw-r--r--gammu/emb/common/gsmstate.c1246
-rw-r--r--gammu/emb/common/gsmstate.h1562
-rw-r--r--gammu/emb/common/misc/cfg.c332
-rw-r--r--gammu/emb/common/misc/cfg.h42
-rw-r--r--gammu/emb/common/misc/coding/coding.c1409
-rw-r--r--gammu/emb/common/misc/coding/coding.h133
-rw-r--r--gammu/emb/common/misc/coding/md5.c298
-rw-r--r--gammu/emb/common/misc/coding/md5.h6
-rw-r--r--gammu/emb/common/misc/misc.c591
-rw-r--r--gammu/emb/common/misc/misc.h137
-rw-r--r--gammu/emb/common/phone/alcatel/alcatel.c3991
-rw-r--r--gammu/emb/common/phone/alcatel/alcatel.h286
-rw-r--r--gammu/emb/common/phone/at/atgen.c3669
-rw-r--r--gammu/emb/common/phone/at/atgen.h110
-rw-r--r--gammu/emb/common/phone/at/siemens.c320
-rw-r--r--gammu/emb/common/phone/at/sonyeric.c411
-rw-r--r--gammu/emb/common/phone/nokia/dct3/dct3comm.h16
-rw-r--r--gammu/emb/common/phone/nokia/dct3/dct3func.c1535
-rw-r--r--gammu/emb/common/phone/nokia/dct3/dct3func.h78
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n6110.c2884
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n6110.h45
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n7110.c1724
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n7110.h45
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n9210.c396
-rw-r--r--gammu/emb/common/phone/nokia/dct3/n9210.h17
-rw-r--r--gammu/emb/common/phone/nokia/dct4/dct4func.c115
-rw-r--r--gammu/emb/common/phone/nokia/dct4/dct4func.h30
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n3320.c271
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n3320.h30
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n3650.c392
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n3650.h30
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n6510.c5782
-rw-r--r--gammu/emb/common/phone/nokia/dct4/n6510.h90
-rw-r--r--gammu/emb/common/phone/nokia/nauto.c144
-rw-r--r--gammu/emb/common/phone/nokia/ncommon.h66
-rw-r--r--gammu/emb/common/phone/nokia/nfunc.c2143
-rw-r--r--gammu/emb/common/phone/nokia/nfunc.h99
-rw-r--r--gammu/emb/common/phone/nokia/nfuncold.c226
-rw-r--r--gammu/emb/common/phone/nokia/nfuncold.h19
-rw-r--r--gammu/emb/common/phone/obex/obexgen.c851
-rw-r--r--gammu/emb/common/phone/obex/obexgen.h38
-rw-r--r--gammu/emb/common/phone/pfunc.c138
-rw-r--r--gammu/emb/common/phone/pfunc.h30
-rw-r--r--gammu/emb/common/phone/symbian/mroutgen.c220
-rw-r--r--gammu/emb/common/phone/symbian/mroutgen.h29
-rw-r--r--gammu/emb/common/protocol/alcatel/alcabus.c255
-rw-r--r--gammu/emb/common/protocol/alcatel/alcabus.h61
-rw-r--r--gammu/emb/common/protocol/at/at.c229
-rw-r--r--gammu/emb/common/protocol/at/at.h36
-rw-r--r--gammu/emb/common/protocol/nokia/fbus2.c444
-rw-r--r--gammu/emb/common/protocol/nokia/fbus2.h38
-rw-r--r--gammu/emb/common/protocol/nokia/mbus2.c252
-rw-r--r--gammu/emb/common/protocol/nokia/mbus2.h28
-rw-r--r--gammu/emb/common/protocol/nokia/phonet.c217
-rw-r--r--gammu/emb/common/protocol/nokia/phonet.h35
-rw-r--r--gammu/emb/common/protocol/obex/obex.c120
-rw-r--r--gammu/emb/common/protocol/obex/obex.h33
-rw-r--r--gammu/emb/common/protocol/protocol.h31
-rw-r--r--gammu/emb/common/protocol/symbian/mrouter.c110
-rw-r--r--gammu/emb/common/protocol/symbian/mrouter.h31
-rw-r--r--gammu/emb/common/service/backup/backgen.h69
-rw-r--r--gammu/emb/common/service/backup/backics.c42
-rw-r--r--gammu/emb/common/service/backup/backics.h17
-rw-r--r--gammu/emb/common/service/backup/backldif.c297
-rw-r--r--gammu/emb/common/service/backup/backldif.h17
-rw-r--r--gammu/emb/common/service/backup/backlmb.c413
-rw-r--r--gammu/emb/common/service/backup/backlmb.h17
-rw-r--r--gammu/emb/common/service/backup/backtext.c2908
-rw-r--r--gammu/emb/common/service/backup/backtext.h17
-rw-r--r--gammu/emb/common/service/backup/backvcf.c75
-rw-r--r--gammu/emb/common/service/backup/backvcf.h17
-rw-r--r--gammu/emb/common/service/backup/backvcs.c106
-rw-r--r--gammu/emb/common/service/backup/backvcs.h17
-rw-r--r--gammu/emb/common/service/backup/gsmback.c280
-rw-r--r--gammu/emb/common/service/backup/gsmback.h48
-rw-r--r--gammu/emb/common/service/gsmcal.c509
-rw-r--r--gammu/emb/common/service/gsmcal.h445
-rw-r--r--gammu/emb/common/service/gsmcall.h185
-rw-r--r--gammu/emb/common/service/gsmdata.c366
-rw-r--r--gammu/emb/common/service/gsmdata.h152
-rw-r--r--gammu/emb/common/service/gsmlogo.c1003
-rw-r--r--gammu/emb/common/service/gsmlogo.h180
-rw-r--r--gammu/emb/common/service/gsmmisc.c262
-rw-r--r--gammu/emb/common/service/gsmmisc.h316
-rw-r--r--gammu/emb/common/service/gsmnet.c444
-rw-r--r--gammu/emb/common/service/gsmnet.h98
-rw-r--r--gammu/emb/common/service/gsmpbk.c370
-rw-r--r--gammu/emb/common/service/gsmpbk.h270
-rw-r--r--gammu/emb/common/service/gsmprof.h104
-rw-r--r--gammu/emb/common/service/gsmring.c1600
-rw-r--r--gammu/emb/common/service/gsmring.h202
-rw-r--r--gammu/emb/common/service/sms/gsmems.c765
-rw-r--r--gammu/emb/common/service/sms/gsmems.h20
-rw-r--r--gammu/emb/common/service/sms/gsmmulti.c1148
-rw-r--r--gammu/emb/common/service/sms/gsmmulti.h271
-rw-r--r--gammu/emb/common/service/sms/gsmsms.c663
-rw-r--r--gammu/emb/common/service/sms/gsmsms.h492
122 files changed, 50851 insertions, 0 deletions
diff --git a/gammu/emb/common/common.pro b/gammu/emb/common/common.pro
new file mode 100644
index 0000000..49fecc6
--- a/dev/null
+++ b/gammu/emb/common/common.pro
@@ -0,0 +1,179 @@
+######################################################################
+# Automatically generated by qmake (1.07a) Fri Jul 30 22:13:34 2004
+######################################################################
+
+TEMPLATE = lib
+DEPENDPATH += device \
+ misc \
+ phone \
+ protocol \
+ service \
+ device/bluetoth \
+ device/irda \
+ device/serial \
+ misc/coding \
+ phone/alcatel \
+ phone/at \
+ phone/nokia \
+ phone/obex \
+ phone/symbian \
+ protocol/alcatel \
+ protocol/at \
+ protocol/nokia \
+ protocol/obex \
+ protocol/symbian \
+ service/backup \
+ service/sms \
+ phone/nokia/dct3 \
+ phone/nokia/dct4
+INCLUDEPATH += . \
+ misc/coding \
+ misc \
+ device \
+ phone/nokia/dct4 \
+ phone/nokia/dct3 \
+ phone/at \
+ phone/alcatel \
+ phone/obex \
+ phone/symbian \
+ protocol \
+ protocol/nokia \
+ protocol/at \
+ protocol/alcatel \
+ protocol/obex \
+ protocol/symbian \
+ device/serial \
+ device/irda \
+ device/bluetoth \
+ service \
+ service/sms \
+ service/backup \
+ phone/nokia \
+ phone
+
+# Input
+HEADERS += config.h \
+ gammu.h \
+ gsmcomon.h \
+ gsmstate.h \
+ device/devfunc.h \
+ misc/cfg.h \
+ misc/misc.h \
+ phone/pfunc.h \
+ protocol/protocol.h \
+ service/gsmcal.h \
+ service/gsmcall.h \
+ service/gsmdata.h \
+ service/gsmlogo.h \
+ service/gsmmisc.h \
+ service/gsmnet.h \
+ service/gsmpbk.h \
+ service/gsmprof.h \
+ service/gsmring.h \
+ device/bluetoth/affix.h \
+ device/bluetoth/blue_w32.h \
+ device/bluetoth/bluetoth.h \
+ device/bluetoth/bluez.h \
+ device/irda/irda.h \
+ device/irda/irda_unx.h \
+ device/irda/irda_w32.h \
+ device/serial/ser_djg.h \
+ device/serial/ser_unx.h \
+ device/serial/ser_w32.h \
+ misc/coding/coding.h \
+ misc/coding/md5.h \
+ phone/alcatel/alcatel.h \
+ phone/at/atgen.h \
+ phone/nokia/ncommon.h \
+ phone/nokia/nfunc.h \
+ phone/nokia/nfuncold.h \
+ phone/obex/obexgen.h \
+ phone/symbian/mroutgen.h \
+ protocol/alcatel/alcabus.h \
+ protocol/at/at.h \
+ protocol/nokia/fbus2.h \
+ protocol/nokia/mbus2.h \
+ protocol/nokia/phonet.h \
+ protocol/obex/obex.h \
+ protocol/symbian/mrouter.h \
+ service/backup/backgen.h \
+ service/backup/backics.h \
+ service/backup/backldif.h \
+ service/backup/backlmb.h \
+ service/backup/backtext.h \
+ service/backup/backvcf.h \
+ service/backup/backvcs.h \
+ service/backup/gsmback.h \
+ service/sms/gsmems.h \
+ service/sms/gsmmulti.h \
+ service/sms/gsmsms.h \
+ phone/nokia/dct3/dct3comm.h \
+ phone/nokia/dct3/dct3func.h \
+ phone/nokia/dct3/n6110.h \
+ phone/nokia/dct3/n7110.h \
+ phone/nokia/dct3/n9210.h \
+ phone/nokia/dct4/dct4func.h \
+ phone/nokia/dct4/n3320.h \
+ phone/nokia/dct4/n3650.h \
+ phone/nokia/dct4/n6510.h
+SOURCES +=gsmcomon.c \
+gsmstate.c \
+misc/misc.c \
+misc/cfg.c \
+misc/coding/coding.c \
+misc/coding/md5.c \
+service/sms/gsmsms.c \
+service/sms/gsmems.c \
+service/sms/gsmmulti.c \
+service/gsmcal.c \
+service/gsmdata.c \
+service/gsmpbk.c \
+service/gsmring.c \
+service/gsmlogo.c \
+service/gsmmisc.c \
+service/gsmnet.c \
+service/backup/gsmback.c \
+service/backup/backldif.c \
+service/backup/backlmb.c \
+service/backup/backtext.c \
+service/backup/backvcs.c \
+service/backup/backvcf.c \
+service/backup/backics.c \
+device/bluetoth/affix.c \
+device/bluetoth/bluez.c \
+device/bluetoth/blue_w32.c \
+device/bluetoth/bluetoth.c \
+device/serial/ser_unx.c \
+device/serial/ser_djg.c \
+device/irda/irda.c \
+device/devfunc.c \
+protocol/at/at.c \
+protocol/alcatel/alcabus.c \
+protocol/nokia/mbus2.c \
+protocol/nokia/fbus2.c \
+protocol/nokia/phonet.c \
+protocol/obex/obex.c \
+protocol/symbian/mrouter.c \
+phone/pfunc.c \
+phone/at/atgen.c \
+phone/at/siemens.c \
+phone/at/sonyeric.c \
+phone/alcatel/alcatel.c \
+phone/nokia/dct3/n6110.c \
+phone/nokia/dct3/n7110.c \
+phone/nokia/dct3/n9210.c \
+phone/nokia/dct3/dct3func.c \
+phone/nokia/dct4/n3320.c \
+phone/nokia/dct4/n3650.c \
+phone/nokia/dct4/n6510.c \
+phone/nokia/dct4/dct4func.c \
+phone/nokia/nauto.c \
+phone/nokia/nfunc.c \
+phone/nokia/nfuncold.c \
+phone/obex/obexgen.c \
+phone/symbian/mroutgen.c
+
+TARGET = microgammu
+DESTDIR = $(QPEDIR)/lib
+OBJECTS_DIR = obj/$(PLATFORM)
+MOC_DIR = moc/$(PLATFORM)
diff --git a/gammu/emb/common/config.h b/gammu/emb/common/config.h
new file mode 100644
index 0000000..f48bb38
--- a/dev/null
+++ b/gammu/emb/common/config.h
@@ -0,0 +1,15 @@
+/* (c) 2003 by Michal Cihar */
+
+/*
+ * This is just wrapper for config.h to allow same path in headers for source
+ * tree and installation.
+ */
+
+#ifndef GAMMU_CONFIG_INCLUDED
+# include "../cfg/config.h"
+# define GAMMU_CONFIG_INCLUDED
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/affix.c b/gammu/emb/common/device/bluetoth/affix.c
new file mode 100644
index 0000000..30a917b
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/affix.c
@@ -0,0 +1,44 @@
+/* (c) 2004 by Marcin Wiacek */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifdef GSM_ENABLE_AFFIX
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <string.h>
+#include <sys/socket.h>
+#include <sys/time.h>
+#include <unistd.h>
+
+#include "../../gsmcomon.h"
+#include "../devfunc.h"
+#include "bluetoth.h"
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device)
+{
+ d->hPhone = fd;
+ return ERR_NONE;
+}
+
+#ifdef BLUETOOTH_RF_SEARCHING
+
+static GSM_Error bluetooth_checkdevice(GSM_StateMachine *s, bdaddr_t *bdaddr, struct search_context *context)
+{
+}
+
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s)
+{
+ return error;
+}
+
+#endif
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/affix.h b/gammu/emb/common/device/bluetoth/affix.h
new file mode 100644
index 0000000..896ecbe
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/affix.h
@@ -0,0 +1,3 @@
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device);
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s);
diff --git a/gammu/emb/common/device/bluetoth/blue_w32.c b/gammu/emb/common/device/bluetoth/blue_w32.c
new file mode 100644
index 0000000..a631c9f
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/blue_w32.c
@@ -0,0 +1,216 @@
+/* (c) 2003-2004 by Marcin Wiacek and Intra */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifdef WIN32
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <string.h>
+#include <windows.h>
+#include <io.h>
+
+#include "../../misc/coding/coding.h"
+#include "../../gsmcomon.h"
+#include "../devfunc.h"
+#include "bluetoth.h"
+#include "blue_w32.h"
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device)
+{
+ GSM_Device_BlueToothData *d = &s->Device.Data.BlueTooth;
+ WSADATA wsaData;
+ SOCKADDR_BTH sab;
+ int i;
+
+ smprintf(s, "Connecting to RF channel %i\n",port);
+
+ WSAStartup(MAKEWORD(1,1), &wsaData);
+
+ d->hPhone = socket(AF_BTH, SOCK_STREAM, BTHPROTO_RFCOMM);
+ if (d->hPhone == INVALID_SOCKET) {
+ i = GetLastError();
+ GSM_OSErrorInfo(s, "Socket in bluetooth_open");
+ if (i == 10041) return ERR_DEVICENODRIVER;//unknown socket type
+ return ERR_UNKNOWN;
+ }
+
+ memset (&sab, 0, sizeof(sab));
+ sab.port = port;
+ sab.addressFamily = AF_BTH;
+ sab.btAddr = 0;
+ for (i=0;i<(int)strlen(device);i++) {
+ if (device[i] >='0' && device[i] <='9') {
+ sab.btAddr = sab.btAddr * 16;
+ sab.btAddr = sab.btAddr + (device[i]-'0');
+ }
+ if (device[i] >='a' && device[i] <='f') {
+ sab.btAddr = sab.btAddr * 16;
+ sab.btAddr = sab.btAddr + (device[i]-'a'+10);
+ }
+ if (device[i] >='A' && device[i] <='F') {
+ sab.btAddr = sab.btAddr * 16;
+ sab.btAddr = sab.btAddr + (device[i]-'A'+10);
+ }
+ }
+ dbgprintf("Remote Bluetooth device is %04x%08x\n",
+ GET_NAP(sab.btAddr), GET_SAP(sab.btAddr));
+
+ if (connect (d->hPhone, (struct sockaddr *)&sab, sizeof(sab)) != 0) {
+ i = GetLastError();
+ GSM_OSErrorInfo(s, "Connect in bluetooth_open");
+ if (i == 10060) return ERR_TIMEOUT; //remote device failed to respond
+ if (i == 10050) return ERR_DEVICENOTWORK; //socket operation connected with dead network
+ //noauth
+ close(d->hPhone);
+ return ERR_UNKNOWN;
+ }
+
+ return ERR_NONE;
+}
+
+#ifdef BLUETOOTH_RF_SEARCHING
+
+#pragma comment(lib, "irprops.lib")
+#pragma comment(lib, "ws2_32.lib")
+
+static GSM_Error bluetooth_checkdevice(GSM_StateMachine *s, char *address, WSAPROTOCOL_INFO *protocolInfo)
+{
+ GSM_Device_BlueToothData *d = &s->Device.Data.BlueTooth;
+ WSAQUERYSET querySet;
+ DWORD flags;
+ GUID protocol;
+ int i, result;
+ BYTE buffer[2000];
+ char addressAsString[1000];
+ DWORD bufferLength, addressSize;
+ WSAQUERYSET *pResults = (WSAQUERYSET*)&buffer;
+ HANDLE handle;
+ GSM_Error error;
+
+ memset(&querySet, 0, sizeof(querySet));
+ querySet.dwSize = sizeof(querySet);
+ protocol = L2CAP_PROTOCOL_UUID;
+ querySet.lpServiceClassId = &protocol;
+ querySet.dwNameSpace = NS_BTH;
+ querySet.lpszContext = address;
+
+ flags = LUP_FLUSHCACHE | LUP_RETURN_NAME |
+ LUP_RETURN_TYPE | LUP_RETURN_ADDR |
+ LUP_RETURN_BLOB | LUP_RETURN_COMMENT;
+
+ result = WSALookupServiceBegin(&querySet, flags, &handle);
+ if (result != 0) return ERR_UNKNOWN;
+
+ bufferLength = sizeof(buffer);
+ while (1) {
+ result = WSALookupServiceNext(handle, flags, &bufferLength, pResults);
+ if (result != 0) break;
+ addressSize = sizeof(addressAsString);
+ addressAsString[0] = 0;
+ if (WSAAddressToString(pResults->lpcsaBuffer->RemoteAddr.lpSockaddr,
+ pResults->lpcsaBuffer->RemoteAddr.iSockaddrLength, protocolInfo,
+ addressAsString,&addressSize)==0) {
+ smprintf(s, "%s - ", addressAsString);
+ }
+ smprintf(s, "\"%s\"\n", pResults->lpszServiceInstanceName);
+ if (addressAsString[0] != 0) {
+ for (i=strlen(addressAsString)-1;i>0;i--) {
+ if (addressAsString[i] == ':') break;
+ }
+ if (bluetooth_checkservicename(s, pResults->lpszServiceInstanceName) == ERR_NONE) {
+ error = bluetooth_connect(s,atoi(addressAsString+i+1),address+1);
+ result = WSALookupServiceEnd(handle);
+ return error;
+ }
+ }
+ }
+ result = WSALookupServiceEnd(handle);
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s)
+{
+ GSM_Device_BlueToothData *d = &s->Device.Data.BlueTooth;
+ WSADATA wsaData;
+ int i, protocolInfoSize, result;
+ WSAPROTOCOL_INFO protocolInfo;
+ HANDLE handle;
+ DWORD flags;
+ WSAQUERYSET querySet;
+ BYTE buffer[2000];
+ char addressAsString[1000];
+ DWORD bufferLength, addressSize;
+ WSAQUERYSET *pResults = (WSAQUERYSET*)&buffer;
+ GSM_Error error;
+
+ if (WSAStartup(MAKEWORD(2,2), &wsaData)!=0x00) return ERR_DEVICENODRIVER;
+
+ d->hPhone = socket(AF_BTH, SOCK_STREAM, BTHPROTO_RFCOMM);
+ if (d->hPhone == INVALID_SOCKET) {
+ i = GetLastError();
+ GSM_OSErrorInfo(s, "Socket in bluetooth_open");
+ if (i == 10041) return ERR_DEVICENODRIVER;//unknown socket type
+ return ERR_UNKNOWN;
+ }
+
+ protocolInfoSize = sizeof(protocolInfo);
+ if (getsockopt(d->hPhone, SOL_SOCKET, SO_PROTOCOL_INFO,
+ (char*)&protocolInfo, &protocolInfoSize) != 0)
+ {
+ close(d->hPhone);
+ return ERR_UNKNOWN;
+ }
+ close(d->hPhone);
+
+ if (!strcmp(s->CurrentConfig->Device,"com2:")) {
+ bufferLength = sizeof(buffer);
+
+ flags = LUP_RETURN_NAME | LUP_CONTAINERS |
+ LUP_RETURN_ADDR | LUP_FLUSHCACHE |
+ LUP_RETURN_TYPE | LUP_RETURN_BLOB | LUP_RES_SERVICE;
+
+ memset(&querySet, 0, sizeof(querySet));
+ querySet.dwSize = sizeof(querySet);
+ querySet.dwNameSpace = NS_BTH;
+
+ result = WSALookupServiceBegin(&querySet, flags, &handle);
+ if (result != 0) return ERR_UNKNOWN;
+
+ while (1) {
+ result = WSALookupServiceNext(handle, flags, &bufferLength, pResults);
+ if (result != 0) break;
+
+ smprintf(s, "\"%s\"", pResults->lpszServiceInstanceName);
+
+ addressSize = sizeof(addressAsString);
+ addressAsString[0] = 0;
+ if (WSAAddressToString(pResults->lpcsaBuffer->RemoteAddr.lpSockaddr,
+ pResults->lpcsaBuffer->RemoteAddr.iSockaddrLength, &protocolInfo,
+ addressAsString,&addressSize)==0) {
+ smprintf(s, " - %s\n", addressAsString);
+ error = bluetooth_checkdevice(s, addressAsString,&protocolInfo);
+ if (error == ERR_NONE) {
+ result = WSALookupServiceEnd(handle);
+ return error;
+ }
+ } else smprintf(s, "\n");
+ }
+
+ result = WSALookupServiceEnd(handle);
+ return ERR_NOTSUPPORTED;
+ } else {
+ return bluetooth_checkdevice(s, s->CurrentConfig->Device,&protocolInfo);
+ }
+}
+
+#endif
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/blue_w32.h b/gammu/emb/common/device/bluetoth/blue_w32.h
new file mode 100644
index 0000000..e457c92
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/blue_w32.h
@@ -0,0 +1,188 @@
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device);
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s);
+
+/* MS Platform SDK */
+
+#ifndef __blue_w32_h
+#define __blue_w32_h
+
+#include <pshpack1.h> // Without it compiled code hangs up BT stack
+
+typedef ULONGLONG BTH_ADDR, *PBTH_ADDR;
+
+#define NAP_MASK ((ULONGLONG) 0xFFFF00000000)
+#define SAP_MASK ((ULONGLONG) 0x0000FFFFFFFF)
+
+#define NAP_BIT_OFFSET (8 * 4)
+#define SAP_BIT_OFFSET (0)
+
+#define GET_NAP(_bth_addr) ((USHORT) (((_bth_addr) & NAP_MASK) >> NAP_BIT_OFFSET))
+#define GET_SAP(_bth_addr) ((ULONG) (((_bth_addr) & SAP_MASK) >> SAP_BIT_OFFSET))
+
+#ifndef AF_BTH
+#define AF_BTH 32
+#endif
+
+typedef struct _SOCKADDR_BTH {
+ USHORT addressFamily; // Always AF_BTH
+ BTH_ADDR btAddr; // Bluetooth device address
+ GUID serviceClassId; // [OPTIONAL] system will query SDP for port
+ ULONG port; // RFCOMM channel or L2CAP PSM
+} SOCKADDR_BTH, *PSOCKADDR_BTH;
+
+#define BTHPROTO_RFCOMM 0x0003
+
+#ifdef BLUETOOTH_RF_SEARCHING
+
+typedef struct _SOCKET_ADDRESS {
+ LPSOCKADDR lpSockaddr ;
+ INT iSockaddrLength ;
+} SOCKET_ADDRESS, *PSOCKET_ADDRESS, FAR * LPSOCKET_ADDRESS ;
+
+typedef struct _CSADDR_INFO {
+ SOCKET_ADDRESS LocalAddr ;
+ SOCKET_ADDRESS RemoteAddr ;
+ INT iSocketType ;
+ INT iProtocol ;
+} CSADDR_INFO, *PCSADDR_INFO, FAR * LPCSADDR_INFO ;
+
+typedef struct _AFPROTOCOLS {
+ INT iAddressFamily;
+ INT iProtocol;
+} AFPROTOCOLS, *PAFPROTOCOLS, *LPAFPROTOCOLS;
+
+typedef enum _WSAEcomparator
+{
+ COMP_EQUAL = 0,
+ COMP_NOTLESS
+} WSAECOMPARATOR, *PWSAECOMPARATOR, *LPWSAECOMPARATOR;
+
+typedef struct _WSAVersion
+{
+ DWORD dwVersion;
+ WSAECOMPARATOR ecHow;
+}WSAVERSION, *PWSAVERSION, *LPWSAVERSION;
+
+typedef struct _WSAQuerySetA
+{
+ DWORD dwSize;
+ LPSTR lpszServiceInstanceName;
+ LPGUID lpServiceClassId;
+ LPWSAVERSION lpVersion;
+ LPSTR lpszComment;
+ DWORD dwNameSpace;
+ LPGUID lpNSProviderId;
+ LPSTR lpszContext;
+ DWORD dwNumberOfProtocols;
+ LPAFPROTOCOLS lpafpProtocols;
+ LPSTR lpszQueryString;
+ DWORD dwNumberOfCsAddrs;
+ LPCSADDR_INFO lpcsaBuffer;
+ DWORD dwOutputFlags;
+ LPBLOB lpBlob;
+} WSAQUERYSET, WSAQUERYSETA, *PWSAQUERYSETA, *LPWSAQUERYSETA;
+
+DEFINE_GUID(L2CAP_PROTOCOL_UUID, 0x00000100, 0x0000, 0x1000, 0x80, 0x00, 0x00, 0x80, 0x5F, 0x9B, 0x34, 0xFB);
+
+#ifndef NS_BTH
+# define NS_BTH 16
+#endif
+
+#define LUP_CONTAINERS 0x0002
+#define LUP_RETURN_NAME 0x0010
+#define LUP_RETURN_TYPE 0x0020
+#define LUP_RETURN_COMMENT 0x0080
+#define LUP_RETURN_ADDR 0x0100
+#define LUP_RETURN_BLOB 0x0200
+#define LUP_FLUSHCACHE 0x1000
+#define LUP_RES_SERVICE 0x8000
+
+#define WSAAPI FAR PASCAL
+
+#ifndef WINSOCK_API_LINKAGE
+#ifdef DECLSPEC_IMPORT
+#define WINSOCK_API_LINKAGE DECLSPEC_IMPORT
+#else
+#define WINSOCK_API_LINKAGE
+#endif
+#endif
+
+WINSOCK_API_LINKAGE INT WSAAPI
+WSALookupServiceBeginA(
+ IN LPWSAQUERYSETA lpqsRestrictions,
+ IN DWORD dwControlFlags,
+ OUT LPHANDLE lphLookup
+ );
+
+#define WSALookupServiceBegin WSALookupServiceBeginA
+
+WINSOCK_API_LINKAGE INT WSAAPI
+WSALookupServiceNextA(
+ IN HANDLE hLookup,
+ IN DWORD dwControlFlags,
+ IN OUT LPDWORD lpdwBufferLength,
+ OUT LPWSAQUERYSETA lpqsResults
+ );
+
+#define WSALookupServiceNext WSALookupServiceNextA
+
+WINSOCK_API_LINKAGE INT WSAAPI WSALookupServiceEnd(IN HANDLE hLookup);
+
+#define MAX_PROTOCOL_CHAIN 7
+
+typedef struct _WSAPROTOCOLCHAIN {
+ int ChainLen; /* the length of the chain, */
+ /* length = 0 means layered protocol, */
+ /* length = 1 means base protocol, */
+ /* length > 1 means protocol chain */
+ DWORD ChainEntries[MAX_PROTOCOL_CHAIN]; /* a list of dwCatalogEntryIds */
+} WSAPROTOCOLCHAIN, FAR * LPWSAPROTOCOLCHAIN;
+
+#define WSAPROTOCOL_LEN 255
+
+typedef struct _WSAPROTOCOL_INFOA {
+ DWORD dwServiceFlags1;
+ DWORD dwServiceFlags2;
+ DWORD dwServiceFlags3;
+ DWORD dwServiceFlags4;
+ DWORD dwProviderFlags;
+ GUID ProviderId;
+ DWORD dwCatalogEntryId;
+ WSAPROTOCOLCHAIN ProtocolChain;
+ int iVersion;
+ int iAddressFamily;
+ int iMaxSockAddr;
+ int iMinSockAddr;
+ int iSocketType;
+ int iProtocol;
+ int iProtocolMaxOffset;
+ int iNetworkByteOrder;
+ int iSecurityScheme;
+ DWORD dwMessageSize;
+ DWORD dwProviderReserved;
+ CHAR szProtocol[WSAPROTOCOL_LEN+1];
+} WSAPROTOCOL_INFOA, FAR * LPWSAPROTOCOL_INFOA;
+
+typedef WSAPROTOCOL_INFOA WSAPROTOCOL_INFO;
+
+#define SO_PROTOCOL_INFOA 0x2004 /* WSAPROTOCOL_INFOA structure */
+#define SO_PROTOCOL_INFO SO_PROTOCOL_INFOA
+
+WINSOCK_API_LINKAGE INT WSAAPI
+WSAAddressToStringA(
+ IN LPSOCKADDR lpsaAddress,
+ IN DWORD dwAddressLength,
+ IN LPWSAPROTOCOL_INFOA lpProtocolInfo,
+ IN OUT LPSTR lpszAddressString,
+ IN OUT LPDWORD lpdwAddressStringLength
+ );
+
+#define WSAAddressToString WSAAddressToStringA
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/bluetoth.c b/gammu/emb/common/device/bluetoth/bluetoth.c
new file mode 100644
index 0000000..c20e04b
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/bluetoth.c
@@ -0,0 +1,85 @@
+/* (c) 2003-2004 by Marcin Wiacek and Marcel Holtmann and others */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifndef DJGPP
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <string.h>
+
+#include "../../misc/coding/coding.h"
+#include "../../gsmcomon.h"
+#include "../devfunc.h"
+#include "bluetoth.h"
+
+#ifdef GSM_ENABLE_BLUEZ
+# include "bluez.h"
+#endif
+#ifdef GSM_ENABLE_AFFIX
+# include "affix.h"
+#endif
+#ifdef WIN32
+# include "blue_w32.h"
+#endif
+
+GSM_Error bluetooth_findrfchannel(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+#ifdef BLUETOOTH_RF_SEARCHING
+ if (!mystrncasecmp(s->CurrentConfig->Connection, "bluerf", 6)) return bluetooth_findchannel(s);
+#endif
+
+ switch (s->ConnectionType) {
+ case GCT_BLUEAT:
+ return bluetooth_connect(s,1,s->CurrentConfig->Device);
+ case GCT_BLUEOBEX:
+ return bluetooth_connect(s,9,s->CurrentConfig->Device);
+ case GCT_BLUEPHONET:
+ error = bluetooth_connect(s,14,s->CurrentConfig->Device); //older Series 40 - 8910, 6310
+ if (error == ERR_NONE) return error;
+ return bluetooth_connect(s,15,s->CurrentConfig->Device); //new Series 40 - 6310i, 6230
+ default:
+ return ERR_UNKNOWN;
+ }
+}
+
+static int bluetooth_read(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ return socket_read(s, buf, nbytes, s->Device.Data.BlueTooth.hPhone);
+}
+
+#ifdef WIN32
+static int bluetooth_write(GSM_StateMachine *s, unsigned char *buf, size_t nbytes)
+#else
+static int bluetooth_write(GSM_StateMachine *s, void *buf, size_t nbytes)
+#endif
+{
+ return socket_write(s, buf, nbytes, s->Device.Data.BlueTooth.hPhone);
+}
+
+static GSM_Error bluetooth_close(GSM_StateMachine *s)
+{
+ return socket_close(s, s->Device.Data.BlueTooth.hPhone);
+}
+
+GSM_Device_Functions BlueToothDevice = {
+ bluetooth_findrfchannel,
+ bluetooth_close,
+ NONEFUNCTION,
+ NONEFUNCTION,
+ NONEFUNCTION,
+ bluetooth_read,
+ bluetooth_write
+};
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/bluetoth.h b/gammu/emb/common/device/bluetoth/bluetoth.h
new file mode 100644
index 0000000..8a7fdf4
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/bluetoth.h
@@ -0,0 +1,15 @@
+
+#ifndef DJGPP
+#ifndef unixbluetooth_h
+#define unixbluetooth_h
+
+typedef struct {
+ int hPhone;
+} GSM_Device_BlueToothData;
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/bluez.c b/gammu/emb/common/device/bluetoth/bluez.c
new file mode 100644
index 0000000..8a4807e
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/bluez.c
@@ -0,0 +1,207 @@
+/* Based on work by Marcel Holtmann and other authors of Bluez */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifdef GSM_ENABLE_BLUEZ
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <string.h>
+#include <sys/socket.h>
+#include <sys/time.h>
+#include <unistd.h>
+#include <bluetooth/bluetooth.h>
+#include <bluetooth/rfcomm.h>
+#include <bluetooth/sdp.h>
+#include <bluetooth/sdp_lib.h>
+
+#include "../../gsmcomon.h"
+#include "../devfunc.h"
+#include "bluetoth.h"
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device)
+{
+ GSM_Device_BlueToothData *d = &s->Device.Data.BlueTooth;
+ struct sockaddr_rc laddr, raddr;
+ bdaddr_t bdaddr;
+ int fd;
+
+ smprintf(s, "Connecting to RF channel %i\n",port);
+
+ fd = socket(PF_BLUETOOTH, SOCK_STREAM, BTPROTO_RFCOMM);
+ if (fd < 0) {
+ dbgprintf("Can't create socket\n");
+ return ERR_DEVICENODRIVER;
+ }
+
+ bacpy(&laddr.rc_bdaddr, BDADDR_ANY);
+ laddr.rc_family = AF_BLUETOOTH;
+ laddr.rc_channel = 0;
+
+ if (bind(fd, (struct sockaddr *)&laddr, sizeof(laddr)) < 0) {
+ dbgprintf("Can't bind socket\n");
+ close(fd);
+ return ERR_DEVICEOPENERROR;
+ }
+
+ str2ba(device, &bdaddr);
+ bacpy(&raddr.rc_bdaddr, &bdaddr);
+ raddr.rc_family = AF_BLUETOOTH;
+ raddr.rc_channel = port;
+
+ if (connect(fd, (struct sockaddr *)&raddr, sizeof(raddr)) < 0) {
+ dbgprintf("Can't connect\n");
+ close(fd);
+ return ERR_DEVICEOPENERROR;
+ }
+
+ d->hPhone = fd;
+ return ERR_NONE;
+}
+
+#ifdef BLUETOOTH_RF_SEARCHING
+
+struct search_context {
+ char *svc;
+ uuid_t group;
+ int tree;
+ uint32_t handle;
+};
+
+static void print_service_desc(void *value, void *user)
+{
+ sdp_data_t *p = (sdp_data_t *)value;
+ int i = 0, proto = 0, *channel = (int *)user;
+
+ for (; p; p = p->next, i++) {
+ switch (p->dtd) {
+ case SDP_UUID16:
+ case SDP_UUID32:
+ case SDP_UUID128:
+ proto = 1;//sdp_uuid_to_proto(&p->val.uuid);
+ break;
+ case SDP_UINT8:
+ if (proto == RFCOMM_UUID) {
+ (*channel) = p->val.uint8;
+ return;
+ }
+ break;
+ }
+ }
+}
+
+void print_access_protos(value, user)
+{
+ sdp_list_t *protDescSeq = (sdp_list_t *)value;
+ int *channel = (int *)user;
+
+ sdp_list_foreach(protDescSeq,print_service_desc,channel);
+}
+
+static GSM_Error bluetooth_checkdevice(GSM_StateMachine *s, bdaddr_t *bdaddr, struct search_context *context)
+{
+ sdp_session_t *sess;
+ sdp_list_t *attrid, *search, *seq, *next, *proto = 0;
+ uint32_t range = 0x0000ffff;
+ char str[20];
+ sdp_record_t *rec;
+ sdp_data_t *d;
+ bdaddr_t interface;
+ struct search_context subcontext;
+ int channel,channel2;
+
+ bacpy(&interface,BDADDR_ANY);
+
+ ba2str(bdaddr, str);
+ smprintf(s,"%s\n", str);
+
+ sess = sdp_connect(&interface, bdaddr, SDP_RETRY_IF_BUSY);
+ if (!sess) {
+ dbgprintf("Failed to connect to SDP server on %s: %s\n", str, strerror(errno));
+ return ERR_UNKNOWN;
+ }
+
+ attrid = sdp_list_append(0, &range);
+ search = sdp_list_append(0, &context->group);
+ if (sdp_service_search_attr_req(sess, search, SDP_ATTR_REQ_RANGE, attrid, &seq)) {
+ dbgprintf("Service Search failed: %s\n", strerror(errno));
+ sdp_close(sess);
+ return ERR_UNKNOWN;
+ }
+ sdp_list_free(attrid, 0);
+ sdp_list_free(search, 0);
+
+ channel2 = -1;
+ for (; seq; seq = next) {
+ rec = (sdp_record_t *) seq->data;
+
+ if (channel2 == -1) {
+ if (!context->tree) {
+ d = sdp_data_get(rec,SDP_ATTR_SVCNAME_PRIMARY);
+
+ if (false) {
+ channel = -1;
+ sdp_list_foreach(proto,print_access_protos,&channel);
+ //sdp_list_free(proto,(sdp_free_func_t)sdp_data_free);
+ }
+ smprintf(s,"Channel %i",channel);
+ if (d) smprintf(s," - \"%s\"",d->val.str);
+ smprintf(s,"\n");
+ if (channel2 == -1 && bluetooth_checkservicename(s, d->val.str) == ERR_NONE) {
+ channel2 = channel;
+ }
+ }
+ if (sdp_get_group_id(rec,&subcontext.group) != -1) {
+ memcpy(&subcontext, context, sizeof(struct search_context));
+ if (subcontext.group.value.uuid16 != context->group.value.uuid16) bluetooth_checkdevice(s,bdaddr,&subcontext);
+ }
+ }
+
+ next = seq->next;
+ free(seq);
+ //sdp_record_free(rec);
+ }
+ sdp_close(sess);
+
+ if (channel2 != -1) return bluetooth_connect(s, channel2, str);
+
+ return ERR_UNKNOWN;
+}
+
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s)
+{
+ inquiry_info ii[20];
+ uint8_t count = 0;
+ int i;
+ struct search_context context;
+ GSM_Error error = ERR_NOTSUPPORTED;
+
+ memset(&context, '\0', sizeof(struct search_context));
+ //sdp_uuid16_create(&(context.group),PUBLIC_BROWSE_GROUP);
+
+ if (!strcmp(s->CurrentConfig->Device,"/dev/ttyS1")) {
+ dbgprintf("Searching for devices\n");
+ if (sdp_general_inquiry(ii, 20, 8, &count) < 0) {
+ return ERR_UNKNOWN;
+ }
+ } else {
+ count = 1;
+ str2ba(s->CurrentConfig->Device,&ii[0].bdaddr);
+ }
+ for (i=0;i<count;i++) {
+ error = bluetooth_checkdevice(s,&ii[i].bdaddr,&context);
+ if (error == ERR_NONE) return error;
+ }
+ return error;
+}
+
+#endif
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/bluetoth/bluez.h b/gammu/emb/common/device/bluetoth/bluez.h
new file mode 100644
index 0000000..896ecbe
--- a/dev/null
+++ b/gammu/emb/common/device/bluetoth/bluez.h
@@ -0,0 +1,3 @@
+
+GSM_Error bluetooth_connect(GSM_StateMachine *s, int port, char *device);
+GSM_Error bluetooth_findchannel(GSM_StateMachine *s);
diff --git a/gammu/emb/common/device/devfunc.c b/gammu/emb/common/device/devfunc.c
new file mode 100644
index 0000000..d31ebbf
--- a/dev/null
+++ b/gammu/emb/common/device/devfunc.c
@@ -0,0 +1,266 @@
+
+#include <string.h>
+#ifdef WIN32
+# include <io.h>
+#else
+# include <errno.h>
+# include <signal.h>
+#endif
+
+#include "../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifdef BLUETOOTH_RF_SEARCHING
+
+GSM_Error bluetooth_checkservicename(GSM_StateMachine *s, char *name)
+{
+ if (s->ConnectionType == GCT_BLUEPHONET && strstr(name,"Nokia PC Suite")!=NULL) return ERR_NONE;
+ if (s->ConnectionType == GCT_BLUEOBEX && strstr(name,"OBEX") !=NULL) return ERR_NONE;
+ if (s->ConnectionType == GCT_BLUEAT && strstr(name,"COM 1") !=NULL) return ERR_NONE;
+ return ERR_UNKNOWN;
+}
+
+#endif
+#endif
+
+#if defined (GSM_ENABLE_BLUETOOTHDEVICE) || defined (GSM_ENABLE_IRDADEVICE)
+
+int socket_read(GSM_StateMachine *s, void *buf, size_t nbytes, int hPhone)
+{
+ fd_set readfds;
+#ifdef WIN32
+ struct timeval timer;
+#endif
+
+ FD_ZERO(&readfds);
+ FD_SET(hPhone, &readfds);
+#ifndef WIN32
+ if (select(hPhone+1, &readfds, NULL, NULL, 0)) {
+ return(read(hPhone, buf, nbytes));
+ }
+#else
+ memset(&timer,0,sizeof(timer));
+ if (select(0, &readfds, NULL, NULL, &timer) != 0) {
+ return(recv(hPhone, buf, nbytes, 0));
+ }
+#endif
+ return 0;
+}
+
+#ifdef WIN32
+int socket_write(GSM_StateMachine *s, unsigned char *buf, size_t nbytes, int hPhone)
+#else
+int socket_write(GSM_StateMachine *s, void *buf, size_t nbytes, int hPhone)
+#endif
+{
+ int ret;
+ size_t actual = 0;
+
+ do {
+ ret = send(hPhone, buf, nbytes - actual, 0);
+ if (ret < 0) {
+ if (actual != nbytes) GSM_OSErrorInfo(s,"socket_write");
+ return actual;
+ }
+ actual += ret;
+ buf += ret;
+ } while (actual < nbytes);
+
+ return actual;
+}
+
+GSM_Error socket_close(GSM_StateMachine *s, int hPhone)
+{
+ shutdown(hPhone, 0);
+#ifdef WIN32
+ closesocket(hPhone); /*FIXME: error checking */
+#else
+ close(hPhone); /*FIXME: error checking */
+#endif
+ return ERR_NONE;
+}
+
+#endif
+
+#ifdef ENABLE_LGPL
+
+GSM_Error lock_device(const char* port, char **lock_device)
+{
+ *lock_device = 0;
+ return ERR_NONE;
+}
+
+bool unlock_device(char **lock_file)
+{
+ return true;
+}
+
+#else
+
+#define max_buf_len 128
+#define lock_path "/var/lock/LCK.."
+
+/* Lock the device. Allocated string with a lock name is returned
+ * in lock_device
+ */
+GSM_Error lock_device(const char* port, char **lock_device)
+{
+#ifndef WIN32
+ char *lock_file = NULL;
+ char buffer[max_buf_len];
+ const char *aux;
+ int fd, len;
+ GSM_Error error = ERR_NONE;
+
+ dbgprintf("Locking device\n");
+
+ aux = strrchr(port, '/');
+ /* Remove leading '/' */
+ if (aux) {
+ aux++;
+ } else {
+ /* No / in port */
+ aux = port;
+ }
+ len = strlen(aux) + strlen(lock_path);
+
+ memset(buffer, 0, sizeof(buffer));
+ lock_file = calloc(len + 1, 1);
+ if (!lock_file) {
+ dbgprintf("Out of memory error while locking device\n");
+ return ERR_MOREMEMORY;
+ }
+ /* I think we don't need to use strncpy, as we should have enough
+ * buffer due to strlen results
+ */
+ strcpy(lock_file, lock_path);
+ strcat(lock_file, aux);
+
+ /* Check for the stale lockfile.
+ * The code taken from minicom by Miquel van Smoorenburg */
+ if ((fd = open(lock_file, O_RDONLY)) >= 0) {
+ char buf[max_buf_len];
+ int pid, n = 0;
+
+ n = read(fd, buf, sizeof(buf) - 1);
+ close(fd);
+ if (n > 0) {
+ pid = -1;
+ if (n == 4)
+ /* Kermit-style lockfile. */
+ pid = *(int *)buf;
+ else {
+ /* Ascii lockfile. */
+ buf[n] = 0;
+ sscanf(buf, "%d", &pid);
+ }
+ if (pid > 0 && kill((pid_t)pid, 0) < 0 && errno == ESRCH) {
+ dbgprintf("Lockfile %s is stale. Overriding it..\n", lock_file);
+ sleep(1);
+ if (unlink(lock_file) == -1) {
+ dbgprintf("Overriding failed, please check the permissions\n");
+ dbgprintf("Cannot lock device\n");
+ error = ERR_PERMISSION;
+ goto failed;
+ }
+ } else {
+ dbgprintf("Device already locked by PID %d.\n", pid);
+ error = ERR_DEVICELOCKED;
+ goto failed;
+ }
+ }
+ /* this must not happen. because we could open the file */
+ /* no wrong permissions are set. only reason could be */
+ /* flock/lockf or a empty lockfile due to a broken binary */
+ /* which is more likely */
+ if (n == 0) {
+ dbgprintf("Unable to read lockfile %s.\n", lock_file);
+ dbgprintf("Please check for reason and remove the lockfile by hand.\n");
+ dbgprintf("Cannot lock device\n");
+ error = ERR_UNKNOWN;
+ goto failed;
+ }
+ }
+
+ /* Try to create a new file, with 0644 mode */
+ fd = open(lock_file, O_CREAT | O_EXCL | O_WRONLY, 0644);
+ if (fd == -1) {
+ if (errno == EEXIST) {
+ dbgprintf("Device seems to be locked by unknown process\n");
+ error = ERR_DEVICEOPENERROR;
+ } else if (errno == EACCES) {
+ dbgprintf("Please check permission on lock directory\n");
+ error = ERR_PERMISSION;
+ } else if (errno == ENOENT) {
+ dbgprintf("Cannot create lockfile %s. Please check for existence of path\n", lock_file);
+ error = ERR_UNKNOWN;
+ } else {
+ dbgprintf("Unknown error with creating lockfile %s\n", lock_file);
+ error = ERR_UNKNOWN;
+ }
+ goto failed;
+ }
+ sprintf(buffer, "%10ld gammu\n", (long)getpid());
+ write(fd, buffer, strlen(buffer));
+ close(fd);
+ *lock_device = lock_file;
+ return ERR_NONE;
+failed:
+ free(lock_file);
+ *lock_device = 0;
+ return error;
+#else
+ *lock_device = 0;
+ return ERR_NONE;
+#endif
+}
+
+/* Removes lock and frees memory */
+bool unlock_device(char **lock_file)
+{
+#ifndef WIN32
+ int err;
+
+ if (!lock_file) {
+ dbgprintf("Cannot unlock device\n");
+ return false;
+ }
+ err = unlink(*lock_file);
+ free(*lock_file);
+ *lock_file = NULL;
+ return (err + 1);
+#else
+ return true;
+#endif
+}
+
+#endif
+
+int FindSerialSpeed(char *buffer)
+{
+ switch (atoi(buffer)) {
+ case 50 : return 50;
+ case 75 : return 75;
+ case 110 : return 110;
+ case 134 : return 134;
+ case 150 : return 150;
+ case 200 : return 200;
+ case 300 : return 300;
+ case 600 : return 600;
+ case 1200 : return 1200;
+ case 1800 : return 1800;
+ case 2400 : return 2400;
+ case 4800 : return 4800;
+ case 9600 : return 9600;
+ case 19200 : return 19200;
+ case 38400 : return 38400;
+ case 57600 : return 57600;
+ case 115200 : return 115200;
+ case 230400 : return 230400;
+ default : return 0;
+ }
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/devfunc.h b/gammu/emb/common/device/devfunc.h
new file mode 100644
index 0000000..6e8fb62
--- a/dev/null
+++ b/gammu/emb/common/device/devfunc.h
@@ -0,0 +1,38 @@
+
+#ifndef device_functions_h
+#define device_functions_h
+
+#include "../gsmstate.h"
+
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+#ifdef BLUETOOTH_RF_SEARCHING
+
+GSM_Error bluetooth_checkservicename(GSM_StateMachine *s, char *name);
+
+#endif
+#endif
+
+#if defined (GSM_ENABLE_BLUETOOTHDEVICE) || defined (GSM_ENABLE_IRDADEVICE)
+
+int socket_read(GSM_StateMachine *s, void *buf, size_t nbytes, int hPhone);
+
+#ifdef WIN32
+int socket_write(GSM_StateMachine *s, unsigned char *buf, size_t nbytes, int hPhone);
+#else
+int socket_write(GSM_StateMachine *s, void *buf, size_t nbytes, int hPhone);
+#endif
+
+GSM_Error socket_close(GSM_StateMachine *s, int hPhone);
+
+#endif
+
+GSM_Error lock_device (const char* port, char **lock_device);
+bool unlock_device (char **lock_file);
+
+int FindSerialSpeed(char *buffer);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/irda/irda.c b/gammu/emb/common/device/irda/irda.c
new file mode 100644
index 0000000..fef50ac
--- a/dev/null
+++ b/gammu/emb/common/device/irda/irda.c
@@ -0,0 +1,187 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Ralf Thelen and MyGnokii */
+/* based on some work from Gnokii and MSDN */
+
+/* You have to include wsock32.lib library to MS VC project to compile it */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_IRDADEVICE
+#ifndef DJGPP
+
+#ifndef WIN32
+# include <stdlib.h>
+# include <unistd.h>
+# include <stdio.h>
+# include <fcntl.h>
+# include <errno.h>
+# include <string.h>
+# include <sys/time.h>
+# include <sys/poll.h>
+# include <sys/socket.h>
+# include <sys/ioctl.h>
+#else
+# include <windows.h>
+# include <io.h>
+#endif
+
+#include "../../gsmcomon.h"
+#include "../devfunc.h"
+#include "irda.h"
+
+static bool irda_discover_device(GSM_StateMachine *state)
+{
+ GSM_Device_IrdaData *d = &state->Device.Data.Irda;
+ struct irda_device_list *list;
+ unsigned char *buf;
+ unsigned int sec;
+ int s, z, len, fd, i;
+ GSM_DateTime Date;
+ bool founddevice = false;
+#ifdef WIN32
+ int index;
+#endif
+
+ fd = socket(AF_IRDA, SOCK_STREAM, 0);
+
+ /* can handle maximally 10 devices during discovering */
+ len = sizeof(struct irda_device_list) + sizeof(struct irda_device_info) * 10;
+ buf = malloc(len);
+ list = (struct irda_device_list *)buf;
+
+ /* Trying to find device during 2 seconds */
+ for (z=0;z<2;z++) {
+ GSM_GetCurrentDateTime (&Date);
+ sec = Date.Second;
+ while (sec==Date.Second) {
+ s = len;
+ memset(buf, 0, s);
+
+ if (getsockopt(fd, SOL_IRLMP, IRLMP_ENUMDEVICES, buf, &s) == 0) {
+ for (i = 0; i < (int)list->numDevice; i++) {
+ dbgprintf("Irda: found device \"%s\" (address %x) - ",list->Device[i].irdaDeviceName,list->Device[i].irdaDeviceID);
+ if (strcmp(GetModelData(NULL,NULL,list->Device[i].irdaDeviceName)->number,"") != 0) {
+ founddevice = true;
+ /* Model AUTO */
+ if (state->CurrentConfig->Model[0]==0) strcpy(state->Phone.Data.Model,GetModelData(NULL,NULL,list->Device[i].irdaDeviceName)->number);
+ state->Phone.Data.ModelInfo = GetModelData(NULL,state->Phone.Data.Model,NULL);
+ }
+ if (founddevice) {
+ dbgprintf("correct\n");
+#ifdef WIN32
+ for(index=0; index <= 3; index++)
+ d->peer.irdaDeviceID[index] = list->Device[i].irdaDeviceID[index];
+#else
+ d->peer.irdaDeviceID = list->Device[i].irdaDeviceID;
+#endif
+ break;
+ }
+ dbgprintf("\n");
+ }
+ }
+ if (founddevice) break;
+ my_sleep(10);
+ GSM_GetCurrentDateTime(&Date);
+ }
+ if (founddevice) break;
+ }
+ free(buf);
+ close(fd);
+
+ return founddevice;
+}
+
+static GSM_Error irda_open (GSM_StateMachine *s)
+{
+ GSM_Device_IrdaData *d = &s->Device.Data.Irda;
+ int fd = -1;
+#ifdef WIN32
+ int Enable9WireMode = 1;
+ WSADATA wsaData;
+
+ WSAStartup(MAKEWORD(1,1), &wsaData);
+#else
+ if (s->ConnectionType == GCT_IRDAAT) return ERR_SOURCENOTAVAILABLE;
+#endif
+
+ /* discovering devices */
+ if (irda_discover_device(s)==false) return ERR_TIMEOUT;
+
+ /* Creating socket */
+ fd = socket(AF_IRDA, SOCK_STREAM, 0);
+
+ d->peer.irdaAddressFamily = AF_IRDA;
+#ifndef WIN32
+ d->peer.sir_lsap_sel = LSAP_ANY;
+#endif
+ switch (s->ConnectionType) {
+ case GCT_IRDAAT:
+ strcpy(d->peer.irdaServiceName, "IrDA:IrCOMM");
+
+#ifdef WIN32
+ if (setsockopt(fd, SOL_IRLMP, IRLMP_9WIRE_MODE, (const char *) &Enable9WireMode,
+ sizeof(int))==SOCKET_ERROR) return ERR_UNKNOWN;
+#endif
+ break;
+ case GCT_IRDAPHONET:
+ strcpy(d->peer.irdaServiceName, "Nokia:PhoNet");
+ break;
+ case GCT_IRDAOBEX:
+ /* IrDA:OBEX not supported by N3650 */
+// strcpy(d->peer.irdaServiceName, "IrDA:OBEX");
+
+ strcpy(d->peer.irdaServiceName, "OBEX");
+
+ /* Alternative server is "OBEX:IrXfer" */
+ break;
+ default:
+ return ERR_UNKNOWN;
+ }
+
+ /* Connecting to service */
+ if (connect(fd, (struct sockaddr *)&d->peer, sizeof(d->peer))) {
+ dbgprintf("Can't connect to service %s\n",d->peer.irdaServiceName);
+ close(fd);
+ return ERR_NOTSUPPORTED;
+ }
+
+ d->hPhone=fd;
+
+ return ERR_NONE;
+}
+
+static int irda_read(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ return socket_read(s, buf, nbytes, s->Device.Data.Irda.hPhone);
+}
+
+#ifdef WIN32
+static int irda_write(GSM_StateMachine *s, unsigned char *buf, size_t nbytes)
+#else
+static int irda_write(GSM_StateMachine *s, void *buf, size_t nbytes)
+#endif
+{
+ return socket_write(s, buf, nbytes, s->Device.Data.Irda.hPhone);
+}
+
+static GSM_Error irda_close(GSM_StateMachine *s)
+{
+ return socket_close(s, s->Device.Data.Irda.hPhone);
+}
+
+GSM_Device_Functions IrdaDevice = {
+ irda_open,
+ irda_close,
+ NONEFUNCTION,
+ NONEFUNCTION,
+ NONEFUNCTION,
+ irda_read,
+ irda_write
+};
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/irda/irda.h b/gammu/emb/common/device/irda/irda.h
new file mode 100644
index 0000000..455e6af
--- a/dev/null
+++ b/gammu/emb/common/device/irda/irda.h
@@ -0,0 +1,22 @@
+
+#ifndef DJGPP
+#ifndef unixirda_h
+#define unixirda_h
+
+#ifndef WIN32
+# include "irda_unx.h"
+#else
+# include "irda_w32.h"
+#endif
+
+typedef struct {
+ int hPhone;
+ struct sockaddr_irda peer;
+} GSM_Device_IrdaData;
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/irda/irda_unx.h b/gammu/emb/common/device/irda/irda_unx.h
new file mode 100644
index 0000000..8dbcb97
--- a/dev/null
+++ b/gammu/emb/common/device/irda/irda_unx.h
@@ -0,0 +1,61 @@
+/* part of irda.h available in Linux kernel source */
+
+/*********************************************************************
+ *
+ * Filename: irda.h
+ * Version:
+ * Description:
+ * Status: Experimental.
+ * Author: Dag Brattli <dagb@cs.uit.no>
+ * Created at: Mon Mar 8 14:06:12 1999
+ * Modified at: Sat Dec 25 16:06:42 1999
+ * Modified by: Dag Brattli <dagb@cs.uit.no>
+ *
+ * Copyright (c) 1999 Dag Brattli, All Rights Reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of
+ * the License, or (at your option) any later version.
+ *
+ * Neither Dag Brattli nor University of Tromsø admit liability nor
+ * provide warranty for any of this software. This material is
+ * provided "AS-IS" and at no charge.
+ *
+ ********************************************************************/
+
+#ifndef __irda_unx_h
+#define __irda_unx_h
+
+#include <sys/types.h>
+#include <sys/socket.h>
+
+#define SOL_IRLMP 266 /* Same as SOL_IRDA for now */
+#define IRLMP_ENUMDEVICES 1 /* Return discovery log */
+#define LSAP_ANY 0xff
+
+struct sockaddr_irda {
+ sa_family_t irdaAddressFamily; /* AF_IRDA */
+ u_int8_t sir_lsap_sel; /* LSAP selector */
+ u_int32_t irdaDeviceID; /* Device address */
+ char irdaServiceName[25]; /* Usually <service>:IrDA:TinyTP */
+};
+
+struct irda_device_info {
+ u_int32_t saddr; /* Address of local interface */
+ u_int32_t irdaDeviceID; /* Address of remote device */
+ char irdaDeviceName[22]; /* Description */
+ u_int8_t charset; /* Charset used for description */
+ u_int8_t hints[2]; /* Hint bits */
+};
+
+struct irda_device_list {
+ u_int32_t numDevice;
+ struct irda_device_info Device[1];
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/irda/irda_w32.h b/gammu/emb/common/device/irda/irda_w32.h
new file mode 100644
index 0000000..daffa37
--- a/dev/null
+++ b/gammu/emb/common/device/irda/irda_w32.h
@@ -0,0 +1,35 @@
+
+/* MS Platform SDK */
+
+#ifndef __irda_w32_h
+#define __irda_w32_h
+
+#define AF_IRDA 26
+#define SOL_IRLMP 0x00FF
+#define IRLMP_ENUMDEVICES 0x00000010
+#define IRLMP_9WIRE_MODE 0x00000016
+
+struct sockaddr_irda {
+ unsigned short irdaAddressFamily;
+ unsigned char irdaDeviceID[4];
+ char irdaServiceName[25];
+};
+
+struct irda_device_info {
+ unsigned char irdaDeviceID[4];
+ char irdaDeviceName[22];
+ unsigned char irdaDeviceHints1;
+ unsigned char irdaDeviceHints2;
+ unsigned char irdaCharSet;
+};
+
+struct irda_device_list {
+ ULONG numDevice;
+ struct irda_device_info Device[1];
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_djg.c b/gammu/emb/common/device/serial/ser_djg.c
new file mode 100644
index 0000000..ac9d7c8
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_djg.c
@@ -0,0 +1,74 @@
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_SERIALDEVICE
+#ifdef DJGPP
+
+#include "../../gsmcomon.h"
+#include "ser_djg.h"
+
+static GSM_Error serial_close(GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_open (GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setparity(GSM_StateMachine *s, bool parity)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setdtrrts(GSM_StateMachine *s, bool dtr, bool rts)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setspeed(GSM_StateMachine *s, int speed)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return ERR_NONE;
+}
+
+static int serial_read(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return 0;
+}
+
+static int serial_write(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ return 0;
+}
+
+GSM_Device_Functions SerialDevice = {
+ serial_open,
+ serial_close,
+ serial_setparity,
+ serial_setdtrrts,
+ serial_setspeed,
+ serial_read,
+ serial_write
+};
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_djg.h b/gammu/emb/common/device/serial/ser_djg.h
new file mode 100644
index 0000000..b35b282
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_djg.h
@@ -0,0 +1,15 @@
+
+#ifdef DJGPP
+#ifndef djgppserial_h
+#define djgppserial_h
+
+typedef struct {
+ int hPhone;
+} GSM_Device_SerialData;
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_unx.c b/gammu/emb/common/device/serial/ser_unx.c
new file mode 100644
index 0000000..2a87b11
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_unx.c
@@ -0,0 +1,291 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* locking device and settings all speeds by Michal Cihar */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_SERIALDEVICE
+#ifndef WIN32
+#ifndef DJGPP
+
+#include <sys/file.h>
+#include <sys/time.h>
+#include <string.h>
+#include <termios.h>
+#include <errno.h>
+
+#include "../../gsmcomon.h"
+#include "ser_unx.h"
+
+static GSM_Error serial_close(GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ /* Restores old settings */
+ tcsetattr(d->hPhone, TCSANOW, &d->old_settings);
+
+ /* Closes device */
+ close(d->hPhone);
+
+ return ERR_NONE;
+}
+
+#ifndef O_NONBLOCK
+# define O_NONBLOCK 0
+#endif
+
+static GSM_Error serial_open (GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ struct termios t;
+ int i;
+
+ /* O_NONBLOCK MUST is required to avoid waiting for DCD */
+ d->hPhone = open(s->CurrentConfig->Device, O_RDWR | O_NOCTTY | O_NONBLOCK);
+ if (d->hPhone < 0) {
+ i = errno;
+ GSM_OSErrorInfo(s,"open in serial_open");
+ if (i == 2) return ERR_DEVICENOTEXIST; //no such file or directory
+ if (i == 13) return ERR_DEVICENOPERMISSION; //permission denied
+ return ERR_DEVICEOPENERROR;
+ }
+
+#ifdef TIOCEXCL
+ /* open() calls from other applications shall fail now */
+ ioctl(d->hPhone, TIOCEXCL, (char *) 0);
+#endif
+
+ if (tcgetattr(d->hPhone, &d->old_settings) == -1) {
+ close(d->hPhone);
+ GSM_OSErrorInfo(s,"tcgetattr in serial_open");
+ return ERR_DEVICEREADERROR;
+ }
+
+ if (tcflush(d->hPhone, TCIOFLUSH) == -1) {
+ serial_close(s);
+ GSM_OSErrorInfo(s,"tcflush in serial_open");
+ return ERR_DEVICEOPENERROR;
+ }
+
+ memcpy(&t, &d->old_settings, sizeof(struct termios));
+
+ /* Opening without parity */
+ t.c_iflag = IGNPAR;
+ t.c_oflag = 0;
+ /* disconnect line, 8 bits, enable receiver,
+ * ignore modem lines,lower modem line after disconnect
+ */
+ t.c_cflag = B0 | CS8 | CREAD | CLOCAL | HUPCL;
+ /* enable hardware (RTS/CTS) flow control (NON POSIX) */
+ /* t.c_cflag |= CRTSCTS; */
+ t.c_lflag = 0;
+ t.c_cc[VMIN] = 1;
+ t.c_cc[VTIME] = 0;
+
+ if (tcsetattr(d->hPhone, TCSANOW, &t) == -1) {
+ serial_close(s);
+ GSM_OSErrorInfo(s,"tcsetattr in serial_open");
+ return ERR_DEVICEOPENERROR;
+ }
+
+ /* Making file descriptor asynchronous. */
+ if (fcntl(d->hPhone, F_SETFL, FASYNC | FNONBLOCK) == -1) {
+ serial_close(s);
+ GSM_OSErrorInfo(s,"fcntl in serial_open");
+ return ERR_DEVICEOPENERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setparity(GSM_StateMachine *s, bool parity)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ struct termios t;
+
+ if (tcgetattr(d->hPhone, &t)) {
+ GSM_OSErrorInfo(s,"tcgetattr in serial_setparity");
+ return ERR_DEVICEREADERROR;
+ }
+
+ if (parity) {
+ t.c_cflag |= (PARENB | PARODD);
+ t.c_iflag = 0;
+ } else {
+ t.c_iflag = IGNPAR;
+ }
+
+ if (tcsetattr(d->hPhone, TCSANOW, &t) == -1){
+ serial_close(s);
+ GSM_OSErrorInfo(s,"tcsetattr in serial_setparity");
+ return ERR_DEVICEPARITYERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setdtrrts(GSM_StateMachine *s, bool dtr, bool rts)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ struct termios t;
+ unsigned int flags;
+
+ if (tcgetattr(d->hPhone, &t)) {
+ GSM_OSErrorInfo(s,"tcgetattr in serial_setdtrrts");
+ return ERR_DEVICEREADERROR;
+ }
+
+#ifdef CRTSCTS
+ /* Disabling hardware flow control */
+ t.c_cflag &= ~CRTSCTS;
+#endif
+
+ if (tcsetattr(d->hPhone, TCSANOW, &t) == -1) {
+ serial_close(s);
+ GSM_OSErrorInfo(s,"tcsetattr in serial_setdtrrts");
+ return ERR_DEVICEDTRRTSERROR;
+ }
+
+ flags = TIOCM_DTR;
+ if (dtr) {
+ ioctl(d->hPhone, TIOCMBIS, &flags);
+ } else {
+ ioctl(d->hPhone, TIOCMBIC, &flags);
+ }
+
+ flags = TIOCM_RTS;
+ if (rts) {
+ ioctl(d->hPhone, TIOCMBIS, &flags);
+ } else {
+ ioctl(d->hPhone, TIOCMBIC, &flags);
+ }
+
+ flags = 0;
+ ioctl(d->hPhone, TIOCMGET, &flags);
+
+ dbgprintf("Serial device:");
+ dbgprintf(" DTR is %s", flags&TIOCM_DTR?"up":"down");
+ dbgprintf(", RTS is %s", flags&TIOCM_RTS?"up":"down");
+ dbgprintf(", CAR is %s", flags&TIOCM_CAR?"up":"down");
+ dbgprintf(", CTS is %s\n", flags&TIOCM_CTS?"up":"down");
+ if (((flags&TIOCM_DTR)==TIOCM_DTR) != dtr) return ERR_DEVICEDTRRTSERROR;
+ if (((flags&TIOCM_RTS)==TIOCM_RTS) != rts) return ERR_DEVICEDTRRTSERROR;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setspeed(GSM_StateMachine *s, int speed)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ struct termios t;
+ int speed2 = B19200;
+
+ if (tcgetattr(d->hPhone, &t)) {
+ GSM_OSErrorInfo(s,"tcgetattr in serial_setspeed");
+ return ERR_DEVICEREADERROR;
+ }
+
+ smprintf(s, "Setting speed to %d\n", speed);
+
+ switch (speed) {
+ case 50: speed2 = B50; break;
+ case 75: speed2 = B75; break;
+ case 110: speed2 = B110; break;
+ case 134: speed2 = B134; break;
+ case 150: speed2 = B150; break;
+ case 200: speed2 = B200; break;
+ case 300: speed2 = B300; break;
+ case 600: speed2 = B600; break;
+ case 1200: speed2 = B1200; break;
+ case 1800: speed2 = B1800; break;
+ case 2400: speed2 = B2400; break;
+ case 4800: speed2 = B4800; break;
+ case 9600: speed2 = B9600; break;
+ case 19200: speed2 = B19200; break;
+ case 38400: speed2 = B38400; break;
+ case 57600: speed2 = B57600; break;
+ case 115200: speed2 = B115200; break;
+ case 230400: speed2 = B230400; break;
+ case 460800: speed2 = B460800; break;
+ case 500000: speed2 = B500000; break;
+ case 576000: speed2 = B576000; break;
+ case 921600: speed2 = B921600; break;
+ case 1000000: speed2 = B1000000; break;
+ case 1152000: speed2 = B1152000; break;
+ case 1500000: speed2 = B1500000; break;
+ case 2000000: speed2 = B2000000; break;
+ case 2500000: speed2 = B2500000; break;
+ case 3000000: speed2 = B3000000; break;
+ case 3500000: speed2 = B3500000; break;
+ case 4000000: speed2 = B4000000; break;
+ }
+
+ /* This should work on all systems because it is done according to POSIX */
+ cfsetispeed(&t, speed2);
+ cfsetospeed(&t, speed2);
+
+ if (tcsetattr(d->hPhone, TCSADRAIN, &t) == -1) {
+ serial_close(s);
+ GSM_OSErrorInfo(s,"tcsetattr in serial_setspeed");
+ return ERR_DEVICECHANGESPEEDERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static int serial_read(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ struct timeval timeout2;
+ fd_set readfds;
+ int actual = 0;
+
+ FD_ZERO(&readfds);
+ FD_SET(d->hPhone, &readfds);
+
+ timeout2.tv_sec = 0;
+ timeout2.tv_usec = 1;
+
+ if (select(d->hPhone+1, &readfds, NULL, NULL, &timeout2)) {
+ actual = read(d->hPhone, buf, nbytes);
+ if (actual == -1) GSM_OSErrorInfo(s,"serial_read");
+ }
+ return actual;
+}
+
+static int serial_write(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ int ret;
+ size_t actual = 0;
+
+ do {
+ ret = write(d->hPhone, (unsigned char *)buf, nbytes - actual);
+ if (ret < 0 && errno == EAGAIN) continue;
+ if (ret < 0) {
+ if (actual != nbytes) GSM_OSErrorInfo(s,"serial_write");
+ return actual;
+ }
+ actual += ret;
+ buf += ret;
+ if (s->ConnectionType == GCT_FBUS2PL2303) my_sleep(1);
+ } while (actual < nbytes);
+ return actual;
+}
+
+GSM_Device_Functions SerialDevice = {
+ serial_open,
+ serial_close,
+ serial_setparity,
+ serial_setdtrrts,
+ serial_setspeed,
+ serial_read,
+ serial_write
+};
+
+#endif
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_unx.h b/gammu/emb/common/device/serial/ser_unx.h
new file mode 100644
index 0000000..ca8fe5a
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_unx.h
@@ -0,0 +1,22 @@
+
+#ifndef WIN32
+#ifndef DJGPP
+#ifndef unixserial_h
+#define unixserial_h
+
+#include <fcntl.h>
+#include <sys/ioctl.h>
+#include <termios.h>
+
+typedef struct {
+ int hPhone;
+ struct termios old_settings;
+} GSM_Device_SerialData;
+
+#endif
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_w32.c b/gammu/emb/common/device/serial/ser_w32.c
new file mode 100644
index 0000000..9fa0135
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_w32.c
@@ -0,0 +1,340 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* based on some work from Gnokii, MSDN and others */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_SERIALDEVICE
+#ifdef WIN32
+
+#include <windows.h>
+#include <string.h>
+#include <stdio.h>
+#include <io.h>
+#include <memory.h>
+
+#include "../../gsmcomon.h"
+#include "ser_w32.h"
+
+static GSM_Error serial_close(GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ /* Disables all monitored events for device */
+ SetCommMask(d->hPhone, 0);
+
+ /* Discards all characters from input/output buffer and terminates
+ * pending read/write operations
+ */
+ PurgeComm(d->hPhone, PURGE_TXABORT | PURGE_RXABORT |
+ PURGE_TXCLEAR | PURGE_RXCLEAR);
+
+ /* Clears the DTR (data-terminal-ready) signal */
+ EscapeCommFunction(d->hPhone, CLRDTR);
+
+ /* Restores old settings */
+ if (SetCommState(d->hPhone, &d->old_settings)==0) {
+ GSM_OSErrorInfo(s, "SetCommState in serial_close");
+ }
+
+ /* Closes device */
+ if (CloseHandle(d->hPhone)==0) {
+ GSM_OSErrorInfo(s, "CloseHandle in serial_close");
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_open (GSM_StateMachine *s)
+{
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+ DCB dcb;
+ unsigned char DeviceName[80],DeviceName2[80];
+ int i;
+#ifdef GSM_ENABLE_FBUS2DKU5
+ HKEY hKey;
+ DWORD DeviceNameLen, KeyNameLen;
+ unsigned char KeyName[100];
+#endif
+
+ strcpy(DeviceName2,s->CurrentConfig->Device);
+
+#ifdef GSM_ENABLE_FBUS2DKU5
+ if (s->ConnectionType == GCT_FBUS2DKU5) {
+ smprintf(s,"Reading DKU5 device\n");
+ DeviceName2[0] = 0;
+ if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, "HARDWARE\\DEVICEMAP\\SERIALCOMM", 0, KEY_QUERY_VALUE, &hKey) != ERROR_SUCCESS) {
+ smprintf(s,"Error opening key\n");
+ return ERR_DEVICENOTWORK;
+ }
+ i = 0;
+ while(1) {
+ DeviceNameLen = 80;
+ KeyNameLen = 100;
+ if (RegEnumValue(hKey,i,KeyName,&KeyNameLen,NULL,NULL,DeviceName2,&DeviceNameLen) != ERROR_SUCCESS) {
+ smprintf(s,"Error reading key value\n");
+ return ERR_DEVICENOTWORK;
+ }
+// smprintf(s,"Key name is %s, value is %s\n",KeyName,DeviceName2);
+ if (!strncmp(KeyName,"\\Device\\AtmelVirtualPort",24)) break;
+ i++;
+ }
+ RegCloseKey(hKey);
+ if (strlen(DeviceName2) == 0) return ERR_DEVICENOTWORK;
+ smprintf(s,"DKU5 device is \"%s\"\n",DeviceName2);
+ //nodriver
+ }
+#endif
+
+ if ((s->ConnectionType == GCT_FBUS2DKU5) ||
+ (!strncmp(DeviceName2,"com",3) && strlen(DeviceName2)>3)) {
+ sprintf(DeviceName,"\\\\.\\COM%i",atoi(DeviceName2+3));
+ } else {
+ strcpy(DeviceName,DeviceName2);
+ }
+
+ smprintf(s,"Device is %s\n",DeviceName);
+
+ /* Allows for reading/writing, no device sharing */
+ d->hPhone = CreateFile(DeviceName,
+ GENERIC_READ | GENERIC_WRITE,
+ 0,
+ 0,
+ OPEN_EXISTING,
+ FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED,
+ NULL);
+
+ if (d->hPhone == INVALID_HANDLE_VALUE) {
+ i = GetLastError();
+ GSM_OSErrorInfo(s, "CreateFile in serial_open");
+ if (i == 2) return ERR_DEVICENOTWORK; //can't find specified file
+ if (i == 5) return ERR_DEVICEBUSY; //access denied
+ if (i == 31) return ERR_DEVICENOTWORK; //attached device not working
+ if (i == 123) return ERR_DEVICENOTEXIST;
+ return ERR_DEVICEOPENERROR;
+ }
+
+ d->old_settings.DCBlength = sizeof(DCB);
+ if (GetCommState(d->hPhone, &d->old_settings)==0) {
+ GSM_OSErrorInfo(s, "ReadDevice in serial_open");
+ return ERR_DEVICEREADERROR;
+ }
+
+ /* When char will be received, we will receive notifications */
+ SetCommMask(d->hPhone, EV_RXCHAR);
+
+ /* Sets size for input/output buffer */
+ SetupComm(d->hPhone, 4096, 4096);
+
+ /* Discards all characters from input/output buffer and terminates
+ * pending read/write operations
+ */
+ PurgeComm(d->hPhone, PURGE_TXABORT | PURGE_RXABORT |
+ PURGE_TXCLEAR | PURGE_RXCLEAR);
+
+ memcpy(&dcb, &d->old_settings, sizeof(DCB));
+
+ dcb.ByteSize = 8;
+ dcb.Parity = NOPARITY;
+ dcb.StopBits = ONESTOPBIT;
+
+ /* No Xon/Xof flow control */
+// dcb.fOutX = false;
+// dcb.fInX = false;
+
+ /* Hardware flow control */
+// dcb.fOutxDsrFlow = true;
+// dcb.fOutxCtsFlow = true;
+// dcb.fDtrControl = DTR_CONTROL_HANDSHAKE;
+// dcb.fRtsControl = RTS_CONTROL_HANDSHAKE;
+
+ /* Initialise the port settings */
+ if (SetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "WriteDevice in serial_open");
+ return ERR_DEVICEOPENERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setparity (GSM_StateMachine *s, bool parity)
+{
+ DCB dcb;
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ dcb.DCBlength = sizeof(DCB);
+ if (GetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "ReadDevice in serial_setparity");
+ return ERR_DEVICEREADERROR;
+ }
+
+ if (parity) {
+ dcb.Parity = ODDPARITY;
+ } else {
+ dcb.Parity = NOPARITY;
+ }
+
+ if (SetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "WriteDevice in serial_setparity");
+ return ERR_DEVICEPARITYERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setdtrrts(GSM_StateMachine *s, bool dtr, bool rts)
+{
+ DCB dcb;
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ dcb.DCBlength = sizeof(DCB);
+ if (GetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "ReadDevice in serial_setdtrrts");
+ return ERR_DEVICEREADERROR;
+ }
+
+ dcb.fOutxDsrFlow = 0;
+ dcb.fDtrControl = DTR_CONTROL_DISABLE;
+ if (dtr) dcb.fDtrControl = DTR_CONTROL_ENABLE;
+
+ dcb.fOutxCtsFlow = 0;
+ dcb.fRtsControl = RTS_CONTROL_DISABLE;
+ if (rts) dcb.fRtsControl = RTS_CONTROL_ENABLE;
+
+ /* no software (Xon/Xof) flow control */
+ dcb.fInX = dcb.fOutX = 0;
+
+ if (SetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "WriteDevice in serial_setdtrrts");
+ return ERR_DEVICEDTRRTSERROR;
+ }
+
+ /* the rest of function checks, if setting was really done */
+
+ dcb.DCBlength = sizeof(DCB);
+ GetCommState(d->hPhone, &dcb);
+
+ dbgprintf("Serial device:");
+ dbgprintf(" DTR is ");
+ switch (dcb.fDtrControl) {
+ case DTR_CONTROL_ENABLE : dbgprintf("up"); break;
+ case DTR_CONTROL_DISABLE : dbgprintf("down"); break;
+ case DTR_CONTROL_HANDSHAKE : dbgprintf("handshake"); break;
+ }
+ dbgprintf(", RTS is ");
+ switch (dcb.fRtsControl) {
+ case RTS_CONTROL_ENABLE : dbgprintf("up"); break;
+ case RTS_CONTROL_DISABLE : dbgprintf("down"); break;
+ case RTS_CONTROL_HANDSHAKE : dbgprintf("handshake"); break;
+ case RTS_CONTROL_TOGGLE : dbgprintf("toggle"); break;
+ }
+ dbgprintf("\n");
+ if ( dtr && dcb.fDtrControl != DTR_CONTROL_ENABLE ) return ERR_DEVICEDTRRTSERROR;
+ if (!dtr && dcb.fDtrControl != DTR_CONTROL_DISABLE) return ERR_DEVICEDTRRTSERROR;
+ if ( rts && dcb.fRtsControl != RTS_CONTROL_ENABLE ) return ERR_DEVICEDTRRTSERROR;
+ if (!rts && dcb.fRtsControl != RTS_CONTROL_DISABLE) return ERR_DEVICEDTRRTSERROR;
+
+ return ERR_NONE;
+}
+
+static GSM_Error serial_setspeed(GSM_StateMachine *s, int speed)
+{
+ DCB dcb;
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ dcb.DCBlength = sizeof(DCB);
+ if (GetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "ReadDevice in serial_setspeed");
+ return ERR_DEVICEREADERROR;
+ }
+
+ dcb.BaudRate = speed;
+
+ if (SetCommState(d->hPhone, &dcb)==0) {
+ GSM_OSErrorInfo(s, "WriteDevice in serial_setspeed");
+ return ERR_DEVICECHANGESPEEDERROR;
+ }
+
+ return ERR_NONE;
+}
+
+static int serial_read(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ COMSTAT ComStat;
+ DWORD ErrorFlags, Length;
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ /* Gets information about a communications error and
+ * current status of device
+ */
+ ClearCommError(d->hPhone, &ErrorFlags, &ComStat);
+ Length = ComStat.cbInQue;
+
+ /* Nothing to read */
+ if (Length <= 0) return Length;
+
+ /* Read without problems */
+ if (ReadFile(d->hPhone, buf, Length, &Length, &d->osRead)) return Length;
+
+ if (GetLastError() != ERROR_IO_PENDING) {
+ GSM_OSErrorInfo(s, "serial_read1");
+ Length = 0;
+ ClearCommError(d->hPhone, &ErrorFlags, &ComStat);
+ return Length;
+ }
+
+ while(1) {
+ if (GetOverlappedResult(d->hPhone,&d->osRead, &Length, TRUE)) break;
+ if (GetLastError() != ERROR_IO_INCOMPLETE) {
+ GSM_OSErrorInfo(s, "serial_read2");
+ /* an error occurred, try to recover */
+ ClearCommError(d->hPhone, &ErrorFlags, &ComStat);
+ break;
+ }
+ }
+ return Length;
+}
+
+static int serial_write(GSM_StateMachine *s, void *buf, size_t nbytes)
+{
+ DWORD BytesWritten,ErrorFlags,BytesSent=0;
+ COMSTAT ComStat;
+ GSM_Device_SerialData *d = &s->Device.Data.Serial;
+
+ if (WriteFile(d->hPhone, buf, nbytes, &BytesSent, &d->osWrite)) return BytesSent;
+
+ if (GetLastError() != ERROR_IO_PENDING) {
+ GSM_OSErrorInfo(s, "serial_write1");
+ ClearCommError(d->hPhone, &ErrorFlags, &ComStat);
+ return BytesSent;
+ }
+
+ while (1) {
+ if (GetOverlappedResult(d->hPhone, &d->osWrite, &BytesWritten, TRUE)) break;
+ if (GetLastError() != ERROR_IO_INCOMPLETE) {
+ GSM_OSErrorInfo(s, "serial_write2");
+ ClearCommError(d->hPhone, &ErrorFlags, &ComStat);
+ break;
+ }
+ BytesSent += BytesWritten;
+ }
+ BytesSent += BytesWritten;
+
+ return BytesSent;
+}
+
+GSM_Device_Functions SerialDevice = {
+ serial_open,
+ serial_close,
+ serial_setparity,
+ serial_setdtrrts,
+ serial_setspeed,
+ serial_read,
+ serial_write
+};
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/device/serial/ser_w32.h b/gammu/emb/common/device/serial/ser_w32.h
new file mode 100644
index 0000000..d226f32
--- a/dev/null
+++ b/gammu/emb/common/device/serial/ser_w32.h
@@ -0,0 +1,19 @@
+
+#ifdef WIN32
+#ifndef winserial_h
+#define winserial_h
+
+#include <windows.h>
+
+typedef struct {
+ HANDLE hPhone;
+ DCB old_settings;
+ OVERLAPPED osWrite,osRead;
+} GSM_Device_SerialData;
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/gammu.h b/gammu/emb/common/gammu.h
new file mode 100644
index 0000000..88d94c5
--- a/dev/null
+++ b/gammu/emb/common/gammu.h
@@ -0,0 +1,23 @@
+/* (c) 2003 by Michal Cihar */
+
+/*
+ * This file includes some core Gammu headers, that you will probably need.
+ */
+
+#include "config.h"
+#include "gsmcomon.h"
+#include "gsmstate.h"
+#include "misc/cfg.h"
+#include "misc/misc.h"
+#include "misc/coding/coding.h"
+#include "service/gsmpbk.h"
+#include "service/gsmring.h"
+#include "service/gsmlogo.h"
+#include "service/gsmnet.h"
+#include "service/backup/gsmback.h"
+#include "phone/pfunc.h"
+#include "phone/nokia/nfunc.h"
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/gsmcomon.c b/gammu/emb/common/gsmcomon.c
new file mode 100644
index 0000000..d094ef3
--- a/dev/null
+++ b/gammu/emb/common/gsmcomon.c
@@ -0,0 +1,260 @@
+/* (c) 2002-2004 by Marcin Wiacek & Michal Cihar */
+
+#include <ctype.h>
+#include <string.h>
+#include <time.h>
+#include <stdio.h>
+#ifndef __OpenBSD__
+# include <wchar.h>
+#endif
+#ifdef WIN32
+# include <windows.h>
+#else
+# include <stdlib.h>
+# include <errno.h>
+# include <fcntl.h>
+# include <signal.h>
+#endif
+
+#include "gsmcomon.h"
+#include "misc/coding/coding.h"
+
+GSM_Error NoneFunction(void)
+{
+ return ERR_NONE;
+}
+
+GSM_Error NotImplementedFunction(void)
+{
+ return ERR_NOTIMPLEMENTED;
+}
+
+GSM_Error NotSupportedFunction(void)
+{
+ return ERR_NOTSUPPORTED;
+}
+
+unsigned char *GetMsg (INI_Section *cfg, unsigned char *default_string)
+{
+ unsigned char *retval, buffer[40], buff2[40], buff[2000];
+ static unsigned char def_str[2000];
+ INI_Entry *e;
+ INI_Section *h;
+ int num;
+ int len;
+
+ if (cfg==NULL) return default_string;
+
+ EncodeUnicode (buff2, "common", 6);
+
+ /* Set all 0x0a to \n */
+ memset(def_str,0,sizeof(def_str));
+ for (num=0;num<((int)strlen(default_string));num++) {
+ if (default_string[num] == 0x0a) {
+ def_str[strlen(def_str)] = '\\';
+ def_str[strlen(def_str)] = 'n';
+ } else def_str[strlen(def_str)] = default_string[num];
+ }
+
+ e = NULL;
+ /* First find our section */
+ for (h = cfg; h != NULL; h = h->Next) {
+ if (mywstrncasecmp(buff2, h->SectionName, 0)) {
+ e = h->SubEntries;
+ break;
+ }
+ }
+ while (e != NULL) {
+ num = -1;
+ DecodeUnicode(e->EntryName,buffer);
+ if (strlen(buffer) == 5 && (buffer[0] == 'F' || buffer[0] == 'f')) {
+ num = atoi(buffer+2);
+ }
+ if (num!=-1) {
+ DecodeUnicode(e->EntryValue,buff);
+ /* Remove quotes */
+ if (buff[0] == '"') {
+ len = strlen(buff);
+ memmove(buff, buff + 1, len - 1);
+ if (buff[len - 2] == '"') buff[len - 2] = 0;
+ }
+ if (strcmp(buff, def_str) == 0) {
+ sprintf(buff,"T%04i",num);
+ EncodeUnicode (buffer, buff, 5);
+ retval = INI_GetValue(cfg, buff2, buffer, true);
+ if (retval) {
+ sprintf(buff,"%s",DecodeUnicodeConsole(retval+2));
+ buff[strlen(buff)-1] = 0;
+ /* Set all \n to 0x0a */
+ memset(def_str,0,sizeof(def_str));
+ num = 0;
+ while (num != (int)strlen(buff)) {
+ if (num < (int)strlen(buff) - 1) {
+ if (buff[num] == '\\' && buff[num+1] == 'n') {
+ def_str[strlen(def_str)] = 0x0a;
+ num+=2;
+ } else {
+ def_str[strlen(def_str)] = buff[num++];
+ }
+ } else {
+ def_str[strlen(def_str)] = buff[num++];
+ }
+ }
+ retval = def_str;
+ } else {
+ retval = default_string;
+ }
+ return retval;
+ }
+ }
+ e = e->Next;
+ }
+ return default_string;
+}
+
+typedef struct {
+ GSM_Error ErrorNum;
+ unsigned char *ErrorText;
+} PrintErrorEntry;
+
+static PrintErrorEntry PrintErrorEntries[] = {
+ {ERR_NONE, "No error."},
+ {ERR_DEVICEOPENERROR, "Error opening device. Unknown/busy or no permissions."},
+ {ERR_DEVICELOCKED, "Error opening device. Device locked."},
+ {ERR_DEVICENOTEXIST, "Error opening device. Not exist."},
+ {ERR_DEVICEBUSY, "Error opening device. Already opened by other application."},
+ {ERR_DEVICENOPERMISSION, "Error opening device. No permissions."},
+ {ERR_DEVICENODRIVER, "Error opening device. No required driver in operating system."},
+ {ERR_DEVICENOTWORK, "Error opening device. Some hardware not connected/wrong configured."},
+ {ERR_DEVICEDTRRTSERROR, "Error setting device DTR or RTS."},
+ {ERR_DEVICECHANGESPEEDERROR, "Error setting device speed. Maybe speed not supported."},
+ {ERR_DEVICEWRITEERROR, "Error writing device."},
+ {ERR_DEVICEREADERROR, "Error during reading device"},
+ {ERR_DEVICEPARITYERROR, "Can't set parity on device"},
+ {ERR_TIMEOUT, "No response in specified timeout. Probably phone not connected."},
+ /* Some missed */
+ {ERR_UNKNOWNRESPONSE, "Unknown response from phone. See readme.txt, how to report it."},
+ /* Some missed */
+ {ERR_UNKNOWNCONNECTIONTYPESTRING,"Unknown connection type string. Check config file."},
+ {ERR_UNKNOWNMODELSTRING, "Unknown model type string. Check config file."},
+ {ERR_SOURCENOTAVAILABLE, "Some required functions not compiled for your OS. Please contact."},
+ {ERR_NOTSUPPORTED, "Function not supported by phone."},
+ {ERR_EMPTY, "Entry is empty"},
+ {ERR_SECURITYERROR, "Security error. Maybe no PIN ?"},
+ {ERR_INVALIDLOCATION, "Invalid location. Maybe too high ?"},
+ {ERR_NOTIMPLEMENTED, "Function not implemented. Help required."},
+ {ERR_FULL, "Memory full."},
+ {ERR_UNKNOWN, "Unknown error."},
+ /* Some missed */
+ {ERR_CANTOPENFILE, "Can't open specified file. Read only ?"},
+ {ERR_MOREMEMORY, "More memory required..."},
+ {ERR_PERMISSION, "Permission to file/device required..."},
+ {ERR_EMPTYSMSC, "Empty SMSC number. Set in phone or use -smscnumber"},
+ {ERR_INSIDEPHONEMENU, "You're inside phone menu (during editing ?). Leave it and try again."},
+ {ERR_WORKINPROGRESS, "Function is during writing. If want help, please contact with authors."},
+ {ERR_PHONEOFF, "Phone is disabled and connected to charger"},
+ {ERR_FILENOTSUPPORTED, "File format not supported by Gammu"},
+ {ERR_BUG, "Nobody is perfect, some bug appeared in protocol implementation. Please contact authors."},
+ {ERR_CANCELED, "Transfer was canceled by phone (you pressed cancel on phone?)."},
+ /* Some missed */
+ {ERR_OTHERCONNECTIONREQUIRED, "Current connection type doesn't support called function."},
+ /* Some missed */
+ {ERR_INVALIDDATETIME, "Invalid date or time specified."},
+ {ERR_MEMORY, "Phone memory error, maybe it is read only"},
+ {ERR_INVALIDDATA, "Invalid data"},
+
+ {0, ""}
+};
+
+unsigned char *print_error(GSM_Error e, FILE *df, INI_Section *cfg)
+{
+ unsigned char *def = NULL;
+ int i = 0;
+
+ while (PrintErrorEntries[i].ErrorNum != 0) {
+ if (PrintErrorEntries[i].ErrorNum == e) {
+ def = PrintErrorEntries[i].ErrorText;
+ break;
+ }
+ i++;
+ }
+ if (def == NULL) def = "Unknown error.";
+ if (df!=NULL && di.dl!=0) fprintf(df,"[ERROR %i: %s]\n",e,def);
+
+ return GetMsg(cfg,def);
+}
+
+const char *GetGammuLocalePath(void)
+{
+#ifdef LOCALE_PATH
+ static const char Buffer[] = LOCALE_PATH;
+ return Buffer;
+#else
+ return NULL;
+#endif
+}
+
+const char *GetGammuVersion(void)
+{
+ static const char Buffer[] = VERSION;
+ return Buffer;
+}
+
+GSM_Error GSM_SetDebugFile(char *info, Debug_Info *privdi)
+{
+ FILE *testfile;
+
+ /* If we should use global file descriptor, use it */
+ if (privdi->use_global) {
+ /* Aren't we the changing the global di? */
+ if (privdi != &di) {
+ if (privdi->df != di.df &&
+ privdi->dl!=0 &&
+ fileno(privdi->df) != 1 &&
+ fileno(privdi->df) != 2)
+ fclose(privdi->df);
+ privdi->df = di.df;
+ return ERR_NONE;
+ }
+ } else {
+ /* If we should not use global file descriptor, don't even try use it */
+ if (privdi->df == di.df) privdi->df = stdout;
+ }
+
+ if (info[0]!=0 && privdi->dl != 0) {
+ switch (privdi->dl) {
+ case DL_BINARY:
+ testfile = fopen(info,"wcb");
+ break;
+ case DL_TEXTERROR:
+ case DL_TEXTERRORDATE:
+ testfile = fopen(info,"ac");
+ if (!testfile) {
+ dbgprintf("Can't open debug file\n");
+ return ERR_CANTOPENFILE;
+ }
+ fseek(testfile, 0, SEEK_END);
+ if (ftell(testfile) > 5000000) {
+ fclose(testfile);
+ testfile = fopen(info,"wc");
+ }
+ break;
+ default:
+ testfile = fopen(info,"wc");
+ }
+ if (!testfile) {
+ dbgprintf("Can't open debug file\n");
+ return ERR_CANTOPENFILE;
+ } else {
+ if (privdi->df && privdi->df != stdout) {
+ fclose(privdi->df);
+ }
+ privdi->df = testfile;
+ }
+ }
+ return ERR_NONE;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/gsmcomon.h b/gammu/emb/common/gsmcomon.h
new file mode 100644
index 0000000..cd36708
--- a/dev/null
+++ b/gammu/emb/common/gsmcomon.h
@@ -0,0 +1,92 @@
+/* (c) 2002-2004 by Marcin Wiacek & Michal Cihar */
+
+#ifndef __gsm_comon_h
+#define __gsm_comon_h
+
+#include <stdio.h>
+#ifndef WIN32
+# include <unistd.h>
+#endif
+
+#include "config.h"
+#include "misc/misc.h"
+#include "misc/cfg.h"
+
+#define MAX_MANUFACTURER_LENGTH 50
+#define MAX_MODEL_LENGTH 50
+#define MAX_VERSION_LENGTH 50
+#define MAX_VERSION_DATE_LENGTH 50
+#define MAX_IMEI_LENGTH 20
+#define PHONE_MAXSMSINFOLDER 200
+#define GSM_MAX_NUMBER_LENGTH 50
+#define GSM_MAXCALENDARTODONOTES 300
+
+/* GSM_Error is used to notify about errors */
+typedef enum {
+/* 1*/ ERR_NONE=1,
+ ERR_DEVICEOPENERROR, /* Error during opening device */
+ ERR_DEVICELOCKED, /* Device locked */
+ ERR_DEVICENOTEXIST,
+ ERR_DEVICEBUSY,
+ ERR_DEVICENOPERMISSION,
+ ERR_DEVICENODRIVER,
+ ERR_DEVICENOTWORK,
+ ERR_DEVICEDTRRTSERROR, /* Error during setting DTR/RTS in device */
+/*10*/ ERR_DEVICECHANGESPEEDERROR, /* Error during changing speed in device */
+ ERR_DEVICEWRITEERROR, /* Error during writing device */
+ ERR_DEVICEREADERROR, /* Error during reading device */
+ ERR_DEVICEPARITYERROR, /* Can't set parity on device */
+ ERR_TIMEOUT, /* Command timed out */
+ ERR_FRAMENOTREQUESTED, /* Frame handled, but not requested in this moment */
+ ERR_UNKNOWNRESPONSE, /* Response not handled by gammu */
+ ERR_UNKNOWNFRAME, /* Frame not handled by gammu */
+ ERR_UNKNOWNCONNECTIONTYPESTRING,/* Unknown connection type given by user */
+ ERR_UNKNOWNMODELSTRING, /* Unknown model given by user */
+/*20*/ ERR_SOURCENOTAVAILABLE, /* Some functions not compiled in your OS */
+ ERR_NOTSUPPORTED, /* Not supported by phone */
+ ERR_EMPTY, /* Empty phonebook entry, ... */
+ ERR_SECURITYERROR, /* Not allowed */
+ ERR_INVALIDLOCATION, /* Too high or too low location... */
+ ERR_NOTIMPLEMENTED, /* Function not implemented */
+ ERR_FULL, /* Memory is full */
+ ERR_UNKNOWN,
+ ERR_CANTOPENFILE, /* Error during opening file */
+ ERR_MOREMEMORY, /* More memory required */
+/*30*/ ERR_PERMISSION, /* No permission */
+ ERR_EMPTYSMSC, /* SMSC number is empty */
+ ERR_INSIDEPHONEMENU, /* Inside phone menu - can't make something */
+ ERR_NOTCONNECTED, /* Phone NOT connected - can't make something */
+ ERR_WORKINPROGRESS, /* Work in progress */
+ ERR_PHONEOFF, /* Phone is disabled and connected to charger */
+ ERR_FILENOTSUPPORTED, /* File format not supported by Gammu */
+ ERR_BUG, /* Found bug in implementation or phone */
+ ERR_CANCELED, /* Action was canceled by user */
+ ERR_NEEDANOTHERANSWER, /* Inside Gammu: phone module need to send another answer frame */
+/*40*/ ERR_OTHERCONNECTIONREQUIRED,
+ ERR_WRONGCRC,
+ ERR_INVALIDDATETIME, /* Invalid date/time */
+ ERR_MEMORY, /* Phone memory error, maybe it is read only */
+ ERR_INVALIDDATA /* Invalid data */
+} GSM_Error;
+
+extern GSM_Error NoneFunction (void);
+extern GSM_Error NotImplementedFunction (void);
+extern GSM_Error NotSupportedFunction (void);
+
+#define NONEFUNCTION (void *) NoneFunction
+#define NOTIMPLEMENTED (void *) NotImplementedFunction
+#define NOTSUPPORTED (void *) NotSupportedFunction
+
+unsigned char *GetMsg (INI_Section *cfg, unsigned char *default_string);
+unsigned char *print_error (GSM_Error e, FILE *df, INI_Section *cfg);
+
+GSM_Error GSM_SetDebugFile(char *info, Debug_Info *privdi);
+
+const char *GetGammuLocalePath(void);
+const char *GetGammuVersion(void);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/gsmstate.c b/gammu/emb/common/gsmstate.c
new file mode 100644
index 0000000..b8f5f89
--- a/dev/null
+++ b/gammu/emb/common/gsmstate.c
@@ -0,0 +1,1246 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* Phones ID (c) partially by Walek */
+
+#include <stdarg.h>
+#include <string.h>
+#include <errno.h>
+
+#include "gsmcomon.h"
+#include "gsmstate.h"
+#include "misc/cfg.h"
+#include "misc/coding/coding.h"
+#include "device/devfunc.h"
+
+static void GSM_RegisterConnection(GSM_StateMachine *s, unsigned int connection,
+ GSM_Device_Functions *device, GSM_Protocol_Functions *protocol)
+{
+ if ((unsigned int)s->ConnectionType == connection) {
+ s->Device.Functions = device;
+ s->Protocol.Functions = protocol;
+ }
+}
+
+static GSM_Error GSM_RegisterAllConnections(GSM_StateMachine *s, char *connection)
+{
+ /* We check here is used connection string type is correct for ANY
+ * OS. If not, we return with error, that string is incorrect at all
+ */
+ s->ConnectionType = 0;
+ if (mystrncasecmp("mbus" ,connection,0)) s->ConnectionType = GCT_MBUS2;
+ if (mystrncasecmp("fbus" ,connection,0)) s->ConnectionType = GCT_FBUS2;
+ if (mystrncasecmp("fbusdlr3" ,connection,0)) s->ConnectionType = GCT_FBUS2DLR3;
+ if (mystrncasecmp("fbusdku5" ,connection,0)) s->ConnectionType = GCT_FBUS2DKU5;
+ if (mystrncasecmp("fbuspl2303" ,connection,0)) s->ConnectionType = GCT_FBUS2PL2303;
+ if (mystrncasecmp("fbusblue" ,connection,0)) s->ConnectionType = GCT_FBUS2BLUE;
+ if (mystrncasecmp("fbusirda" ,connection,0)) s->ConnectionType = GCT_FBUS2IRDA;
+ if (mystrncasecmp("phonetblue" ,connection,0)) s->ConnectionType = GCT_PHONETBLUE;
+ if (mystrncasecmp("mrouterblue" ,connection,0)) s->ConnectionType = GCT_MROUTERBLUE;
+ if (mystrncasecmp("irdaphonet" ,connection,0)) s->ConnectionType = GCT_IRDAPHONET;
+ if (mystrncasecmp("irdaat" ,connection,0)) s->ConnectionType = GCT_IRDAAT;
+ if (mystrncasecmp("irdaobex" ,connection,0)) s->ConnectionType = GCT_IRDAOBEX;
+ if (mystrncasecmp("blueobex" ,connection,0)) s->ConnectionType = GCT_BLUEOBEX;
+ if (mystrncasecmp("bluefbus" ,connection,0)) s->ConnectionType = GCT_BLUEFBUS2;
+ if (mystrncasecmp("bluephonet" ,connection,0)) s->ConnectionType = GCT_BLUEPHONET;
+ if (mystrncasecmp("blueat" ,connection,0)) s->ConnectionType = GCT_BLUEAT;
+ if (mystrncasecmp("bluerfobex" ,connection,0)) s->ConnectionType = GCT_BLUEOBEX;
+ if (mystrncasecmp("bluerffbus" ,connection,0)) s->ConnectionType = GCT_BLUEFBUS2;
+ if (mystrncasecmp("bluerfphonet",connection,0)) s->ConnectionType = GCT_BLUEPHONET;
+ if (mystrncasecmp("bluerfat" ,connection,0)) s->ConnectionType = GCT_BLUEAT;
+
+ /* These are for compatibility only */
+ if (mystrncasecmp("atblue" ,connection,0)) s->ConnectionType = GCT_BLUEAT;
+ if (mystrncasecmp("dlr3blue" ,connection,0)) s->ConnectionType = GCT_BLUEFBUS2;
+ if (mystrncasecmp("irda" ,connection,0)) s->ConnectionType = GCT_IRDAPHONET;
+ if (mystrncasecmp("dlr3" ,connection,0)) s->ConnectionType = GCT_FBUS2DLR3;
+ if (mystrncasecmp("infrared" ,connection,0)) s->ConnectionType = GCT_FBUS2IRDA;
+
+ if (mystrncasecmp("at" ,connection,2)) {
+ /* Use some resonable default, when no speed defined */
+ if (strlen(connection) == 2) {
+ s->Speed = 19200;
+ } else {
+ s->Speed = FindSerialSpeed(connection+2);
+ }
+ if (s->Speed != 0) s->ConnectionType = GCT_AT;
+ }
+ if (s->ConnectionType==0) return ERR_UNKNOWNCONNECTIONTYPESTRING;
+
+ /* We check now if user gave connection type compiled & available
+ * for used OS (if not, we return, that source not available)
+ */
+ s->Device.Functions = NULL;
+ s->Protocol.Functions = NULL;
+#ifdef GSM_ENABLE_MBUS2
+ GSM_RegisterConnection(s, GCT_MBUS2, &SerialDevice, &MBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2
+ GSM_RegisterConnection(s, GCT_FBUS2, &SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2DLR3
+ GSM_RegisterConnection(s, GCT_FBUS2DLR3, &SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2DKU5
+ GSM_RegisterConnection(s, GCT_FBUS2DKU5, &SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2PL2303
+ GSM_RegisterConnection(s, GCT_FBUS2PL2303,&SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2BLUE
+ GSM_RegisterConnection(s, GCT_FBUS2BLUE, &SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_FBUS2IRDA
+ GSM_RegisterConnection(s, GCT_FBUS2IRDA, &SerialDevice, &FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_PHONETBLUE
+ GSM_RegisterConnection(s, GCT_PHONETBLUE,&SerialDevice, &PHONETProtocol);
+#endif
+#ifdef GSM_ENABLE_MROUTERBLUE
+ GSM_RegisterConnection(s, GCT_MROUTERBLUE,&SerialDevice, &MROUTERProtocol);
+#endif
+#ifdef GSM_ENABLE_IRDAPHONET
+ GSM_RegisterConnection(s, GCT_IRDAPHONET,&IrdaDevice, &PHONETProtocol);
+#endif
+#ifdef GSM_ENABLE_BLUEFBUS2
+ GSM_RegisterConnection(s, GCT_BLUEFBUS2, &BlueToothDevice,&FBUS2Protocol);
+#endif
+#ifdef GSM_ENABLE_BLUEPHONET
+ GSM_RegisterConnection(s, GCT_BLUEPHONET,&BlueToothDevice,&PHONETProtocol);
+#endif
+#ifdef GSM_ENABLE_BLUEAT
+ GSM_RegisterConnection(s, GCT_BLUEAT, &BlueToothDevice,&ATProtocol);
+#endif
+#ifdef GSM_ENABLE_AT
+ GSM_RegisterConnection(s, GCT_AT, &SerialDevice, &ATProtocol);
+#endif
+#ifdef GSM_ENABLE_IRDAAT
+ GSM_RegisterConnection(s, GCT_IRDAAT, &IrdaDevice, &ATProtocol);
+#endif
+#ifdef GSM_ENABLE_IRDAOBEX
+ GSM_RegisterConnection(s, GCT_IRDAOBEX, &IrdaDevice, &OBEXProtocol);
+#endif
+#ifdef GSM_ENABLE_BLUEOBEX
+ GSM_RegisterConnection(s, GCT_BLUEOBEX, &BlueToothDevice,&OBEXProtocol);
+#endif
+ if (s->Device.Functions==NULL || s->Protocol.Functions==NULL)
+ return ERR_SOURCENOTAVAILABLE;
+ return ERR_NONE;
+}
+
+static void GSM_RegisterModule(GSM_StateMachine *s,GSM_Phone_Functions *phone)
+{
+ /* Auto model */
+ if (s->CurrentConfig->Model[0] == 0) {
+ if (strstr(phone->models,GetModelData(NULL,s->Phone.Data.Model,NULL)->model) != NULL) {
+ smprintf(s,"[Module - \"%s\"]\n",phone->models);
+ s->Phone.Functions = phone;
+ }
+ } else {
+ if (strstr(phone->models,s->CurrentConfig->Model) != NULL) {
+ smprintf(s,"[Module - \"%s\"]\n",phone->models);
+ s->Phone.Functions = phone;
+ }
+ }
+}
+
+GSM_Error GSM_RegisterAllPhoneModules(GSM_StateMachine *s)
+{
+ OnePhoneModel *model;
+
+ /* Auto model */
+ if (s->CurrentConfig->Model[0] == 0) {
+ model = GetModelData(NULL,s->Phone.Data.Model,NULL);
+#ifdef GSM_ENABLE_ALCATEL
+ if (model->model[0] != 0 && IsPhoneFeatureAvailable(model, F_ALCATEL)) {
+ smprintf(s,"[Module - \"%s\"]\n",ALCATELPhone.models);
+ s->Phone.Functions = &ALCATELPhone;
+ return ERR_NONE;
+ }
+#endif
+#ifdef GSM_ENABLE_ATGEN
+ /* With ATgen and auto model we can work with unknown models too */
+ if (s->ConnectionType==GCT_AT || s->ConnectionType==GCT_BLUEAT || s->ConnectionType==GCT_IRDAAT) {
+ smprintf(s,"[Module - \"%s\"]\n",ATGENPhone.models);
+ s->Phone.Functions = &ATGENPhone;
+ return ERR_NONE;
+ }
+#endif
+ if (model->model[0] == 0) return ERR_UNKNOWNMODELSTRING;
+ }
+ s->Phone.Functions=NULL;
+#ifdef GSM_ENABLE_ATGEN
+ /* AT module can have the same models ID to "normal" Nokia modules */
+ if (s->ConnectionType==GCT_AT || s->ConnectionType==GCT_BLUEAT || s->ConnectionType==GCT_IRDAAT) {
+ GSM_RegisterModule(s,&ATGENPhone);
+ if (s->Phone.Functions!=NULL) return ERR_NONE;
+ }
+#endif
+#ifdef GSM_ENABLE_OBEXGEN
+ GSM_RegisterModule(s,&OBEXGENPhone);
+#endif
+#ifdef GSM_ENABLE_MROUTERGEN
+ GSM_RegisterModule(s,&MROUTERGENPhone);
+#endif
+#ifdef GSM_ENABLE_NOKIA3320
+ GSM_RegisterModule(s,&N3320Phone);
+#endif
+#ifdef GSM_ENABLE_NOKIA3650
+ GSM_RegisterModule(s,&N3650Phone);
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ GSM_RegisterModule(s,&N6110Phone);
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ GSM_RegisterModule(s,&N6510Phone);
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ GSM_RegisterModule(s,&N7110Phone);
+#endif
+#ifdef GSM_ENABLE_NOKIA9210
+ GSM_RegisterModule(s,&N9210Phone);
+#endif
+#ifdef GSM_ENABLE_ALCATEL
+ GSM_RegisterModule(s,&ALCATELPhone);
+#endif
+ if (s->Phone.Functions==NULL) return ERR_UNKNOWNMODELSTRING;
+ return ERR_NONE;
+}
+
+GSM_Error GSM_InitConnection(GSM_StateMachine *s, int ReplyNum)
+{
+ GSM_Error error;
+ GSM_DateTime time;
+ int i;
+ char Buffer[80];
+
+ for (i=0;i<s->ConfigNum;i++) {
+ s->CurrentConfig = &s->Config[i];
+
+ s->Speed = 0;
+ s->ReplyNum = ReplyNum;
+ s->Phone.Data.ModelInfo = GetModelData("unknown",NULL,NULL);
+ s->Phone.Data.Manufacturer[0] = 0;
+ s->Phone.Data.Model[0] = 0;
+ s->Phone.Data.Version[0] = 0;
+ s->Phone.Data.VerDate[0] = 0;
+ s->Phone.Data.VerNum = 0;
+ s->Phone.Data.StartInfoCounter = 0;
+ s->Phone.Data.SentMsg = NULL;
+
+ s->Phone.Data.HardwareCache[0] = 0;
+ s->Phone.Data.ProductCodeCache[0] = 0;
+ s->Phone.Data.EnableIncomingCall = false;
+ s->Phone.Data.EnableIncomingSMS = false;
+ s->Phone.Data.EnableIncomingCB = false;
+ s->Phone.Data.EnableIncomingUSSD = false;
+ s->User.UserReplyFunctions = NULL;
+ s->User.IncomingCall = NULL;
+ s->User.IncomingSMS = NULL;
+ s->User.IncomingCB = NULL;
+ s->User.IncomingUSSD = NULL;
+ s->User.SendSMSStatus = NULL;
+ s->LockFile = NULL;
+ s->opened = false;
+ s->Phone.Functions = NULL;
+
+ s->di = di;
+ s->di.use_global = s->CurrentConfig->UseGlobalDebugFile;
+ GSM_SetDebugLevel(s->CurrentConfig->DebugLevel, &s->di);
+ error=GSM_SetDebugFile(s->CurrentConfig->DebugFile, &s->di);
+ if (error != ERR_NONE) return error;
+
+ if (s->di.dl == DL_TEXTALL || s->di.dl == DL_TEXT || s->di.dl == DL_TEXTERROR ||
+ s->di.dl == DL_TEXTALLDATE || s->di.dl == DL_TEXTDATE || s->di.dl == DL_TEXTERRORDATE) {
+ smprintf(s,"[Gammu - version %s built %s %s]\n",VERSION,__TIME__,__DATE__);
+ smprintf(s,"[Connection - \"%s\"]\n",s->CurrentConfig->Connection);
+ smprintf(s,"[Model type - \"%s\"]\n",s->CurrentConfig->Model);
+ smprintf(s,"[Device - \"%s\"]\n",s->CurrentConfig->Device);
+
+ Buffer[0] = 0;
+ if (strlen(GetOS()) != 0) sprintf(Buffer,"%s",GetOS());
+ if (strlen(GetCompiler()) != 0) {
+ if (Buffer[0] != 0) strcat(Buffer+strlen(Buffer),", ");
+ strcat(Buffer+strlen(Buffer),GetCompiler());
+ }
+ if (Buffer[0] != 0) smprintf(s,"[OS/compiler - %s]\n",Buffer);
+ }
+ if (s->di.dl==DL_BINARY) {
+ smprintf(s,"%c",((unsigned char)strlen(VERSION)));
+ smprintf(s,"%s",VERSION);
+ }
+
+ error=GSM_RegisterAllConnections(s, s->CurrentConfig->Connection);
+ if (error!=ERR_NONE) return error;
+
+ /* Model auto */
+ if (s->CurrentConfig->Model[0]==0) {
+ if (mystrncasecmp(s->CurrentConfig->LockDevice,"yes",0)) {
+ error = lock_device(s->CurrentConfig->Device, &(s->LockFile));
+ if (error != ERR_NONE) return error;
+ }
+
+ /* Irda devices can set now model to some specific and
+ * we don't have to make auto detection later */
+ error=s->Device.Functions->OpenDevice(s);
+ if (i != s->ConfigNum - 1) {
+ if (error == ERR_DEVICEOPENERROR) continue;
+ if (error == ERR_DEVICELOCKED) continue;
+ if (error == ERR_DEVICENOTEXIST) continue;
+ if (error == ERR_DEVICEBUSY) continue;
+ if (error == ERR_DEVICENOPERMISSION) continue;
+ if (error == ERR_DEVICENODRIVER) continue;
+ if (error == ERR_DEVICENOTWORK) continue;
+ }
+ if (error!=ERR_NONE) {
+ if (s->LockFile!=NULL) unlock_device(&(s->LockFile));
+ return error;
+ }
+
+ s->opened = true;
+
+ error=s->Protocol.Functions->Initialise(s);
+ if (error!=ERR_NONE) return error;
+
+ /* If still auto model, try to get model by asking phone for it */
+ if (s->Phone.Data.Model[0]==0) {
+ smprintf(s,"[Module - \"auto\"]\n");
+ switch (s->ConnectionType) {
+#ifdef GSM_ENABLE_ATGEN
+ case GCT_AT:
+ case GCT_BLUEAT:
+ case GCT_IRDAAT:
+ s->Phone.Functions = &ATGENPhone;
+ break;
+#endif
+#ifdef GSM_ENABLE_OBEXGEN
+ case GCT_IRDAOBEX:
+ case GCT_BLUEOBEX:
+ s->Phone.Functions = &OBEXGENPhone;
+ break;
+#endif
+#ifdef GSM_ENABLE_MROUTERGEN
+ case GCT_MROUTERBLUE:
+ s->Phone.Functions = &MROUTERGENPhone;
+ break;
+#endif
+#if defined(GSM_ENABLE_NOKIA_DCT3) || defined(GSM_ENABLE_NOKIA_DCT4)
+ case GCT_MBUS2:
+ case GCT_FBUS2:
+ case GCT_FBUS2DLR3:
+ case GCT_FBUS2DKU5:
+ case GCT_FBUS2PL2303:
+ case GCT_FBUS2BLUE:
+ case GCT_FBUS2IRDA:
+ case GCT_PHONETBLUE:
+ case GCT_IRDAPHONET:
+ case GCT_BLUEFBUS2:
+ case GCT_BLUEPHONET:
+ s->Phone.Functions = &NAUTOPhone;
+ break;
+#endif
+ default:
+ s->Phone.Functions = NULL;
+ }
+ if (s->Phone.Functions == NULL) return ERR_UNKNOWN;
+
+ /* Please note, that AT module need to send first
+ * command for enabling echo
+ */
+ error=s->Phone.Functions->Initialise(s);
+ if (error == ERR_TIMEOUT && i != s->ConfigNum - 1) continue;
+ if (error != ERR_NONE) return error;
+
+ error=s->Phone.Functions->GetModel(s);
+ if (error == ERR_TIMEOUT && i != s->ConfigNum - 1) continue;
+ if (error != ERR_NONE) return error;
+ }
+ }
+
+ /* Switching to "correct" module */
+ error=GSM_RegisterAllPhoneModules(s);
+ if (error!=ERR_NONE) return error;
+
+ /* We didn't open device earlier ? Make it now */
+ if (!s->opened) {
+ if (mystrncasecmp(s->CurrentConfig->LockDevice,"yes",0)) {
+ error = lock_device(s->CurrentConfig->Device, &(s->LockFile));
+ if (error != ERR_NONE) return error;
+ }
+
+ error=s->Device.Functions->OpenDevice(s);
+ if (i != s->ConfigNum - 1) {
+ if (error == ERR_DEVICEOPENERROR) continue;
+ if (error == ERR_DEVICELOCKED) continue;
+ if (error == ERR_DEVICENOTEXIST) continue;
+ if (error == ERR_DEVICEBUSY) continue;
+ if (error == ERR_DEVICENOPERMISSION) continue;
+ if (error == ERR_DEVICENODRIVER) continue;
+ if (error == ERR_DEVICENOTWORK) continue;
+ }
+ if (error!=ERR_NONE) {
+ if (s->LockFile!=NULL) unlock_device(&(s->LockFile));
+ return error;
+ }
+
+ s->opened = true;
+
+ error=s->Protocol.Functions->Initialise(s);
+ if (error!=ERR_NONE) return error;
+ }
+
+ error=s->Phone.Functions->Initialise(s);
+ if (error == ERR_TIMEOUT && i != s->ConfigNum - 1) continue;
+ if (error != ERR_NONE) return error;
+
+ if (mystrncasecmp(s->CurrentConfig->StartInfo,"yes",0)) {
+ s->Phone.Functions->ShowStartInfo(s,true);
+ s->Phone.Data.StartInfoCounter = 30;
+ }
+
+ if (mystrncasecmp(s->CurrentConfig->SyncTime,"yes",0)) {
+ GSM_GetCurrentDateTime (&time);
+ s->Phone.Functions->SetDateTime(s,&time);
+ }
+
+ /* For debug it's good to have firmware and real model version and manufacturer */
+ error=s->Phone.Functions->GetManufacturer(s);
+ if (error == ERR_TIMEOUT && i != s->ConfigNum - 1) continue;
+ if (error != ERR_NONE) return error;
+ error=s->Phone.Functions->GetModel(s);
+ if (error != ERR_NONE) return error;
+ error=s->Phone.Functions->GetFirmware(s);
+ if (error != ERR_NONE) return error;
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWN;
+}
+
+int GSM_ReadDevice (GSM_StateMachine *s, bool wait)
+{
+ unsigned char buff[255];
+ int res = 0, count;
+
+ unsigned int i;
+ GSM_DateTime Date;
+
+ GSM_GetCurrentDateTime (&Date);
+ i=Date.Second;
+ while (i==Date.Second) {
+ res = s->Device.Functions->ReadDevice(s, buff, 255);
+ if (!wait) break;
+ if (res > 0) break;
+ my_sleep(5);
+ GSM_GetCurrentDateTime(&Date);
+ }
+
+ for (count = 0; count < res; count++)
+ s->Protocol.Functions->StateMachine(s,buff[count]);
+
+ return res;
+}
+
+GSM_Error GSM_TerminateConnection(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if (!s->opened) return ERR_UNKNOWN;
+
+ smprintf(s,"[Closing]\n");
+
+ if (mystrncasecmp(s->CurrentConfig->StartInfo,"yes",0)) {
+ if (s->Phone.Data.StartInfoCounter > 0) s->Phone.Functions->ShowStartInfo(s,false);
+ }
+
+ if (s->Phone.Functions != NULL) {
+ error=s->Phone.Functions->Terminate(s);
+ if (error!=ERR_NONE) return error;
+ }
+
+ error=s->Protocol.Functions->Terminate(s);
+ if (error!=ERR_NONE) return error;
+
+ error = s->Device.Functions->CloseDevice(s);
+ if (error!=ERR_NONE) return error;
+
+ s->Phone.Data.ModelInfo = NULL;
+ s->Phone.Data.Manufacturer[0] = 0;
+ s->Phone.Data.Model[0] = 0;
+ s->Phone.Data.Version[0] = 0;
+ s->Phone.Data.VerDate[0] = 0;
+ s->Phone.Data.VerNum = 0;
+
+ if (s->LockFile!=NULL) unlock_device(&(s->LockFile));
+
+ if (!s->di.use_global && s->di.dl!=0 && fileno(s->di.df) != 1 && fileno(s->di.df) != 2) fclose(s->di.df);
+
+ s->opened = false;
+
+ return ERR_NONE;
+}
+
+GSM_Error GSM_WaitForOnce(GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type, int time)
+{
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+ GSM_Protocol_Message sentmsg;
+ int i;
+
+ i=0;
+ do {
+ if (length != 0) {
+ sentmsg.Length = length;
+ sentmsg.Type = type;
+ sentmsg.Buffer = (unsigned char *)malloc(length);
+ memcpy(sentmsg.Buffer,buffer,length);
+ Phone->SentMsg = &sentmsg;
+ }
+
+ /* Some data received. Reset timer */
+ if (GSM_ReadDevice(s,true)!=0) i=0;
+
+ if (length != 0) {
+ free (sentmsg.Buffer);
+ Phone->SentMsg = NULL;
+ }
+
+ /* Request completed */
+ if (Phone->RequestID==ID_None) return Phone->DispatchError;
+
+ i++;
+ } while (i<time);
+
+ return ERR_TIMEOUT;
+}
+
+GSM_Error GSM_WaitFor (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type, int time,
+ GSM_Phone_RequestID request)
+{
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+ GSM_Error error;
+ int reply;
+
+ if (mystrncasecmp(s->CurrentConfig->StartInfo,"yes",0)) {
+ if (Phone->StartInfoCounter > 0) {
+ Phone->StartInfoCounter--;
+ if (Phone->StartInfoCounter == 0) s->Phone.Functions->ShowStartInfo(s,false);
+ }
+ }
+
+ Phone->RequestID = request;
+ Phone->DispatchError = ERR_TIMEOUT;
+
+ for (reply=0;reply<s->ReplyNum;reply++) {
+ if (reply!=0) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl == DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl == DL_TEXTERRORDATE)
+ {
+ smprintf(s, "[Retrying %i type 0x%02X]\n", reply, type);
+ }
+ }
+ error = s->Protocol.Functions->WriteMessage(s, buffer, length, type);
+ if (error!=ERR_NONE) return error;
+
+ error = GSM_WaitForOnce(s, buffer, length, type, time);
+ if (error != ERR_TIMEOUT) return error;
+ }
+
+ return Phone->DispatchError;
+}
+
+static GSM_Error CheckReplyFunctions(GSM_StateMachine *s, GSM_Reply_Function *Reply, int *reply)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ GSM_Protocol_Message *msg = s->Phone.Data.RequestMsg;
+ bool execute;
+ bool available = false;
+ int i = 0;
+
+ while (Reply[i].requestID!=ID_None) {
+ execute=false;
+ /* Binary frames like in Nokia */
+ if (strlen(Reply[i].msgtype) < 2) {
+ if (Reply[i].msgtype[0]==msg->Type) {
+ if (Reply[i].subtypechar!=0) {
+ if (Reply[i].subtypechar<=msg->Length) {
+ if (msg->Buffer[Reply[i].subtypechar]==Reply[i].subtype)
+ execute=true;
+ }
+ } else execute=true;
+ }
+ } else {
+ if (strncmp(Reply[i].msgtype,msg->Buffer,strlen(Reply[i].msgtype))==0) {
+ execute=true;
+ }
+ }
+
+ if (execute) {
+ *reply=i;
+ if (Reply[i].requestID == ID_IncomingFrame ||
+ Reply[i].requestID == Data->RequestID ||
+ Data->RequestID == ID_EachFrame) {
+ return ERR_NONE;
+ }
+ available=true;
+ }
+ i++;
+ }
+
+ if (available) {
+ return ERR_FRAMENOTREQUESTED;
+ } else {
+ return ERR_UNKNOWNFRAME;
+ }
+}
+
+GSM_Error GSM_DispatchMessage(GSM_StateMachine *s)
+{
+ GSM_Error error = ERR_UNKNOWNFRAME;
+ GSM_Protocol_Message *msg = s->Phone.Data.RequestMsg;
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+ bool disp = false;
+ GSM_Reply_Function *Reply;
+ int reply, i;
+
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "RECEIVED frame ");
+ smprintf(s, "type 0x%02X/length 0x%02X/%i", msg->Type, msg->Length, msg->Length);
+ DumpMessage(s->di.use_global ? di.df : s->di.df, s->di.dl, msg->Buffer, msg->Length);
+ if (msg->Length == 0) smprintf(s, "\n");
+ fflush(s->di.df);
+ }
+ if (s->di.dl==DL_BINARY) {
+ smprintf(s,"%c",0x02); /* Receiving */
+ smprintf(s,"%c",msg->Type);
+ smprintf(s,"%c",msg->Length/256);
+ smprintf(s,"%c",msg->Length%256);
+ for (i=0;i<msg->Length;i++) {
+ smprintf(s,"%c",msg->Buffer[i]);
+ }
+ }
+
+ Reply=s->User.UserReplyFunctions;
+ if (Reply!=NULL) error=CheckReplyFunctions(s,Reply,&reply);
+
+ if (error==ERR_UNKNOWNFRAME) {
+ Reply=s->Phone.Functions->ReplyFunctions;
+ error=CheckReplyFunctions(s,Reply,&reply);
+ }
+
+ if (error==ERR_NONE) {
+ error=Reply[reply].Function(*msg, s);
+ if (Reply[reply].requestID==Phone->RequestID) {
+ if (error == ERR_NEEDANOTHERANSWER) {
+ error = ERR_NONE;
+ } else {
+ Phone->RequestID=ID_None;
+ }
+ }
+ }
+
+ if (strcmp(s->Phone.Functions->models,"NAUTO")) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ disp = true;
+ switch (error) {
+ case ERR_UNKNOWNRESPONSE:
+ smprintf(s, "\nUNKNOWN response");
+ break;
+ case ERR_UNKNOWNFRAME:
+ smprintf(s, "\nUNKNOWN frame");
+ break;
+ case ERR_FRAMENOTREQUESTED:
+ smprintf(s, "\nFrame not request now");
+ break;
+ default:
+ disp = false;
+ }
+ }
+
+ if (error == ERR_UNKNOWNFRAME || error == ERR_FRAMENOTREQUESTED) {
+ error = ERR_TIMEOUT;
+ }
+ }
+
+ if (disp) {
+ smprintf(s,". If you can, PLEASE report it (see readme.txt). THANK YOU\n");
+ if (Phone->SentMsg != NULL) {
+ smprintf(s,"LAST SENT frame ");
+ smprintf(s, "type 0x%02X/length %i", Phone->SentMsg->Type, Phone->SentMsg->Length);
+ DumpMessage(s->di.use_global ? di.df : s->di.df, s->di.dl, Phone->SentMsg->Buffer, Phone->SentMsg->Length);
+ }
+ smprintf(s, "RECEIVED frame ");
+ smprintf(s, "type 0x%02X/length 0x%02X/%i", msg->Type, msg->Length, msg->Length);
+ DumpMessage(s->di.use_global ? di.df : s->di.df, s->di.dl, msg->Buffer, msg->Length);
+ smprintf(s, "\n");
+ }
+
+ return error;
+}
+
+INI_Section *GSM_FindGammuRC(void)
+{
+ INI_Section *ini_file;
+ char *HomeDrive,*HomePath,*FileName=malloc(1);
+ int FileNameUsed=1;
+
+ FileName[0] = 0;
+#if defined(WIN32) || defined(DJGPP)
+ HomeDrive = getenv("HOMEDRIVE");
+ if (HomeDrive) {
+ FileName = realloc(FileName,FileNameUsed+strlen(HomeDrive)+1);
+ FileName = strcat(FileName, HomeDrive);
+ FileNameUsed += strlen(HomeDrive)+1;
+ }
+ HomePath = getenv("HOMEPATH");
+ if (HomePath) {
+ FileName = realloc(FileName,FileNameUsed+strlen(HomePath)+1);
+ FileName = strcat(FileName, HomePath);
+ FileNameUsed += strlen(HomePath)+1;
+ }
+ FileName = realloc(FileName,FileNameUsed+8+1);
+ strcat(FileName, "\\gammurc");
+#else
+ HomeDrive = NULL;
+ HomePath = getenv("HOME");
+ if (HomePath) {
+ FileName = realloc(FileName,FileNameUsed+strlen(HomePath)+1);
+ FileName = strcat(FileName, HomePath);
+ FileNameUsed += strlen(HomePath)+1;
+ }
+ FileName = realloc(FileName,FileNameUsed+9+1);
+ strcat(FileName, "/.gammurc");
+#endif
+// dbgprintf("\"%s\"\n",FileName);
+
+ ini_file = INI_ReadFile(FileName, false);
+ free(FileName);
+ if (ini_file == NULL) {
+#if defined(WIN32) || defined(DJGPP)
+ ini_file = INI_ReadFile("gammurc", false);
+ if (ini_file == NULL) return NULL;
+#else
+ ini_file = INI_ReadFile("/etc/gammurc", false);
+ if (ini_file == NULL) return NULL;
+#endif
+ }
+
+ return ini_file;
+}
+
+bool GSM_ReadConfig(INI_Section *cfg_info, GSM_Config *cfg, int num)
+{
+ INI_Section *h;
+ unsigned char section[50];
+ bool found = false;
+
+#if defined(WIN32) || defined(DJGPP)
+ char *DefaultPort = "com2:";
+#else
+ char *DefaultPort = "/dev/ttyS1";
+#endif
+ char *DefaultModel = "";
+ char *DefaultConnection = "fbus";
+ char *DefaultSynchronizeTime = "no";
+ char *DefaultDebugFile = "";
+ char *DefaultDebugLevel = "";
+ char *DefaultLockDevice = "no";
+ char *DefaultStartInfo = "no";
+ char *Temp;
+
+ /* By default all debug output will go to one filedescriptor */
+ bool DefaultUseGlobalDebugFile = true;
+
+ cfg->Device = DefaultPort;
+ cfg->Connection = DefaultConnection;
+ cfg->SyncTime = DefaultSynchronizeTime;
+ cfg->DebugFile = DefaultDebugFile;
+ strcpy(cfg->Model,DefaultModel);
+ strcpy(cfg->DebugLevel,DefaultDebugLevel);
+ cfg->LockDevice = DefaultLockDevice;
+ cfg->StartInfo = DefaultStartInfo;
+ cfg->DefaultDevice = true;
+ cfg->DefaultModel = true;
+ cfg->DefaultConnection = true;
+ cfg->DefaultSyncTime = true;
+ cfg->DefaultDebugFile = true;
+ cfg->DefaultDebugLevel = true;
+ cfg->DefaultLockDevice = true;
+ cfg->DefaultStartInfo = true;
+
+ cfg->UseGlobalDebugFile = DefaultUseGlobalDebugFile;
+
+ if (cfg_info==NULL) return false;
+
+ if (num == 0) {
+ sprintf(section,"gammu");
+ } else {
+ sprintf(section,"gammu%i",num);
+ }
+ for (h = cfg_info; h != NULL; h = h->Next) {
+ if (mystrncasecmp(section, h->SectionName, strlen(section))) {
+ found = true;
+ break;
+ }
+ }
+ if (!found) return false;
+
+ cfg->Device = INI_GetValue(cfg_info, section, "port", false);
+ if (!cfg->Device) {
+ free(cfg->Device);
+ cfg->Device = strdup(DefaultPort);
+ } else {
+ cfg->DefaultDevice = false;
+ }
+ cfg->Connection = INI_GetValue(cfg_info, section, "connection", false);
+ if (!cfg->Connection) {
+ free(cfg->Connection);
+ cfg->Connection = strdup(DefaultConnection);
+ } else {
+ cfg->DefaultConnection = false;
+ }
+ cfg->SyncTime = INI_GetValue(cfg_info, section, "synchronizetime", false);
+ if (!cfg->SyncTime) {
+ free(cfg->SyncTime);
+ cfg->SyncTime = strdup(DefaultSynchronizeTime);
+ } else {
+ cfg->DefaultSyncTime = false;
+ }
+ cfg->DebugFile = INI_GetValue(cfg_info, section, "logfile", false);
+ if (!cfg->DebugFile) {
+ free(cfg->DebugFile);
+ cfg->DebugFile = strdup(DefaultDebugFile);
+ } else {
+ cfg->DefaultDebugFile = false;
+ }
+ cfg->LockDevice = INI_GetValue(cfg_info, section, "use_locking", false);
+ if (!cfg->LockDevice) {
+ free(cfg->LockDevice);
+ cfg->LockDevice = strdup(DefaultLockDevice);
+ } else {
+ cfg->DefaultLockDevice = false;
+ }
+ Temp = INI_GetValue(cfg_info, section, "model", false);
+ if (!Temp) {
+ strcpy(cfg->Model,DefaultModel);
+ } else {
+ cfg->DefaultModel = false;
+ strcpy(cfg->Model,Temp);
+ }
+ Temp = INI_GetValue(cfg_info, section, "logformat", false);
+ if (!Temp) {
+ strcpy(cfg->DebugLevel,DefaultDebugLevel);
+ } else {
+ cfg->DefaultDebugLevel = false;
+ strcpy(cfg->DebugLevel,Temp);
+ }
+ cfg->StartInfo = INI_GetValue(cfg_info, section, "startinfo", false);
+ if (!cfg->StartInfo) {
+ free(cfg->StartInfo);
+ cfg->StartInfo = strdup(DefaultStartInfo);
+ } else {
+ cfg->DefaultStartInfo = false;
+ }
+ return true;
+}
+
+static OnePhoneModel allmodels[] = {
+#ifdef GSM_ENABLE_NOKIA6510
+ {"1100", "RH-18" ,"", {0}},
+ {"1100a","RH-38" ,"", {0}},
+ {"1100b","RH-36" ,"", {0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"2100" ,"NAM-2" ,"", {F_NOWAP,F_NOCALLER,F_RING_SM,F_CAL33,F_POWER_BATT,F_PROFILES33,F_NOCALLINFO,F_NODTMF,0}},//quess
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ {"3100" ,"RH-19" ,"", {F_PBKTONEGAL,F_PBKSMSLIST,0}},
+ {"3100b","RH-50" ,"", {F_PBKTONEGAL,F_PBKSMSLIST,0}},
+ {"3108", "RH-6", "Nokia 3108", {0}}, //does it have irda ?
+ {"3200", "RH-30" ,"Nokia 3200", {F_PBKTONEGAL,0}},
+ {"3200a","RH-31" ,"Nokia 3200", {F_PBKTONEGAL,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"3210" ,"NSE-8" ,"", {F_NOWAP,F_NOCALLER,F_NOCALENDAR,F_NOPBKUNICODE,F_POWER_BATT,F_PROFILES51,F_NOPICTUREUNI,F_NOCALLINFO,F_NODTMF,0}},
+ {"3210" ,"NSE-9" ,"", {F_NOWAP,F_NOCALLER,F_NOCALENDAR,F_NOPBKUNICODE,F_POWER_BATT,F_PROFILES51,F_NOPICTUREUNI,F_NOCALLINFO,F_NODTMF,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ {"3300" ,"NEM-1" ,"Nokia 3300", {F_PBKTONEGAL,0}},
+ {"3300" ,"NEM-2" ,"Nokia 3300", {F_PBKTONEGAL,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"3310" ,"NHM-5" ,"", {F_NOWAP,F_NOCALLER,F_RING_SM,F_CAL33,F_POWER_BATT,F_PROFILES33,F_NOCALLINFO,F_NODTMF,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA3320
+ {"3320" ,"NPC-1" ,"Nokia 3320", {F_CAL62,F_DAYMONTH,0}},//fixme
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"3330" ,"NHM-6" ,"", {F_NOCALLER,F_RING_SM,F_CAL33,F_PROFILES33,F_NOPICTUREUNI,F_NOCALLINFO,F_NODTMF,0}},
+ {"3390" ,"NPB-1" ,"", {F_NOWAP,F_NOCALLER,F_RING_SM,F_CAL33,F_PROFILES33,F_NOPICTUREUNI,F_NOCALLINFO,F_NODTMF,0}},
+ {"3410" ,"NHM-2" ,"", {F_RING_SM,F_CAL33,F_PROFILES33,F_NOCALLINFO,F_NODTMF,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ {"3510" ,"NHM-8" ,"", {F_CAL35,F_PBK35,F_NOGPRSPOINT,F_VOICETAGS,0}},
+ {"3510i","RH-9" ,"", {F_CAL35,F_PBK35,F_NOGPRSPOINT,F_VOICETAGS,0}},
+ {"3530" ,"RH-9" ,"", {F_CAL35,F_PBK35,F_NOGPRSPOINT,F_VOICETAGS,0}},
+ {"3589i","RH-44" ,"", {F_VOICETAGS,0}},
+ {"3590" ,"NPM-8" ,"", {0}},//irda?
+ {"3595" ,"NPM-10" ,"", {0}},//irda?
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"3610" ,"NAM-1" ,"", {F_NOCALLER,F_RING_SM,F_CAL33,F_POWER_BATT,F_PROFILES33,F_NOCALLINFO,F_NODTMF,0}},//quess
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA3650)
+ {"3650" ,"NHL-8" ,"Nokia 3650", {0}},
+ {"NGAGE","NEM-4" ,"", {F_RADIO,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"5100" ,"NPM-6" ,"Nokia 5100", {F_PBKTONEGAL,F_TODO66,F_RADIO,0}},
+ {"5100" ,"NPM-6U","Nokia 5100", {F_PBKTONEGAL,F_TODO66,F_RADIO,0}},
+ {"5100" ,"NPM-6X","Nokia 5100", {F_PBKTONEGAL,F_TODO66,F_RADIO,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"5110" ,"NSE-1" ,"", {F_NOWAP,F_NOCALLER,F_NORING,F_NOPICTURE,F_NOSTARTUP,F_NOCALENDAR,F_NOPBKUNICODE,F_PROFILES51,F_MAGICBYTES,F_DISPSTATUS,0}},
+ {"5110i","NSE-2" ,"", {F_NOWAP,F_NOCALLER,F_NORING,F_NOPICTURE,F_NOSTARTUP,F_NOCALENDAR,F_NOPBKUNICODE,F_PROFILES51,F_MAGICBYTES,F_DISPSTATUS,0}},
+ {"5130" ,"NSK-1" ,"", {F_NOWAP,F_NOCALLER,F_NORING,F_NOPICTURE,F_NOSTARTUP,F_NOCALENDAR,F_NOPBKUNICODE,F_PROFILES51,F_MAGICBYTES,F_DISPSTATUS,0}},
+ {"5190" ,"NSB-1" ,"", {F_NOWAP,F_NOCALLER,F_NORING,F_NOPICTURE,F_NOSTARTUP,F_NOCALENDAR,F_NOPBKUNICODE,F_PROFILES51,F_MAGICBYTES,F_DISPSTATUS,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6110)
+ {"5210" ,"NSM-5" ,"Nokia 5210", {F_CAL52,F_NOSTARTANI,F_NOPICTUREUNI,F_NODTMF,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"5510" ,"NPM-5" ,"", {F_NOCALLER,F_PROFILES33,F_NOPICTUREUNI,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"6100" ,"NPL-2" ,"Nokia 6100", {F_PBKTONEGAL,F_TODO66,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ {"6110" ,"NSE-3" ,"", {F_NOWAP,F_NOPICTURE,F_NOSTARTANI,F_NOPBKUNICODE,F_MAGICBYTES,F_DISPSTATUS,0}},
+ {"6130" ,"NSK-3" ,"", {F_NOWAP,F_NOPICTURE,F_NOSTARTANI,F_NOPBKUNICODE,F_MAGICBYTES,F_DISPSTATUS,0}},
+ {"6150" ,"NSM-1" ,"", {F_NOWAP,F_NOSTARTANI,F_NOPBKUNICODE,F_MAGICBYTES,F_DISPSTATUS,F_NOPICTUREUNI,0}},
+ {"6190" ,"NSB-3" ,"", {F_NOWAP,F_NOPICTURE,F_NOSTARTANI,F_NOPBKUNICODE,F_MAGICBYTES,F_DISPSTATUS,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"6200" ,"NPL-3" ,"Nokia 6200", {F_PBKTONEGAL,0}},
+ {"6220" ,"RH-20" ,"Nokia 6220", {F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKSMSLIST,F_PBKUSER,F_WAPMMSPROXY,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA7110)
+ {"6210" ,"NPE-3" ,"Nokia 6210", {F_VOICETAGS,F_CAL62,0}},
+ {"6250" ,"NHM-3" ,"Nokia 6250", {F_VOICETAGS,F_CAL62,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"6230" ,"RH-12" ,"Nokia 6230", {F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKSMSLIST,F_PBKUSER,0}},
+ {"6310" ,"NPE-4" ,"Nokia 6310", {F_TODO63,F_CAL65,F_NOMIDI,F_NOMMS,F_VOICETAGS,0}},
+ {"6310i","NPL-1" ,"Nokia 6310i",{F_TODO63,F_CAL65,F_NOMIDI,F_BLUETOOTH,F_NOMMS,F_VOICETAGS,0}},
+ {"6385" ,"NHP-2AX","Nokia 6385",{F_TODO63,F_CAL65,F_NOMIDI,F_NOMMS,F_VOICETAGS,0}},
+ {"6510" ,"NPM-9" ,"Nokia 6510", {F_TODO63,F_CAL65,F_NOMIDI,F_RADIO,F_NOFILESYSTEM,F_NOMMS,F_VOICETAGS,0}},
+ {"6610" ,"NHL-4U","Nokia 6610", {F_PBKTONEGAL,F_TODO66,F_RADIO,0}},
+ {"6800" ,"NSB-9" ,"Nokia 6800", {F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKSMSLIST,0}},
+ {"6800" ,"NHL-6" ,"Nokia 6800", {F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKSMSLIST,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA7110)
+ {"7110" ,"NSE-5" ,"Nokia 7110", {F_CAL62,0}},
+ {"7190" ,"NSB-5" ,"Nokia 7190", {F_CAL62,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"7210" ,"NHL-4" ,"Nokia 7210", {F_PBKTONEGAL,F_TODO66,F_RADIO,0}},
+ {"7250" ,"NHL-4J","Nokia 7250", {F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKIMG,0}},
+ {"7250i","NHL-4JX","Nokia 7250i",{F_PBKTONEGAL,F_TODO66,F_RADIO,F_PBKIMG,0}},
+ {"7600", "NMM-3", "Nokia 7600", {F_TODO66,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN)
+ {"7650" ,"NHL-2" ,"Nokia 7650", {0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6110)
+ {"8210" ,"NSM-3" ,"Nokia 8210", {F_NOWAP,F_NOSTARTANI,F_NOPBKUNICODE,F_NOPICTUREUNI,0}},
+ {"8250" ,"NSM-3D","Nokia 8250", {F_NOWAP,F_NOSTARTANI,F_CAL82,F_NOPICTUREUNI,0}},
+ {"8290" ,"NSB-7" ,"Nokia 8290", {F_NOWAP,F_NOSTARTANI,F_NOPBKUNICODE,F_NOPICTUREUNI,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"8310" ,"NHM-7" ,"Nokia 8310", {F_CAL62,F_NOMIDI,F_RADIO,F_NOFILESYSTEM,F_NOMMS,F_VOICETAGS,0}},
+ {"8390" ,"NSB-8" ,"Nokia 8390", {F_CAL62,F_NOMIDI,F_RADIO,F_NOFILESYSTEM,F_NOMMS,F_VOICETAGS,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6110)
+ {"8850" ,"NSM-2" ,"Nokia 8850", {0}},
+ {"8855" ,"NSM-4" ,"Nokia 8855", {0}},
+ {"8890" ,"NSB-6" ,"Nokia 8890", {0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_NOKIA6510)
+ {"8910" ,"NHM-4" ,"Nokia 8910", {F_CAL62,F_NOMIDI,F_NOFILESYSTEM,F_NOMMS,0}},
+ {"8910i","NHM-4" ,"Nokia 8910i",{F_CAL62,F_NOMIDI,F_NOFILESYSTEM,F_NOMMS,0}},
+#endif
+#ifdef GSM_ENABLE_NOKIA9210
+ {"9210" ,"RAE-3" ,"", {0}},
+ {"9210i","RAE-5" ,"", {0}},
+#endif
+#ifdef GSM_ENABLE_ATGEN
+ {"at" , "at", "", {0}},
+ {"M20" , "M20", "", {F_M20SMS,F_SLOWWRITE,0}},
+ {"MC35" , "MC35", "", {0}},
+ {"S25", "S25", "SIEMENS S25", {0}},
+ {"C35i" , "C35i", "", {0}},
+ {"S35i" , "S35i", "", {0}},
+ {"M35i" , "M35i", "", {0}},
+ {"S40" , "Siemens S40", "", {0}},
+ {"C45" , "C45", "", {0}},
+ {"S45" , "S45", "", {0}},
+ {"ME45" , "ME45", "", {0}},
+ {"SL45" , "SL45", "", {0}},
+ {"SL45i" , "SL45i", "", {0}},
+ {"M50" , "M50", "", {0}},
+ {"S45" , "6618" , "", {0}},
+ {"ME45" , "3618" , "", {0}},
+ {"S55" , "S55" , "", {0}},
+ {"T28s", "1101101-BVT28s","", {0}},
+ {"R320s" , "1101201-BV R320s","", {0}},
+ {"R380s", "7100101-BVR380s" ,"", {0}},
+ {"R520m", "1130101-BVR520m" ,"", {0}},
+ {"T39m", "1130102-BVT39m" ,"", {0}},
+ {"T65", "1101901-BVT65" , "", {0}},
+ {"T68", "1130201-BVT68" , "", {0}},
+ {"T68i", "1130202-BVT68" , "", {0}},
+ {"R600", "102001-BVR600" , "", {0}},
+ {"T200", "1130501-BVT200" ,"", {0}},
+ {"T300", "1130601-BVT300" ,"T300", {0}},
+ {"T310", "1130602-BVT310" ,"", {0}},
+ {"P800", "7130501-BVP800" ,"", {0}},
+ {"iPAQ" , "iPAQ" , "" , {0}},
+ {"A2D" , "A2D" , "" , {0}},
+ {"9210" , "RAE-3", "Nokia Communicator GSM900/1800",{0}},
+ {"myV-65", "myV-65 GPRS", "", {F_SMSME900,0}},
+#endif
+#if defined(GSM_ENABLE_ATGEN) || defined(GSM_ENABLE_ALCATEL)
+ {"BE5", "ONE TOUCH 500","", {F_ALCATEL,F_SMSONLYSENT,F_BROKENCPBS,0}},
+ {"BH4", "ONE TOUCH 535","ALCATEL OT535", {F_ALCATEL,F_SMSONLYSENT,0}},
+ {"BF5", "ONE TOUCH 715","ALCATEL OT715", {F_ALCATEL,F_SMSONLYSENT,F_BROKENCPBS,0}},
+#endif
+ {"unknown", "" ,"", {0}}
+};
+
+OnePhoneModel *GetModelData(char *model, char *number, char *irdamodel)
+{
+ int i = 0;
+
+ while (strcmp(allmodels[i].number,"") != 0) {
+ if (model !=NULL) {
+ if (strcmp (model, allmodels[i].model) == 0) {
+ return (&allmodels[i]);
+ }
+ }
+ if (number !=NULL) {
+ if (strcmp (number, allmodels[i].number) == 0) {
+ return (&allmodels[i]);
+ }
+ }
+ if (irdamodel !=NULL) {
+ if (strcmp (irdamodel, allmodels[i].irdamodel) == 0) {
+ return (&allmodels[i]);
+ }
+ }
+ i++;
+ }
+ return (&allmodels[i]);
+}
+
+bool IsPhoneFeatureAvailable(OnePhoneModel *model, Feature feature)
+{
+ int i = 0;
+ bool retval = false;
+
+ while (model->features[i] != 0) {
+ if (model->features[i] == feature) {
+ retval = true;
+ break;
+ }
+ i++;
+ }
+ return retval;
+}
+
+void GSM_DumpMessageLevel2(GSM_StateMachine *s, unsigned char *message, int messagesize, int type)
+{
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s,"SENDING frame ");
+ smprintf(s,"type 0x%02X/length 0x%02X/%i", type, messagesize, messagesize);
+ DumpMessage(s->di.use_global ? di.df : s->di.df, s->di.dl, message, messagesize);
+ if (messagesize == 0) smprintf(s,"\n");
+ if (s->di.df) fflush(s->di.df);
+ }
+}
+
+void GSM_DumpMessageLevel3(GSM_StateMachine *s, unsigned char *message, int messagesize, int type)
+{
+ int i;
+
+ if (s->di.dl==DL_BINARY) {
+ smprintf(s,"%c",0x01); /* Sending */
+ smprintf(s,"%c",type);
+ smprintf(s,"%c",messagesize/256);
+ smprintf(s,"%c",messagesize%256);
+ for (i=0;i<messagesize;i++) smprintf(s,"%c",message[i]);
+ }
+}
+
+#ifdef __GNUC__
+__attribute__((format(printf, 2, 3)))
+#endif
+int smprintf(GSM_StateMachine *s, const char *format, ...)
+{
+ va_list argp;
+ int result=0;
+ unsigned char buffer[2000];
+ Debug_Level dl;
+ FILE *df;
+
+ va_start(argp, format);
+ if (s == NULL) {
+ dl = di.dl;
+ df = di.df;
+ } else {
+ dl = s->di.dl;
+ if (s->di.use_global) {
+ df = di.df;
+ } else {
+ df = s->di.df;
+ }
+ }
+
+ if (dl != 0) {
+ result = vsprintf(buffer, format, argp);
+ result = smfprintf(df, dl, "%s", buffer);
+ }
+
+ va_end(argp);
+ return result;
+}
+
+void GSM_OSErrorInfo(GSM_StateMachine *s, char *description)
+{
+#ifdef WIN32
+ int i;
+ unsigned char *lpMsgBuf;
+
+ /* We don't use errno in win32 - GetLastError gives better info */
+ if (GetLastError()!=-1) {
+ if (s->di.dl == DL_TEXTERROR || s->di.dl == DL_TEXT || s->di.dl == DL_TEXTALL ||
+ s->di.dl == DL_TEXTERRORDATE || s->di.dl == DL_TEXTDATE || s->di.dl == DL_TEXTALLDATE) {
+ FormatMessage(
+ FORMAT_MESSAGE_ALLOCATE_BUFFER |
+ FORMAT_MESSAGE_FROM_SYSTEM |
+ FORMAT_MESSAGE_IGNORE_INSERTS,
+ NULL,
+ GetLastError(),
+ MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
+ (LPTSTR) &lpMsgBuf,
+ 0,
+ NULL
+ );
+ for (i=0;i<(int)strlen(lpMsgBuf);i++) {
+ if (lpMsgBuf[i] == 13 || lpMsgBuf[i] == 10) {
+ lpMsgBuf[i] = ' ';
+ }
+ }
+ smprintf(s,"[System error - %s, %i, \"%s\"]\n",description,GetLastError(),(LPCTSTR)lpMsgBuf);
+ LocalFree(lpMsgBuf);
+ }
+ }
+ return;
+#endif
+
+ if (errno!=-1) {
+ if (s->di.dl == DL_TEXTERROR || s->di.dl == DL_TEXT || s->di.dl == DL_TEXTALL ||
+ s->di.dl == DL_TEXTERRORDATE || s->di.dl == DL_TEXTDATE || s->di.dl == DL_TEXTALLDATE) {
+ smprintf(s,"[System error - %s, %i, \"%s\"]\n",description,errno,strerror(errno));
+ }
+ }
+}
+
+#ifdef GSM_ENABLE_BACKUP
+
+void GSM_GetPhoneFeaturesForBackup(GSM_StateMachine *s, GSM_Backup_Info *info)
+{
+ GSM_Error error;
+ GSM_MemoryStatus MemStatus;
+ GSM_ToDoStatus ToDoStatus;
+ GSM_CalendarEntry Note;
+ GSM_WAPBookmark Bookmark;
+ GSM_MultiWAPSettings WAPSettings;
+ GSM_FMStation FMStation;
+ GSM_GPRSAccessPoint GPRSPoint;
+// GSM_Profile Profile;
+
+ if (info->PhonePhonebook) {
+ MemStatus.MemoryType = MEM_ME;
+ error=s->Phone.Functions->GetMemoryStatus(s, &MemStatus);
+ if (error==ERR_NONE && MemStatus.MemoryUsed != 0) {
+ } else {
+ info->PhonePhonebook = false;
+ }
+ }
+ if (info->SIMPhonebook) {
+ MemStatus.MemoryType = MEM_SM;
+ error=s->Phone.Functions->GetMemoryStatus(s, &MemStatus);
+ if (error==ERR_NONE && MemStatus.MemoryUsed != 0) {
+ } else {
+ info->SIMPhonebook = false;
+ }
+ }
+ if (info->Calendar) {
+ error=s->Phone.Functions->GetNextCalendar(s,&Note,true);
+ if (error!=ERR_NONE) info->Calendar = false;
+ }
+ if (info->ToDo) {
+ error=s->Phone.Functions->GetToDoStatus(s,&ToDoStatus);
+ if (error == ERR_NONE && ToDoStatus.Used != 0) {
+ } else {
+ info->ToDo = false;
+ }
+ }
+ if (info->WAPBookmark) {
+ Bookmark.Location = 1;
+ error=s->Phone.Functions->GetWAPBookmark(s,&Bookmark);
+ if (error == ERR_NONE) {
+ } else {
+ info->WAPBookmark = false;
+ }
+ }
+ if (info->WAPSettings) {
+ WAPSettings.Location = 1;
+ error=s->Phone.Functions->GetWAPSettings(s,&WAPSettings);
+ if (error == ERR_NONE) {
+ } else {
+ info->WAPSettings = false;
+ }
+ }
+ if (info->MMSSettings) {
+ WAPSettings.Location = 1;
+ error=s->Phone.Functions->GetMMSSettings(s,&WAPSettings);
+ if (error == ERR_NONE) {
+ } else {
+ info->WAPSettings = false;
+ }
+ }
+ if (info->FMStation) {
+ FMStation.Location = 1;
+ error = s->Phone.Functions->GetFMStation(s,&FMStation);
+ if (error == ERR_NONE || error == ERR_EMPTY) {
+ } else {
+ info->FMStation = false;
+ }
+ }
+ if (info->GPRSPoint) {
+ GPRSPoint.Location = 1;
+ error = s->Phone.Functions->GetGPRSAccessPoint(s,&GPRSPoint);
+ if (error == ERR_NONE || error == ERR_EMPTY) {
+ } else {
+ info->GPRSPoint = false;
+ }
+ }
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/gsmstate.h b/gammu/emb/common/gsmstate.h
new file mode 100644
index 0000000..cb17623
--- a/dev/null
+++ b/gammu/emb/common/gsmstate.h
@@ -0,0 +1,1562 @@
+/* (c) 2002-2004 by Marcin Wiacek & Michal Cihar */
+
+#ifndef __gsm_state_h
+#define __gsm_state_h
+
+#include <time.h>
+
+#include "config.h"
+#include "misc/cfg.h"
+
+#ifdef GSM_ENABLE_NOKIA3320
+# include "phone/nokia/dct4/n3320.h"
+#endif
+#ifdef GSM_ENABLE_NOKIA3650
+# include "phone/nokia/dct4/n3650.h"
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+# include "phone/nokia/dct3/n6110.h"
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+# include "phone/nokia/dct4/n6510.h"
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+# include "phone/nokia/dct3/n7110.h"
+#endif
+#ifdef GSM_ENABLE_NOKIA9210
+# include "phone/nokia/dct3/n9210.h"
+#endif
+#ifdef GSM_ENABLE_ATGEN
+# include "phone/at/atgen.h"
+#endif
+#ifdef GSM_ENABLE_ALCATEL
+# include "phone/alcatel/alcatel.h"
+#endif
+#ifdef GSM_ENABLE_OBEXGEN
+# include "phone/obex/obexgen.h"
+#endif
+#ifdef GSM_ENABLE_MROUTERGEN
+# include "phone/symbian/mroutgen.h"
+#endif
+
+#ifndef GSM_USED_MBUS2
+# undef GSM_ENABLE_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# undef GSM_ENABLE_FBUS2
+#endif
+#ifndef GSM_USED_FBUS2DLR3
+# undef GSM_ENABLE_FBUS2DLR3
+#endif
+#ifndef GSM_USED_FBUS2DKU5
+# undef GSM_ENABLE_FBUS2DKU5
+#endif
+#ifndef GSM_USED_FBUS2PL2303
+# undef GSM_ENABLE_FBUS2PL2303
+#endif
+#ifndef GSM_USED_FBUS2BLUE
+# undef GSM_ENABLE_FBUS2BLUE
+#endif
+#ifndef GSM_USED_FBUS2IRDA
+# undef GSM_ENABLE_FBUS2IRDA
+#endif
+#ifndef GSM_USED_PHONETBLUE
+# undef GSM_ENABLE_PHONETBLUE
+#endif
+#ifndef GSM_USED_AT
+# undef GSM_ENABLE_AT
+#endif
+#ifndef GSM_USED_IRDAOBEX
+# undef GSM_ENABLE_IRDAOBEX
+#endif
+#ifndef GSM_USED_BLUEOBEX
+# undef GSM_ENABLE_BLUEOBEX
+#endif
+#ifndef GSM_USED_ALCABUS
+# undef GSM_ENABLE_ALCABUS
+#endif
+#ifndef GSM_USED_IRDAPHONET
+# undef GSM_ENABLE_IRDAPHONET
+#endif
+#ifndef GSM_USED_BLUEFBUS2
+# undef GSM_ENABLE_BLUEFBUS2
+#endif
+#ifndef GSM_USED_BLUEPHONET
+# undef GSM_ENABLE_BLUEPHONET
+#endif
+#ifndef GSM_USED_BLUEAT
+# undef GSM_ENABLE_BLUEAT
+#endif
+#ifndef GSM_USED_IRDAAT
+# undef GSM_ENABLE_IRDAAT
+#endif
+#ifndef GSM_USED_MROUTERBLUE
+# undef GSM_ENABLE_MROUTERBLUE
+#endif
+
+#if defined(GSM_ENABLE_NOKIA3320) || defined(GSM_ENABLE_NOKIA6110) || defined(GSM_ENABLE_NOKIA7110) || defined(GSM_ENABLE_NOKIA9210)
+# define GSM_ENABLE_NOKIA_DCT3
+#endif
+#if defined(GSM_ENABLE_NOKIA3650) || defined(GSM_ENABLE_NOKIA6510)
+# define GSM_ENABLE_NOKIA_DCT4
+#endif
+
+#include "protocol/protocol.h"
+#if defined(GSM_ENABLE_FBUS2) || defined(GSM_ENABLE_FBUS2IRDA) || defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2BLUE) || defined(GSM_ENABLE_BLUEFBUS2) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2PL2303)
+# include "protocol/nokia/fbus2.h"
+#endif
+#ifdef GSM_ENABLE_MBUS2
+# include "protocol/nokia/mbus2.h"
+#endif
+#if defined(GSM_ENABLE_PHONETBLUE) || defined(GSM_ENABLE_IRDAPHONET) || defined(GSM_ENABLE_BLUEPHONET)
+# include "protocol/nokia/phonet.h"
+#endif
+#if defined(GSM_ENABLE_AT) || defined(GSM_ENABLE_BLUEAT) || defined(GSM_ENABLE_IRDAAT)
+# include "protocol/at/at.h"
+#endif
+#ifdef GSM_ENABLE_ALCABUS
+# include "protocol/alcatel/alcabus.h"
+#endif
+#if defined(GSM_ENABLE_IRDAOBEX) || defined(GSM_ENABLE_BLUEOBEX)
+# include "protocol/obex/obex.h"
+#endif
+#if defined(GSM_ENABLE_MROUTERBLUE)
+# include "protocol/symbian/mrouter.h"
+#endif
+
+#define GSM_ENABLE_SERIALDEVICE
+#ifndef GSM_USED_SERIALDEVICE
+# undef GSM_ENABLE_SERIALDEVICE
+#endif
+#define GSM_ENABLE_IRDADEVICE
+#ifndef GSM_USED_IRDADEVICE
+# undef GSM_ENABLE_IRDADEVICE
+#endif
+#define GSM_ENABLE_BLUETOOTHDEVICE
+#ifndef GSM_USED_BLUETOOTHDEVICE
+# undef GSM_ENABLE_BLUETOOTHDEVICE
+#endif
+
+#ifdef DJGPP
+# undef GSM_ENABLE_IRDADEVICE
+# undef GSM_ENABLE_IRDAPHONET
+# undef GSM_ENABLE_IRDAOBEX
+# undef GSM_ENABLE_IRDAAT
+# undef GSM_ENABLE_FBUS2IRDA
+
+# undef GSM_ENABLE_BLUETOOTHDEVICE
+# undef GSM_ENABLE_BLUEPHONET
+# undef GSM_ENABLE_BLUEOBEX
+# undef GSM_ENABLE_BLUEAT
+# undef GSM_ENABLE_BLUEFBUS2
+# undef GSM_ENABLE_PHONETBLUE
+# undef GSM_ENABLE_FBUS2BLUE
+# undef GSM_ENABLE_MROUTERBLUE
+#endif
+
+#ifndef WIN32
+# ifdef ENABLE_LGPL
+# undef GSM_ENABLE_IRDADEVICE
+# undef GSM_ENABLE_IRDAPHONET
+# undef GSM_ENABLE_IRDAOBEX
+# undef GSM_ENABLE_IRDAAT
+# undef GSM_ENABLE_FBUS2IRDA
+
+# undef GSM_ENABLE_BLUETOOTHDEVICE
+# undef GSM_ENABLE_BLUEPHONET
+# undef GSM_ENABLE_BLUEOBEX
+# undef GSM_ENABLE_BLUEAT
+# undef GSM_ENABLE_BLUEFBUS2
+# undef GSM_ENABLE_PHONETBLUE
+# undef GSM_ENABLE_FBUS2BLUE
+# undef GSM_ENABLE_MROUTERBLUE
+# endif
+#endif
+
+#ifdef GSM_ENABLE_SERIALDEVICE
+# include "device/serial/ser_w32.h"
+# include "device/serial/ser_unx.h"
+# include "device/serial/ser_djg.h"
+#endif
+#ifdef GSM_ENABLE_IRDADEVICE
+# include "device/irda/irda.h"
+#endif
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+# include "device/bluetoth/bluetoth.h"
+#endif
+
+#include "service/gsmpbk.h"
+#include "service/gsmnet.h"
+#include "service/gsmring.h"
+#include "service/gsmcal.h"
+#include "service/gsmdata.h"
+#include "service/gsmlogo.h"
+#include "service/gsmmisc.h"
+#include "service/gsmprof.h"
+#include "service/gsmcall.h"
+#include "service/sms/gsmsms.h"
+#include "service/sms/gsmems.h"
+#include "service/sms/gsmmulti.h"
+#include "service/backup/gsmback.h"
+
+typedef struct _GSM_StateMachine GSM_StateMachine;
+typedef struct _GSM_User GSM_User;
+typedef struct _OnePhoneModel OnePhoneModel;
+
+/* ------------------------- Device layer ---------------------------------- */
+
+/**
+ * Device functions, each device has to provide these.
+ */
+typedef struct {
+ /**
+ * Opens device.
+ */
+ GSM_Error (*OpenDevice) (GSM_StateMachine *s);
+ /**
+ * Closes device.
+ */
+ GSM_Error (*CloseDevice) (GSM_StateMachine *s);
+ /**
+ * Sets parity for device.
+ */
+ GSM_Error (*DeviceSetParity) (GSM_StateMachine *s, bool parity);
+ /**
+ * Sets dtr (data to read) and rts (ready to send) flags.
+ */
+ GSM_Error (*DeviceSetDtrRts) (GSM_StateMachine *s, bool dtr, bool rts);
+ /**
+ * Sets device speed.
+ */
+ GSM_Error (*DeviceSetSpeed) (GSM_StateMachine *s, int speed);
+ /**
+ * Attempts to read nbytes from device.
+ */
+ int (*ReadDevice) (GSM_StateMachine *s, void *buf, size_t nbytes);
+ /**
+ * Attempts to read nbytes from device.
+ */
+ int (*WriteDevice) (GSM_StateMachine *s, void *buf, size_t nbytes);
+} GSM_Device_Functions;
+
+#ifdef GSM_ENABLE_SERIALDEVICE
+ extern GSM_Device_Functions SerialDevice;
+#endif
+#ifdef GSM_ENABLE_IRDADEVICE
+ extern GSM_Device_Functions IrdaDevice;
+#endif
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+ extern GSM_Device_Functions BlueToothDevice;
+#endif
+
+/**
+ * Structure containing device specific data and pointer to device functions -
+ * @ref GSM_Device_Functions. The data are in a union, so you can use only
+ * one device at one time.
+ */
+typedef struct {
+ union {
+ char fake;
+#ifdef GSM_ENABLE_SERIALDEVICE
+ GSM_Device_SerialData Serial;
+#endif
+#ifdef GSM_ENABLE_IRDADEVICE
+ GSM_Device_IrdaData Irda;
+#endif
+#ifdef GSM_ENABLE_BLUETOOTHDEVICE
+ GSM_Device_BlueToothData BlueTooth;
+#endif
+ } Data;
+ GSM_Device_Functions *Functions;
+} GSM_Device;
+
+/* ---------------------- Protocol layer ----------------------------------- */
+
+/**
+ * Protocol functions, each protocol has to implement these.
+ */
+typedef struct {
+ /**
+ * Writes message to device.
+ */
+ GSM_Error (*WriteMessage) (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type);
+ /**
+ * This one is called when character is received from device.
+ */
+ GSM_Error (*StateMachine) (GSM_StateMachine *s, unsigned char rx_char);
+ /**
+ * Protocol initialisation.
+ */
+ GSM_Error (*Initialise) (GSM_StateMachine *s);
+ /**
+ * Protocol termination.
+ */
+ GSM_Error (*Terminate) (GSM_StateMachine *s);
+} GSM_Protocol_Functions;
+
+#ifdef GSM_ENABLE_MBUS2
+ extern GSM_Protocol_Functions MBUS2Protocol;
+#endif
+#if defined(GSM_ENABLE_FBUS2) || defined(GSM_ENABLE_FBUS2IRDA) || defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2BLUE) || defined(GSM_ENABLE_BLUEFBUS2) || defined(GSM_ENABLE_FBUS2PL2303)
+ extern GSM_Protocol_Functions FBUS2Protocol;
+#endif
+#if defined(GSM_ENABLE_PHONETBLUE) || defined(GSM_ENABLE_IRDAPHONET) || defined(GSM_ENABLE_BLUEPHONET)
+ extern GSM_Protocol_Functions PHONETProtocol;
+#endif
+#if defined(GSM_ENABLE_AT) || defined(GSM_ENABLE_BLUEAT) || defined(GSM_ENABLE_IRDAAT)
+ extern GSM_Protocol_Functions ATProtocol;
+#endif
+#ifdef GSM_ENABLE_ALCABUS
+ extern GSM_Protocol_Functions ALCABUSProtocol;
+#endif
+#if defined(GSM_ENABLE_IRDAOBEX) || defined(GSM_ENABLE_BLUEOBEX)
+ extern GSM_Protocol_Functions OBEXProtocol;
+#endif
+#if defined(GSM_ENABLE_MROUTERBLUE)
+ extern GSM_Protocol_Functions MROUTERProtocol;
+#endif
+
+/**
+ * Structure containing protocol specific data and pointer to protocol
+ * functions - @ref GSM_Protocol_Functions. The data are in a structure, so
+ * you may use more protocols at once and switch between them.
+ */
+typedef struct {
+ struct {
+ char fake;
+#ifdef GSM_ENABLE_MBUS2
+ GSM_Protocol_MBUS2Data MBUS2;
+#endif
+#if defined(GSM_ENABLE_FBUS2) || defined(GSM_ENABLE_FBUS2IRDA) || defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2PL2303) || defined(GSM_ENABLE_FBUS2BLUE) || defined(GSM_ENABLE_BLUEFBUS2)
+ GSM_Protocol_FBUS2Data FBUS2;
+#endif
+#if defined(GSM_ENABLE_PHONETBLUE) || defined(GSM_ENABLE_IRDAPHONET) || defined(GSM_ENABLE_BLUEPHONET)
+ GSM_Protocol_PHONETData PHONET;
+#endif
+#if defined(GSM_ENABLE_AT) || defined(GSM_ENABLE_BLUEAT) || defined(GSM_ENABLE_IRDAAT)
+ GSM_Protocol_ATData AT;
+#endif
+#ifdef GSM_ENABLE_ALCABUS
+ GSM_Protocol_ALCABUSData ALCABUS;
+#endif
+#if defined(GSM_ENABLE_IRDAOBEX) || defined(GSM_ENABLE_BLUEOBEX)
+ GSM_Protocol_OBEXData OBEX;
+#endif
+#if defined(GSM_ENABLE_MROUTERBLUE)
+ GSM_Protocol_MROUTERData MROUTER;
+#endif
+ } Data;
+ GSM_Protocol_Functions *Functions;
+} GSM_Protocol;
+
+/* -------------------------- Phone layer ---------------------------------- */
+
+/**
+ * Phone requests identificators, these are used for internally identifying
+ * which operation is being performed.
+ */
+typedef enum {
+ ID_None=1,
+ ID_GetModel,
+ ID_GetFirmware,
+ ID_EnableSecurity,
+ ID_GetIMEI,
+ ID_GetDateTime,
+ ID_GetAlarm,
+ ID_GetMemory,
+ ID_GetMemoryStatus,
+ ID_GetSMSC,
+ ID_GetSMSMessage,
+ ID_EnableEcho,
+ ID_EnableErrorInfo,
+ ID_SetOBEX,
+ ID_GetSignalQuality,
+ ID_GetBatteryCharge,
+ ID_GetSMSFolders,
+ ID_GetSMSFolderStatus,
+ ID_GetSMSStatus,
+ ID_AddSMSFolder,
+ ID_GetNetworkInfo,
+ ID_GetRingtone,
+ ID_DialVoice,
+ ID_GetCalendarNotesInfo,
+ ID_GetCalendarNote,
+ ID_GetSecurityCode,
+ ID_GetWAPBookmark,
+ ID_GetBitmap,
+ ID_SaveSMSMessage,
+ ID_CancelCall,
+ ID_SetDateTime,
+ ID_SetAlarm,
+ ID_DisableConnectFunc,
+ ID_EnableConnectFunc,
+ ID_AnswerCall,
+ ID_SetBitmap,
+ ID_SetRingtone,
+ ID_DeleteSMSMessage,
+ ID_DeleteCalendarNote,
+ ID_SetPath,
+ ID_SetSMSC,
+ ID_SetProfile,
+ ID_SetMemory,
+ ID_DeleteMemory,
+ ID_SetCalendarNote,
+ ID_SetIncomingSMS,
+ ID_SetIncomingCB,
+ ID_GetCalendarNotePos,
+ ID_Initialise,
+ ID_GetConnectSet,
+ ID_SetWAPBookmark,
+ ID_GetLocale,
+ ID_SetLocale,
+ ID_GetCalendarSettings,
+ ID_SetCalendarSettings,
+ ID_GetGPRSPoint,
+ ID_SetGPRSPoint,
+ ID_EnableGPRSPoint,
+ ID_DeleteWAPBookmark,
+ ID_Netmonitor,
+ ID_HoldCall,
+ ID_UnholdCall,
+ ID_ConferenceCall,
+ ID_SplitCall,
+ ID_TransferCall,
+ ID_SwitchCall,
+ ID_GetManufactureMonth,
+ ID_GetProductCode,
+ ID_GetOriginalIMEI,
+ ID_GetHardware,
+ ID_GetPPM,
+ ID_GetSMSMode,
+ ID_GetSMSMemories,
+ ID_GetManufacturer,
+ ID_SetMemoryType,
+ ID_SetMemoryCharset,
+ ID_GetMMSSettings,
+ ID_SetSMSParameters,
+ ID_GetFMStation,
+ ID_SetFMStation,
+ ID_GetLanguage,
+ ID_Reset,
+ ID_GetToDo,
+ ID_PressKey,
+ ID_DeleteAllToDo,
+ ID_SetLight,
+ ID_Divert,
+ ID_SetToDo,
+ ID_PlayTone,
+ ID_GetChatSettings,
+ ID_GetSyncMLSettings,
+ ID_GetSyncMLName,
+ ID_GetSecurityStatus,
+ ID_EnterSecurityCode,
+ ID_GetProfile,
+ ID_GetRingtonesInfo,
+ ID_MakeAuthentication,
+ ID_GetSpeedDial,
+ ID_ResetPhoneSettings,
+ ID_SendDTMF,
+ ID_GetDisplayStatus,
+ ID_SetAutoNetworkLogin,
+ ID_SetConnectSet,
+ ID_SetMMSSettings,
+ ID_GetSIMIMSI,
+ ID_GetFileInfo,
+ ID_FileSystemStatus,
+ ID_GetFile,
+ ID_AddFile,
+ ID_AddFolder,
+ ID_DeleteFile,
+#ifdef GSM_ENABLE_ALCATEL
+ /* AT mode */
+ ID_SetFlowControl,
+ ID_AlcatelConnect,
+ ID_AlcatelProtocol,
+
+ /* Binary mode */
+ ID_AlcatelAttach,
+ ID_AlcatelDetach,
+ ID_AlcatelCommit,
+ ID_AlcatelCommit2,
+ ID_AlcatelEnd,
+ ID_AlcatelClose,
+ ID_AlcatelStart,
+ ID_AlcatelSelect1,
+ ID_AlcatelSelect2,
+ ID_AlcatelSelect3,
+ ID_AlcatelBegin1,
+ ID_AlcatelBegin2,
+ ID_AlcatelGetIds1,
+ ID_AlcatelGetIds2,
+ ID_AlcatelGetCategories1,
+ ID_AlcatelGetCategories2,
+ ID_AlcatelGetCategoryText1,
+ ID_AlcatelGetCategoryText2,
+ ID_AlcatelAddCategoryText1,
+ ID_AlcatelAddCategoryText2,
+ ID_AlcatelGetFields1,
+ ID_AlcatelGetFields2,
+ ID_AlcatelGetFieldValue1,
+ ID_AlcatelGetFieldValue2,
+ ID_AlcatelDeleteItem1,
+ ID_AlcatelDeleteItem2,
+ ID_AlcatelDeleteField,
+ ID_AlcatelCreateField,
+ ID_AlcatelUpdateField,
+#endif
+ ID_IncomingFrame,
+
+ ID_User1,
+ ID_User2,
+ ID_User3,
+ ID_User4,
+ ID_User5,
+ ID_User6,
+ ID_User7,
+ ID_User8,
+ ID_User9,
+ ID_User10,
+
+ ID_EachFrame
+} GSM_Phone_RequestID;
+
+/**
+ * Phone related data are stored here.
+ */
+typedef struct {
+ /**
+ * Phone IMEI (or serial number).
+ */
+ char IMEI[MAX_IMEI_LENGTH];
+ /**
+ * Phone manufacturer as reported by phone.
+ */
+ char Manufacturer[MAX_MANUFACTURER_LENGTH];
+ /**
+ * Phone model as reported by phone.
+ */
+ char Model[MAX_MODEL_LENGTH];
+ /**
+ * Model information, pointer to static @ref allmodels array.
+ */
+ OnePhoneModel *ModelInfo;
+ /**
+ * Phone version as reported by phone. It doesn't have to be numerical
+ * at all.
+ */
+ char Version[MAX_VERSION_LENGTH];
+ /**
+ * Phone version date, might be empty for some models.
+ */
+ char VerDate[MAX_VERSION_DATE_LENGTH];
+ /**
+ * Phone version as number, if applicable.
+ */
+ double VerNum;
+ /**
+ * Cache for hardware version used by some modules.
+ */
+ char HardwareCache[50];
+ /**
+ * Cache for product code version used by some modules.
+ */
+ char ProductCodeCache[50];
+
+ /**
+ * Counter used for disabling startup info on phone, see
+ * @ref GSM_Phone_Functions::ShowStartInfo . After this is 0, the startup info is hidden.
+ */
+ int StartInfoCounter;
+
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_GPRSAccessPoint *GPRSPoint;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SpeedDial *SpeedDial;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_DateTime *DateTime;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_Alarm *Alarm;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_MemoryEntry *Memory;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_MemoryStatus *MemoryStatus;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SMSC *SMSC;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_MultiSMSMessage *GetSMSMessage;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SMSMessage *SaveSMSMessage;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SMSMemoryStatus *SMSStatus;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SMSFolders *SMSFolders;
+ /**
+ * Used internally by phone drivers.
+ */
+ int *VoiceRecord;
+ /**
+ * Used internally by phone drivers.
+ */
+ int CallID;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SignalQuality *SignalQuality;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_BatteryCharge *BatteryCharge;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_NetworkInfo *NetworkInfo;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_Ringtone *Ringtone;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_CalendarEntry *Cal;
+ /**
+ * Used internally by phone drivers.
+ */
+ unsigned char *SecurityCode;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_WAPBookmark *WAPBookmark;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_MultiWAPSettings *WAPSettings;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_Bitmap *Bitmap;
+ /**
+ * Used internally by phone drivers.
+ */
+ unsigned char *Netmonitor;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_MultiCallDivert *Divert;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_ToDoEntry *ToDo;
+ /**
+ * Used internally by phone drivers.
+ */
+ bool PressKey;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_SecurityCodeType *SecurityStatus;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_Profile *Profile;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_AllRingtonesInfo *RingtonesInfo;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_DisplayFeatures *DisplayFeatures;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_FMStation *FMStation;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_Locale *Locale;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_CalendarSettings *CalendarSettings;
+ /**
+ * Used internally by phone drivers.
+ */
+ unsigned char *PhoneString;
+ /**
+ * Used internally by phone drivers.
+ */
+ int StartPhoneString;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_File *FileInfo;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_File *File;
+ /**
+ * Pointer to structure used internally by phone drivers.
+ */
+ GSM_FileSystemStatus *FileSystemStatus;
+ GSM_ChatSettings *ChatSettings;
+ GSM_SyncMLSettings *SyncMLSettings;
+
+ /**
+ * Should phone notify about incoming calls?
+ */
+ bool EnableIncomingCall;
+ /**
+ * Should phone notify about incoming SMSes?
+ */
+ bool EnableIncomingSMS;
+ /**
+ * Should phone notify about incoming CBs?
+ */
+ bool EnableIncomingCB;
+ /**
+ * Should phone notify about incoming USSDs?
+ */
+ bool EnableIncomingUSSD;
+
+ /**
+ * Last message received from phone.
+ */
+ GSM_Protocol_Message *RequestMsg;
+ /**
+ * Last message sent by Gammu.
+ */
+ GSM_Protocol_Message *SentMsg;
+ /**
+ * What operation is being performed now, see @ref GSM_Phone_RequestID
+ * for possible values.
+ */
+ GSM_Phone_RequestID RequestID;
+ /**
+ * Error returned by function in phone module.
+ */
+ GSM_Error DispatchError;
+
+ /**
+ * Structure with private phone modules data.
+ */
+ struct {
+ int fake;
+#ifdef GSM_ENABLE_NOKIA3320
+ GSM_Phone_N3320Data N3320;
+#endif
+#ifdef GSM_ENABLE_NOKIA3650
+ GSM_Phone_N3650Data N3650;
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ GSM_Phone_N6110Data N6110;
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ GSM_Phone_N6510Data N6510;
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ GSM_Phone_N7110Data N7110;
+#endif
+#ifdef GSM_ENABLE_ATGEN
+ GSM_Phone_ATGENData ATGEN;
+#endif
+#ifdef GSM_ENABLE_ALCATEL
+ GSM_Phone_ALCATELData ALCATEL;
+#endif
+#ifdef GSM_ENABLE_OBEXGEN
+ GSM_Phone_OBEXGENData OBEXGEN;
+#endif
+#ifdef GSM_ENABLE_MROUTERGEN
+ GSM_Phone_MROUTERGENData MROUTERGEN;
+#endif
+ } Priv;
+} GSM_Phone_Data;
+
+/**
+ * Structure for defining reply functions.
+ *
+ * Function is called when requestID matches current operation or is
+ * ID_IncomingFrame and msgtype matches start message and (if msgtype is just
+ * one character) subtypechar is zero or subtypechar-th character of message
+ * matches subtype.
+ *
+ * Should be used in array with last element containing ID_None as requestID.
+ */
+typedef struct {
+ /**
+ * Pointer to function that should be executed.
+ */
+ GSM_Error (*Function) (GSM_Protocol_Message msg, GSM_StateMachine *s);
+ /**
+ * Message type, if it is longer than 1 character, it disables subtype
+ * checking.
+ */
+ unsigned char *msgtype;
+ /**
+ * Which character of message should be checked as subtype. Zero to
+ * disable subtype checking.
+ */
+ int subtypechar;
+ /**
+ * Subtype to be checked.
+ */
+ unsigned char subtype;
+ /**
+ * Phone request when this can be called, use ID_IncomingFrame when
+ * you want to use this in any state.
+ */
+ GSM_Phone_RequestID requestID;
+} GSM_Reply_Function;
+
+/**
+ * Structure defining phone functions.
+ */
+typedef struct {
+ /**
+ * Names of supported models separated by |. Must contain at least one
+ * name.
+ */
+ char *models;
+ /**
+ * Array of reply functions for the phone, see
+ * @ref GSM_Reply_Function for details about it.
+ */
+ GSM_Reply_Function *ReplyFunctions;
+ /**
+ * Initializes phone.
+ */
+ GSM_Error (*Initialise) (GSM_StateMachine *s);
+ /**
+ * Terminates phone communication.
+ */
+ GSM_Error (*Terminate) (GSM_StateMachine *s);
+ /**
+ * Dispatches messages from phone, at the end it should call
+ * @ref GSM_DispatchMessage.
+ */
+ GSM_Error (*DispatchMessage) (GSM_StateMachine *s);
+ /**
+ * Enables showing information on phone display.
+ */
+ GSM_Error (*ShowStartInfo) (GSM_StateMachine *s, bool enable);
+ /**
+ * Reads manufacturer from phone.
+ */
+ GSM_Error (*GetManufacturer) (GSM_StateMachine *s);
+ /**
+ * Reads model from phone.
+ */
+ GSM_Error (*GetModel) (GSM_StateMachine *s);
+ /**
+ * Reads firmware information from phone.
+ */
+ GSM_Error (*GetFirmware) (GSM_StateMachine *s);
+ /**
+ * Reads IMEI/serial number from phone.
+ */
+ GSM_Error (*GetIMEI) (GSM_StateMachine *s);
+ /**
+ * Gets date and time from phone.
+ */
+ GSM_Error (*GetOriginalIMEI) (GSM_StateMachine *s, char *value);
+ /**
+ * Gets month when device was manufactured.
+ */
+ GSM_Error (*GetManufactureMonth)(GSM_StateMachine *s, char *value);
+ /**
+ * Gets product code of device.
+ */
+ GSM_Error (*GetProductCode) (GSM_StateMachine *s, char *value);
+ /**
+ * Gets hardware information about device.
+ */
+ GSM_Error (*GetHardware) (GSM_StateMachine *s, char *value);
+ /**
+ * Gets PPM (Post Programmable Memory) info from phone
+ * (in other words for Nokia get, which language pack is in phone)
+ */
+ GSM_Error (*GetPPM) (GSM_StateMachine *s, char *value);
+ /**
+ * Gets SIM IMSI from phone.
+ */
+ GSM_Error (*GetSIMIMSI) (GSM_StateMachine *s, char *IMSI);
+ /**
+ * Reads date and time from phone.
+ */
+ GSM_Error (*GetDateTime) (GSM_StateMachine *s, GSM_DateTime *date_time);
+ /**
+ * Sets date and time in phone.
+ */
+ GSM_Error (*SetDateTime) (GSM_StateMachine *s, GSM_DateTime *date_time);
+ /**
+ * Reads alarm set in phone.
+ */
+ GSM_Error (*GetAlarm) (GSM_StateMachine *s, GSM_Alarm *alarm);
+ /**
+ * Sets alarm in phone.
+ */
+ GSM_Error (*SetAlarm) (GSM_StateMachine *s, GSM_Alarm *alarm);
+ /**
+ * Gets locale from phone.
+ */
+ GSM_Error (*GetLocale) (GSM_StateMachine *s, GSM_Locale *locale);
+ /**
+ * Sets locale of phone.
+ */
+ GSM_Error (*SetLocale) (GSM_StateMachine *s, GSM_Locale *locale);
+ /**
+ * Emulates key press or key release.
+ */
+ GSM_Error (*PressKey) (GSM_StateMachine *s, GSM_KeyCode Key, bool Press);
+ /**
+ * Performs phone reset.
+ */
+ GSM_Error (*Reset) (GSM_StateMachine *s, bool hard);
+ /**
+ * Resets phone settings.
+ */
+ GSM_Error (*ResetPhoneSettings) (GSM_StateMachine *s, GSM_ResetSettingsType Type);
+ /**
+ * Enters security code (PIN, PUK,...) .
+ */
+ GSM_Error (*EnterSecurityCode) (GSM_StateMachine *s, GSM_SecurityCode Code);
+ /**
+ * Queries whether some security code needs to be entered./
+ */
+ GSM_Error (*GetSecurityStatus) (GSM_StateMachine *s, GSM_SecurityCodeType *Status);
+ /**
+ * Acquired display status.
+ */
+ GSM_Error (*GetDisplayStatus) (GSM_StateMachine *s, GSM_DisplayFeatures *features);
+ /**
+ * Enables network auto login.
+ */
+ GSM_Error (*SetAutoNetworkLogin)(GSM_StateMachine *s);
+ /**
+ * Gets information about batery charge and phone charging state.
+ */
+ GSM_Error (*GetBatteryCharge) (GSM_StateMachine *s, GSM_BatteryCharge *bat);
+ /**
+ * Reads signal quality (strength and error rate).
+ */
+ GSM_Error (*GetSignalQuality) (GSM_StateMachine *s, GSM_SignalQuality *sig);
+ /**
+ * Gets network information.
+ */
+ GSM_Error (*GetNetworkInfo) (GSM_StateMachine *s, GSM_NetworkInfo *netinfo);
+ /**
+ * Reads category from phone.
+ */
+ GSM_Error (*GetCategory) (GSM_StateMachine *s, GSM_Category *Category);
+ /**
+ * Adds category to phone.
+ */
+ GSM_Error (*AddCategory) (GSM_StateMachine *s, GSM_Category *Category);
+ /**
+ * Reads category status (number of used entries) from phone.
+ */
+ GSM_Error (*GetCategoryStatus) (GSM_StateMachine *s, GSM_CategoryStatus *Status);
+ /**
+ * Gets memory (phonebooks or calls) status (eg. number of used and
+ * free entries).
+ */
+ GSM_Error (*GetMemoryStatus) (GSM_StateMachine *s, GSM_MemoryStatus *status);
+ /**
+ * Reads entry from memory (phonebooks or calls). Which entry should
+ * be read is defined in entry.
+ */
+ GSM_Error (*GetMemory) (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+ /**
+ * Reads entry from memory (phonebooks or calls). Which entry should
+ * be read is defined in entry. This can be easily used for reading all entries.
+ */
+ GSM_Error (*GetNextMemory) (GSM_StateMachine *s, GSM_MemoryEntry *entry, bool start);
+ /**
+ * Sets memory (phonebooks or calls) entry.
+ */
+ GSM_Error (*SetMemory) (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+ /**
+ * Deletes memory (phonebooks or calls) entry.
+ */
+ GSM_Error (*AddMemory) (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+ /**
+ * Deletes memory (phonebooks or calls) entry.
+ */
+ GSM_Error (*DeleteMemory) (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+ /**
+ * Deletes all memory (phonebooks or calls) entries of specified type.
+ */
+ GSM_Error (*DeleteAllMemory) (GSM_StateMachine *s, GSM_MemoryType MemoryType);
+ /**
+ * Gets speed dial.
+ */
+ GSM_Error (*GetSpeedDial) (GSM_StateMachine *s, GSM_SpeedDial *Speed);
+ /**
+ * Sets speed dial.
+ */
+ GSM_Error (*SetSpeedDial) (GSM_StateMachine *s, GSM_SpeedDial *Speed);
+ /**
+ * Gets SMS Service Center number and SMS settings.
+ */
+ GSM_Error (*GetSMSC) (GSM_StateMachine *s, GSM_SMSC *smsc);
+ /**
+ * Sets SMS Service Center number and SMS settings.
+ */
+ GSM_Error (*SetSMSC) (GSM_StateMachine *s, GSM_SMSC *smsc);
+ /**
+ * Gets information about SMS memory (read/unread/size of memory for
+ * both SIM and phone).
+ */
+ GSM_Error (*GetSMSStatus) (GSM_StateMachine *s, GSM_SMSMemoryStatus *status);
+ /**
+ * Reads SMS message.
+ */
+ GSM_Error (*GetSMS) (GSM_StateMachine *s, GSM_MultiSMSMessage *sms);
+ /**
+ * Reads next (or first if start set) SMS message. This might be
+ * faster for some phones than using @ref GetSMS for each message.
+ */
+ GSM_Error (*GetNextSMS) (GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start);
+ /**
+ * Sets SMS.
+ */
+ GSM_Error (*SetSMS) (GSM_StateMachine *s, GSM_SMSMessage *sms);
+ /**
+ * Adds SMS to specified folder.
+ */
+ GSM_Error (*AddSMS) (GSM_StateMachine *s, GSM_SMSMessage *sms);
+ /**
+ * Deletes SMS.
+ */
+ GSM_Error (*DeleteSMS) (GSM_StateMachine *s, GSM_SMSMessage *sms);
+ /**
+ * Sends SMS.
+ */
+ GSM_Error (*SendSMS) (GSM_StateMachine *s, GSM_SMSMessage *sms);
+ /**
+ * Sends SMS already saved in phone.
+ */
+ GSM_Error (*SendSavedSMS) (GSM_StateMachine *s, int Folder, int Location);
+ /**
+ * Enable/disable notification on incoming SMS.
+ */
+ GSM_Error (*SetIncomingSMS) (GSM_StateMachine *s, bool enable);
+ /**
+ * Gets network information from phone.
+ */
+ GSM_Error (*SetIncomingCB) (GSM_StateMachine *s, bool enable);
+ /**
+ * Returns SMS folders information.
+ */
+ GSM_Error (*GetSMSFolders) (GSM_StateMachine *s, GSM_SMSFolders *folders);
+ /**
+ * Creates SMS folder.
+ */
+ GSM_Error (*AddSMSFolder) (GSM_StateMachine *s, unsigned char *name);
+ /**
+ * Deletes SMS folder.
+ */
+ GSM_Error (*DeleteSMSFolder) (GSM_StateMachine *s, int ID);
+ /**
+ * Dials number and starts voice call.
+ */
+ GSM_Error (*DialVoice) (GSM_StateMachine *s, char *Number, GSM_CallShowNumber ShowNumber);
+ /**
+ * Accept current incoming call.
+ */
+ GSM_Error (*AnswerCall) (GSM_StateMachine *s, int ID, bool all);
+ /**
+ * Deny current incoming call.
+ */
+ GSM_Error (*CancelCall) (GSM_StateMachine *s, int ID, bool all);
+ /**
+ * Holds call.
+ */
+ GSM_Error (*HoldCall) (GSM_StateMachine *s, int ID);
+ /**
+ * Unholds call.
+ */
+ GSM_Error (*UnholdCall) (GSM_StateMachine *s, int ID);
+ /**
+ * Initiates conference call.
+ */
+ GSM_Error (*ConferenceCall) (GSM_StateMachine *s, int ID);
+ /**
+ * Splits call.
+ */
+ GSM_Error (*SplitCall) (GSM_StateMachine *s, int ID);
+ /**
+ * Transfers call.
+ */
+ GSM_Error (*TransferCall) (GSM_StateMachine *s, int ID, bool next);
+ /**
+ * Switches call.
+ */
+ GSM_Error (*SwitchCall) (GSM_StateMachine *s, int ID, bool next);
+ /**
+ * Gets call diverts.
+ */
+ GSM_Error (*GetCallDivert) (GSM_StateMachine *s, GSM_MultiCallDivert *divert);
+ /**
+ * Sets call diverts.
+ */
+ GSM_Error (*SetCallDivert) (GSM_StateMachine *s, GSM_MultiCallDivert *divert);
+ /**
+ * Cancels all diverts.
+ */
+ GSM_Error (*CancelAllDiverts) (GSM_StateMachine *s);
+ /**
+ * Activates/deactivates noticing about incoming calls.
+ */
+ GSM_Error (*SetIncomingCall) (GSM_StateMachine *s, bool enable);
+ /**
+ * Activates/deactivates noticing about incoming USSDs (UnStructured Supplementary Services).
+ */
+ GSM_Error (*SetIncomingUSSD) (GSM_StateMachine *s, bool enable);
+ /**
+ * Sends DTMF (Dual Tone Multi Frequency) tone.
+ */
+ GSM_Error (*SendDTMF) (GSM_StateMachine *s, char *sequence);
+ /**
+ * Gets ringtone from phone.
+ */
+ GSM_Error (*GetRingtone) (GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone);
+ /**
+ * Sets ringtone in phone.
+ */
+ GSM_Error (*SetRingtone) (GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength);
+ /**
+ * Acquires ringtone informaiton.
+ */
+ GSM_Error (*GetRingtonesInfo) (GSM_StateMachine *s, GSM_AllRingtonesInfo *Info);
+ /**
+ * Deletes user defined ringtones from phone.
+ */
+ GSM_Error (*DeleteUserRingtones)(GSM_StateMachine *s);
+ /**
+ * Plays tone.
+ */
+ GSM_Error (*PlayTone) (GSM_StateMachine *s, int Herz, unsigned char Volume, bool start);
+ /**
+ * Reads WAP bookmark.
+ */
+ GSM_Error (*GetWAPBookmark) (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+ /**
+ * Sets WAP bookmark.
+ */
+ GSM_Error (*SetWAPBookmark) (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+ /**
+ * Deletes WAP bookmark.
+ */
+ GSM_Error (*DeleteWAPBookmark) (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+ /**
+ * Acquires WAP settings.
+ */
+ GSM_Error (*GetWAPSettings) (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+ /**
+ * Changes WAP settings.
+ */
+ GSM_Error (*SetWAPSettings) (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+ /**
+ * Acquires MMS settings.
+ */
+ GSM_Error (*GetMMSSettings) (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+ /**
+ * Changes MMS settings.
+ */
+ GSM_Error (*SetMMSSettings) (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+ /**
+ * Acquires SyncML settings.
+ */
+ GSM_Error (*GetSyncMLSettings) (GSM_StateMachine *s, GSM_SyncMLSettings *settings);
+ /**
+ * Changes SyncML settings.
+ */
+ GSM_Error (*SetSyncMLSettings) (GSM_StateMachine *s, GSM_SyncMLSettings *settings);
+ /**
+ * Acquires chat/presence settings.
+ */
+ GSM_Error (*GetChatSettings) (GSM_StateMachine *s, GSM_ChatSettings *settings);
+ /**
+ * Changes chat/presence settings.
+ */
+ GSM_Error (*SetChatSettings) (GSM_StateMachine *s, GSM_ChatSettings *settings);
+ /**
+ * Gets bitmap.
+ */
+ GSM_Error (*GetBitmap) (GSM_StateMachine *s, GSM_Bitmap *Bitmap);
+ /**
+ * Sets bitmap.
+ */
+ GSM_Error (*SetBitmap) (GSM_StateMachine *s, GSM_Bitmap *Bitmap);
+ /**
+ * Gets status of ToDos (count of used entries).
+ */
+ GSM_Error (*GetToDoStatus) (GSM_StateMachine *s, GSM_ToDoStatus *status);
+ /**
+ * Reads ToDo from phone.
+ */
+ GSM_Error (*GetToDo) (GSM_StateMachine *s, GSM_ToDoEntry *ToDo);
+ /**
+ * Reads ToDo from phone.
+ */
+ GSM_Error (*GetNextToDo) (GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool start);
+ /**
+ * Sets ToDo in phone.
+ */
+ GSM_Error (*SetToDo) (GSM_StateMachine *s, GSM_ToDoEntry *ToDo);
+ /**
+ * Adds ToDo in phone.
+ */
+ GSM_Error (*AddToDo) (GSM_StateMachine *s, GSM_ToDoEntry *ToDo);
+ /**
+ * Deletes ToDo entry in phone.
+ */
+ GSM_Error (*DeleteToDo) (GSM_StateMachine *s, GSM_ToDoEntry *ToDo);
+ /**
+ * Deletes all todo entries in phone.
+ */
+ GSM_Error (*DeleteAllToDo) (GSM_StateMachine *s);
+ /**
+ * Retrieves calendar status (number of used entries).
+ */
+ GSM_Error (*GetCalendarStatus) (GSM_StateMachine *s, GSM_CalendarStatus *Status);
+ /**
+ * Retrieves calendar entry.
+ */
+ GSM_Error (*GetCalendar) (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+ /**
+ * Retrieves calendar entry. This is useful for continuous reading of all
+ * calendar entries.
+ */
+ GSM_Error (*GetNextCalendar) (GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start);
+ /**
+ * Sets calendar entry
+ */
+ GSM_Error (*SetCalendar) (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+ /**
+ * Adds calendar entry.
+ */
+ GSM_Error (*AddCalendar) (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+ /**
+ * Deletes calendar entry.
+ */
+ GSM_Error (*DeleteCalendar) (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+ /**
+ * Deletes all calendar entries.
+ */
+ GSM_Error (*DeleteAllCalendar) (GSM_StateMachine *s);
+ /**
+ * Reads calendar settings.
+ */
+ GSM_Error (*GetCalendarSettings)(GSM_StateMachine *s, GSM_CalendarSettings *settings);
+ /**
+ * Sets calendar settings.
+ */
+ GSM_Error (*SetCalendarSettings)(GSM_StateMachine *s, GSM_CalendarSettings *settings);
+ /**
+ * Gets note.
+ */
+ GSM_Error (*GetNote) (GSM_StateMachine *s, GSM_NoteEntry *Note, bool refresh);
+ /**
+ * Reads profile.
+ */
+ GSM_Error (*GetProfile) (GSM_StateMachine *s, GSM_Profile *Profile);
+ /**
+ * Updates profile.
+ */
+ GSM_Error (*SetProfile) (GSM_StateMachine *s, GSM_Profile *Profile);
+ /**
+ * Reads FM station.
+ */
+ GSM_Error (*GetFMStation) (GSM_StateMachine *s, GSM_FMStation *FMStation);
+ /**
+ * Sets FM station.
+ */
+ GSM_Error (*SetFMStation) (GSM_StateMachine *s, GSM_FMStation *FMStation);
+ /**
+ * Clears defined FM stations.
+ */
+ GSM_Error (*ClearFMStations) (GSM_StateMachine *s);
+ /**
+ * Gets next filename from filesystem.
+ */
+ GSM_Error (*GetNextFileFolder) (GSM_StateMachine *s, GSM_File *File, bool start);
+ /**
+ * Gets file part from filesystem.
+ */
+ GSM_Error (*GetFilePart) (GSM_StateMachine *s, GSM_File *File);
+ /**
+ * Adds file part to filesystem.
+ */
+ GSM_Error (*AddFilePart) (GSM_StateMachine *s, GSM_File *File, int *Pos);
+ /**
+ * Acquires filesystem status.
+ */
+ GSM_Error (*GetFileSystemStatus)(GSM_StateMachine *s, GSM_FileSystemStatus *Status);
+ /**
+ * Deletes file from filessytem.
+ */
+ GSM_Error (*DeleteFile) (GSM_StateMachine *s, unsigned char *ID);
+ /**
+ * Adds folder to filesystem.
+ */
+ GSM_Error (*AddFolder) (GSM_StateMachine *s, GSM_File *File);
+ /**
+ * Gets GPRS access point.
+ */
+ GSM_Error (*GetGPRSAccessPoint) (GSM_StateMachine *s, GSM_GPRSAccessPoint *point);
+ /**
+ * Sets GPRS access point.
+ */
+ GSM_Error (*SetGPRSAccessPoint) (GSM_StateMachine *s, GSM_GPRSAccessPoint *point);
+} GSM_Phone_Functions;
+
+ extern GSM_Phone_Functions NAUTOPhone;
+#ifdef GSM_ENABLE_NOKIA3320
+ extern GSM_Phone_Functions N3320Phone;
+#endif
+#ifdef GSM_ENABLE_NOKIA3650
+ extern GSM_Phone_Functions N3650Phone;
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ extern GSM_Phone_Functions N6110Phone;
+#endif
+#ifdef GSM_ENABLE_NOKIA6510
+ extern GSM_Phone_Functions N6510Phone;
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ extern GSM_Phone_Functions N7110Phone;
+#endif
+#ifdef GSM_ENABLE_NOKIA9210
+ extern GSM_Phone_Functions N9210Phone;
+#endif
+#ifdef GSM_ENABLE_ATGEN
+ extern GSM_Phone_Functions ATGENPhone;
+#endif
+#ifdef GSM_ENABLE_ALCATEL
+ extern GSM_Phone_Functions ALCATELPhone;
+#endif
+#ifdef GSM_ENABLE_OBEXGEN
+ extern GSM_Phone_Functions OBEXGENPhone;
+#endif
+#ifdef GSM_ENABLE_MROUTERGEN
+ extern GSM_Phone_Functions MROUTERGENPhone;
+#endif
+
+typedef struct {
+ GSM_Phone_Data Data;
+ GSM_Phone_Functions *Functions;
+} GSM_Phone;
+
+/* --------------------------- User layer ---------------------------------- */
+
+struct _GSM_User {
+ GSM_Reply_Function *UserReplyFunctions;
+
+ void (*IncomingCall) (char *Device, GSM_Call call);
+ void (*IncomingSMS) (char *Device, GSM_SMSMessage sms);
+ void (*IncomingCB) (char *Device, GSM_CBMessage cb);
+ void (*IncomingUSSD) (char *Device, char *Text);
+ void (*SendSMSStatus) (char *Device, int status, int MessageReference);
+};
+
+/* --------------------------- Statemachine layer -------------------------- */
+
+typedef enum {
+ GCT_MBUS2=1,
+ GCT_FBUS2,
+ GCT_FBUS2DLR3,
+ GCT_FBUS2DKU5,
+ GCT_FBUS2PL2303,
+ GCT_FBUS2BLUE,
+ GCT_FBUS2IRDA,
+ GCT_PHONETBLUE,
+ GCT_AT,
+ GCT_MROUTERBLUE,
+
+ GCT_IRDAOBEX,
+ GCT_IRDAAT,
+ GCT_IRDAPHONET,
+ GCT_BLUEFBUS2,
+ GCT_BLUEAT,
+ GCT_BLUEPHONET,
+ GCT_BLUEOBEX
+} GSM_ConnectionType;
+
+typedef struct {
+ /* Config file (or Registry or...) variables */
+ char Model[50]; /* Model from config file */
+ char DebugLevel[50]; /* Debug level */
+ char *Device; /* Device name from config file */
+ char *Connection; /* Connection type as string */
+ char *SyncTime; /* Synchronize time on startup? */
+ char *LockDevice; /* Lock device ? (Unix) */
+ char *DebugFile; /* Name of debug file */
+ char *Localize; /* Name of localisation file */
+ char *StartInfo; /* Display something during start ? */
+ bool UseGlobalDebugFile;/* Should we use global debug file? */
+ bool DefaultModel;
+ bool DefaultDebugLevel;
+ bool DefaultDevice;
+ bool DefaultConnection;
+ bool DefaultSyncTime;
+ bool DefaultLockDevice;
+ bool DefaultDebugFile;
+ bool DefaultLocalize;
+ bool DefaultStartInfo;
+} GSM_Config;
+
+#define MAX_CONFIG_NUM 5
+
+struct _GSM_StateMachine {
+ GSM_ConnectionType ConnectionType; /* Type of connection as int */
+ char *LockFile; /* Lock file name for Unix */
+ Debug_Info di; /* Debug information */
+ bool opened; /* Is connection opened ? */
+ GSM_Config Config[MAX_CONFIG_NUM + 1];
+ GSM_Config *CurrentConfig; /* Config file (or Registry or...) variables */
+ int ConfigNum;
+ INI_Section *msg; /* Localisation strings structure */
+ int ReplyNum; /* How many times make sth. */
+ int Speed; /* For some protocols used speed */
+
+ GSM_Device Device;
+ GSM_Protocol Protocol;
+ GSM_Phone Phone;
+ GSM_User User;
+};
+
+/* ------------------------ Other general definitions ---------------------- */
+
+GSM_Error GSM_RegisterAllPhoneModules (GSM_StateMachine *s);
+
+GSM_Error GSM_InitConnection (GSM_StateMachine *s, int ReplyNum);
+GSM_Error GSM_TerminateConnection (GSM_StateMachine *s);
+
+int GSM_ReadDevice (GSM_StateMachine *s, bool wait);
+
+GSM_Error GSM_WaitForOnce (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type, int time);
+
+GSM_Error GSM_WaitFor (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type, int time,
+ GSM_Phone_RequestID request);
+
+GSM_Error GSM_DispatchMessage (GSM_StateMachine *s);
+
+INI_Section *GSM_FindGammuRC(void);
+
+bool GSM_ReadConfig (INI_Section *cfg_info, GSM_Config *cfg, int num);
+
+void GSM_DumpMessageLevel2 (GSM_StateMachine *s, unsigned char *message, int messagesize, int type);
+void GSM_DumpMessageLevel3 (GSM_StateMachine *s, unsigned char *message, int messagesize, int type);
+
+/* ---------------------- Phone features ----------------------------------- */
+
+typedef enum {
+ /* n6110.c */
+ F_CAL33 = 1, /* Calendar,3310 style - 10 reminders, Unicode, 3 coding types */
+ F_CAL52, /* Calendar,5210 style - full Unicode, etc. */
+ F_CAL82, /* Calendar,8250 style - "normal", but with Unicode */
+ F_RING_SM, /* Ringtones returned in SM format - 33xx */
+ F_NORING, /* No ringtones */
+ F_NOPBKUNICODE, /* No phonebook in Unicode */
+ F_NOWAP, /* No WAP */
+ F_NOCALLER, /* No caller groups */
+ F_NOPICTURE, /* No Picture Images */
+ F_NOPICTUREUNI, /* No Picture Images text in Unicode */
+ F_NOSTARTUP, /* No startup logo */
+ F_NOCALENDAR, /* No calendar */
+ F_NOSTARTANI, /* Startup logo is not animated */
+ F_POWER_BATT, /* Network and battery level get from netmonitor */
+ F_PROFILES33, /* Phone profiles in 3310 style */
+ F_PROFILES51, /* Phone profiles in 5110 style */
+ F_MAGICBYTES, /* Phone can make authentication with magic bytes */
+ F_NODTMF, /* Phone can't send DTMF */
+ F_DISPSTATUS, /* Phone return display status */
+ F_NOCALLINFO,
+
+ /* n3320.c */
+ F_DAYMONTH, /* Day and month reversed in pbk, when compare to GSM models */
+
+ /* n6510.c */
+ F_PBK35, /* Phonebook in 3510 style with ringtones ID */
+ F_PBKIMG, /* Phonebook in 7250 style with picture ID */
+ F_PBKTONEGAL, /* Phonebook with selecting ringtones from gallery */
+ F_PBKSMSLIST, /* Phonebook with SMS list */
+ F_PBKUSER, /* Phonebook with user ID */
+ F_RADIO, /* Phone with FM radio */
+ F_TODO63, /* ToDo in 6310 style - 0x55 msg type */
+ F_TODO66, /* ToDo in 6610 style - like calendar, with date and other */
+ F_NOMIDI, /* No ringtones in MIDI */
+ F_BLUETOOTH, /* Bluetooth support */
+ F_NOFILESYSTEM, /* No images, ringtones, java saved in special filesystem */
+ F_NOMMS, /* No MMS sets in phone */
+ F_NOGPRSPOINT, /* GPRS point are not useable */
+ F_CAL35, /* Calendar,3510 style - Reminder,Call,Birthday */
+ F_CAL65, /* Calendar,6510 style - CBMM, method 3 */
+ F_WAPMMSPROXY, /* WAP & MMS settings contains first & second proxy */
+
+ /* n6510.c && n7110.c */
+ F_VOICETAGS, /* Voice tags available */
+ F_CAL62, /* Calendar,6210 style - Call,Birthday,Memo,Meeting */
+
+ /* AT modules */
+ F_SMSONLYSENT, /* Phone supports only sent/unsent messages */
+ F_BROKENCPBS, /* CPBS on some memories can hang phone */
+ F_M20SMS, /* Siemens M20 like SMS handling */
+ F_SLOWWRITE, /* Use slower writing which some phone need */
+ F_SMSME900, /* SMS in ME start from location 900 - case of Sagem */
+ F_ALCATEL /* Phone supports Alcatel protocol */
+} Feature;
+
+/* For models table */
+struct _OnePhoneModel {
+ char *model;
+ char *number;
+ char *irdamodel;
+ Feature features[12];
+};
+
+bool IsPhoneFeatureAvailable (OnePhoneModel *model, Feature feature);
+OnePhoneModel *GetModelData (char *model, char *number, char *irdamodel);
+
+#ifdef __GNUC__
+__attribute__((format(printf, 2, 3)))
+#endif
+int smprintf(GSM_StateMachine *s, const char *format, ...);
+
+void GSM_OSErrorInfo(GSM_StateMachine *s, char *description);
+
+#ifdef GSM_ENABLE_BACKUP
+void GSM_GetPhoneFeaturesForBackup(GSM_StateMachine *s, GSM_Backup_Info *info);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/misc/cfg.c b/gammu/emb/common/misc/cfg.c
new file mode 100644
index 0000000..1c74874
--- a/dev/null
+++ b/gammu/emb/common/misc/cfg.c
@@ -0,0 +1,332 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include <errno.h>
+#ifndef __OpenBSD__
+# include <wchar.h>
+# include <wctype.h>
+#endif
+
+#include "../config.h"
+#include "coding/coding.h"
+#include "cfg.h"
+#include "misc.h"
+
+/*
+ * Read information from file in Windows INI format style
+ */
+INI_Section *INI_ReadFile(char *FileName, bool Unicode)
+{
+ FILE *f;
+ bool FFEEUnicode=false;
+ int level = -1, buffer1used, buffer2used;
+ int bufferused, i, buffused=1000,buffread=1000, num;
+ unsigned char ch[3], *buffer = NULL;
+ unsigned char *buffer2 = NULL, *buffer1 = NULL, buff[1000];
+ INI_Section *INI_info = NULL, *INI_head = NULL, *heading;
+ INI_Entry *entry;
+
+ f = fopen(FileName,"rb");
+ if (f == NULL) return NULL;
+
+ num = 0;
+ while(1) {
+ /* We read one line from file */
+ bufferused = 0;
+ while (1) {
+ if (buffused == buffread) {
+ buffused = fread(buff,1,1000,f);
+ buffread = 0;
+ if (buffused == 0) {
+ free(buffer); free(buffer1); free(buffer2);
+ fclose(f);
+ return INI_head;
+ }
+ }
+ if (Unicode) {
+ if (num == 0) {
+ if (buffused == buffread) continue;
+ ch[0] = buff[buffread++];
+ num = 1;
+ }
+ if (num == 1) {
+ if (buffused == buffread) continue;
+ ch[1] = buff[buffread++];
+ num = 0;
+ }
+ if (level == -1) {
+ if (ch[0] == 0xFF && ch[1] == 0xFE) FFEEUnicode = true;
+ level = 0;
+ continue;
+ }
+ if (FFEEUnicode) {
+ ch[2] = ch[0]; ch[0] = ch[1]; ch[1] = ch[2];
+ }
+ } else {
+ if (buffused == buffread) continue;
+ ch[0] = 0;
+ ch[1] = buff[buffread++];
+ if (level == -1) level = 0;
+ }
+ if ((ch[0] == 0 && ch[1] == 13) ||
+ (ch[0] == 0 && ch[1] == 10)) {
+ break;
+ }
+ buffer = realloc(buffer,bufferused+2);
+ buffer[bufferused] = ch[0];
+ buffer[bufferused+1] = ch[1];
+ bufferused = bufferused + 2;
+ }
+// printf("line \"%s\"\n",DecodeUnicodeConsole(buffer));
+
+ buffer1used = 0;
+ buffer2used = 0;
+ if (level == 1) level = 0;
+ if (level == 3 || level == 4 || level == 5) level = 2;
+
+ /* We parse read line */
+ for (i=0;i<bufferused/2;i++) {
+ ch[0] = buffer[i*2];
+ ch[1] = buffer[i*2+1];
+ if (level == 0) { //search for name of section
+ if (ch[0] == 0 && ch[1] == '[') level = 1;
+ if (ch[0] == 0 && ch[1] == ';') break;
+ if (ch[0] == 0 && ch[1] == '#') break;
+ continue;
+ }
+ if (level == 1) { //section name
+ if (ch[0] == 0 && ch[1] == ']') {
+ if (buffer1used == 0) break;
+ if (Unicode) {
+ buffer1 = realloc(buffer1,buffer1used+2);
+ buffer1[buffer1used] = 0;
+ buffer1[buffer1used+1] = 0;
+ buffer1used = buffer1used + 2;
+ } else {
+ buffer1 = realloc(buffer1,buffer1used+1);
+ buffer1[buffer1used] = 0x00;
+ buffer1used = buffer1used + 1;
+ }
+ heading = (INI_Section *)malloc(sizeof(*heading));
+ if (heading == NULL) {
+ free(buffer); free(buffer1); free(buffer2);
+ fclose(f);
+ return NULL;
+ }
+ heading->SectionName = (char *)malloc(buffer1used);
+ memcpy(heading->SectionName,buffer1,buffer1used);
+ heading->Prev = INI_info;
+ heading->Next = NULL;
+ if (INI_info != NULL) {
+ INI_info->Next = heading;
+ } else {
+ INI_head = heading;
+ }
+ INI_info = heading;
+ INI_info->SubEntries = NULL;
+ level = 2;
+// printf("[%s]\n",DecodeUnicodeConsole(buffer1));
+ break;
+ }
+ if (Unicode) {
+ buffer1 = realloc(buffer1,buffer1used+2);
+ buffer1[buffer1used] = ch[0];
+ buffer1[buffer1used+1] = ch[1];
+ buffer1used = buffer1used + 2;
+ } else {
+ buffer1 = realloc(buffer1,buffer1used+1);
+ buffer1[buffer1used] = ch[1];
+ buffer1used = buffer1used + 1;
+ }
+ continue;
+ }
+ if (level == 2) { //search for key name
+ if (ch[0] == 0 && ch[1] == ';') break;
+ if (ch[0] == 0 && ch[1] == '#') break;
+ if (ch[0] == 0 && ch[1] == '[') {
+ level = 1;
+ continue;
+ }
+ if (Unicode) {
+ if (myiswspace(ch)) continue;
+ } else {
+ if (isspace((int) ch[1])) continue;
+ }
+ level = 3;
+ }
+ if (level == 3) { //key name
+ if (ch[0] == 0 && ch[1] == '=') {
+ if (buffer1used == 0) break;
+ while(1) {
+ if (Unicode) {
+ if (!myiswspace(buffer1+(buffer1used-2))) break;
+ buffer1used = buffer1used - 2;
+ } else {
+ if (!isspace((int)buffer1[buffer1used-1])) break;
+ buffer1used = buffer1used - 1;
+ }
+ }
+ level = 4;
+ continue;
+ }
+ if (Unicode) {
+ buffer1 = realloc(buffer1,buffer1used+2);
+ buffer1[buffer1used] = ch[0];
+ buffer1[buffer1used+1] = ch[1];
+ buffer1used = buffer1used + 2;
+ } else {
+ buffer1 = realloc(buffer1,buffer1used+1);
+ buffer1[buffer1used] = ch[1];
+ buffer1used = buffer1used + 1;
+ }
+ }
+ if (level == 4) { //search for key value
+ if (Unicode) {
+ if (myiswspace(ch)) continue;
+ } else {
+ if (isspace((int) ch[1])) continue;
+ }
+ level = 5;
+ }
+ if (level == 5) { //key value
+ if (Unicode) {
+ buffer2 = realloc(buffer2,buffer2used+2);
+ buffer2[buffer2used] = ch[0];
+ buffer2[buffer2used+1] = ch[1];
+ buffer2used = buffer2used + 2;
+ } else {
+ buffer2 = realloc(buffer2,buffer2used+1);
+ buffer2[buffer2used] = ch[1];
+ buffer2used = buffer2used + 1;
+ }
+ }
+ }
+ if (level == 5) {
+ if (buffer2used == 0) continue;
+
+ entry = (INI_Entry *)malloc(sizeof(*entry));
+ if (entry == NULL) {
+ free(buffer); free(buffer1); free(buffer2);
+ fclose(f);
+ return NULL;
+ }
+ if (Unicode) {
+ buffer1 = realloc(buffer1,buffer1used+2);
+ buffer1[buffer1used] = 0;
+ buffer1[buffer1used+1] = 0;
+ buffer1used = buffer1used + 2;
+ buffer2 = realloc(buffer2,buffer2used+2);
+ buffer2[buffer2used] = 0;
+ buffer2[buffer2used+1] = 0;
+ buffer2used = buffer2used + 2;
+ } else {
+ buffer1 = realloc(buffer1,buffer1used+1);
+ buffer1[buffer1used] = 0x00;
+ buffer1used = buffer1used + 1;
+ buffer2 = realloc(buffer2,buffer2used+1);
+ buffer2[buffer2used] = 0x00;
+ buffer2used = buffer2used + 1;
+ }
+// printf("\"%s\"=\"%s\"\n",buffer1,buffer2);
+// printf("\"%s\"=",DecodeUnicodeConsole(buffer1));
+// printf("\"%s\"\n",DecodeUnicodeConsole(buffer2));
+
+ entry->EntryName = (char *)malloc(buffer1used);
+ memcpy(entry->EntryName,buffer1,buffer1used);
+
+ entry->EntryValue = (char *)malloc(buffer2used);
+ memcpy(entry->EntryValue,buffer2,buffer2used);
+
+ entry->Prev = NULL;
+ entry->Next = INI_info->SubEntries;
+ if (INI_info->SubEntries != NULL) INI_info->SubEntries->Prev = entry;
+ INI_info->SubEntries = entry;
+ }
+ }
+ free(buffer); free(buffer1); free(buffer2);
+ fclose(f);
+ return INI_head;
+}
+
+/*
+ * Search for key value in file in Windows INI format style
+ * Returns found value or NULL
+ */
+unsigned char *INI_GetValue(INI_Section *cfg, unsigned char *section, unsigned char *key, bool Unicode)
+{
+ INI_Section *sec;
+ INI_Entry *ent;
+
+ if (cfg == NULL || section == NULL || key == NULL) return NULL;
+
+ if (Unicode) {
+ /* Search for section */
+ sec = cfg;
+ while (sec != NULL) {
+ if (mywstrncasecmp(section, sec->SectionName, 0)) {
+ /* Search for key inside section */
+ ent = sec->SubEntries;
+ while (ent != NULL) {
+ if (mywstrncasecmp(key,ent->EntryName,0)) {
+ return ent->EntryValue;
+ }
+ ent = ent->Next;
+ }
+ }
+ sec = sec->Next;
+ }
+ } else {
+ /* Search for section */
+ sec = cfg;
+ while (sec != NULL) {
+ if (mystrncasecmp(section, sec->SectionName, 0)) {
+ /* Search for key inside section */
+ ent = sec->SubEntries;
+ while (ent != NULL) {
+ if (mystrncasecmp(key,ent->EntryName,0)) {
+ return ent->EntryValue;
+ }
+ ent = ent->Next;
+ }
+ }
+ sec = sec->Next;
+ }
+ }
+ return NULL;
+}
+
+/* Return last value in specified section */
+INI_Entry *INI_FindLastSectionEntry(INI_Section *file_info, unsigned char *section, bool Unicode)
+{
+ INI_Section *h;
+ INI_Entry *e;
+
+ e = NULL;
+ /* First find our section */
+ for (h = file_info; h != NULL; h = h->Next) {
+ if (Unicode) {
+ if (mywstrncasecmp(section, h->SectionName, 0)) {
+ e = h->SubEntries;
+ break;
+ }
+ } else {
+ if (mystrncasecmp(section, h->SectionName, 0)) {
+ e = h->SubEntries;
+ break;
+ }
+ }
+ }
+
+ if (e == NULL) return NULL;
+
+ /* Goes into last value in section */
+ while (e->Next != NULL) e = e->Next;
+ return e;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/misc/cfg.h b/gammu/emb/common/misc/cfg.h
new file mode 100644
index 0000000..edb9b09
--- a/dev/null
+++ b/gammu/emb/common/misc/cfg.h
@@ -0,0 +1,42 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef _cfg_h
+#define _cfg_h
+
+#include "misc.h"
+
+/* -------------------------------- structures ----------------------------- */
+
+typedef struct _INI_Entry INI_Entry;
+
+/*
+ * Structure used to save value for single key in INI style file
+ */
+struct _INI_Entry {
+ INI_Entry *Next, *Prev;
+ unsigned char *EntryName;
+ unsigned char *EntryValue;
+};
+
+typedef struct _INI_Section INI_Section;
+
+/*
+ * Structure used to save section in INI style file
+ */
+struct _INI_Section {
+ INI_Section *Next, *Prev;
+ INI_Entry *SubEntries;
+ unsigned char *SectionName;
+};
+
+/* ------------------------- function prototypes --------------------------- */
+
+INI_Section *INI_ReadFile (char *FileName, bool Unicode);
+INI_Entry *INI_FindLastSectionEntry (INI_Section *file_info, unsigned char *section, bool Unicode);
+unsigned char *INI_GetValue (INI_Section *cfg, unsigned char *section, unsigned char *key, bool Unicode);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/misc/coding/coding.c b/gammu/emb/common/misc/coding/coding.c
new file mode 100644
index 0000000..62543ac
--- a/dev/null
+++ b/gammu/emb/common/misc/coding/coding.c
@@ -0,0 +1,1409 @@
+/* (c) 2002-2004 by Marcin Wiacek, Michal Cihar and others */
+/* based on some work from MyGnokii and Gnokii */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include <locale.h>
+#ifndef __OpenBSD__
+# include <wctype.h>
+#endif
+#ifdef WIN32
+# include "windows.h"
+#endif
+
+#include "../misc.h"
+#include "coding.h"
+
+unsigned int UnicodeLength(const unsigned char *str)
+{
+ unsigned int len = 0;
+
+ if (str == NULL) return 0;
+
+ while(str[len*2] != 0 || str[len*2+1] != 0) len++;
+
+ return len;
+}
+
+/* Convert Unicode char saved in src to dest */
+unsigned int EncodeWithUnicodeAlphabet(const unsigned char *src, wchar_t *dest)
+{
+ char retval;
+
+ switch (retval = mbtowc(dest, src, MB_CUR_MAX)) {
+ case -1 :
+ case 0 : return 1;
+ default : return retval;
+ }
+}
+
+/* Convert Unicode char saved in src to dest */
+unsigned int DecodeWithUnicodeAlphabet(wchar_t src, unsigned char *dest)
+{
+ int retval;
+
+ switch (retval = wctomb(dest, src)) {
+ case -1:
+ *dest = '?';
+ return 1;
+ default:
+ return retval;
+ }
+}
+
+void DecodeUnicode (const unsigned char *src, unsigned char *dest)
+{
+ int i=0,o=0;
+ wchar_t wc;
+
+ while (src[(2*i)+1]!=0x00 || src[2*i]!=0x00) {
+ wc = src[(2*i)+1] | (src[2*i] << 8);
+ o += DecodeWithUnicodeAlphabet(wc, dest + o);
+ i++;
+ }
+ dest[o]=0;
+}
+
+/* Decode Unicode string and return as function result */
+unsigned char *DecodeUnicodeString (const unsigned char *src)
+{
+ static char dest[500];
+
+ DecodeUnicode(src,dest);
+ return dest;
+}
+
+/* Decode Unicode string to UTF8 or other console charset
+ * and return as function result
+ */
+unsigned char *DecodeUnicodeConsole(const unsigned char *src)
+{
+ static char dest[500];
+
+ if (di.coding[0] != 0) {
+ if (!strcmp(di.coding,"utf8")) {
+ EncodeUTF8(dest, src);
+ } else {
+#ifdef WIN32
+ setlocale(LC_ALL, di.coding);
+#endif
+ DecodeUnicode(src,dest);
+ }
+ } else {
+#ifdef WIN32
+ setlocale(LC_ALL, ".OCP");
+#endif
+ DecodeUnicode(src,dest);
+#ifdef WIN32
+ setlocale(LC_ALL, ".ACP");
+#endif
+ }
+ return dest;
+}
+
+/* Encode string to Unicode. Len is number of input chars */
+void EncodeUnicode (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i_len = 0, o_len;
+ wchar_t wc;
+
+ for (o_len = 0; i_len < len; o_len++) {
+ i_len += EncodeWithUnicodeAlphabet(&src[i_len], &wc);
+ dest[o_len*2] = (wc >> 8) & 0xff;
+ dest[(o_len*2)+1] = wc & 0xff;
+ }
+ dest[o_len*2] = 0;
+ dest[(o_len*2)+1] = 0;
+}
+
+unsigned char EncodeWithBCDAlphabet(int value)
+{
+ div_t division;
+
+ division=div(value,10);
+ return ( ( (value-division.quot*10) & 0x0f) << 4) | (division.quot & 0xf);
+}
+
+int DecodeWithBCDAlphabet(unsigned char value)
+{
+ return 10*(value & 0x0f)+(value >> 4);
+}
+
+void DecodeBCD (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current=0,digit;
+
+ for (i = 0; i < len; i++) {
+ digit=src[i] & 0x0f;
+ if (digit<10) dest[current++]=digit + '0';
+ digit=src[i] >> 4;
+ if (digit<10) dest[current++]=digit + '0';
+ }
+ dest[current++]=0;
+}
+
+void EncodeBCD (unsigned char *dest, const unsigned char *src, int len, bool fill)
+{
+ int i,current=0;
+
+ for (i = 0; i < len; i++) {
+ if (i & 0x01) {
+ dest[current]=dest[current] | ((src[i]-'0') << 4);
+ current++;
+ } else {
+ dest[current]=src[i]-'0';
+ }
+ }
+
+ /* When fill is set: if number consist of odd number of digits,
+ we fill last bits in last byte with 0x0f
+ */
+ if (fill && (len & 0x01)) dest[current]=dest[current] | 0xf0;
+}
+
+int DecodeWithHexBinAlphabet (unsigned char mychar)
+{
+ if (mychar>='A' && mychar<='F') return mychar-'A'+10;
+ if (mychar>='a' && mychar<='f') return mychar-'a'+10;
+ if (mychar>='0' && mychar<='9') return mychar-'0';
+ return -1;
+}
+
+unsigned char EncodeWithHexBinAlphabet (int digit)
+{
+ if (digit >= 0 && digit <= 9) return '0'+(digit);
+ if (digit >=10 && digit <=15) return 'A'+(digit-10);
+ return 0;
+}
+
+void DecodeHexUnicode (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current=0;
+ bool first = false;
+
+ if (len != 0 && src[0] == '0' && src[1] == '0') {
+ first = true;
+ } else if (len != 0 && src[2] == '0' && src[3] == '0') {
+ first = false;
+ } else {
+ first = (10 * (src[0] - '0') + (src[1] - '0')) < (10 * (src[2] - '0')+ (src[3] - '0'));
+ }
+ for (i = 0; i < len/4 ; i++) {
+ if (first) {
+ dest[current++] = DecodeWithHexBinAlphabet(src[i*4+0])*16+
+ DecodeWithHexBinAlphabet(src[i*4+1]);
+ dest[current++] = DecodeWithHexBinAlphabet(src[i*4+2])*16+
+ DecodeWithHexBinAlphabet(src[i*4+3]);
+ } else {
+ dest[current++] = DecodeWithHexBinAlphabet(src[i*4+2])*16+
+ DecodeWithHexBinAlphabet(src[i*4+3]);
+ dest[current++] = DecodeWithHexBinAlphabet(src[i*4+0])*16+
+ DecodeWithHexBinAlphabet(src[i*4+1]);
+ }
+ }
+ dest[current++] = 0;
+ dest[current++] = 0;
+}
+
+void EncodeHexUnicode (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current=0;
+
+ for (i = 0; i < len; i++) {
+ dest[current++] = EncodeWithHexBinAlphabet(src[2*i] >> 0x04);
+ dest[current++] = EncodeWithHexBinAlphabet(src[2*i] & 0x0f);
+ dest[current++] = EncodeWithHexBinAlphabet(src[2*i+1] >> 0x04);
+ dest[current++] = EncodeWithHexBinAlphabet(src[2*i+1] & 0x0f);
+ }
+ dest[current++] = 0;
+}
+
+void DecodeHexBin (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current=0;
+
+ for (i = 0; i < len/2 ; i++) {
+ dest[current++] = DecodeWithHexBinAlphabet(src[i*2])*16+
+ DecodeWithHexBinAlphabet(src[i*2+1]);
+ }
+ dest[current++] = 0;
+}
+
+void EncodeHexBin (unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current=0;
+
+ for (i = 0; i < len; i++) {
+ dest[current++] = EncodeWithHexBinAlphabet(src[i] >> 0x04);
+ dest[current++] = EncodeWithHexBinAlphabet(src[i] & 0x0f);
+ }
+ dest[current++] = 0;
+}
+
+/* ETSI GSM 03.38, section 6.2.1: Default alphabet for SMS messages */
+static unsigned char GSM_DefaultAlphabetUnicode[128+1][2] =
+{
+ {0x00,0x40},{0x00,0xa3},{0x00,0x24},{0x00,0xA5},
+ {0x00,0xE8},{0x00,0xE9},{0x00,0xF9},{0x00,0xEC},/*0x08*/
+ {0x00,0xF2},{0x00,0xC7},{0x00,'\n'},{0x00,0xD8},
+ {0x00,0xF8},{0x00,'\r'},{0x00,0xC5},{0x00,0xE5},
+ {0x03,0x94},{0x00,0x5f},{0x03,0xA6},{0x03,0x93},
+ {0x03,0x9B},{0x03,0xA9},{0x03,0xA0},{0x03,0xA8},
+ {0x03,0xA3},{0x03,0x98},{0x03,0x9E},{0x00,0xb9},
+ {0x00,0xC6},{0x00,0xE6},{0x00,0xDF},{0x00,0xC9},/*0x20*/
+ {0x00,' ' },{0x00,'!' },{0x00,'\"'},{0x00,'#' },
+ {0x00,0xA4},{0x00,'%' },{0x00,'&' },{0x00,'\''},
+ {0x00,'(' },{0x00,')' },{0x00,'*' },{0x00,'+' },
+ {0x00,',' },{0x00,'-' },{0x00,'.' },{0x00,'/' },/*0x30*/
+ {0x00,'0' },{0x00,'1' },{0x00,'2' },{0x00,'3' },
+ {0x00,'4' },{0x00,'5' },{0x00,'6' },{0x00,'7' },
+ {0x00,'8' },{0x00,'9' },{0x00,':' },{0x00,';' },
+ {0x00,'<' },{0x00,'=' },{0x00,'>' },{0x00,'?' },/*0x40*/
+ {0x00,0xA1},{0x00,'A' },{0x00,'B' },{0x00,'C' },
+ {0x00,'D' },{0x00,'E' },{0x00,'F' },{0x00,'G' },
+ {0x00,'H' },{0x00,'I' },{0x00,'J' },{0x00,'K' },
+ {0x00,'L' },{0x00,'M' },{0x00,'N' },{0x00,'O' },
+ {0x00,'P' },{0x00,'Q' },{0x00,'R' },{0x00,'S' },
+ {0x00,'T' },{0x00,'U' },{0x00,'V' },{0x00,'W' },
+ {0x00,'X' },{0x00,'Y' },{0x00,'Z' },{0x00,0xC4},
+ {0x00,0xD6},{0x00,0xD1},{0x00,0xDC},{0x00,0xA7},
+ {0x00,0xBF},{0x00,'a' },{0x00,'b' },{0x00,'c' },
+ {0x00,'d' },{0x00,'e' },{0x00,'f' },{0x00,'g' },
+ {0x00,'h' },{0x00,'i' },{0x00,'j' },{0x00,'k' },
+ {0x00,'l' },{0x00,'m' },{0x00,'n' },{0x00,'o' },
+ {0x00,'p' },{0x00,'q' },{0x00,'r' },{0x00,'s' },
+ {0x00,'t' },{0x00,'u' },{0x00,'v' },{0x00,'w' },
+ {0x00,'x' },{0x00,'y' },{0x00,'z' },{0x00,0xE4},
+ {0x00,0xF6},{0x00,0xF1},{0x00,0xFC},{0x00,0xE0},
+ {0x00,0x00}
+};
+
+/* ETSI GSM 3.38
+ * Some sequences of 2 default alphabet chars (for example,
+ * 0x1b, 0x65) are visible as one single additional char (for example,
+ * 0x1b, 0x65 gives Euro char saved in Unicode as 0x20, 0xAC)
+ * This table contains:
+ * 1. two first chars means sequence of chars from GSM default alphabet
+ * 2. two second is target (encoded) char saved in Unicode
+ */
+static unsigned char GSM_DefaultAlphabetCharsExtension[][4] =
+{
+ {0x1b,0x14,0x00,0x5e}, /* ^ */
+ {0x1b,0x28,0x00,0x7b}, /* { */
+ {0x1b,0x29,0x00,0x7d}, /* } */
+ {0x1b,0x2f,0x00,0x5c}, /* \ */
+ {0x1b,0x3c,0x00,0x5b}, /* [ */
+ {0x1b,0x3d,0x00,0x7E}, /* ~ */
+ {0x1b,0x3e,0x00,0x5d}, /* ] */
+ {0x1b,0x40,0x00,0x7C}, /* | */
+ {0x1b,0x65,0x20,0xAC}, /* Euro */
+ {0x00,0x00,0x00,0x00}
+};
+
+void DecodeDefault (unsigned char *dest, const unsigned char *src, int len, bool UseExtensions, unsigned char *ExtraAlphabet)
+{
+ int i,current=0,j;
+ bool FoundSpecial = false;
+
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, src, len);
+#endif
+
+ for (i = 0; i < len; i++) {
+ FoundSpecial = false;
+ if ((i < (len-1)) && UseExtensions) {
+ j=0;
+ while (GSM_DefaultAlphabetCharsExtension[j][0]!=0x00) {
+ if (GSM_DefaultAlphabetCharsExtension[j][0]==src[i] &&
+ GSM_DefaultAlphabetCharsExtension[j][1]==src[i+1]) {
+ FoundSpecial = true;
+ dest[current++] = GSM_DefaultAlphabetCharsExtension[j][2];
+ dest[current++] = GSM_DefaultAlphabetCharsExtension[j][3];
+ i++;
+ break;
+ }
+ j++;
+ }
+ }
+ if (ExtraAlphabet!=NULL && !FoundSpecial) {
+ j = 0;
+ while (ExtraAlphabet[j] != 0x00 || ExtraAlphabet[j+1] != 0x00 || ExtraAlphabet[j+2] != 0x00) {
+ if (ExtraAlphabet[j] == src[i]) {
+ dest[current++] = ExtraAlphabet[j+1];
+ dest[current++] = ExtraAlphabet[j+2];
+ FoundSpecial = true;
+ break;
+ }
+ j=j+3;
+ }
+ }
+ if (!FoundSpecial) {
+ dest[current++] = GSM_DefaultAlphabetUnicode[src[i]][0];
+ dest[current++] = GSM_DefaultAlphabetUnicode[src[i]][1];
+ }
+ }
+ dest[current++]=0;
+ dest[current++]=0;
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, dest, UnicodeLength(dest)*2);
+#endif
+}
+
+/* There are many national chars with "adds". In phone they're normally
+ * changed to "plain" Latin chars. We have such functionality too.
+ * This table is automatically created from convert.txt file (see
+ * /docs/developers) using --makeconverttable. It contains such chars
+ * to replace in order:
+ * 1. original char (Unicode) 2. destination char (Unicode)
+ */
+static unsigned char ConvertTable[] =
+"\x00\xc0\x00\x41\x00\xe0\x00\x61\x00\xc1\x00\x41\x00\xe1\x00\x61\x00\xc2\x00\x41\x00\xe2\x00\x61\x00\xc3\x00\x41\x00\xe3\x00\x61\x1e\xa0\x00\x41\x1e\xa1\x00\x61\x1e\xa2\x00\x41\x1e\xa3\x00\x61\x1e\xa4\x00\x41\x1e\xa5\x00\x61\x1e\xa6\x00\x41\x1e\xa7\x00\x61\x1e\xa8\x00\x41\x1e\xa9\x00\x61\x1e\xaa\x00\x41\x1e\xab\x00\x61\x1e\xac\x00\x41\x1e\xad\x00\x61\x1e\xae\x00\x41\x1e\xaf\x00\x61\x1e\xb0\x00\x41\x1e\xb1\x00\x61\x1e\xb2\x00\x41\x1e\xb3\x00\x61\x1e\xb4\x00\x41\x1e\xb5\x00\x61\x1e\xb6\x00\x41\x1e\xb7\x00\x61\x01\xcd\x00\x41\x01\xce\x00\x61\x01\x00\x00\x41\x01\x01\x00\x61\x01\x02\x00\x41\x01\x03\x00\x61\x01\x04\x00\x41\x01\x05\x00\x61\x01\xfb\x00\x61\x01\x06\x00\x43\x01\x07\x00\x63\x01\x08\x00\x43\x01\x09\x00\x63\x01\x0a\x00\x43\x01\x0b\x00\x63\x01\x0c\x00\x43\x01\x0d\x00\x63\x00\xe7"\
+"\x00\x63\x01\x0e\x00\x44\x01\x0f\x00\x64\x01\x10\x00\x44\x01\x11\x00\x64\x00\xc8\x00\x45\x00\xca\x00\x45\x00\xea\x00\x65\x00\xcb\x00\x45\x00\xeb\x00\x65\x1e\xb8\x00\x45\x1e\xb9\x00\x65\x1e\xba\x00\x45\x1e\xbb\x00\x65\x1e\xbc\x00\x45\x1e\xbd\x00\x65\x1e\xbe\x00\x45\x1e\xbf\x00\x65\x1e\xc0\x00\x45\x1e\xc1\x00\x65\x1e\xc2\x00\x45\x1e\xc3\x00\x65\x1e\xc4\x00\x45\x1e\xc5\x00\x65\x1e\xc6\x00\x45\x1e\xc7\x00\x65\x01\x12\x00\x45\x01\x13\x00\x65\x01\x14\x00\x45\x01\x15\x00\x65\x01\x16\x00\x45\x01\x17\x00\x65\x01\x18\x00\x45\x01\x19\x00\x65\x01\x1a\x00\x45\x01\x1b\x00\x65\x01\x1c\x00\x47\x01\x1d\x00\x67\x01\x1e\x00\x47\x01\x1f\x00\x67\x01\x20\x00\x47\x01\x21\x00\x67\x01\x22\x00\x47\x01\x23\x00\x67\x01\x24\x00\x48\x01\x25\x00\x68\x01\x26\x00\x48\x01\x27\x00\x68\x00\xcc\x00\x49\x00\xcd\x00\x49\x00\xed"\
+"\x00\x69\x00\xce\x00\x49\x00\xee\x00\x69\x00\xcf\x00\x49\x00\xef\x00\x69\x01\x28\x00\x49\x01\x29\x00\x69\x01\x2a\x00\x49\x01\x2b\x00\x69\x01\x2c\x00\x49\x01\x2d\x00\x69\x01\x2e\x00\x49\x01\x2f\x00\x69\x01\x30\x00\x49\x01\x31\x00\x69\x01\xcf\x00\x49\x01\xd0\x00\x69\x1e\xc8\x00\x49\x1e\xc9\x00\x69\x1e\xca\x00\x49\x1e\xcb\x00\x69\x01\x34\x00\x4a\x01\x35\x00\x6a\x01\x36\x00\x4b\x01\x37\x00\x6b\x01\x39\x00\x4c\x01\x3a\x00\x6c\x01\x3b\x00\x4c\x01\x3c\x00\x6c\x01\x3d\x00\x4c\x01\x3e\x00\x6c\x01\x3f\x00\x4c\x01\x40\x00\x6c\x01\x41\x00\x4c\x01\x42\x00\x6c\x01\x43\x00\x4e\x01\x44\x00\x6e\x01\x45\x00\x4e\x01\x46\x00\x6e\x01\x47\x00\x4e\x01\x48\x00\x6e\x01\x49\x00\x6e\x00\xd2\x00\x4f\x00\xd3\x00\x4f\x00\xf3\x00\x6f\x00\xd4\x00\x4f\x00\xf4\x00\x6f\x00\xd5\x00\x4f\x00\xf5\x00\x6f\x01\x4c\x00\x4f\x01\x4d"\
+"\x00\x6f\x01\x4e\x00\x4f\x01\x4f\x00\x6f\x01\x50\x00\x4f\x01\x51\x00\x6f\x01\xa0\x00\x4f\x01\xa1\x00\x6f\x01\xd1\x00\x4f\x01\xd2\x00\x6f\x1e\xcc\x00\x4f\x1e\xcd\x00\x6f\x1e\xce\x00\x4f\x1e\xcf\x00\x6f\x1e\xd0\x00\x4f\x1e\xd1\x00\x6f\x1e\xd2\x00\x4f\x1e\xd3\x00\x6f\x1e\xd4\x00\x4f\x1e\xd5\x00\x6f\x1e\xd6\x00\x4f\x1e\xd7\x00\x6f\x1e\xd8\x00\x4f\x1e\xd9\x00\x6f\x1e\xda\x00\x4f\x1e\xdb\x00\x6f\x1e\xdc\x00\x4f\x1e\xdd\x00\x6f\x1e\xde\x00\x4f\x1e\xdf\x00\x6f\x1e\xe0\x00\x4f\x1e\xe1\x00\x6f\x1e\xe2\x00\x4f\x1e\xe3\x00\x6f\x01\x54\x00\x52\x01\x55\x00\x72\x01\x56\x00\x52\x01\x57\x00\x72\x01\x58\x00\x52\x01\x59\x00\x72\x01\x5a\x00\x53\x01\x5b\x00\x73\x01\x5c\x00\x53\x01\x5d\x00\x73\x01\x5e\x00\x53\x01\x5f\x00\x73\x01\x60\x00\x53\x01\x61\x00\x73\x01\x62\x00\x54\x01\x63\x00\x74\x01\x64\x00\x54\x01\x65"\
+"\x00\x74\x01\x66\x00\x54\x01\x67\x00\x74\x00\xd9\x00\x55\x00\xda\x00\x55\x00\xfa\x00\x75\x00\xdb\x00\x55\x00\xfb\x00\x75\x01\x68\x00\x55\x01\x69\x00\x75\x01\x6a\x00\x55\x01\x6b\x00\x75\x01\x6c\x00\x55\x01\x6d\x00\x75\x01\x6e\x00\x55\x01\x6f\x00\x75\x01\x70\x00\x55\x01\x71\x00\x75\x01\x72\x00\x55\x01\x73\x00\x75\x01\xaf\x00\x55\x01\xb0\x00\x75\x01\xd3\x00\x55\x01\xd4\x00\x75\x01\xd5\x00\x55\x01\xd6\x00\x75\x01\xd7\x00\x55\x01\xd8\x00\x75\x01\xd9\x00\x55\x01\xda\x00\x75\x01\xdb\x00\x55\x01\xdc\x00\x75\x1e\xe4\x00\x55\x1e\xe5\x00\x75\x1e\xe6\x00\x55\x1e\xe7\x00\x75\x1e\xe8\x00\x55\x1e\xe9\x00\x75\x1e\xea\x00\x55\x1e\xeb\x00\x75\x1e\xec\x00\x55\x1e\xed\x00\x75\x1e\xee\x00\x55\x1e\xef\x00\x75\x1e\xf0\x00\x55\x1e\xf1\x00\x75\x01\x74\x00\x57\x01\x75\x00\x77\x1e\x80\x00\x57\x1e\x81\x00\x77\x1e\x82"\
+"\x00\x57\x1e\x83\x00\x77\x1e\x84\x00\x57\x1e\x85\x00\x77\x00\xdd\x00\x59\x00\xfd\x00\x79\x00\xff\x00\x79\x01\x76\x00\x59\x01\x77\x00\x79\x01\x78\x00\x59\x1e\xf2\x00\x59\x1e\xf3\x00\x75\x1e\xf4\x00\x59\x1e\xf5\x00\x79\x1e\xf6\x00\x59\x1e\xf7\x00\x79\x1e\xf8\x00\x59\x1e\xf9\x00\x79\x01\x79\x00\x5a\x01\x7a\x00\x7a\x01\x7b\x00\x5a\x01\x7c\x00\x7a\x01\x7d\x00\x5a\x01\x7e\x00\x7a\x01\xfc\x00\xc6\x01\xfd\x00\xe6\x01\xfe\x00\xd8\x01\xff\x00\xf8\x00\x00";
+
+void EncodeDefault(unsigned char *dest, const unsigned char *src, int *len, bool UseExtensions, unsigned char *ExtraAlphabet)
+{
+ int i,current=0,j,z;
+ char ret;
+ bool FoundSpecial,FoundNormal;
+
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, src, (*len)*2);
+#endif
+
+ for (i = 0; i < *len; i++) {
+ FoundSpecial = false;
+ j = 0;
+ while (GSM_DefaultAlphabetCharsExtension[j][0]!=0x00 && UseExtensions) {
+ if (src[i*2] == GSM_DefaultAlphabetCharsExtension[j][2] &&
+ src[i*2+1] == GSM_DefaultAlphabetCharsExtension[j][3]) {
+ dest[current++] = GSM_DefaultAlphabetCharsExtension[j][0];
+ dest[current++] = GSM_DefaultAlphabetCharsExtension[j][1];
+ FoundSpecial = true;
+ break;
+ }
+ j++;
+ }
+ if (!FoundSpecial) {
+ ret = '?';
+ FoundNormal = false;
+ j = 0;
+ while (GSM_DefaultAlphabetUnicode[j][1]!=0x00) {
+ if (src[i*2] == GSM_DefaultAlphabetUnicode[j][0] &&
+ src[i*2+1] == GSM_DefaultAlphabetUnicode[j][1]) {
+ ret = j;
+ FoundNormal = true;
+ break;
+ }
+ j++;
+ }
+ if (ExtraAlphabet!=NULL && !FoundNormal) {
+ j = 0;
+ while (ExtraAlphabet[j] != 0x00 || ExtraAlphabet[j+1] != 0x00 || ExtraAlphabet[j+2] != 0x00) {
+ if (ExtraAlphabet[j+1] == src[i*2] &&
+ ExtraAlphabet[j+2] == src[i*2 + 1]) {
+ ret = ExtraAlphabet[j];
+ FoundSpecial = true;
+ break;
+ }
+ j=j+3;
+ }
+ }
+ if (!FoundNormal && !FoundSpecial) {
+ j = 0;
+ FoundNormal = false;
+ while (ConvertTable[j*4] != 0x00 ||
+ ConvertTable[j*4+1] != 0x00) {
+ if (src[i*2] == ConvertTable[j*4] &&
+ src[i*2+1] == ConvertTable[j*4+1]) {
+ z = 0;
+ while (GSM_DefaultAlphabetUnicode[z][1]!=0x00) {
+ if (ConvertTable[j*4+2] == GSM_DefaultAlphabetUnicode[z][0] &&
+ ConvertTable[j*4+3] == GSM_DefaultAlphabetUnicode[z][1]) {
+ ret = z;
+ FoundNormal = true;
+ break;
+ }
+ z++;
+ }
+ if (FoundNormal) break;
+ }
+ j++;
+ }
+ }
+ dest[current++]=ret;
+ }
+ }
+ dest[current]=0;
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, dest, current);
+#endif
+
+ *len = current;
+}
+
+/* You don't have to use ConvertTable here - 1 char is replaced there by 1 char */
+void FindDefaultAlphabetLen(const unsigned char *src, int *srclen, int *smslen, int maxlen)
+{
+ int current=0,j,i;
+ bool FoundSpecial;
+
+ i = 0;
+ while (src[i*2] != 0x00 || src[i*2+1] != 0x00) {
+ FoundSpecial = false;
+ j = 0;
+ while (GSM_DefaultAlphabetCharsExtension[j][0]!=0x00) {
+ if (src[i*2] == GSM_DefaultAlphabetCharsExtension[j][2] &&
+ src[i*2+1] == GSM_DefaultAlphabetCharsExtension[j][3]) {
+ FoundSpecial = true;
+ if (current+2 > maxlen) {
+ *srclen = i;
+ *smslen = current;
+ return;
+ }
+ current+=2;
+ break;
+ }
+ j++;
+ }
+ if (!FoundSpecial) {
+ if (current+1 > maxlen) {
+ *srclen = i;
+ *smslen = current;
+ return;
+ }
+ current++;
+ }
+ i++;
+ }
+ *srclen = i;
+ *smslen = current;
+}
+
+#ifndef ENABLE_LGPL
+# define ByteMask ((1 << Bits) - 1)
+#endif
+
+int GSM_UnpackEightBitsToSeven(int offset, int in_length, int out_length,
+ unsigned char *input, unsigned char *output)
+{
+#ifndef ENABLE_LGPL
+ /* (c) by Pavel Janik and Pawel Kot */
+
+ unsigned char *OUTPUT = output; /* Current pointer to the output buffer */
+ unsigned char *INPUT = input; /* Current pointer to the input buffer */
+ unsigned char Rest = 0x00;
+ int Bits;
+
+ Bits = offset ? offset : 7;
+
+ while ((INPUT - input) < in_length) {
+
+ *OUTPUT = ((*INPUT & ByteMask) << (7 - Bits)) | Rest;
+ Rest = *INPUT >> Bits;
+
+ /* If we don't start from 0th bit, we shouldn't go to the
+ next char. Under *OUTPUT we have now 0 and under Rest -
+ _first_ part of the char. */
+ if ((INPUT != input) || (Bits == 7)) OUTPUT++;
+ INPUT++;
+
+ if ((OUTPUT - output) >= out_length) break;
+
+ /* After reading 7 octets we have read 7 full characters but
+ we have 7 bits as well. This is the next character */
+ if (Bits == 1) {
+ *OUTPUT = Rest;
+ OUTPUT++;
+ Bits = 7;
+ Rest = 0x00;
+ } else {
+ Bits--;
+ }
+ }
+
+ return OUTPUT - output;
+#else
+ return 0;
+#endif
+}
+
+int GSM_PackSevenBitsToEight(int offset, unsigned char *input, unsigned char *output, int length)
+{
+#ifndef ENABLE_LGPL
+ /* (c) by Pavel Janik and Pawel Kot */
+
+ unsigned char *OUTPUT = output; /* Current pointer to the output buffer */
+ unsigned char *INPUT = input; /* Current pointer to the input buffer */
+ int Bits; /* Number of bits directly copied to
+ * the output buffer */
+ Bits = (7 + offset) % 8;
+
+ /* If we don't begin with 0th bit, we will write only a part of the
+ first octet */
+ if (offset) {
+ *OUTPUT = 0x00;
+ OUTPUT++;
+ }
+
+ while ((INPUT - input) < length) {
+ unsigned char Byte = *INPUT;
+
+ *OUTPUT = Byte >> (7 - Bits);
+ /* If we don't write at 0th bit of the octet, we should write
+ a second part of the previous octet */
+ if (Bits != 7)
+ *(OUTPUT-1) |= (Byte & ((1 << (7-Bits)) - 1)) << (Bits+1);
+
+ Bits--;
+
+ if (Bits == -1) Bits = 7; else OUTPUT++;
+
+ INPUT++;
+ }
+ return (OUTPUT - output);
+#else
+ return 0;
+#endif
+}
+
+void GSM_UnpackSemiOctetNumber(unsigned char *retval, unsigned char *Number, bool semioctet)
+{
+ unsigned char Buffer[50] = "";
+ int length = Number[0];
+
+ if (semioctet) {
+ /* Convert number of semioctets to number of chars */
+ if (length % 2) length++;
+ length=length / 2 + 1;
+ }
+
+ /*without leading byte with format of number*/
+ length--;
+
+ switch (Number[1]) {
+ case NUMBER_ALPHANUMERIC:
+ if (length > 6) length++;
+ dbgprintf("Alphanumeric number, length %i\n",length);
+ GSM_UnpackEightBitsToSeven(0, length, length, Number+2, Buffer);
+ Buffer[length]=0;
+ break;
+ case NUMBER_INTERNATIONAL:
+ dbgprintf("International number\n");
+ Buffer[0]='+';
+ DecodeBCD(Buffer+1,Number+2, length);
+ break;
+ default:
+ dbgprintf("Default number %02x\n",Number[1]);
+ DecodeBCD (Buffer, Number+2, length);
+ break;
+ }
+
+ EncodeUnicode(retval,Buffer,strlen(Buffer));
+}
+
+/**
+ * Packing some phone numbers (SMSC, SMS destination and others)
+ *
+ * See GSM 03.40 9.1.1:
+ * 1 byte - length of number given in semioctets or bytes (when given in
+ * bytes, includes one byte for byte with number format).
+ * Returned by function (set semioctet to true, if want result
+ * in semioctets).
+ * 1 byte - format of number (see GSM_NumberType in coding.h). Returned
+ * in unsigned char *Output.
+ * n bytes - 2n or 2n-1 semioctets with number. Returned in unsigned char
+ * *Output.
+ *
+ * 1 semioctet = 4 bits = half of byte
+ */
+int GSM_PackSemiOctetNumber(unsigned char *Number, unsigned char *Output, bool semioctet)
+{
+ unsigned char format, buffer[50];
+ int length, i;
+
+ length=UnicodeLength(Number);
+ memcpy(buffer,DecodeUnicodeString(Number),length+1);
+
+ /* Checking for format number */
+ format = NUMBER_UNKNOWN;
+ for (i=0;i<length;i++) {
+ /* first byte is '+'. Number can be international */
+ if (i==0 && buffer[i]=='+') {
+ format=NUMBER_INTERNATIONAL;
+ } else {
+ /*char is not number. It must be alphanumeric*/
+ if (!isdigit(buffer[i])) format=NUMBER_ALPHANUMERIC;
+ }
+ }
+
+ /**
+ * First byte is used for saving type of number. See GSM 03.40
+ * section 9.1.2.5
+ */
+ Output[0]=format;
+
+ /* After number type we will have number. GSM 03.40 section 9.1.2 */
+ switch (format) {
+ case NUMBER_ALPHANUMERIC:
+ length=GSM_PackSevenBitsToEight(0, buffer, Output+1, strlen(buffer))*2;
+ if (strlen(buffer)==7) length--;
+ break;
+ case NUMBER_INTERNATIONAL:
+ length--;
+ EncodeBCD (Output+1, buffer+1, length, true);
+ break;
+ default:
+ EncodeBCD (Output+1, buffer, length, true);
+ break;
+ }
+
+ if (semioctet) return length;
+
+ /* Convert number of semioctets to number of chars */
+ if (length % 2) length++;
+ return length / 2 + 1;
+}
+
+void CopyUnicodeString(unsigned char *Dest, unsigned char *Source)
+{
+ int j = 0;
+
+ while (Source[j]!=0x00 || Source[j+1]!=0x00) {
+ Dest[j] = Source[j];
+ Dest[j+1] = Source[j+1];
+ j=j+2;
+ }
+ Dest[j] = 0;
+ Dest[j+1] = 0;
+}
+
+/* Changes minor/major order in Unicode string */
+void ReverseUnicodeString(unsigned char *String)
+{
+ int j = 0;
+ unsigned char byte1, byte2;
+
+ while (String[j]!=0x00 || String[j+1]!=0x00) {
+ byte1 = String[j];
+ byte2 = String[j+1];
+ String[j+1] = byte1;
+ String[j] = byte2;
+ j=j+2;
+ }
+ String[j] = 0;
+ String[j+1] = 0;
+}
+
+/* All input is in Unicode. First char can show Unicode minor/major order.
+ Output is Unicode string in Gammu minor/major order */
+void ReadUnicodeFile(unsigned char *Dest, unsigned char *Source)
+{
+ int j = 0, current = 0;
+
+ if (Source[0] == 0xFF && Source[1] == 0xFE) j = 2;
+ if (Source[0] == 0xFE && Source[1] == 0xFF) j = 2;
+
+ while (Source[j]!=0x00 || Source[j+1]!=0x00) {
+ if (Source[0] == 0xFF) {
+ Dest[current++] = Source[j+1];
+ Dest[current++] = Source[j];
+ } else {
+ Dest[current++] = Source[j];
+ Dest[current++] = Source[j+1];
+ }
+ j=j+2;
+ }
+ Dest[current++] = 0;
+ Dest[current++] = 0;
+}
+
+int GetBit(unsigned char *Buffer, int BitNum)
+{
+ return Buffer[BitNum/8] & 1<<(7-(BitNum%8));
+}
+
+int SetBit(unsigned char *Buffer, int BitNum)
+{
+ return Buffer[BitNum/8] |= 1<<(7-(BitNum%8));
+}
+
+int ClearBit(unsigned char *Buffer, int BitNum)
+{
+ return Buffer[BitNum/8] &= 255 - (1 << (7-(BitNum%8)));
+}
+
+void BufferAlign(unsigned char *Destination, int *CurrentBit)
+{
+ int i=0;
+
+ while(((*CurrentBit) + i) % 8 != 0) {
+ ClearBit(Destination, (*CurrentBit)+i);
+ i++;
+ }
+
+ (*CurrentBit) = (*CurrentBit) + i;
+}
+
+void BufferAlignNumber(int *CurrentBit)
+{
+ int i=0;
+
+ while(((*CurrentBit) + i) % 8 != 0) {
+ i++;
+ }
+
+ (*CurrentBit) = (*CurrentBit) + i;
+}
+
+void AddBuffer(unsigned char *Destination,
+ int *CurrentBit,
+ unsigned char *Source,
+ int BitsToProcess)
+{
+ int i=0;
+
+ while (i!=BitsToProcess) {
+ if (GetBit(Source, i)) {
+ SetBit(Destination, (*CurrentBit)+i);
+ } else {
+ ClearBit(Destination, (*CurrentBit)+i);
+ }
+ i++;
+ }
+ (*CurrentBit) = (*CurrentBit) + BitsToProcess;
+}
+
+void AddBufferByte(unsigned char *Destination,
+ int *CurrentBit,
+ unsigned char Source,
+ int BitsToProcess)
+{
+ unsigned char Byte;
+
+ Byte = Source;
+
+ AddBuffer(Destination, CurrentBit, &Byte, BitsToProcess);
+}
+
+void GetBuffer(unsigned char *Source,
+ int *CurrentBit,
+ unsigned char *Destination,
+ int BitsToProcess)
+{
+ int i=0;
+
+ while (i!=BitsToProcess) {
+ if (GetBit(Source, (*CurrentBit)+i)) {
+ SetBit(Destination, i);
+ } else {
+ ClearBit(Destination, i);
+ }
+ i++;
+ }
+ (*CurrentBit) = (*CurrentBit) + BitsToProcess;
+}
+
+void GetBufferInt(unsigned char *Source,
+ int *CurrentBit,
+ int *integer,
+ int BitsToProcess)
+{
+ int l=0,z=128,i=0;
+
+ while (i!=BitsToProcess) {
+ if (GetBit(Source, (*CurrentBit)+i)) l=l+z;
+ z=z/2;
+ i++;
+ }
+ *integer=l;
+ (*CurrentBit) = (*CurrentBit) + i;
+}
+
+void GetBufferI(unsigned char *Source,
+ int *CurrentBit,
+ int *result,
+ int BitsToProcess)
+{
+ int l=0,z,i=0;
+
+ z = 1<<(BitsToProcess-1);
+
+ while (i!=BitsToProcess) {
+ if (GetBit(Source, (*CurrentBit)+i)) l=l+z;
+ z=z>>1;
+ i++;
+ }
+ *result=l;
+ (*CurrentBit) = (*CurrentBit) + i;
+}
+
+/* Unicode char 0x00 0x01 makes blinking in some Nokia phones.
+ * We replace single ~ chars into it. When user give double ~, it's replaced
+ * to single ~
+ */
+void EncodeUnicodeSpecialNOKIAChars(unsigned char *dest, const unsigned char *src, int len)
+{
+ int i,current = 0;
+ bool special=false;
+
+ for (i = 0; i < len; i++) {
+ if (special) {
+ if (src[i*2] == 0x00 && src[i*2+1] == '~') {
+ dest[current++] = 0x00;
+ dest[current++] = '~';
+ } else {
+ dest[current++] = 0x00;
+ dest[current++] = 0x01;
+ dest[current++] = src[i*2];
+ dest[current++] = src[i*2+1];
+ }
+ special = false;
+ } else {
+ if (src[i*2] == 0x00 && src[i*2+1] == '~') {
+ special = true;
+ } else {
+ dest[current++] = src[i*2];
+ dest[current++] = src[i*2+1];
+ }
+ }
+ }
+ if (special) {
+ dest[current++] = 0x00;
+ dest[current++] = 0x01;
+ }
+ dest[current++] = 0x00;
+ dest[current++] = 0x00;
+}
+
+void DecodeUnicodeSpecialNOKIAChars(unsigned char *dest, const unsigned char *src, int len)
+{
+ int i=0,current=0;
+
+ for (i=0;i<len;i++) {
+ switch (src[2*i]) {
+ case 0x00:
+ switch (src[2*i+1]) {
+ case 0x01:
+ dest[current++] = 0x00;
+ dest[current++] = '~';
+ break;
+ case '~':
+ dest[current++] = 0x00;
+ dest[current++] = '~';
+ dest[current++] = 0x00;
+ dest[current++] = '~';
+ break;
+ default:
+ dest[current++] = src[i*2];
+ dest[current++] = src[i*2+1];
+ }
+ break;
+ default:
+ dest[current++] = src[i*2];
+ dest[current++] = src[i*2+1];
+ }
+ }
+ dest[current++] = 0x00;
+ dest[current++] = 0x00;
+}
+
+bool mystrncasecmp(unsigned const char *a, unsigned const char *b, int num)
+{
+ int i;
+
+ if (a == NULL || b == NULL) return false;
+
+ num--;
+
+ for (i = 0; i != num; i++) {
+ if (a[i] == 0x00 && b[i] == 0x00) return true;
+ if (a[i] == 0x00 || b[i] == 0x00) return false;
+ if (tolower(a[i]) != tolower(b[i])) return false;
+ }
+ return true;
+}
+
+/* Compares two Unicode strings without regarding to case.
+ * Return true, when they're equal
+ */
+bool mywstrncasecmp(unsigned const char *a, unsigned const char *b, int num)
+{
+ int i;
+ wchar_t wc,wc2;
+
+ if (a == NULL || b == NULL) return false;
+
+ num--;
+
+ for (i = 0; i != num; i++) {
+ if ((a[i*2] == 0x00 && a[i*2+1] == 0x00) && (b[i*2] == 0x00 && b[i*2+1] == 0x00)) return true;
+ if ((a[i*2] == 0x00 && a[i*2+1] == 0x00) || (b[i*2] == 0x00 && b[i*2+1] == 0x00)) return false;
+ wc = a[i*2+1] | (a[i*2] << 8);
+ wc2 = b[i*2+1] | (b[i*2] << 8);
+ if (mytowlower(wc) != mytowlower(wc2)) return false;
+ }
+ return true;
+}
+
+/* wcscmp in Mandrake 9.0 is wrong */
+bool mywstrncmp(unsigned const char *a, unsigned const char *b, int num)
+{
+ int i=0;
+
+ while (1) {
+ if (a[i*2] != b[i*2] || a[i*2+1] != b[i*2+1]) return false;
+ if (a[i*2] == 0x00 && a[i*2+1] == 0x00) return true;
+ i++;
+ if (num == i) return true;
+ }
+}
+
+/* FreeBSD boxes 4.7-STABLE does't have it, although it's ANSI standard */
+bool myiswspace(unsigned const char *src)
+{
+#ifndef HAVE_ISWSPACE
+ int o;
+ unsigned char dest[10];
+#endif
+ wchar_t wc;
+
+ wc = src[1] | (src[0] << 8);
+
+#ifndef HAVE_ISWSPACE
+ o = DecodeWithUnicodeAlphabet(wc, dest);
+ if (o == 1) {
+ if (isspace(((int)dest[0]))!=0) return true;
+ return false;
+ }
+ return false;
+#else
+ return iswspace(wc);
+#endif
+}
+
+/* FreeBSD boxes 4.7-STABLE does't have it, although it's ANSI standard */
+int mytowlower(wchar_t c)
+{
+#ifndef HAVE_TOWLOWER
+ unsigned char dest[10];
+
+ DecodeWithUnicodeAlphabet(c, dest);
+ return tolower(dest[0]);
+#else
+ return towlower(c);
+#endif
+}
+
+/*
+ * Following code is based on wcsstr from the GNU C Library, original
+ * comment follows:
+ */
+/*
+ * The original strstr() file contains the following comment:
+ *
+ * My personal strstr() implementation that beats most other algorithms.
+ * Until someone tells me otherwise, I assume that this is the
+ * fastest implementation of strstr() in C.
+ * I deliberately chose not to comment it. You should have at least
+ * as much fun trying to understand it, as I had to write it :-).
+ *
+ * Stephen R. van den Berg, berg@pool.informatik.rwth-aachen.de */
+
+unsigned char *mystrstr (const unsigned char *haystack, const unsigned char *needle)
+{
+/* One crazy define to convert unicode used in Gammu to standard wchar_t */
+#define tolowerwchar(x) (mytowlower((wchar_t)( (((&(x))[0] & 0xff) << 8) | (((&(x))[1] & 0xff)) )))
+ register wchar_t b, c;
+
+ if ((b = tolowerwchar(*needle)) != L'\0') {
+ haystack -= 2; /* possible ANSI violation */
+ do {
+ haystack += 2;
+ if ((c = tolowerwchar(*haystack)) == L'\0')
+ goto ret0;
+ } while (c != b);
+
+ needle += 2;
+ if ((c = tolowerwchar(*needle)) == L'\0')
+ goto foundneedle;
+ needle += 2;
+ goto jin;
+
+ for (;;) {
+ register wchar_t a;
+ register const unsigned char *rhaystack, *rneedle;
+
+ do {
+ haystack += 2;
+ if ((a = tolowerwchar(*haystack)) == L'\0')
+ goto ret0;
+ if (a == b)
+ break;
+ haystack += 2;
+ if ((a = tolowerwchar(*haystack)) == L'\0')
+ goto ret0;
+shloop: ;
+ } while (a != b);
+
+jin: haystack += 2;
+ if ((a = tolowerwchar(*haystack)) == L'\0')
+ goto ret0;
+
+ if (a != c)
+ goto shloop;
+
+ rhaystack = haystack + 2;
+ haystack -= 2;
+ rneedle = needle;
+ if (tolowerwchar(*rhaystack) == (a = tolowerwchar(*rneedle)))
+ do {
+ if (a == L'\0')
+ goto foundneedle;
+ rhaystack += 2;
+ needle += 2;
+ if (tolowerwchar(*rhaystack) != (a = tolowerwchar(*needle)))
+ break ;
+ if (a == L'\0')
+ goto foundneedle;
+ rhaystack += 2;
+ needle += 2;
+ } while (tolowerwchar(*rhaystack) == (a = tolowerwchar(*needle)));
+
+ needle = rneedle; /* took the register-poor approach */
+
+ if (a == L'\0')
+ break;
+ }
+ }
+foundneedle:
+ return (unsigned char *)haystack;
+ret0:
+ return NULL;
+#undef tolowerwchar
+}
+
+void MyGetLine(unsigned char *Buffer, int *Pos, unsigned char *OutBuffer, int MaxLen)
+{
+ OutBuffer[0] = 0;
+ if (Buffer == NULL) return;
+ while (1) {
+ if ((*Pos) >= MaxLen) return;
+ switch (Buffer[*Pos]) {
+ case 0x00:
+ return;
+ case 0x0A:
+ if (strlen(OutBuffer) != 0) return;
+ break;
+ case 0x0D:
+ if (strlen(OutBuffer) != 0) return;
+ break;
+ default :
+ OutBuffer[strlen(OutBuffer) + 1] = 0;
+ OutBuffer[strlen(OutBuffer)] = Buffer[*Pos];
+ }
+ (*Pos)++;
+ }
+}
+
+void StringToDouble(char *text, double *d)
+{
+ bool before=true;
+ double multiply = 1;
+ unsigned int i;
+
+ *d = 0;
+ for (i=0;i<strlen(text);i++) {
+ if (isdigit(text[i])) {
+ if (before) {
+ (*d)=(*d)*10+(text[i]-'0');
+ } else {
+ multiply=multiply*0.1;
+ (*d)=(*d)+(text[i]-'0')*multiply;
+ }
+ }
+ if (text[i]=='.' || text[i]==',') before=false;
+ }
+}
+
+/* When char can be converted, convert it from Unicode to UTF8 */
+bool EncodeWithUTF8Alphabet(unsigned char mychar1, unsigned char mychar2, unsigned char *ret1, unsigned char *ret2)
+{
+ unsigned char mychar3,mychar4;
+ int j=0;
+
+ if (mychar1>0x00 || mychar2>128) {
+ mychar3=0x00;
+ mychar4=128;
+ while (true) {
+ if (mychar3==mychar1) {
+ if (mychar4+64>=mychar2) {
+ *ret1=j+0xc2;
+ *ret2=0x80+(mychar2-mychar4);
+ return true;
+ }
+ }
+ if (mychar4==192) {
+ mychar3++;
+ mychar4=0;
+ } else {
+ mychar4=mychar4+64;
+ }
+ j++;
+ }
+ }
+ return false;
+}
+
+/* Make UTF8 string from Unicode input string */
+bool EncodeUTF8QuotedPrintable(unsigned char *dest, const unsigned char *src)
+{
+ int i,j=0;
+ unsigned char mychar1, mychar2;
+ bool retval = false;
+
+ for (i = 0; i < (int)(UnicodeLength(src)); i++) {
+ if (EncodeWithUTF8Alphabet(src[i*2],src[i*2+1],&mychar1,&mychar2)) {
+ sprintf(dest+j, "=%02X=%02X",mychar1,mychar2);
+ j = j+6;
+ retval = true;
+ } else {
+ j += DecodeWithUnicodeAlphabet(((wchar_t)(src[i*2]*256+src[i*2+1])), dest + j);
+ }
+ }
+ dest[j++]=0;
+ return retval;
+}
+
+bool EncodeUTF8(unsigned char *dest, const unsigned char *src)
+{
+ int i,j=0;
+ unsigned char mychar1, mychar2;
+ bool retval = false;
+
+ for (i = 0; i < (int)(UnicodeLength(src)); i++) {
+ if (EncodeWithUTF8Alphabet(src[i*2],src[i*2+1],&mychar1,&mychar2)) {
+ sprintf(dest+j, "%c%c",mychar1,mychar2);
+ j = j+2;
+ retval = true;
+ } else {
+ j += DecodeWithUnicodeAlphabet(((wchar_t)(src[i*2]*256+src[i*2+1])), dest + j);
+ }
+ }
+ dest[j++]=0;
+ return retval;
+}
+
+/* Decode UTF8 char to Unicode char */
+wchar_t DecodeWithUTF8Alphabet(unsigned char mychar3, unsigned char mychar4)
+{
+ unsigned char mychar1, mychar2;
+ int j;
+
+ mychar1=0x00;
+ mychar2=128;
+ for(j=0;j<mychar3-0xc2;j++) {
+ if (mychar2==192) {
+ mychar1++;
+ mychar2 = 0;
+ } else {
+ mychar2 = mychar2+64;
+ }
+ }
+ mychar2 = mychar2+(mychar4-0x80);
+ return mychar2 | (mychar1 << 8);
+}
+
+/* Make Unicode string from UTF8 string */
+void DecodeUTF8QuotedPrintable(unsigned char *dest, const unsigned char *src, int len)
+{
+ int i=0,j=0;
+ unsigned char mychar1, mychar2;
+ wchar_t ret;
+
+ while (i<=len) {
+ if (len-6>=i) {
+ /* Need to have correct chars */
+ if (src[i] =='=' && DecodeWithHexBinAlphabet(src[i+1])!=-1
+ && DecodeWithHexBinAlphabet(src[i+2])!=-1 &&
+ src[i+3]=='=' && DecodeWithHexBinAlphabet(src[i+4])!=-1 &&
+ DecodeWithHexBinAlphabet(src[i+5])!=-1) {
+ mychar1 = 16*DecodeWithHexBinAlphabet(src[i+1])+DecodeWithHexBinAlphabet(src[i+2]);
+ mychar2 = 16*DecodeWithHexBinAlphabet(src[i+4])+DecodeWithHexBinAlphabet(src[i+5]);
+ ret = DecodeWithUTF8Alphabet(mychar1,mychar2);
+ i = i+6;
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ }
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ }
+ dest[j++] = (ret >> 8) & 0xff;
+ dest[j++] = ret & 0xff;
+ }
+ dest[j++] = 0;
+ dest[j++] = 0;
+}
+
+void DecodeUTF8(unsigned char *dest, const unsigned char *src, int len)
+{
+ int i=0,j=0;
+ wchar_t ret;
+
+ while (i<=len) {
+ if (len-2>=i) {
+ if (src[i] >= 0xC2) {
+ ret = DecodeWithUTF8Alphabet(src[i],src[i+1]);
+ i = i+2;
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ }
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ }
+ dest[j++] = (ret >> 8) & 0xff;
+ dest[j++] = ret & 0xff;
+ }
+ dest[j++] = 0;
+ dest[j++] = 0;
+}
+
+void DecodeUTF7(unsigned char *dest, const unsigned char *src, int len)
+{
+ int i=0,j=0,z,p;
+ wchar_t ret;
+
+ while (i<=len) {
+ if (len-5>=i) {
+ if (src[i] == '+') {
+ z=0;
+ while (src[z+i+1] != '-' && z+i+1<len) z++;
+ p=DecodeBASE64(src+i+1, dest+j, z);
+ if (p%2 != 0) p--;
+ j+=p;
+ i+=z+2;
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ dest[j++] = (ret >> 8) & 0xff;
+ dest[j++] = ret & 0xff;
+ }
+ } else {
+ i+=EncodeWithUnicodeAlphabet(&src[i], &ret);
+ dest[j++] = (ret >> 8) & 0xff;
+ dest[j++] = ret & 0xff;
+ }
+ }
+ dest[j++] = 0;
+ dest[j++] = 0;
+}
+
+/*
+Bob Trower 08/04/01
+Copyright (c) Trantor Standard Systems Inc., 2001
+
+Permission is hereby granted, free of charge, to any person
+obtaining a copy of this software and associated
+documentation files (the "Software"), to deal in the
+Software without restriction, including without limitation
+the rights to use, copy, modify, merge, publish, distribute,
+sublicense, and/or sell copies of the Software, and to
+permit persons to whom the Software is furnished to do so,
+subject to the following conditions:
+
+The above copyright notice and this permission notice shall
+be included in all copies or substantial portions of the
+Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
+PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS
+OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
+OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
+OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
+SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+*/
+
+static void EncodeBASE64Block(unsigned char in[3], unsigned char out[4], int len)
+{
+ /* BASE64 translation Table as described in RFC1113 */
+ unsigned char cb64[]="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
+
+ out[0] = cb64[ in[0] >> 2 ];
+ out[1] = cb64[ ((in[0] & 0x03) << 4) | ((in[1] & 0xf0) >> 4) ];
+ out[2] = (unsigned char) (len > 1 ? cb64[ ((in[1] & 0x0f) << 2) | ((in[2] & 0xc0) >> 6) ] : '=');
+ out[3] = (unsigned char) (len > 2 ? cb64[ in[2] & 0x3f ] : '=');
+}
+
+void EncodeBASE64(const unsigned char *Input, unsigned char *Output, int Length)
+{
+ unsigned char in[3], out[4];
+ int i, pos = 0, len, outpos = 0;
+
+ while (pos < Length) {
+ len = 0;
+ for (i = 0; i < 3; i++) {
+ in[i] = 0;
+ if (pos < Length) {
+ in[i] = Input[pos];
+ len++;
+ pos++;
+ }
+ }
+ if(len) {
+ EncodeBASE64Block(in, out, len);
+ for (i = 0; i < 4; i++) Output[outpos++] = out[i];
+ }
+ }
+
+ Output[outpos++] = 0;
+}
+
+static void DecodeBASE64Block(unsigned char in[4], unsigned char out[3])
+{
+ out[0] = (unsigned char) (in[0] << 2 | in[1] >> 4);
+ out[1] = (unsigned char) (in[1] << 4 | in[2] >> 2);
+ out[2] = (unsigned char) (((in[2] << 6) & 0xc0) | in[3]);
+}
+
+int DecodeBASE64(const unsigned char *Input, unsigned char *Output, int Length)
+{
+ unsigned char cd64[]="|$$$}rstuvwxyz{$$$$$$$>?@ABCDEFGHIJKLMNOPQRSTUVW$$$$$$XYZ[\\]^_`abcdefghijklmnopq";
+ unsigned char in[4], out[3], v;
+ int i, len, pos = 0, outpos = 0;
+
+ while (pos < Length) {
+ len = 0;
+ for(i = 0; i < 4; i++) {
+ v = 0;
+ while(v == 0) {
+ if (pos >= Length) break;
+ v = (unsigned char) Input[pos++];
+ v = (unsigned char) ((v < 43 || v > 122) ? 0 : cd64[ v - 43 ]);
+ if (v) v = (unsigned char) ((v == '$') ? 0 : v - 61);
+ }
+ if(pos<=Length) {
+ if (v) {
+ len++;
+ in[i] = (unsigned char) (v - 1);
+ }
+ }
+ }
+ if (len) {
+ DecodeBASE64Block(in, out);
+ for(i = 0; i < len - 1; i++) Output[outpos++] = out[i];
+ }
+ }
+ Output[outpos] = 0;
+ return outpos;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
+
diff --git a/gammu/emb/common/misc/coding/coding.h b/gammu/emb/common/misc/coding/coding.h
new file mode 100644
index 0000000..d0c334d
--- a/dev/null
+++ b/gammu/emb/common/misc/coding/coding.h
@@ -0,0 +1,133 @@
+/* (c) 2002-2004 by Marcin Wiacek and others */
+
+#ifndef __coding_h
+#define __coding_h
+
+#include <stdlib.h>
+
+#include "../misc.h"
+
+#ifdef __OpenBSD__
+typedef int wint_t;
+#endif
+
+/* ---------------------------- Unicode ------------------------------------ */
+bool mywstrncasecmp (unsigned const char *a, unsigned const char *b, int num);
+unsigned char *mystrstr (unsigned const char *haystack, unsigned const char *needle);
+bool mywstrncmp (unsigned const char *a, unsigned const char *b, int num);
+bool myiswspace (unsigned const char *src);
+int mytowlower (wchar_t c);
+
+unsigned int EncodeWithUnicodeAlphabet (const unsigned char *value, wchar_t *dest);
+unsigned int DecodeWithUnicodeAlphabet (wchar_t value, unsigned char *dest);
+
+unsigned int UnicodeLength (const unsigned char *str);
+unsigned char *DecodeUnicodeString (const unsigned char *src);
+unsigned char *DecodeUnicodeConsole (const unsigned char *src);
+void DecodeUnicode (const unsigned char *src, unsigned char *dest);
+void EncodeUnicode (unsigned char *dest, const unsigned char *src, int len);
+
+void CopyUnicodeString (unsigned char *Dest, unsigned char *Source);
+void ReverseUnicodeString (unsigned char *String);
+
+void ReadUnicodeFile (unsigned char *Dest, unsigned char *Source);
+
+void DecodeUnicodeSpecialNOKIAChars (unsigned char *dest, const unsigned char *src, int len);
+void EncodeUnicodeSpecialNOKIAChars (unsigned char *dest, const unsigned char *src, int len);
+
+/* ------------------------------- BCD ------------------------------------- */
+unsigned char EncodeWithBCDAlphabet (int value);
+int DecodeWithBCDAlphabet (unsigned char value);
+
+void DecodeBCD (unsigned char *dest, const unsigned char *src, int len);
+void EncodeBCD (unsigned char *dest, const unsigned char *src, int len, bool fill);
+
+/* ------------------------------ UTF7 ------------------------------------- */
+void DecodeUTF7 (unsigned char *dest, const unsigned char *src, int len);
+
+/* ------------------------------ UTF8 ------------------------------------- */
+wchar_t DecodeWithUTF8Alphabet (unsigned char mychar3, unsigned char mychar4);
+bool EncodeWithUTF8Alphabet (unsigned char mychar1, unsigned char mychar2, unsigned char *ret1, unsigned char *ret2);
+
+bool EncodeUTF8QuotedPrintable (unsigned char *dest, const unsigned char *src);
+void DecodeUTF8QuotedPrintable (unsigned char *dest, const unsigned char *src, int len);
+
+bool EncodeUTF8 (unsigned char *dest, const unsigned char *src);
+void DecodeUTF8 (unsigned char *dest, const unsigned char *src, int len);
+
+/* ------------------------------- BASE64 ---------------------------------- */
+void EncodeBASE64 (const unsigned char *Input, unsigned char *Output, int Length);
+int DecodeBASE64 (const unsigned char *Input, unsigned char *Output, int Length);
+
+/* ----------------------------- HexBin ------------------------------------ */
+void DecodeHexBin (unsigned char *dest, const unsigned char *src, int len);
+void EncodeHexBin (unsigned char *dest, const unsigned char *src, int len);
+
+/* ----------------------------- HexUnicode -------------------------------- */
+void DecodeHexUnicode (unsigned char *dest, const unsigned char *src, int len);
+void EncodeHexUnicode (unsigned char *dest, const unsigned char *src, int len);
+
+/* ---------------------- DefaultAlphabet for SMS -------------------------- */
+void EncodeDefault (unsigned char *dest, const unsigned char *src, int *len, bool UseExtensions, unsigned char *ExtraAlphabet);
+void DecodeDefault (unsigned char *dest, const unsigned char *src, int len, bool UseExtensions, unsigned char *ExtraAlphabet);
+void FindDefaultAlphabetLen (const unsigned char *src, int *srclen, int *smslen, int maxlen);
+
+int GSM_PackSevenBitsToEight (int offset, unsigned char *input, unsigned char *output, int length);
+int GSM_UnpackEightBitsToSeven (int offset, int in_length, int out_length,
+ unsigned char *input, unsigned char *output);
+
+/* ----------------- Phone numbers according to GSM specs ------------------ */
+
+/**
+ * Enum to handle types of phones numbers like
+ * specified in GSM 03.40 section 9.1.2.5
+ */
+typedef enum {
+ /**
+ * Unknown number type
+ */
+ NUMBER_UNKNOWN = 0x81,
+ /**
+ * International number (full number with code of country)
+ */
+ NUMBER_INTERNATIONAL = 0x91,
+ /**
+ * Alphanumeric number (with chars too)
+ */
+ NUMBER_ALPHANUMERIC = 0xD0
+
+ /* specification give also other values */
+} GSM_NumberType;
+
+void GSM_UnpackSemiOctetNumber (unsigned char *retval, unsigned char *Number, bool semioctet);
+int GSM_PackSemiOctetNumber (unsigned char *Number, unsigned char *Output, bool semioctet);
+
+/* ---------------------------- Bits --------------------------------------- */
+
+void BufferAlign (unsigned char *Destination, int *CurrentBit);
+void BufferAlignNumber(int *CurrentBit);
+
+void AddBuffer (unsigned char *Destination, int *CurrentBit, unsigned char *Source, int BitsToProcess);
+void AddBufferByte(unsigned char *Destination, int *CurrentBit, unsigned char Source, int BitsToProcess);
+
+void GetBuffer (unsigned char *Source, int *CurrentBit, unsigned char *Destination, int BitsToProcess);
+void GetBufferInt (unsigned char *Source, int *CurrentBit, int *integer, int BitsToProcess);
+void GetBufferI (unsigned char *Source, int *CurrentBit, int *result, int BitsToProcess);
+
+int GetBit (unsigned char *Buffer, int BitNum);
+int SetBit (unsigned char *Buffer, int BitNum);
+int ClearBit (unsigned char *Buffer, int BitNum);
+
+/* ---------------------------- Other -------------------------------------- */
+
+void StringToDouble (char *text, double *d);
+
+bool mystrncasecmp (unsigned const char *a, unsigned const char *b, int num);
+
+void MyGetLine(unsigned char *Buffer, int *Pos, unsigned char *OutBuffer, int MaxLen);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/misc/coding/md5.c b/gammu/emb/common/misc/coding/md5.c
new file mode 100644
index 0000000..30fe33f
--- a/dev/null
+++ b/gammu/emb/common/misc/coding/md5.c
@@ -0,0 +1,298 @@
+/* Taken from ReHash (see http://www.reichlsoft.de.vu/) and released
+ * under GPL/LGPL with permission from ReHash author
+ * Dominik Reichl <dominik.reichl@t-online.de>, Germany
+ */
+
+/*
+ **********************************************************************
+ ** MD5.cpp **
+ ** **
+ ** - Style modified by Tony Ray, January 2001 **
+ ** Added support for randomizing initialization constants **
+ ** - Style modified by Dominik Reichl, April 2003 **
+ ** Optimized code **
+ ** **
+ ** MD5.c **
+ ** RSA Data Security, Inc. MD5 Message Digest Algorithm **
+ ** Created: 2/17/90 RLR **
+ ** Revised: 1/91 SRD,AJ,BSK,JT Reference C Version **
+ **********************************************************************
+ */
+
+/*
+ **********************************************************************
+ ** MD5.h -- Header file for implementation of MD5 **
+ ** RSA Data Security, Inc. MD5 Message Digest Algorithm **
+ ** Created: 2/17/90 RLR **
+ ** Revised: 12/27/90 SRD,AJ,BSK,JT Reference C version **
+ ** Revised (for MD5): RLR 4/27/91 **
+ ** -- G modified to have y&~z instead of y&z **
+ ** -- FF, GG, HH modified to add in last register done **
+ ** -- Access pattern: round 2 works mod 5, round 3 works mod 3 **
+ ** -- distinct additive constant for each step **
+ ** -- round 4 added, working mod 7 **
+ **********************************************************************
+ */
+
+/*
+ **********************************************************************
+ ** Copyright (C) 1990, RSA Data Security, Inc. All rights reserved. **
+ ** **
+ ** License to copy and use this software is granted provided that **
+ ** it is identified as the "RSA Data Security, Inc. MD5 Message **
+ ** Digest Algorithm" in all material mentioning or referencing this **
+ ** software or this function. **
+ ** **
+ ** License is also granted to make and use derivative works **
+ ** provided that such works are identified as "derived from the RSA **
+ ** Data Security, Inc. MD5 Message Digest Algorithm" in all **
+ ** material mentioning or referencing the derived work. **
+ ** **
+ ** RSA Data Security, Inc. makes no representations concerning **
+ ** either the merchantability of this software or the suitability **
+ ** of this software for any particular purpose. It is provided "as **
+ ** is" without express or implied warranty of any kind. **
+ ** **
+ ** These notices must be retained in any copies of any part of this **
+ ** documentation and/or software. **
+ **********************************************************************
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "md5.h"
+
+/* Typedef a 32 bit type */
+#ifndef UINT4
+typedef unsigned long int UINT4;
+#endif
+
+/* Data structure for MD5 (Message Digest) computation */
+typedef struct {
+ UINT4 i[2]; /* Number of _bits_ handled mod 2^64 */
+ UINT4 buf[4]; /* Scratch buffer */
+ unsigned char in[64]; /* Input buffer */
+ unsigned char digest[16]; /* Actual digest after MD5Final call */
+} MD5_CTX;
+
+/* Padding */
+static unsigned char MD5_PADDING[64] = {
+ 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
+};
+
+/* MD5_F, MD5_G and MD5_H are basic MD5 functions: selection, majority, parity */
+#define MD5_F(x, y, z) (((x) & (y)) | ((~x) & (z)))
+#define MD5_G(x, y, z) (((x) & (z)) | ((y) & (~z)))
+#define MD5_H(x, y, z) ((x) ^ (y) ^ (z))
+#define MD5_I(x, y, z) ((y) ^ ((x) | (~z)))
+
+/* ROTATE_LEFT rotates x left n bits */
+#ifndef ROTATE_LEFT
+#define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32-(n))))
+#endif
+
+/* MD5_FF, MD5_GG, MD5_HH, and MD5_II transformations for rounds 1, 2, 3, and 4 */
+/* Rotation is separate from addition to prevent recomputation */
+#define MD5_FF(a, b, c, d, x, s, ac) {(a) += MD5_F ((b), (c), (d)) + (x) + (UINT4)(ac); (a) = ROTATE_LEFT ((a), (s)); (a) += (b); }
+#define MD5_GG(a, b, c, d, x, s, ac) {(a) += MD5_G ((b), (c), (d)) + (x) + (UINT4)(ac); (a) = ROTATE_LEFT ((a), (s)); (a) += (b); }
+#define MD5_HH(a, b, c, d, x, s, ac) {(a) += MD5_H ((b), (c), (d)) + (x) + (UINT4)(ac); (a) = ROTATE_LEFT ((a), (s)); (a) += (b); }
+#define MD5_II(a, b, c, d, x, s, ac) {(a) += MD5_I ((b), (c), (d)) + (x) + (UINT4)(ac); (a) = ROTATE_LEFT ((a), (s)); (a) += (b); }
+
+/* Constants for transformation */
+#define MD5_S11 7 /* Round 1 */
+#define MD5_S12 12
+#define MD5_S13 17
+#define MD5_S14 22
+#define MD5_S21 5 /* Round 2 */
+#define MD5_S22 9
+#define MD5_S23 14
+#define MD5_S24 20
+#define MD5_S31 4 /* Round 3 */
+#define MD5_S32 11
+#define MD5_S33 16
+#define MD5_S34 23
+#define MD5_S41 6 /* Round 4 */
+#define MD5_S42 10
+#define MD5_S43 15
+#define MD5_S44 21
+
+/* Basic MD5 step. MD5_Transform buf based on in */
+static void MD5_Transform (UINT4 *buf, UINT4 *in)
+{
+ UINT4 a = buf[0], b = buf[1], c = buf[2], d = buf[3];
+
+ /* Round 1 */
+ MD5_FF ( a, b, c, d, in[ 0], MD5_S11, (UINT4) 3614090360u); /* 1 */
+ MD5_FF ( d, a, b, c, in[ 1], MD5_S12, (UINT4) 3905402710u); /* 2 */
+ MD5_FF ( c, d, a, b, in[ 2], MD5_S13, (UINT4) 606105819u); /* 3 */
+ MD5_FF ( b, c, d, a, in[ 3], MD5_S14, (UINT4) 3250441966u); /* 4 */
+ MD5_FF ( a, b, c, d, in[ 4], MD5_S11, (UINT4) 4118548399u); /* 5 */
+ MD5_FF ( d, a, b, c, in[ 5], MD5_S12, (UINT4) 1200080426u); /* 6 */
+ MD5_FF ( c, d, a, b, in[ 6], MD5_S13, (UINT4) 2821735955u); /* 7 */
+ MD5_FF ( b, c, d, a, in[ 7], MD5_S14, (UINT4) 4249261313u); /* 8 */
+ MD5_FF ( a, b, c, d, in[ 8], MD5_S11, (UINT4) 1770035416u); /* 9 */
+ MD5_FF ( d, a, b, c, in[ 9], MD5_S12, (UINT4) 2336552879u); /* 10 */
+ MD5_FF ( c, d, a, b, in[10], MD5_S13, (UINT4) 4294925233u); /* 11 */
+ MD5_FF ( b, c, d, a, in[11], MD5_S14, (UINT4) 2304563134u); /* 12 */
+ MD5_FF ( a, b, c, d, in[12], MD5_S11, (UINT4) 1804603682u); /* 13 */
+ MD5_FF ( d, a, b, c, in[13], MD5_S12, (UINT4) 4254626195u); /* 14 */
+ MD5_FF ( c, d, a, b, in[14], MD5_S13, (UINT4) 2792965006u); /* 15 */
+ MD5_FF ( b, c, d, a, in[15], MD5_S14, (UINT4) 1236535329u); /* 16 */
+
+ /* Round 2 */
+ MD5_GG ( a, b, c, d, in[ 1], MD5_S21, (UINT4) 4129170786u); /* 17 */
+ MD5_GG ( d, a, b, c, in[ 6], MD5_S22, (UINT4) 3225465664u); /* 18 */
+ MD5_GG ( c, d, a, b, in[11], MD5_S23, (UINT4) 643717713u); /* 19 */
+ MD5_GG ( b, c, d, a, in[ 0], MD5_S24, (UINT4) 3921069994u); /* 20 */
+ MD5_GG ( a, b, c, d, in[ 5], MD5_S21, (UINT4) 3593408605u); /* 21 */
+ MD5_GG ( d, a, b, c, in[10], MD5_S22, (UINT4) 38016083u); /* 22 */
+ MD5_GG ( c, d, a, b, in[15], MD5_S23, (UINT4) 3634488961u); /* 23 */
+ MD5_GG ( b, c, d, a, in[ 4], MD5_S24, (UINT4) 3889429448u); /* 24 */
+ MD5_GG ( a, b, c, d, in[ 9], MD5_S21, (UINT4) 568446438u); /* 25 */
+ MD5_GG ( d, a, b, c, in[14], MD5_S22, (UINT4) 3275163606u); /* 26 */
+ MD5_GG ( c, d, a, b, in[ 3], MD5_S23, (UINT4) 4107603335u); /* 27 */
+ MD5_GG ( b, c, d, a, in[ 8], MD5_S24, (UINT4) 1163531501u); /* 28 */
+ MD5_GG ( a, b, c, d, in[13], MD5_S21, (UINT4) 2850285829u); /* 29 */
+ MD5_GG ( d, a, b, c, in[ 2], MD5_S22, (UINT4) 4243563512u); /* 30 */
+ MD5_GG ( c, d, a, b, in[ 7], MD5_S23, (UINT4) 1735328473u); /* 31 */
+ MD5_GG ( b, c, d, a, in[12], MD5_S24, (UINT4) 2368359562u); /* 32 */
+
+ /* Round 3 */
+ MD5_HH ( a, b, c, d, in[ 5], MD5_S31, (UINT4) 4294588738u); /* 33 */
+ MD5_HH ( d, a, b, c, in[ 8], MD5_S32, (UINT4) 2272392833u); /* 34 */
+ MD5_HH ( c, d, a, b, in[11], MD5_S33, (UINT4) 1839030562u); /* 35 */
+ MD5_HH ( b, c, d, a, in[14], MD5_S34, (UINT4) 4259657740u); /* 36 */
+ MD5_HH ( a, b, c, d, in[ 1], MD5_S31, (UINT4) 2763975236u); /* 37 */
+ MD5_HH ( d, a, b, c, in[ 4], MD5_S32, (UINT4) 1272893353u); /* 38 */
+ MD5_HH ( c, d, a, b, in[ 7], MD5_S33, (UINT4) 4139469664u); /* 39 */
+ MD5_HH ( b, c, d, a, in[10], MD5_S34, (UINT4) 3200236656u); /* 40 */
+ MD5_HH ( a, b, c, d, in[13], MD5_S31, (UINT4) 681279174u); /* 41 */
+ MD5_HH ( d, a, b, c, in[ 0], MD5_S32, (UINT4) 3936430074u); /* 42 */
+ MD5_HH ( c, d, a, b, in[ 3], MD5_S33, (UINT4) 3572445317u); /* 43 */
+ MD5_HH ( b, c, d, a, in[ 6], MD5_S34, (UINT4) 76029189u); /* 44 */
+ MD5_HH ( a, b, c, d, in[ 9], MD5_S31, (UINT4) 3654602809u); /* 45 */
+ MD5_HH ( d, a, b, c, in[12], MD5_S32, (UINT4) 3873151461u); /* 46 */
+ MD5_HH ( c, d, a, b, in[15], MD5_S33, (UINT4) 530742520u); /* 47 */
+ MD5_HH ( b, c, d, a, in[ 2], MD5_S34, (UINT4) 3299628645u); /* 48 */
+
+ /* Round 4 */
+ MD5_II ( a, b, c, d, in[ 0], MD5_S41, (UINT4) 4096336452u); /* 49 */
+ MD5_II ( d, a, b, c, in[ 7], MD5_S42, (UINT4) 1126891415u); /* 50 */
+ MD5_II ( c, d, a, b, in[14], MD5_S43, (UINT4) 2878612391u); /* 51 */
+ MD5_II ( b, c, d, a, in[ 5], MD5_S44, (UINT4) 4237533241u); /* 52 */
+ MD5_II ( a, b, c, d, in[12], MD5_S41, (UINT4) 1700485571u); /* 53 */
+ MD5_II ( d, a, b, c, in[ 3], MD5_S42, (UINT4) 2399980690u); /* 54 */
+ MD5_II ( c, d, a, b, in[10], MD5_S43, (UINT4) 4293915773u); /* 55 */
+ MD5_II ( b, c, d, a, in[ 1], MD5_S44, (UINT4) 2240044497u); /* 56 */
+ MD5_II ( a, b, c, d, in[ 8], MD5_S41, (UINT4) 1873313359u); /* 57 */
+ MD5_II ( d, a, b, c, in[15], MD5_S42, (UINT4) 4264355552u); /* 58 */
+ MD5_II ( c, d, a, b, in[ 6], MD5_S43, (UINT4) 2734768916u); /* 59 */
+ MD5_II ( b, c, d, a, in[13], MD5_S44, (UINT4) 1309151649u); /* 60 */
+ MD5_II ( a, b, c, d, in[ 4], MD5_S41, (UINT4) 4149444226u); /* 61 */
+ MD5_II ( d, a, b, c, in[11], MD5_S42, (UINT4) 3174756917u); /* 62 */
+ MD5_II ( c, d, a, b, in[ 2], MD5_S43, (UINT4) 718787259u); /* 63 */
+ MD5_II ( b, c, d, a, in[ 9], MD5_S44, (UINT4) 3951481745u); /* 64 */
+
+ buf[0] += a;
+ buf[1] += b;
+ buf[2] += c;
+ buf[3] += d;
+}
+
+// Set pseudoRandomNumber to zero for RFC MD5 implementation
+void MD5Init (MD5_CTX *mdContext, unsigned long pseudoRandomNumber)
+{
+ mdContext->i[0] = mdContext->i[1] = (UINT4)0;
+
+ /* Load magic initialization constants */
+ mdContext->buf[0] = (UINT4)0x67452301 + (pseudoRandomNumber * 11);
+ mdContext->buf[1] = (UINT4)0xefcdab89 + (pseudoRandomNumber * 71);
+ mdContext->buf[2] = (UINT4)0x98badcfe + (pseudoRandomNumber * 37);
+ mdContext->buf[3] = (UINT4)0x10325476 + (pseudoRandomNumber * 97);
+}
+
+void MD5Update (MD5_CTX *mdContext, unsigned char *inBuf, unsigned int inLen)
+{
+ UINT4 in[16];
+ int mdi = 0;
+ unsigned int i = 0, ii = 0;
+
+ /* Compute number of bytes mod 64 */
+ mdi = (int)((mdContext->i[0] >> 3) & 0x3F);
+
+ /* Update number of bits */
+ if ((mdContext->i[0] + ((UINT4)inLen << 3)) < mdContext->i[0]) mdContext->i[1]++;
+ mdContext->i[0] += ((UINT4)inLen << 3);
+ mdContext->i[1] += ((UINT4)inLen >> 29);
+
+ while (inLen--) {
+ /* Add new character to buffer, increment mdi */
+ mdContext->in[mdi++] = *inBuf++;
+
+ /* Transform if necessary */
+ if (mdi == 0x40) {
+ for (i = 0, ii = 0; i < 16; i++, ii += 4)
+ in[i] = (((UINT4)mdContext->in[ii+3]) << 24) |
+ (((UINT4)mdContext->in[ii+2]) << 16) |
+ (((UINT4)mdContext->in[ii+1]) << 8) |
+ ((UINT4)mdContext->in[ii]);
+
+ MD5_Transform (mdContext->buf, in);
+ mdi = 0;
+ }
+ }
+}
+
+void MD5Final (MD5_CTX *mdContext)
+{
+ UINT4 in[16];
+ int mdi = 0;
+ unsigned int i = 0, ii = 0, padLen = 0;
+
+ /* Save number of bits */
+ in[14] = mdContext->i[0];
+ in[15] = mdContext->i[1];
+
+ /* Compute number of bytes mod 64 */
+ mdi = (int)((mdContext->i[0] >> 3) & 0x3F);
+
+ /* Pad out to 56 mod 64 */
+ padLen = (mdi < 56) ? (56 - mdi) : (120 - mdi);
+ MD5Update (mdContext, MD5_PADDING, padLen);
+
+ /* Append length in bits and transform */
+ for (i = 0, ii = 0; i < 14; i++, ii += 4)
+ in[i] = (((UINT4)mdContext->in[ii+3]) << 24) |
+ (((UINT4)mdContext->in[ii+2]) << 16) |
+ (((UINT4)mdContext->in[ii+1]) << 8) |
+ ((UINT4)mdContext->in[ii]);
+ MD5_Transform (mdContext->buf, in);
+
+ /* Store buffer in digest */
+ for (i = 0, ii = 0; i < 4; i++, ii += 4) {
+ mdContext->digest[ii] = (unsigned char)( mdContext->buf[i] & 0xFF);
+ mdContext->digest[ii+1] = (unsigned char)((mdContext->buf[i] >> 8) & 0xFF);
+ mdContext->digest[ii+2] = (unsigned char)((mdContext->buf[i] >> 16) & 0xFF);
+ mdContext->digest[ii+3] = (unsigned char)((mdContext->buf[i] >> 24) & 0xFF);
+ }
+}
+
+void CalculateMD5(unsigned char *buffer, int length, unsigned char *checksum)
+{
+ int i;
+ MD5_CTX m_md5;
+
+ MD5Init(&m_md5, 0);
+ MD5Update(&m_md5, buffer, length);
+ MD5Final(&m_md5);
+
+ for (i = 0; i < 16; i++) sprintf(checksum+i*2,"%02X", m_md5.digest[i]);
+}
diff --git a/gammu/emb/common/misc/coding/md5.h b/gammu/emb/common/misc/coding/md5.h
new file mode 100644
index 0000000..9da9229
--- a/dev/null
+++ b/gammu/emb/common/misc/coding/md5.h
@@ -0,0 +1,6 @@
+#ifndef __md5_h
+#define __md5_h
+
+void CalculateMD5(unsigned char *buffer, int length, unsigned char *checksum);
+
+#endif
diff --git a/gammu/emb/common/misc/misc.c b/gammu/emb/common/misc/misc.c
new file mode 100644
index 0000000..c2f09e4
--- a/dev/null
+++ b/gammu/emb/common/misc/misc.c
@@ -0,0 +1,591 @@
+/* (c) 2002-2004 by Marcin Wiacek and Michal Cihar */
+/* Checking used compiler (c) 2002 by Michal Cihar */
+
+#include <string.h>
+#include <ctype.h>
+#include <time.h>
+#include <stdarg.h>
+#include <stdio.h>
+#include <locale.h>
+#include <sys/timeb.h>
+#ifdef WIN32
+# include "windows.h"
+#endif
+
+#include "../gsmstate.h"
+#include "misc.h"
+
+/* Based on article in Polish PC-Kurier 8/1998 page 104
+ * Archive on http://www.pckurier.pl
+ */
+char *DayOfWeek (int year, int month, int day)
+{
+ int p,q,r,w;
+ static char DayOfWeekChar[10];
+
+ p=(14-month) / 12;
+ q=month+12*p-2;
+ r=year-p;
+ w=(day+(31*q) / 12 + r + r / 4 - r / 100 + r / 400) % 7;
+ strcpy(DayOfWeekChar,"");
+ switch (w) {
+ case 0: strcpy(DayOfWeekChar,"Sun"); break;
+ case 1: strcpy(DayOfWeekChar,"Mon"); break;
+ case 2: strcpy(DayOfWeekChar,"Tue"); break;
+ case 3: strcpy(DayOfWeekChar,"Wed"); break;
+ case 4: strcpy(DayOfWeekChar,"Thu"); break;
+ case 5: strcpy(DayOfWeekChar,"Fri"); break;
+ case 6: strcpy(DayOfWeekChar,"Sat"); break;
+ }
+ return DayOfWeekChar;
+}
+
+void Fill_GSM_DateTime(GSM_DateTime *Date, time_t timet)
+{
+ struct tm *now;
+
+ now = localtime(&timet);
+ Date->Year = now->tm_year;
+ Date->Month = now->tm_mon+1;
+ Date->Day = now->tm_mday;
+ Date->Hour = now->tm_hour;
+ Date->Minute = now->tm_min;
+ Date->Second = now->tm_sec;
+}
+
+void GSM_GetCurrentDateTime (GSM_DateTime *Date)
+{
+ Fill_GSM_DateTime(Date, time(NULL));
+ if (Date->Year<1900) {
+ if (Date->Year>90) Date->Year = Date->Year+1900;
+ else Date->Year = Date->Year+2000;
+ }
+}
+
+time_t Fill_Time_T(GSM_DateTime DT, int TZ)
+{
+ struct tm tm_starttime;
+ unsigned char buffer[30];
+
+ dbgprintf(" StartTime : %02i-%02i-%04i %02i:%02i:%02i\n",
+ DT.Day,DT.Month,DT.Year,DT.Hour,DT.Minute,DT.Second);
+
+ if (TZ != 0) {
+#if defined(WIN32) || defined(__SVR4)
+ sprintf(buffer,"TZ=PST+%i",TZ);
+ putenv(buffer);
+#else
+ sprintf(buffer,"PST+%i",TZ);
+ setenv("TZ",buffer,1);
+#endif
+ }
+ tzset();
+
+ memset(&tm_starttime, 0, sizeof(tm_starttime));
+ tm_starttime.tm_year = DT.Year - 1900;
+ tm_starttime.tm_mon = DT.Month - 1;
+ tm_starttime.tm_mday = DT.Day;
+ tm_starttime.tm_hour = DT.Hour;
+ tm_starttime.tm_min = DT.Minute;
+ tm_starttime.tm_sec = DT.Second;
+ tm_starttime.tm_isdst = 0;
+
+ return mktime(&tm_starttime);
+}
+
+void GetTimeDifference(unsigned long diff, GSM_DateTime *DT, bool Plus, int multi)
+{
+ time_t t_time;
+
+ t_time = Fill_Time_T(*DT,8);
+
+ if (Plus) {
+ t_time += diff*multi;
+ } else {
+ t_time -= diff*multi;
+ }
+
+ Fill_GSM_DateTime(DT, t_time);
+ DT->Year = DT->Year + 1900;
+ dbgprintf(" EndTime : %02i-%02i-%04i %02i:%02i:%02i\n",
+ DT->Day,DT->Month,DT->Year,DT->Hour,DT->Minute,DT->Second);
+}
+
+char *OSDateTime (GSM_DateTime dt, bool TimeZone)
+{
+ struct tm timeptr;
+ static char retval[200],retval2[200];
+ int p,q,r,w;
+
+#ifdef WIN32
+ setlocale(LC_ALL, ".OCP");
+#endif
+
+ /* Based on article in Polish PC-Kurier 8/1998 page 104
+ * Archive on http://www.pckurier.pl
+ */
+ p=(14-dt.Month) / 12;
+ q=dt.Month+12*p-2;
+ r=dt.Year-p;
+ w=(dt.Day+(31*q) / 12 + r + r / 4 - r / 100 + r / 400) % 7;
+
+ timeptr.tm_yday = 0; /* FIXME */
+ timeptr.tm_isdst = -1; /* FIXME */
+ timeptr.tm_year = dt.Year - 1900;
+ timeptr.tm_mon = dt.Month - 1;
+ timeptr.tm_mday = dt.Day;
+ timeptr.tm_hour = dt.Hour;
+ timeptr.tm_min = dt.Minute;
+ timeptr.tm_sec = dt.Second;
+ timeptr.tm_wday = w;
+#ifdef _BSD_SOURCE
+ timeptr.tm_zone = NULL;
+#endif
+
+#ifdef WIN32
+ strftime(retval2, 200, "%#c", &timeptr);
+#else
+ strftime(retval2, 200, "%c", &timeptr);
+#endif
+ if (TimeZone) {
+ if (dt.Timezone >= 0) {
+ sprintf(retval," +%02i00",dt.Timezone);
+ } else {
+ sprintf(retval," -%02i00",dt.Timezone);
+ }
+ strcat(retval2,retval);
+ }
+ /* If don't have weekday name, include it */
+ strftime(retval, 200, "%A", &timeptr);
+ if (strstr(retval2,retval)==NULL) {
+ /* Check for abbreviated weekday */
+ strftime(retval, 200, "%a", &timeptr);
+ if (strstr(retval2,retval)==NULL) {
+ strcat(retval2," (");
+ strcat(retval2,retval);
+ strcat(retval2,")");
+ }
+ }
+
+#ifdef WIN32
+ setlocale(LC_ALL, ".ACP");
+#endif
+
+ return retval2;
+}
+
+char *OSDate (GSM_DateTime dt)
+{
+ struct tm timeptr;
+ static char retval[200],retval2[200];
+ int p,q,r,w;
+
+#ifdef WIN32
+ setlocale(LC_ALL, ".OCP");
+#endif
+
+ /* Based on article in Polish PC-Kurier 8/1998 page 104
+ * Archive on http://www.pckurier.pl
+ */
+ p=(14-dt.Month) / 12;
+ q=dt.Month+12*p-2;
+ r=dt.Year-p;
+ w=(dt.Day+(31*q) / 12 + r + r / 4 - r / 100 + r / 400) % 7;
+
+ timeptr.tm_yday = 0; /* FIXME */
+ timeptr.tm_isdst = -1; /* FIXME */
+ timeptr.tm_year = dt.Year - 1900;
+ timeptr.tm_mon = dt.Month - 1;
+ timeptr.tm_mday = dt.Day;
+ timeptr.tm_hour = dt.Hour;
+ timeptr.tm_min = dt.Minute;
+ timeptr.tm_sec = dt.Second;
+ timeptr.tm_wday = w;
+#ifdef _BSD_SOURCE
+ timeptr.tm_zone = NULL;
+#endif
+
+#ifdef WIN32
+ strftime(retval2, 200, "%#x", &timeptr);
+#else
+ strftime(retval2, 200, "%x", &timeptr);
+#endif
+ /* If don't have weekday name, include it */
+ strftime(retval, 200, "%A", &timeptr);
+ if (strstr(retval2,retval)==NULL) {
+ /* Check also for short name */
+ strftime(retval, 200, "%a", &timeptr);
+ if (strstr(retval2,retval)==NULL) {
+ strcat(retval2," (");
+ strcat(retval2,retval);
+ strcat(retval2,")");
+ }
+ }
+
+#ifdef WIN32
+ setlocale(LC_ALL, ".ACP");
+#endif
+
+ return retval2;
+}
+
+bool CheckDate(GSM_DateTime *date)
+{
+ /* FIXME: This could also check if day is correct for selected month */
+ return date->Year != 0 &&
+ date->Month >= 1 && date->Month <= 12 &&
+ date->Day >= 1 && date->Day <= 31;
+}
+
+bool CheckTime(GSM_DateTime *date)
+{
+ return date->Hour <= 23 && date->Hour >= 0 &&
+ date->Minute <= 59 && date->Minute >= 0 &&
+ date->Second <= 59 && date->Second >= 0;
+}
+
+int GetLine(FILE *File, char *Line, int count)
+{
+ int num;
+
+ if (fgets(Line, count, File) != NULL) {
+ num = strlen(Line) - 1;
+ while(1) {
+ if (Line[num] != '\n' && Line[num] != '\r') break;
+ if (num == 0) break;
+ Line[num--] = 0;
+ }
+ return strlen(Line);
+ }
+ return -1;
+}
+
+void SplitLines(unsigned char *message, int messagesize, GSM_Lines *lines, unsigned char *whitespaces, int spaceslen, bool eot)
+{
+ int i,number=0,j;
+ bool whitespace=true, nowwhite;
+
+ for (i=0;i<MAX_LINES*2;i++) lines->numbers[i]=0;
+
+ for (i=0;i<messagesize;i++) {
+ nowwhite = false;
+ for (j=0;j<spaceslen;j++) {
+ if (whitespaces[j] == message[i]) {
+ nowwhite = true;
+ break;
+ }
+ }
+ if (whitespace) {
+ if (!nowwhite) {
+ lines->numbers[number]=i;
+ number++;
+ whitespace=false;
+ }
+ } else {
+ if (nowwhite) {
+ lines->numbers[number]=i;
+ number++;
+ whitespace=true;
+ }
+
+ }
+ }
+ if (eot && !whitespace) lines->numbers[number]=messagesize;
+}
+
+char *GetLineString(unsigned char *message, GSM_Lines lines, int start)
+{
+ static char retval[800];
+
+ memcpy(retval,message + lines.numbers[start*2-2],lines.numbers[start*2-2+1]-lines.numbers[start*2-2]);
+ retval[lines.numbers[start*2-2+1]-lines.numbers[start*2-2]]=0;
+
+ return retval;
+}
+
+void CopyLineString(unsigned char *dest, unsigned char *src, GSM_Lines lines, int start)
+{
+ memcpy(dest,GetLineString(src, lines, start),strlen(GetLineString(src, lines, start)));
+ dest[strlen(GetLineString(src, lines, start))] = 0;
+}
+
+Debug_Info di = {0,NULL,false,""};
+
+#ifdef DEBUG
+#ifdef __GNUC__
+__attribute__((format(printf, 1, 2)))
+#endif
+int dbgprintf(const char *format, ...)
+{
+ va_list argp;
+ int result;
+ static unsigned char nextline[2000]="";
+ unsigned char buffer[2000];
+ GSM_DateTime date_time;
+
+ if (di.df != NULL && (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE)) {
+ va_start(argp, format);
+ result = vsprintf(buffer, format, argp);
+ strcat(nextline, buffer);
+ if (strstr(buffer, "\n")) {
+ if (di.dl == DL_TEXTALLDATE) {
+ GSM_GetCurrentDateTime(&date_time);
+ fprintf(di.df,"%s %4d/%02d/%02d %02d:%02d:%02d: %s",
+ DayOfWeek(date_time.Year, date_time.Month, date_time.Day),
+ date_time.Year, date_time.Month, date_time.Day,
+ date_time.Hour, date_time.Minute, date_time.Second,nextline);
+ } else {
+ fprintf(di.df,"%s",nextline);
+ }
+ strcpy(nextline, "");
+ }
+ fflush(di.df);
+ va_end(argp);
+ return result;
+ }
+ return 0;
+}
+#endif
+
+/* assumption: if \n is present it is always the last char,
+ * string never of the form "......\n..."
+ */
+#ifdef __GNUC__
+__attribute__((format(printf, 3, 4)))
+#endif
+int smfprintf(FILE *f, Debug_Level dl, const char *format, ...)
+{
+ va_list argp;
+ int result=0;
+ static unsigned char prevline[2000] = "", nextline[2000]="";
+ static unsigned int linecount=0;
+ unsigned char buffer[2000];
+ GSM_DateTime date_time;
+
+ if (f == NULL) return 0;
+ va_start(argp, format);
+ result = vsprintf(buffer, format, argp);
+ strcat(nextline, buffer);
+ if (strstr(buffer, "\n")) {
+ if (ftell(f) < 5000000) {
+ GSM_GetCurrentDateTime(&date_time);
+ if (linecount > 0) {
+ if (dl == DL_TEXTALLDATE || dl == DL_TEXTERRORDATE || dl == DL_TEXTDATE) {
+ fprintf(f,"%s %4d/%02d/%02d %02d:%02d:%02d: <%i> %s",
+ DayOfWeek(date_time.Year, date_time.Month, date_time.Day),
+ date_time.Year, date_time.Month, date_time.Day,
+ date_time.Hour, date_time.Minute, date_time.Second,linecount,prevline);
+ } else {
+ fprintf(f,"%s",prevline);
+ }
+ }
+ linecount=0;
+ if (dl == DL_TEXTALLDATE || dl == DL_TEXTERRORDATE || dl == DL_TEXTDATE) {
+ fprintf(f,"%s %4d/%02d/%02d %02d:%02d:%02d: %s",
+ DayOfWeek(date_time.Year, date_time.Month, date_time.Day),
+ date_time.Year, date_time.Month, date_time.Day,
+ date_time.Hour, date_time.Minute, date_time.Second,nextline);
+ } else {
+ fprintf(f,"%s",nextline);
+ }
+ strcpy(prevline, nextline);
+ }
+ strcpy(nextline, "");
+ fflush(f);
+ }
+ va_end(argp);
+ return result;
+}
+
+bool GSM_SetDebugLevel(char *info, Debug_Info *di)
+{
+ if (!strcmp(info,"nothing")) {di->dl = 0; return true;}
+ if (!strcmp(info,"text")) {di->dl = DL_TEXT; return true;}
+ if (!strcmp(info,"textall")) {di->dl = DL_TEXTALL; return true;}
+ if (!strcmp(info,"binary")) {di->dl = DL_BINARY; return true;}
+ if (!strcmp(info,"errors")) {di->dl = DL_TEXTERROR; return true;}
+ if (!strcmp(info,"textdate")) {di->dl = DL_TEXTDATE; return true;}
+ if (!strcmp(info,"textalldate")) {di->dl = DL_TEXTALLDATE; return true;}
+ if (!strcmp(info,"errorsdate")) {di->dl = DL_TEXTERRORDATE; return true;}
+ return false;
+}
+
+/* Dumps a message */
+void DumpMessage(FILE *df, Debug_Level dl, const unsigned char *message, int messagesize)
+{
+ int i,j=0,len=16;
+ unsigned char buffer[200];
+
+ if (df==NULL || messagesize == 0) return;
+
+ smfprintf(df, dl, "\n");
+
+ memset(buffer,0x20,sizeof(buffer));
+ buffer[len*5-1]=0;
+
+ for (i = 0; i < messagesize; i++) {
+ sprintf(buffer+j*4,"%02X",message[i]);
+ buffer[j*4+2] = 0x20;
+ if (isprint(message[i]) && message[i]!=0x09) {
+ if (j != len-1) buffer[j*4+2] = message[i];
+ buffer[(len-1)*4+j+3] = message[i];
+ } else {
+ buffer[(len-1)*4+j+3] = '.';
+ }
+ if (j != len-1 && i != messagesize-1) buffer[j*4+3] = '|';
+ if (j == len-1) {
+ smfprintf(df, dl, "%s\n", buffer);
+ memset(buffer,0x20,sizeof(buffer));
+ buffer[len*5-1]=0;
+ j = 0;
+ } else {
+ j++;
+ }
+ }
+ if (j != 0) smfprintf(df, dl, "%s\n", buffer);
+}
+
+char *GetOS(void)
+{
+#ifdef WIN32
+ OSVERSIONINFOEX Ver;
+ bool Extended = true;
+#endif
+ static char Buffer[100] = {0x00};
+
+#ifdef WIN32
+ memset(&Ver,sizeof(OSVERSIONINFOEX),0);
+ Ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
+
+ if (!GetVersionEx((OSVERSIONINFO *)&Ver)) {
+ Extended = false;
+ Ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
+ if (!GetVersionEx((OSVERSIONINFO *)&Ver)) {
+//#ifdef _MSC_VER
+// Ver.dwMajorVersion = _winmajor;
+// Ver.dwMinorVersion = _winminor;
+// Ver.dwBuildNumber = _osver;
+//#else
+ sprintf(Buffer, "Windows");
+ return Buffer;
+//#endif
+ }
+ }
+
+ /* ----------------- 9x family ------------------ */
+
+ /* no info about Win95 SP1, Win95 OSR2.1, Win95 OSR2.5.... */
+ if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 0 && Ver.dwBuildNumber == 950) {
+ sprintf(Buffer,"Win 95");
+ } else if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 0 && Ver.dwBuildNumber == 1111) {
+ sprintf(Buffer,"Win 95 OSR2.x");
+
+ /* no info about Win98 SP1.... */
+ } else if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 10 && Ver.dwBuildNumber == 1998) {
+ sprintf(Buffer,"Win 98");
+ } else if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 10 && Ver.dwBuildNumber == 2222) {
+ sprintf(Buffer,"Win 98 SE");
+
+ } else if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 90 && Ver.dwBuildNumber == 3000) {
+ sprintf(Buffer,"Win ME");
+
+ /* ---------------- NT family ------------------- */
+
+ } else if (Ver.dwMajorVersion == 4 && Ver.dwMinorVersion == 0 && Ver.dwBuildNumber == 1381) {
+ sprintf(Buffer,"Win NT 4.0");
+
+ } else if (Ver.dwMajorVersion == 5 && Ver.dwMinorVersion == 0 && Ver.dwBuildNumber == 2195) {
+ sprintf(Buffer,"Win 2000");
+
+ } else if (Ver.dwMajorVersion == 5 && Ver.dwMinorVersion == 1 && Ver.dwBuildNumber == 2600) {
+ sprintf(Buffer,"Win XP");
+#if _MSC_VER > 1200 //6.0 has it undeclared
+ if (Extended) {
+ if (Ver.wSuiteMask & VER_SUITE_PERSONAL) {
+ sprintf(Buffer+strlen(Buffer)," Home");
+ } else {
+ sprintf(Buffer+strlen(Buffer)," Pro");
+ }
+ }
+#endif
+
+ } else if (Ver.dwMajorVersion == 5 && Ver.dwMinorVersion == 2) {
+ sprintf(Buffer,"Win 2003");
+
+ } else {
+ sprintf(Buffer, "Windows %i.%i.%i",Ver.dwMajorVersion,Ver.dwMinorVersion,Ver.dwBuildNumber);
+ }
+
+ if (Extended && Ver.wServicePackMajor != 0) {
+ sprintf(Buffer+strlen(Buffer)," SP%i",Ver.wServicePackMajor);
+ }
+#elif defined(linux) || defined(__linux) || defined(__linux__)
+ sprintf(Buffer, "Linux");
+#elif defined(__FreeBSD__)
+ sprintf(Buffer, "FreeBSD");
+#elif defined(__NetBSD__)
+ sprintf(Buffer, "NetBSD");
+#elif defined(__OpenBSD__)
+ sprintf(Buffer, "OpenBSD");
+#elif defined(__GNU__)
+ sprintf(Buffer, "GNU/Hurd");
+#elif defined(sun) || defined(__sun) || defined(__sun__)
+# ifdef __SVR4
+ sprintf(Buffer, "Sun Solaris");
+# else
+ sprintf(Buffer, "SunOS");
+# endif
+#elif defined(hpux) || defined(__hpux) || defined(__hpux__)
+ sprintf(Buffer, "HP-UX");
+#elif defined(ultrix) || defined(__ultrix) || defined(__ultrix__)
+ sprintf(Buffer, "DEC Ultrix");
+#elif defined(sgi) || defined(__sgi)
+ sprintf(Buffer, "SGI Irix");
+#elif defined(__osf__)
+ sprintf(Buffer, "OSF Unix");
+#elif defined(bsdi) || defined(__bsdi__)
+ sprintf(Buffer, "BSDI Unix");
+#elif defined(_AIX)
+ sprintf(Buffer, "AIX Unix");
+#elif defined(_UNIXWARE)
+ sprintf(Buffer, "SCO Unixware");
+#elif defined(DGUX)
+ sprintf(Buffer, "DG Unix");
+#elif defined(__QNX__)
+ sprintf(Buffer, "QNX");
+#endif
+ return Buffer;
+}
+
+char *GetCompiler(void)
+{
+ static char Buffer[100] = {0x00};
+
+#ifdef WIN32
+# ifdef _MSC_VER
+ if (_MSC_VER == 1200) { //?
+ sprintf(Buffer, "MS VC 6.0");
+ } else if (_MSC_VER == 1300) {
+ sprintf(Buffer, "MS VC .NET 2002");
+ } else if (_MSC_VER == 1310) {
+ sprintf(Buffer, "MS VC .NET 2003");
+ } else {
+ sprintf(Buffer, "MS VC %i",_MSC_VER);
+ }
+# elif defined(__BORLANDC__)
+ sprintf(Buffer, "Borland C++ %i",__BORLANDC__);
+# endif
+#elif defined(DJGPP)
+ sprintf(Buffer, "djgpp");
+#elif defined(__GNUC__)
+ sprintf(Buffer, "gcc %i.%i", __GNUC__, __GNUC_MINOR__);
+#elif defined(__SUNPRO_CC)
+ sprintf(Buffer, "Sun C++ %x", __SUNPRO_CC);
+#endif
+
+ return Buffer;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/misc/misc.h b/gammu/emb/common/misc/misc.h
new file mode 100644
index 0000000..8b46170
--- a/dev/null
+++ b/gammu/emb/common/misc/misc.h
@@ -0,0 +1,137 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef __misc_h
+#define __misc_h
+
+#include <stdio.h>
+#include <time.h>
+#ifdef WIN32
+# include <windows.h>
+#endif
+
+#include "../config.h"
+
+#ifndef __cplusplus
+#ifndef false
+# define false 0
+#endif
+#ifndef true
+# define true !0
+#endif
+#ifndef bool
+# define bool char
+#endif
+#endif /* __cplusplus */
+
+#ifdef WIN32
+# define my_sleep(x) ((x)<1000 ? Sleep(1) : Sleep((x)/1000))
+#else
+# define my_sleep(x) usleep(x)
+#endif
+
+#undef MAX
+#define MAX(a,b) ((a)>(b) ? (a) : (b))
+#undef MIN
+#define MIN(a,b) ((a)<(b) ? (a) : (b))
+
+/* ------------------------------------------------------------------------- */
+
+#define MAX_LINES 50
+
+int GetLine(FILE *File, char *Line, int count);
+
+typedef struct {
+ int numbers[MAX_LINES*2];
+} GSM_Lines;
+
+void SplitLines(unsigned char *message, int messagesize, GSM_Lines *lines, unsigned char *whitespaces, int spaceslen, bool eot);
+char *GetLineString(unsigned char *message, GSM_Lines lines, int start);
+void CopyLineString(unsigned char *dest, unsigned char *src, GSM_Lines lines, int start);
+
+/* ------------------------------------------------------------------------- */
+
+typedef enum {
+ DL_BINARY = 1, /* Binary transmission dump */
+ DL_TEXT, /* Text transmission dump */
+ DL_TEXTALL, /* Everything */
+ DL_TEXTERROR, /* Only errors */
+ DL_TEXTDATE, /* Text transmission dump */
+ DL_TEXTALLDATE, /* Everything */
+ DL_TEXTERRORDATE /* Only errors */
+} Debug_Level;
+
+typedef struct {
+ Debug_Level dl;
+ FILE *df;
+ bool use_global;
+ char *coding;
+} Debug_Info;
+
+extern Debug_Info di;
+
+#ifdef DEBUG
+#ifdef __GNUC__
+__attribute__((format(printf, 1, 2)))
+#endif
+int dbgprintf(const char *format, ...);
+#else
+# ifndef WIN32
+# define dbgprintf(a...) do { } while (0)
+# else
+# define dbgprintf
+# endif
+#endif
+
+#ifdef __GNUC__
+__attribute__((format(printf, 3, 4)))
+#endif
+int smfprintf(FILE *f, Debug_Level dl, const char *format, ...);
+
+void DumpMessage(FILE *df, Debug_Level dl, const unsigned char *message, int messagesize);
+
+bool GSM_SetDebugLevel(char *info, Debug_Info *di);
+
+/* ------------------------------------------------------------------------- */
+
+/**
+ * Structure used for saving date and time
+ */
+typedef struct {
+ /**
+ * The difference between local time and GMT in hours
+ */
+ int Timezone;
+
+ unsigned int Second;
+ unsigned int Minute;
+ unsigned int Hour;
+
+ unsigned int Day;
+ /**
+ * January = 1, February = 2, etc.
+ */
+ unsigned int Month;
+ /**
+ * Complete year number. Not 03, but 2003
+ */
+ unsigned int Year;
+} GSM_DateTime;
+
+void GSM_GetCurrentDateTime (GSM_DateTime *Date);
+char *OSDateTime (GSM_DateTime dt, bool TimeZone);
+char *OSDate (GSM_DateTime dt);
+char *DayOfWeek (int year, int month, int day);
+time_t Fill_Time_T (GSM_DateTime DT, int TZ);
+void GetTimeDifference (unsigned long diff, GSM_DateTime *DT, bool Plus, int multi);
+void Fill_GSM_DateTime (GSM_DateTime *Date, time_t timet);
+bool CheckDate (GSM_DateTime *date);
+bool CheckTime (GSM_DateTime *date);
+
+char *GetCompiler(void);
+char *GetOS(void);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/alcatel/alcatel.c b/gammu/emb/common/phone/alcatel/alcatel.c
new file mode 100644
index 0000000..f004ad4
--- a/dev/null
+++ b/gammu/emb/common/phone/alcatel/alcatel.c
@@ -0,0 +1,3991 @@
+/* (c) 2002-2004 by Michal Cihar */
+
+/*
+ * High level functions for communication with Alcatel One Touch 501 and
+ * compatible mobile phone.
+ *
+ * This code implements functions to communicate with Alcatel phones,
+ * currently seem to work:
+ * - BE5 series (501/701)
+ * - BF5 series (715)
+ * - BH4 series (535/735)
+ * For some functions it uses normal AT mode (not implemented here, look at
+ * ../at/atgen.[ch]) for others it switches into binary mode and initialises
+ * underlaying protocol (see ../../protocol/alcatel/alcabus.[ch]) and
+ * communicates over it. Don't ask me why Alcatel uses such silly thing...
+ *
+ * Notes for future features:
+ * - max phone number length is 61 (BE5)
+ * - max name length is 50 (BE5)
+ */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_ALCATEL
+#ifdef GSM_ENABLE_ATGEN
+
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+#include "../../misc/misc.h"
+#include "../../service/sms/gsmsms.h"
+#include "../pfunc.h"
+#include "alcatel.h"
+
+/* Timeout for GSM_WaitFor calls. */
+#define ALCATEL_TIMEOUT 64
+
+/* Some magic numbers for protocol follow */
+
+/* synchronisation types (for everything except begin transfer): */
+#define ALCATEL_SYNC_TYPE_CALENDAR 0x64
+#define ALCATEL_SYNC_TYPE_TODO 0x68
+#define ALCATEL_SYNC_TYPE_CONTACTS 0x6C
+
+/* synchronisation types (for begin transfer): */
+#define ALCATEL_BEGIN_SYNC_CALENDAR 0x00
+#define ALCATEL_BEGIN_SYNC_TODO 0x02
+#define ALCATEL_BEGIN_SYNC_CONTACTS 0x01
+
+/* category types */
+#define ALCATEL_LIST_TODO_CAT 0x9B
+#define ALCATEL_LIST_CONTACTS_CAT 0x96
+
+
+/* We need lot of ATGEN functions, because Alcatel is an AT device. */
+
+extern GSM_Reply_Function ALCATELReplyFunctions[];
+extern GSM_Reply_Function ATGENReplyFunctions[];
+
+extern GSM_Error ATGEN_Initialise (GSM_StateMachine *s);
+extern GSM_Error ATGEN_Terminate (GSM_StateMachine *s);
+extern GSM_Error ATGEN_GetIMEI (GSM_StateMachine *s);
+extern GSM_Error ATGEN_GetFirmware (GSM_StateMachine *s);
+extern GSM_Error ATGEN_GetModel (GSM_StateMachine *s);
+extern GSM_Error ATGEN_GetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time);
+extern GSM_Error ATGEN_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+extern GSM_Error ATGEN_GetNextMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry, bool start);
+extern GSM_Error ATGEN_SetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+extern GSM_Error ATGEN_AddMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+extern GSM_Error ATGEN_DeleteMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry);
+extern GSM_Error ATGEN_GetMemoryStatus (GSM_StateMachine *s, GSM_MemoryStatus *Status);
+extern GSM_Error ATGEN_GetSMSC (GSM_StateMachine *s, GSM_SMSC *smsc);
+extern GSM_Error ATGEN_SetSMSC (GSM_StateMachine *s, GSM_SMSC *smsc);
+extern GSM_Error ATGEN_GetSMSFolders (GSM_StateMachine *s, GSM_SMSFolders *folders);
+extern GSM_Error ATGEN_GetSMSStatus (GSM_StateMachine *s, GSM_SMSMemoryStatus *status);
+extern GSM_Error ATGEN_GetSMS (GSM_StateMachine *s, GSM_MultiSMSMessage *sms);
+extern GSM_Error ATGEN_GetNextSMS (GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start);
+extern GSM_Error ATGEN_SendSavedSMS (GSM_StateMachine *s, int Folder, int Location);
+extern GSM_Error ATGEN_SendSMS (GSM_StateMachine *s, GSM_SMSMessage *sms);
+extern GSM_Error ATGEN_DeleteSMS (GSM_StateMachine *s, GSM_SMSMessage *sms);
+extern GSM_Error ATGEN_AddSMS (GSM_StateMachine *s, GSM_SMSMessage *sms);
+extern GSM_Error ATGEN_GetBatteryCharge (GSM_StateMachine *s, GSM_BatteryCharge *bat);
+extern GSM_Error ATGEN_GetSignalQuality (GSM_StateMachine *s, GSM_SignalQuality *sig);
+extern GSM_Error ATGEN_DialVoice (GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber);
+extern GSM_Error ATGEN_AnswerCall (GSM_StateMachine *s, int ID, bool all);
+extern GSM_Error ATGEN_CancelCall (GSM_StateMachine *s, int ID, bool all);
+extern GSM_Error ATGEN_SetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time);
+extern GSM_Error ATGEN_EnterSecurityCode (GSM_StateMachine *s, GSM_SecurityCode Code);
+extern GSM_Error ATGEN_GetSecurityStatus (GSM_StateMachine *s, GSM_SecurityCodeType *Status);
+extern GSM_Error ATGEN_ResetPhoneSettings (GSM_StateMachine *s, GSM_ResetSettingsType Type);
+extern GSM_Error ATGEN_SendDTMF (GSM_StateMachine *s, char *sequence);
+extern GSM_Error ATGEN_GetSIMIMSI (GSM_StateMachine *s, char *IMSI);
+extern GSM_Error ATGEN_HandleCMSError (GSM_StateMachine *s);
+extern GSM_Error ATGEN_GetNetworkInfo (GSM_StateMachine *s, GSM_NetworkInfo *netinfo);
+extern GSM_Error ATGEN_Reset (GSM_StateMachine *s, bool hard);
+extern GSM_Error ATGEN_PressKey (GSM_StateMachine *s, GSM_KeyCode Key, bool Press);
+extern GSM_Error ATGEN_GetDisplayStatus (GSM_StateMachine *s, GSM_DisplayFeatures *features);
+extern GSM_Error ATGEN_SetAutoNetworkLogin (GSM_StateMachine *s);
+extern GSM_Error ATGEN_DeleteAllMemory (GSM_StateMachine *s, GSM_MemoryType type);
+
+extern GSM_Error ATGEN_DispatchMessage (GSM_StateMachine *s);
+extern GSM_Error ATGEN_SetIncomingCB (GSM_StateMachine *s, bool enable);
+extern GSM_Error ATGEN_SetIncomingSMS (GSM_StateMachine *s, bool enable);
+
+/**
+ * Alcatel uses some 8-bit characters in contacts, calendar etc.. This table
+ * attempts to decode it, it is probably not complete, here are just chars
+ * that I found...
+ */
+unsigned char GSM_AlcatelAlphabet[] =
+{
+/* in phone unicode description */
+ 0x80, 0x00,0x20, /* empty */
+ 0x81, 0x00,0x20, /* empty */
+ 0x82, 0x00,0x20, /* empty */
+ 0x83, 0x00,0x20, /* empty */
+
+ 0x84, 0x00,0xe7, /* c cedilla */
+ 0x85, 0x20,0x26, /* ... */
+ 0x86, 0x03,0xc0, /* pi */
+ 0x87, 0x01,0x3e, /* l caron */
+ 0x88, 0x00,0xc0, /* A grave */
+ 0x89, 0x00,0xc1, /* A acute */
+ 0x8a, 0x00,0xc2, /* A circumflex */
+ 0x8b, 0x00,0xc3, /* A tilde */
+ 0x8c, 0x00,0xc8, /* E grave */
+ 0x8d, 0x00,0xca, /* E circumflex */
+ 0x8e, 0x00,0xcb, /* E diaresis */
+ 0x8f, 0x00,0xcc, /* I grave */
+ 0x90, 0x00,0xcd, /* I acute */
+ 0x91, 0x00,0xd0, /* ETH */
+ 0x92, 0x00,0xd2, /* O grave */
+ 0x93, 0x00,0xd3, /* O acute */
+ 0x94, 0x00,0xd4, /* O circumflex */
+ 0x95, 0x00,0xd5, /* O tilde */
+ 0x96, 0x00,0xd9, /* U grave */
+ 0x97, 0x00,0xda, /* U acute */
+ 0x98, 0x00,0xe1, /* a acute */
+ 0x99, 0x00,0xe2, /* a circumflex */
+ 0x9a, 0x00,0xe3, /* a tilde */
+ 0x9b, 0x00,0xea, /* e circumflex */
+ 0x9c, 0x00,0xeb, /* e diaresis */
+ 0x9d, 0x00,0xed, /* i acute */
+ 0x9e, 0x00,0xee, /* i circumflex */
+ 0x9f, 0x00,0xef, /* i diaresis */
+ 0xa0, 0x00,0xf3, /* o acute */
+ 0xa1, 0x00,0xf4, /* o circumflex */
+ 0xa2, 0x00,0xf5, /* o tilde */
+ 0xa3, 0x00,0xfa, /* u acute */
+ 0xa4, 0x00,0xa2, /* cent */
+ 0xa5, 0x00,0x5b, /* [ */
+ 0xa6, 0x01,0x59, /* r caron */
+ 0xa7, 0x01,0x0d, /* c caron */
+ 0xa8, 0x01,0x61, /* s caron */
+ 0xa9, 0x01,0x1b, /* e caron */
+ 0xaa, 0x01,0x6f, /* u ring */
+ 0xab, 0x00,0xfd, /* y acute */
+ 0xac, 0x00,0xf0, /* eth */
+ 0xad, 0x01,0x07, /* c acute */
+ 0xae, 0x01,0x19, /* e ogonek */
+ 0xaf, 0x01,0x05, /* a ogonek */
+ 0xb0, 0x01,0x7c, /* z dot */
+ 0xb1, 0x01,0x7a, /* z acute */
+ 0xb2, 0x01,0x5b, /* s acute */
+ 0xb3, 0x01,0x44, /* n acute */
+ 0xb4, 0x01,0x42, /* l stroke */
+
+ 0xb5, 0x00,0x20, /* empty */
+
+ 0xb6, 0x01,0x48, /* n caron */
+ 0xb7, 0x01,0x65, /* t caron */
+
+ 0xb8, 0x00,0x20, /* empty */
+
+ 0xb9, 0x01,0x7e, /* z caron */
+ 0xba, 0x01,0xe7, /* g caron */
+
+ 0xbb, 0x00,0x20, /* empty */
+ 0xbc, 0x00,0x20, /* empty */
+
+ 0xbd, 0x1e,0x20, /* G macron */
+ 0xbe, 0x1e,0x21, /* g macron */
+ 0xbf, 0x01,0x5e, /* S cedilla */
+ 0xc0, 0x01,0x5f, /* s cedilla */
+ 0xc1, 0x01,0x2f, /* i ogonek */ /* FIXME: not sure with this, it look like normal i */
+ 0xc2, 0x01,0x31, /* i dotless */
+ 0xc3, 0x01,0x68, /* U tilde */
+ 0xc4, 0x01,0x50, /* O dbl acute */
+ 0xc5, 0x01,0x69, /* u tilde */
+ 0xc6, 0x01,0x51, /* o dbl acute */
+ 0xc7, 0x27,0xa9, /* => */
+ 0xc8, 0x27,0xa8, /* filled => */
+ 0xc9, 0x00,0xd7, /* x */
+ 0xca, 0x00,0x5d, /* ] */
+ 0xcb, 0x26,0x0f, /* phone */
+ 0xcc, 0x01,0x0f, /* d caron */
+
+ 0xcd, 0x00,0x20, /* empty */
+
+ 0xce, 0x00,0x7e, /* ~ */
+ 0xcf, 0x00,0x5c, /* \ */
+ 0xd0, 0x00,0x5e, /* ^ */
+
+ 0xd1, 0x00,0x20, /* empty */
+
+ 0xd2, 0x00,0x7b, /* { */
+ 0xd3, 0x00,0x7c, /* | */
+ 0xd4, 0x00,0x7d, /* } */
+
+ 0xd5, 0x00,0x20, /* empty */
+
+ 0xd6, 0x01,0x63, /* t cedilla */
+
+ 0xd7, 0x00,0x20, /* empty */
+ 0xd8, 0x00,0x20, /* empty */
+ 0xd9, 0x00,0x20, /* empty */
+ 0xda, 0x00,0x20, /* empty */
+ 0xdb, 0x00,0x20, /* empty */
+ 0xdc, 0x00,0x20, /* empty */
+ 0xdd, 0x00,0x20, /* empty */
+ 0xde, 0x00,0x20, /* empty */
+ 0xdf, 0x00,0x20, /* empty */
+ 0xe0, 0x00,0x20, /* empty */
+
+ 0xe1, 0x00,0x20, /* two candles */ /* FIXME */
+
+ 0xe2, 0x00,0x20, /* empty */
+ 0xe3, 0x00,0x20, /* empty */
+ 0xe4, 0x00,0x20, /* empty */
+
+ 0xe5, 0x01,0xce, /* a caron */
+ 0xe6, 0x01,0x01, /* a macron */
+ 0xe7, 0x01,0x13, /* e macron */
+ 0xe8, 0x01,0x2b, /* i macron */
+ 0xe9, 0x01,0x4d, /* o macron */
+ 0xea, 0x01,0x6b, /* u macron */
+ 0xeb, 0x00,0x41, /* A */
+ 0xec, 0x00,0x40, /* @ */
+ 0xed, 0x00,0x20, /* some strange char :-) */ /* FIXME */
+
+ 0xee, 0x00,0x20, /* big key stroken */ /* FIXME */
+ 0xef, 0x00,0x20, /* big key */ /* FIXME */
+
+ 0xf0, 0x00,0x20, /* empty */
+
+ 0xf1, 0x00,0x31, /* 1 */
+ 0xf2, 0x00,0x21, /* bold ! */
+ 0xf3, 0x26,0x0e, /* black phone */
+ 0xf4, 0x00,0x26, /* & */
+ 0xf5, 0x23,0x7e, /* bell */
+ 0xf6, 0x26,0x6a, /* note */
+
+ 0xf7, 0x27,0x13, /* okay inv */ /* FIXME */
+ 0xf8, 0x27,0x13, /* okay */
+
+ 0xf9, 0x00,0x20, /* empty */
+
+ 0xfa, 0x00,0x20, /* key */ /* FIXME */
+
+ 0xfb, 0x00,0x20, /* empty */
+
+ 0xfc, 0x20,0xac, /* Euro */
+ 0xfd, 0x21,0x97, /* NE arrow */
+ 0xfe, 0x21,0x98, /* SE arrow */
+
+ 0xff, 0x00,0x20, /* empty */
+
+ 0x00, 0x00,0x00
+};
+
+/* This is being called from atgen */
+GSM_Error ALCATEL_ProtocolVersionReply (GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ char *str, *str2;
+/*
+ * Reply received here looks like:
+ * 1 "AT+CPROT=?"
+ * 2 "+CPROT: 0,"V1.0",1"
+ * 3 "+CPROT: 16,"V1.1",16"
+ * 4 "OK"
+ */
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ str = strstr(msg.Buffer, "\"V");
+ if (str == NULL) return ERR_UNKNOWNRESPONSE;
+ str += 2;
+ while((str2 = strstr(str, "\"V")) != NULL) str = str2 + 2;
+ if (strncmp(str, "1.0", 3) == 0) {
+ s->Phone.Data.Priv.ALCATEL.ProtocolVersion = V_1_0;
+ } else if (strncmp(str, "1.1", 3) == 0) {
+ s->Phone.Data.Priv.ALCATEL.ProtocolVersion = V_1_1;
+ } else {
+ smprintf(s, "Unknown protocol version. Please send debug log and phone info to author.\n");
+ return ERR_NOTIMPLEMENTED;
+ }
+ return ERR_NONE;
+ case AT_Reply_Error:
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+static GSM_Error ALCATEL_SetBinaryMode(GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+
+ if (Priv->Mode == ModeBinary) return ERR_NONE;
+
+ dbgprintf ("Changing to binary mode\n");
+
+ error=GSM_WaitFor (s, "AT+IFC=2,2\r", 11, 0x02, 4, ID_SetFlowControl);
+ if (error != ERR_NONE) return error;
+
+ error=GSM_WaitFor (s, "AT+CPROT=?\r", 11, 0x02, 4, ID_AlcatelProtocol);
+ if (error != ERR_NONE) return error;
+
+ if (Priv->ProtocolVersion == V_1_0) {
+ error=GSM_WaitFor (s, "AT+CPROT=16,\"V1.0\",16\r", 22, 0x00, 4, ID_AlcatelConnect);
+ } else {
+ error=GSM_WaitFor (s, "AT+CPROT=16,\"V1.1\",16\r", 22, 0x00, 4, ID_AlcatelConnect);
+ }
+
+ if (error == ERR_TIMEOUT && s->Speed != 19200) {
+ smprintf(s, "HINT: Try changing speed to 19200, it is sometimes needed for Alcatel binary mode.\n");
+ }
+
+ if (error != ERR_NONE) return error;
+
+ dbgprintf ("Changing protocol to Alcabus\n");
+
+ s->Protocol.Functions = &ALCABUSProtocol;
+ error = s->Protocol.Functions->Initialise(s);
+ if (error != ERR_NONE) {
+ s->Protocol.Functions = &ATProtocol;
+ return error;
+ }
+ s->Phone.Functions->ReplyFunctions = ALCATELReplyFunctions;
+ Priv->Mode = ModeBinary;
+ Priv->BinaryItem = 0;
+ Priv->BinaryType = 0;
+ Priv->BinaryState = StateAttached;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GoToBinaryState(GSM_StateMachine *s, GSM_Alcatel_BinaryState state, GSM_Alcatel_BinaryType type, int item) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char attach_buffer[] = {0x00, 0x00, 0x7C ,0x20};
+ unsigned char detach_buffer[] = {0x00, 0x01, 0x7C ,0x00};
+ unsigned char start_buffer[] =
+ {0x00, 0x04, 0x7C, 0x80, /* 4 byte database id follows */
+ 0x12, 0x34, 0x56, 0x78};
+ unsigned char end_buffer[] =
+ {0x00, 0x04, 0x7C, 0x82,
+ 0x00, /* type */
+ 0x00, 0x00, 0x00, 0x00}; /* TimeStamp */
+ unsigned char close_buffer[] =
+ {0x00, 0x04,
+ 0x00, /*type */
+ 0x23, 0x01};
+ unsigned char select1_buffer[] =
+ {0x00, 0x00,
+ 0x00, /*type */
+ 0x20};
+ unsigned char select2_buffer[] =
+ {0x00, 0x04,
+ 0x00, /*type */
+ 0x22, 0x01, 0x00};
+ unsigned char begin_buffer[] =
+ {0x00, 0x04, 0x7C, 0x81,
+ 0x00, /*type */
+ 0x00, 0x85, 0x00};
+ unsigned char commit_buffer[] =
+ {0x00, 0x04,
+ 0x00, /*type */
+ 0x20, 0x01};
+
+ smprintf(s, "Alcatel state switcher: %d -> %d, %d -> %d, %d -> %d\n", Priv->BinaryState, state, Priv->BinaryType, type, Priv->BinaryItem, item);
+ error = ALCATEL_SetBinaryMode(s);
+ if (error != ERR_NONE) return error;
+
+ /* Do we need to do anything? */
+ if ((state == Priv->BinaryState) && (type == Priv->BinaryType) && (item == Priv->BinaryItem)) return ERR_NONE;
+
+ /* We're editing, but the next state is not the same. so commit editing */
+ if (Priv->BinaryState == StateEdit) {
+ /* Something has changed, we will have to reread fields! */
+ Priv->CurrentFieldsItem = -1;
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ commit_buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ commit_buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ commit_buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+ dbgprintf ("Commiting edited record\n");
+ error=GSM_WaitFor (s, commit_buffer, 5, 0x02, ALCATEL_TIMEOUT, ID_AlcatelCommit);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelCommit2);
+ if (error != ERR_NONE) return error;
+ Priv->BinaryState = StateSession;
+ Priv->BinaryItem = 0;
+ }
+
+ /* Do we want to edit something of same type? */
+ if ((state == StateEdit) && (type == Priv->BinaryType)) {
+ /* Edit state doesn't need any switching, it is needed only for
+ * indication that e have to commit record before we switch to other
+ * mode.
+ */
+ Priv->BinaryState = StateEdit;
+ Priv->BinaryItem = item;
+ return ERR_NONE;
+ }
+
+ /* Now we can be only in Attached or Session state, so if states and types matches, just keep them as they are */
+ if ((state == Priv->BinaryState) && (type == Priv->BinaryType)) {
+ return ERR_NONE;
+ }
+
+ /* Do we need to close session? */
+ if (Priv->BinaryState == StateSession) {
+ dbgprintf ("Ending session\n");
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ end_buffer[4] = ALCATEL_BEGIN_SYNC_CALENDAR;
+ break;
+ case TypeContacts:
+ end_buffer[4] = ALCATEL_BEGIN_SYNC_CONTACTS;
+ break;
+ case TypeToDo:
+ end_buffer[4] = ALCATEL_BEGIN_SYNC_TODO;
+ break;
+ }
+ error=GSM_WaitFor (s, end_buffer, 9, 0x02, ALCATEL_TIMEOUT, ID_AlcatelEnd);
+ if (error != ERR_NONE) return error;
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ close_buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ close_buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ close_buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+ dbgprintf ("Closing session\n");
+ error=GSM_WaitFor (s, close_buffer, 5, 0x02, ALCATEL_TIMEOUT, ID_AlcatelClose);
+ if (error != ERR_NONE) return error;
+
+ dbgprintf ("Detaching binary mode\n");
+ GSM_WaitFor (s, detach_buffer, 4, 0x02, ALCATEL_TIMEOUT, ID_AlcatelDetach);
+
+ Priv->BinaryState = StateAttached;
+ Priv->BinaryType = 0;
+ }
+
+ /* Do we need to open session? */
+ if (state == StateSession || state == StateEdit) {
+ dbgprintf ("Starting session for %s\n",
+ (type == TypeCalendar ? "Calendar" :
+ (type == TypeToDo ? "Todo" :
+ (type == TypeContacts ? "Contacts" :
+ "Unknown!"))));
+ /* Fill up buffers */
+ switch (type) {
+ case TypeCalendar:
+ select1_buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ select2_buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ begin_buffer[4] = ALCATEL_BEGIN_SYNC_CALENDAR;
+ break;
+ case TypeContacts:
+ select1_buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ select2_buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ begin_buffer[4] = ALCATEL_BEGIN_SYNC_CONTACTS;
+ break;
+ case TypeToDo:
+ select1_buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ select2_buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ begin_buffer[4] = ALCATEL_BEGIN_SYNC_TODO;
+ break;
+ }
+ dbgprintf ("Attaching in binary mode\n");
+
+ /* Communicate */
+ error=GSM_WaitFor (s, attach_buffer, 4, 0x02, ALCATEL_TIMEOUT, ID_AlcatelAttach);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Start session\n");
+ error=GSM_WaitFor (s, start_buffer, 8, 0x02, ALCATEL_TIMEOUT, ID_AlcatelStart);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Select type\n");
+ error=GSM_WaitFor (s, select1_buffer, 4, 0x02, ALCATEL_TIMEOUT, ID_AlcatelSelect1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, select2_buffer, 6, 0x02, ALCATEL_TIMEOUT, ID_AlcatelSelect2);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelSelect3);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Begin transfer\n");
+ error=GSM_WaitFor (s, begin_buffer, 8, 0x02, ALCATEL_TIMEOUT, ID_AlcatelBegin1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelBegin2);
+ if (error != ERR_NONE) return error;
+
+ Priv->BinaryState = StateSession;
+ Priv->BinaryType = type;
+ /* Do we want to edit something of same type? */
+ if ((state == StateEdit) && (type == Priv->BinaryType)) {
+ Priv->BinaryState = StateEdit;
+ Priv->BinaryItem = item;
+ return ERR_NONE;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_SetATMode(GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+
+ if (Priv->Mode == ModeAT) return ERR_NONE;
+
+ error = ALCATEL_GoToBinaryState(s, StateAttached, 0, 0);
+ if (error != ERR_NONE) return error;
+
+ error = s->Protocol.Functions->Terminate(s);
+ if (error != ERR_NONE) return error;
+
+ dbgprintf ("Changing protocol to AT\n");
+ s->Protocol.Functions = &ATProtocol;
+ s->Phone.Functions->ReplyFunctions = ATGENReplyFunctions;
+ Priv->Mode = ModeAT;
+
+ my_sleep(100);
+
+ /* In case we don't send AT command short after closing binary mode,
+ * phone takes VERY long to react next time. The error code in
+ * intetionally ignored.
+ */
+ GSM_WaitFor (s, "AT\r", 3, 0x00, 0, ID_IncomingFrame);
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_Initialise(GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+
+ Priv->Mode = ModeAT;
+
+ Priv->CalendarItems = NULL;
+ Priv->ContactsItems = NULL;
+ Priv->ToDoItems = NULL;
+ Priv->CalendarItemsCount = 0;
+ Priv->ToDoItemsCount = 0;
+ Priv->ContactsItemsCount = 0;
+ Priv->CurrentFields[0] = 0;
+ Priv->CurrentFieldsCount = 0;
+ Priv->CurrentFieldsItem = 0;
+ Priv->CurrentFieldsType = 0;
+ Priv->ProtocolVersion = V_1_0;
+ Priv->CurrentFieldsItem = -1;
+
+ Priv->CurrentCategoriesCount = 0;
+ Priv->CurrentCategoriesType = 0;
+
+ s->Protocol.Functions = &ATProtocol;
+ s->Phone.Functions->ReplyFunctions = ATGENReplyFunctions;
+
+ if (ATGEN_Initialise(s) != ERR_NONE || GSM_WaitFor (s, "AT\r", 3, 0x00, 2, ID_IncomingFrame) != ERR_NONE) {
+ smprintf(s,"AT initialisation failed, trying to stop binary mode...\n");
+ s->Protocol.Functions = &ALCABUSProtocol;
+ error = s->Protocol.Functions->Terminate(s);
+ s->Protocol.Functions = &ATProtocol;
+
+ error = ATGEN_Initialise(s);
+ if (error != ERR_NONE) return error;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_Terminate(GSM_StateMachine *s)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ free(Priv->CalendarItems);
+ free(Priv->ContactsItems);
+ free(Priv->ToDoItems);
+ error = ALCATEL_SetATMode(s);
+ return ATGEN_Terminate(s);
+}
+
+/* finds whether id is set in the phone */
+static GSM_Error ALCATEL_IsIdAvailable(GSM_StateMachine *s, int id) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ if (id > ALCATEL_MAX_LOCATION) return ERR_INVALIDLOCATION;
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ Priv->CurrentList = &(Priv->CalendarItems);
+ Priv->CurrentCount = &(Priv->CalendarItemsCount);
+ break;
+ case TypeContacts:
+ Priv->CurrentList = &(Priv->ContactsItems);
+ Priv->CurrentCount = &(Priv->ContactsItemsCount);
+ break;
+ case TypeToDo:
+ Priv->CurrentList = &(Priv->ToDoItems);
+ Priv->CurrentCount = &(Priv->ToDoItemsCount);
+ break;
+ }
+
+ for (i=0; i<*Priv->CurrentCount; i++) {
+ if ((*Priv->CurrentList)[i] == id) return ERR_NONE;
+ }
+
+ return ERR_EMPTY;
+}
+
+/* finds next id that is available in the phone */
+static GSM_Error ALCATEL_GetNextId(GSM_StateMachine *s, int *id) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i = 0;
+ int next = ALCATEL_MAX_LOCATION;
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ Priv->CurrentList = &(Priv->CalendarItems);
+ Priv->CurrentCount = &(Priv->CalendarItemsCount);
+ break;
+ case TypeContacts:
+ Priv->CurrentList = &(Priv->ContactsItems);
+ Priv->CurrentCount = &(Priv->ContactsItemsCount);
+ break;
+ case TypeToDo:
+ Priv->CurrentList = &(Priv->ToDoItems);
+ Priv->CurrentCount = &(Priv->ToDoItemsCount);
+ break;
+ }
+
+ for (i=0; i<*Priv->CurrentCount; i++) {
+ if (((*Priv->CurrentList)[i] > *id) && ((*Priv->CurrentList)[i] < next )) {
+ next = (*Priv->CurrentList)[i];
+ }
+ }
+ if (next == ALCATEL_MAX_LOCATION) {
+ return ERR_EMPTY;
+ } else {
+ *id = next;
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error ALCATEL_ReplyGetIds(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int count,i,pos;
+
+ count = msg.Buffer[10];
+ *Priv->CurrentCount += count;
+
+ *Priv->CurrentList = (int *)realloc(*Priv->CurrentList, (*Priv->CurrentCount + 1)* sizeof(int));
+ if (*Priv->CurrentList == NULL) return ERR_MOREMEMORY;
+
+ for (i = 0; i < count; i++) {
+ pos = 11 + (4 * i);
+ (*Priv->CurrentList)[*Priv->CurrentCount - count + i] = msg.Buffer[pos + 3] +
+ (msg.Buffer[pos + 2] << 8) +
+ (msg.Buffer[pos + 1] << 16) +
+ (msg.Buffer[pos] << 24);
+ }
+ (*Priv->CurrentList)[*Priv->CurrentCount] = 0;
+
+ /* If last byte is 0, then we transmitted all items */
+ Priv->TransferCompleted = msg.Buffer[4 + msg.Buffer[4]] == 0;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetAvailableIds(GSM_StateMachine *s, bool refresh)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ int i;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00, /*type */
+ 0x2F, 0x01};
+
+ if (Priv->BinaryState != StateSession) return ERR_UNKNOWN;
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ Priv->CurrentList = &(Priv->CalendarItems);
+ Priv->CurrentCount = &(Priv->CalendarItemsCount);
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ Priv->CurrentList = &(Priv->ContactsItems);
+ Priv->CurrentCount = &(Priv->ContactsItemsCount);
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ Priv->CurrentList = &(Priv->ToDoItems);
+ Priv->CurrentCount = &(Priv->ToDoItemsCount);
+ break;
+ }
+
+ if (*Priv->CurrentList != NULL) {
+ if (!refresh) return ERR_NONE;
+ free(*Priv->CurrentList);
+ *Priv->CurrentList = NULL;
+ }
+ smprintf(s,"Reading items list\n");
+
+ *Priv->CurrentCount = 0;
+ Priv->TransferCompleted = false;
+
+ error=GSM_WaitFor (s, buffer, 5, 0x02, ALCATEL_TIMEOUT, ID_AlcatelGetIds1);
+ if (error != ERR_NONE) return error;
+
+ while (!Priv->TransferCompleted) {
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelGetIds2);
+ if (error != ERR_NONE) return error;
+ }
+
+ i = 0;
+ smprintf(s,"Received %d ids: ", *Priv->CurrentCount);
+ for (i=0; i < *Priv->CurrentCount; i++) {
+ smprintf(s,"%x ", (*Priv->CurrentList)[i]);
+ }
+ smprintf(s,"\n");
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_ReplyGetFields(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ if (msg.Buffer[14] > GSM_PHONEBOOK_ENTRIES) {
+ smprintf(s, "WARNING: Field list truncated, you should increase GSM_PHONEBOOK_ENTRIES to at least %d\n", msg.Buffer[14]);
+ Priv->CurrentFieldsCount = GSM_PHONEBOOK_ENTRIES;
+ } else {
+ Priv->CurrentFieldsCount = msg.Buffer[14];
+ }
+
+ Priv->CurrentFields[Priv->CurrentFieldsCount] = 0;
+
+ for (i = 0; i < Priv->CurrentFieldsCount; i++) {
+ Priv->CurrentFields[i] = msg.Buffer[15 + i];
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetFields(GSM_StateMachine *s, int id) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ int i;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x30, 0x01,
+ 0x00, 0x00, 0x00, 0x00}; /* item */
+
+ if (Priv->BinaryState != StateSession) return ERR_UNKNOWN;
+ if ((Priv->CurrentFieldsItem == id) && (Priv->CurrentFieldsType == Priv->BinaryType)) return ERR_NONE;
+
+ smprintf(s,"Reading item fields (%d)\n", id);
+
+ buffer[5] = (id >> 24);
+ buffer[6] = ((id >> 16) & 0xff);
+ buffer[7] = ((id >> 8) & 0xff);
+ buffer[8] = (id & 0xff);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+
+ Priv->CurrentFieldsItem = id;
+ Priv->CurrentFieldsType = Priv->BinaryType;
+
+ error=GSM_WaitFor (s, buffer, 9, 0x02, ALCATEL_TIMEOUT, ID_AlcatelGetFields1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelGetFields2);
+ if (error != ERR_NONE) return error;
+
+ i = 0;
+ smprintf(s,"Received %d fields: ", Priv->CurrentFieldsCount);
+ for (i=0; i < Priv->CurrentFieldsCount; i++) {
+ smprintf(s,"%x ", Priv->CurrentFields[i]);
+ }
+ smprintf(s,"\n");
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_ReplyGetFieldValue(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ unsigned char *buffer = &(msg.Buffer[16]);
+
+ if (buffer[1] == 0x05 && buffer[2] == 0x67) {
+ /* date */
+ Priv->ReturnType = Alcatel_date;
+ Priv->ReturnDateTime.Day = buffer[4];
+ Priv->ReturnDateTime.Month = buffer[5];
+ Priv->ReturnDateTime.Year = buffer[7] + (buffer[6] << 8);
+ Priv->ReturnDateTime.Timezone = 0; /* FIXME: how to acquire this? */
+
+ Priv->ReturnDateTime.Hour = 0;
+ Priv->ReturnDateTime.Minute = 0;
+ Priv->ReturnDateTime.Second = 0;
+ } else if (buffer[1] == 0x06 && buffer[2] == 0x68) {
+ /* time */
+ Priv->ReturnType = Alcatel_time;
+ Priv->ReturnDateTime.Hour = buffer[4];
+ Priv->ReturnDateTime.Minute = buffer[5];
+ Priv->ReturnDateTime.Second = buffer[6];
+
+ Priv->ReturnDateTime.Day = 0;
+ Priv->ReturnDateTime.Month = 0;
+ Priv->ReturnDateTime.Year = 0;
+ Priv->ReturnDateTime.Timezone = 0;
+ } else if (buffer[1] == 0x08 && buffer[2] == 0x3C) {
+ /* string */
+ Priv->ReturnType = Alcatel_string;
+ if (GSM_PHONEBOOK_TEXT_LENGTH < buffer[3])
+ smprintf(s, "WARNING: Text truncated, you should increase GSM_PHONEBOOK_TEXT_LENGTH to at least %d\n", buffer[3] + 1);
+ if (Priv->ProtocolVersion == V_1_0) {
+ DecodeDefault( Priv->ReturnString, buffer + 4, MIN(GSM_PHONEBOOK_TEXT_LENGTH, buffer[3]), false, GSM_AlcatelAlphabet);
+ } else if(Priv->ProtocolVersion == V_1_1 && (buffer[4] & 0x80)) {
+ memcpy(Priv->ReturnString, buffer + 5, buffer[3]);
+ Priv->ReturnString[buffer[3] + 1] = 0;
+ Priv->ReturnString[buffer[3] + 2] = 0;
+ ReverseUnicodeString(Priv->ReturnString);
+ } else {
+ DecodeDefault( Priv->ReturnString, buffer + 4, MIN(GSM_PHONEBOOK_TEXT_LENGTH, buffer[3]), false, GSM_AlcatelAlphabet);
+ }
+ } else if (buffer[1] == 0x07 && buffer[2] == 0x3C) {
+ /* phone */
+ Priv->ReturnType = Alcatel_phone;
+ if (GSM_PHONEBOOK_TEXT_LENGTH < buffer[3])
+ smprintf(s, "WARNING: Text truncated, you should increase GSM_PHONEBOOK_TEXT_LENGTH to at least %d\n", buffer[3] + 1);
+ if (Priv->ProtocolVersion == V_1_0) {
+ DecodeDefault( Priv->ReturnString, buffer + 4, MIN(GSM_PHONEBOOK_TEXT_LENGTH, buffer[3]), false, GSM_AlcatelAlphabet);
+ } else if(Priv->ProtocolVersion == V_1_1 && (buffer[4] & 0x80)) {
+ memcpy(Priv->ReturnString, buffer + 5, buffer[3]);
+ Priv->ReturnString[buffer[3] + 1] = 0;
+ Priv->ReturnString[buffer[3] + 2] = 0;
+ ReverseUnicodeString(Priv->ReturnString);
+ } else {
+ DecodeDefault( Priv->ReturnString, buffer + 4, MIN(GSM_PHONEBOOK_TEXT_LENGTH, buffer[3]), false, GSM_AlcatelAlphabet);
+ }
+ } else if (buffer[1] == 0x03 && buffer[2] == 0x3B) {
+ /* boolean */
+ Priv->ReturnType = Alcatel_bool;
+ Priv->ReturnInt = buffer[3];
+ } else if (buffer[1] == 0x02 && buffer[2] == 0x3A) {
+ /* integer */
+ Priv->ReturnType = Alcatel_int;
+ Priv->ReturnInt = buffer[6] + (buffer[5] << 8) + (buffer[4] << 16) + (buffer[3] << 24);
+ } else if (buffer[1] == 0x04 && buffer[2] == 0x38) {
+ /* enumeration */
+ Priv->ReturnType = Alcatel_enum;
+ Priv->ReturnInt = buffer[3];
+ } else if (buffer[1] == 0x00 && buffer[2] == 0x38) {
+ /* byte */
+ Priv->ReturnType = Alcatel_byte;
+ Priv->ReturnInt = buffer[3];
+ } else {
+ smprintf(s, "WARNING: Uknown data type received (%02X,%02X)\n", buffer[1], buffer[2]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetFieldValue(GSM_StateMachine *s, int id, int field)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x1f, 0x01,
+ 0x00, 0x00, 0x00, 0x00, /* here follows 4byte id */
+ 0x00}; /* field */
+
+ smprintf(s,"Reading item value (%08x.%02x)\n", id, field);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+
+ buffer[5] = (id >> 24);
+ buffer[6] = ((id >> 16) & 0xff);
+ buffer[7] = ((id >> 8) & 0xff);
+ buffer[8] = (id & 0xff);
+ buffer[9] = (field & 0xff);
+
+ error=GSM_WaitFor (s, buffer, 10, 0x02, ALCATEL_TIMEOUT, ID_AlcatelGetFieldValue1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelGetFieldValue2);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_ReplyGetCategories(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ /* Did we get any category? */
+ if (msg.Buffer[4] == 6) {
+ Priv->CurrentCategoriesCount = 0;
+ return ERR_NONE;
+ }
+ if (msg.Buffer[12] > ALCATEL_MAX_CATEGORIES) {
+ smprintf(s, "WARNING: Field list truncated, you should increase ALCATEL_MAX_CATEGORIES to at least %d\n", msg.Buffer[12]);
+ Priv->CurrentCategoriesCount = ALCATEL_MAX_CATEGORIES;
+ } else {
+ Priv->CurrentCategoriesCount = msg.Buffer[12];
+ }
+
+ for (i = 0; i < Priv->CurrentCategoriesCount; i++) {
+ Priv->CurrentCategories[i] = msg.Buffer[13 + i];
+ Priv->CurrentCategoriesCache[i][0] = '\000';
+ Priv->CurrentCategoriesCache[i][1] = '\000';
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetAvailableCategoryIds(GSM_StateMachine *s) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ int i;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00 /*type */,
+ 0x0b,
+ 0x00 /* list */};
+
+ if (Priv->BinaryState != StateSession) return ERR_UNKNOWN;
+ if (Priv->CurrentCategoriesType == Priv->BinaryType) return ERR_NONE;
+ switch (Priv->BinaryType) {
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ buffer[4] = ALCATEL_LIST_CONTACTS_CAT;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ buffer[4] = ALCATEL_LIST_TODO_CAT;
+ break;
+ default:
+ return ERR_NOTSUPPORTED;
+ }
+
+ Priv->CurrentCategoriesType = Priv->BinaryType;
+
+ smprintf(s,"Reading category list\n");
+
+ error=GSM_WaitFor (s, buffer, 5, 0x02, ALCATEL_TIMEOUT, ID_AlcatelGetCategories1);
+ if (error != ERR_NONE) return error;
+
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelGetCategories2);
+ if (error != ERR_NONE) return error;
+
+ i = 0;
+ smprintf(s,"Received %d ids: ", Priv->CurrentCategoriesCount);
+ for (i=0; i < Priv->CurrentCategoriesCount; i++) {
+ smprintf(s,"%i ", Priv->CurrentCategories[i]);
+ }
+ smprintf(s,"\n");
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_IsCategoryIdAvailable(GSM_StateMachine *s, int id) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i = 0;
+
+ if (Priv->CurrentCategoriesType != Priv->BinaryType) return ERR_UNKNOWN;
+
+ for (i = 0; i< Priv->CurrentCategoriesCount; i++) {
+ if (Priv->CurrentCategories[i] == id) return ERR_NONE;
+ }
+ return ERR_EMPTY;
+}
+
+static GSM_Error ALCATEL_ReplyAddCategoryText(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ Priv->ReturnInt = msg.Buffer[12];
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_AddCategoryText(GSM_StateMachine *s, const unsigned char *str) {
+ unsigned char buffer[200] = {0x00, 0x04, 0x00 /*type*/, 0x0d, 0x00 /*list*/, 0x0b };
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ int len;
+
+ smprintf(s,"Creating category\n");
+ len = UnicodeLength(str);
+ EncodeDefault(buffer + 8, str, &len, true, GSM_AlcatelAlphabet);
+ buffer[6] = len + 1;
+ buffer[7] = len;
+
+ switch (Priv->BinaryType) {
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ buffer[4] = ALCATEL_LIST_CONTACTS_CAT;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ buffer[4] = ALCATEL_LIST_TODO_CAT;
+ break;
+ default:
+ return ERR_NOTSUPPORTED;
+ }
+
+ error=GSM_WaitFor (s, buffer, 8 + len, 0x02, ALCATEL_TIMEOUT, ID_AlcatelAddCategoryText1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelAddCategoryText2);
+ if (error != ERR_NONE) return error;
+
+ /* Refresh list */
+ Priv->CurrentCategoriesType = 0;
+ return ALCATEL_GetAvailableCategoryIds(s);
+}
+
+static GSM_Error ALCATEL_ReplyGetCategoryText(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int len;
+
+ len = msg.Buffer[14];
+ if (len > GSM_MAX_CATEGORY_NAME_LENGTH) {
+ smprintf(s, "WARNING: Category name truncated, you should increase GSM_MAX_CATEGORY_NAME_LENGTH to at least %d\n", len);
+ }
+ if (Priv->ProtocolVersion == V_1_0) {
+ DecodeDefault( Priv->ReturnString, msg.Buffer + 15, MIN(GSM_MAX_CATEGORY_NAME_LENGTH, len), false, GSM_AlcatelAlphabet);
+ } else if(Priv->ProtocolVersion == V_1_1 && (msg.Buffer[15] & 0x80)) {
+ memcpy(Priv->ReturnString, msg.Buffer + 16, len);
+ Priv->ReturnString[len + 1] = 0;
+ Priv->ReturnString[len + 2] = 0;
+ ReverseUnicodeString(Priv->ReturnString);
+ } else {
+ DecodeDefault( Priv->ReturnString, msg.Buffer + 15, MIN(GSM_MAX_CATEGORY_NAME_LENGTH, len), false, GSM_AlcatelAlphabet);
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetCategoryText(GSM_StateMachine *s, int id) {
+ unsigned char buffer[] = {0x00, 0x04, 0x00 /*type*/, 0x0c, 0x00 /*list*/, 0x0A, 0x01, 0x00 /*item*/ };
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+
+ if (Priv->CurrentCategoriesCache[id][0] != '\000' || Priv->CurrentCategoriesCache[id][1] != '\000') {
+ CopyUnicodeString(Priv->ReturnString, Priv->CurrentCategoriesCache[id]);
+ return ERR_NONE;
+ }
+
+ smprintf(s,"Reading category %d\n", id);
+
+ switch (Priv->BinaryType) {
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ buffer[4] = ALCATEL_LIST_CONTACTS_CAT;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ buffer[4] = ALCATEL_LIST_TODO_CAT;
+ break;
+ default:
+ return ERR_NOTSUPPORTED;
+ }
+
+ buffer[7] = (id & 0xff);
+
+ error=GSM_WaitFor (s, buffer, 8, 0x02, ALCATEL_TIMEOUT, ID_AlcatelGetCategoryText1);
+ if (error != ERR_NONE) return error;
+ error=GSM_WaitFor (s, 0, 0, 0x00, ALCATEL_TIMEOUT, ID_AlcatelGetCategoryText2);
+ if (error != ERR_NONE) return error;
+
+ CopyUnicodeString(Priv->CurrentCategoriesCache[id], Priv->ReturnString);
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_DeleteField(GSM_StateMachine *s, int id, int field) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x26, 0x01,
+ 0x00, 0x00, 0x00, 0x00, /* here follows 4byte id */
+ 0x65, 0x01,
+ 0x00, /* field */
+ 0x01};
+
+ smprintf(s,"Deleting field (%08x.%02x)\n", id, field);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+
+ buffer[5] = (id >> 24);
+ buffer[6] = ((id >> 16) & 0xff);
+ buffer[7] = ((id >> 8) & 0xff);
+ buffer[8] = (id & 0xff);
+ buffer[11] = (field & 0xff);
+
+ error=GSM_WaitFor (s, buffer, 13, 0x02, ALCATEL_TIMEOUT, ID_AlcatelDeleteField);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_DeleteItem(GSM_StateMachine *s, int id) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char buffer[] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x27, 0x01,
+ 0x00, 0x00, 0x00, 0x00, /* here follows 4byte id */
+ 0x42};
+
+ smprintf(s,"Deleting item (%08x)\n", id);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+
+ buffer[5] = (id >> 24);
+ buffer[6] = ((id >> 16) & 0xff);
+ buffer[7] = ((id >> 8) & 0xff);
+ buffer[8] = (id & 0xff);
+
+ error=GSM_WaitFor (s, buffer, 10, 0x02, ALCATEL_TIMEOUT, ID_AlcatelDeleteItem1);
+ if (error != ERR_NONE) return error;
+
+ error=GSM_WaitFor (s, 0, 0, 0x0, ALCATEL_TIMEOUT, ID_AlcatelDeleteItem2);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_ReplyDeleteItem(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ if (msg.Buffer[8] != 0x25) return ERR_UNKNOWNRESPONSE;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_BuildWriteBuffer(unsigned char * buffer, GSM_Alcatel_FieldType type, int field, void *data) {
+ int len;
+
+ buffer[1] = field & 0xff;
+
+ switch(type) {
+ case Alcatel_date:
+ if (!CheckDate((GSM_DateTime *)data)) return ERR_INVALIDDATETIME;
+
+ buffer[3] = 0x05;
+ buffer[4] = 0x67;
+
+ buffer[0] = 0x09;
+ buffer[5] = 0x04;
+ buffer[6] = ((GSM_DateTime *)data)->Day & 0xff;
+ buffer[7] = ((GSM_DateTime *)data)->Month & 0xff;
+ buffer[8] = ((GSM_DateTime *)data)->Year >> 8;
+ buffer[9] = ((GSM_DateTime *)data)->Year & 0xff;
+ buffer[10] = 0x00;
+ break;
+ case Alcatel_time:
+ if (!CheckTime((GSM_DateTime *)data)) return ERR_INVALIDDATETIME;
+
+ buffer[3] = 0x06;
+ buffer[4] = 0x68;
+
+ buffer[0] = 0x08;
+ buffer[5] = 0x03;
+ buffer[6] = ((GSM_DateTime *)data)->Hour & 0xff;
+ buffer[7] = ((GSM_DateTime *)data)->Minute & 0xff;
+ buffer[8] = ((GSM_DateTime *)data)->Second & 0xff;
+ buffer[9] = 0x00;
+ break;
+ case Alcatel_string:
+ buffer[3] = 0x08;
+ buffer[4] = 0x3c;
+
+ len = MIN(UnicodeLength((char *)data),62);
+ EncodeDefault(buffer + 6, (char *)data, &len, true, GSM_AlcatelAlphabet);
+ buffer[5] = len;
+ buffer[0] = 5 + len;
+ buffer[6 + len] = 0x00;
+ break;
+ case Alcatel_phone:
+ buffer[3] = 0x07;
+ buffer[4] = 0x3c;
+
+ len = MIN(UnicodeLength((char *)data),50);
+ EncodeDefault(buffer + 6, (char *)data, &len, true, GSM_AlcatelAlphabet);
+ buffer[5] = len;
+ buffer[0] = 5 + len;
+ buffer[6 + len] = 0x00;
+ break;
+ case Alcatel_enum:
+ buffer[3] = 0x04;
+ buffer[4] = 0x38;
+
+ buffer[0] = 0x05;
+ buffer[5] = *(int *)data & 0xff;
+ buffer[6] = 0x00;
+ break;
+ case Alcatel_bool:
+ buffer[3] = 0x03;
+ buffer[4] = 0x3b;
+
+ buffer[0] = 0x05;
+ buffer[5] = *(int *)data & 0xff;
+ buffer[6] = 0x00;
+ break;
+ case Alcatel_int:
+ buffer[3] = 0x02;
+ buffer[4] = 0x3a;
+
+ buffer[0] = 0x08;
+ buffer[5] = *(unsigned int *)data >> 24;
+ buffer[6] = (*(unsigned int *)data >> 16) & 0xff;
+ buffer[7] = (*(unsigned int *)data >> 8) & 0xff;
+ buffer[8] = *(unsigned int *)data & 0xff;
+ buffer[9] = 0x00;
+ break;
+ case Alcatel_byte:
+ buffer[3] = 0x00;
+ buffer[4] = 0x38;
+
+ buffer[0] = 0x05;
+ buffer[5] = *(int *)data & 0xff;
+ buffer[6] = 0x00;
+ break;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_CreateField(GSM_StateMachine *s, GSM_Alcatel_FieldType type, int field, void *data) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char buffer[200] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x25, 0x01, 0x65,
+ 0x00, /* length of remaining part */
+ 0x00, /* field */
+ 0x37}; /* data follows here */
+
+ smprintf(s,"Creating field (%02x)\n", field);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+ error = ALCATEL_BuildWriteBuffer(buffer + 6, type, field, data);
+ if (error != ERR_NONE) return error;
+
+ error = GSM_WaitFor (s, buffer, 8 + buffer[6], 0x02, ALCATEL_TIMEOUT, ID_AlcatelCreateField);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_UpdateField(GSM_StateMachine *s, GSM_Alcatel_FieldType type, int id, int field, void *data) {
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+ unsigned char buffer[200] =
+ {0x00, 0x04,
+ 0x00, /* type */
+ 0x26, 0x01,
+ 0x00, 0x00, 0x00, 0x00, /* id */
+ 0x65,
+ 0x00, /* length of remaining part */
+ 0x00, /* field */
+ 0x37}; /* data follows here */
+
+ smprintf(s,"Updating field (%08x.%02x)\n", id, field);
+
+ buffer[5] = (id >> 24);
+ buffer[6] = ((id >> 16) & 0xff);
+ buffer[7] = ((id >> 8) & 0xff);
+ buffer[8] = (id & 0xff);
+
+ switch (Priv->BinaryType) {
+ case TypeCalendar:
+ buffer[2] = ALCATEL_SYNC_TYPE_CALENDAR;
+ break;
+ case TypeContacts:
+ buffer[2] = ALCATEL_SYNC_TYPE_CONTACTS;
+ break;
+ case TypeToDo:
+ buffer[2] = ALCATEL_SYNC_TYPE_TODO;
+ break;
+ }
+ error = ALCATEL_BuildWriteBuffer(buffer + 10, type, field, data);
+ if (error != ERR_NONE) return error;
+
+ error = GSM_WaitFor (s, buffer, 12 + buffer[10], 0x02, ALCATEL_TIMEOUT, ID_AlcatelUpdateField);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetManufacturer(GSM_StateMachine *s)
+{
+ strcpy(s->Phone.Data.Manufacturer, "Alcatel");
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetIMEI (GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetIMEI(s);
+}
+
+static GSM_Error ALCATEL_GetFirmware(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetFirmware(s);
+}
+
+static GSM_Error ALCATEL_GetModel(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetModel(s);
+}
+
+static GSM_Error ALCATEL_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetDateTime(s, date_time);
+}
+
+static GSM_Error ALCATEL_GetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+ int j = 0;
+
+ if (entry->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, entry->Location))!= ERR_NONE) {
+ entry->EntriesNum = 0;
+ return error;
+ }
+ if ((error = ALCATEL_GetFields(s, entry->Location))!= ERR_NONE) return error;
+
+ entry->EntriesNum = Priv->CurrentFieldsCount;
+
+ for (i=0; i<Priv->CurrentFieldsCount; i++) {
+ if ((error = ALCATEL_GetFieldValue(s, entry->Location, Priv->CurrentFields[i]))!= ERR_NONE) return error;
+ entry->Entries[i].VoiceTag = 0;
+ entry->Entries[i].SMSList[0] = 0;
+ switch (Priv->CurrentFields[i]) {
+ case 0:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 0, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_LastName;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 1:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 1, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_FirstName;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 2:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 2, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Company;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 3:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 3, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_JobTitle;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 4:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 4, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Note;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 5:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 5, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Category;
+ entry->Entries[i - j].Number = Priv->ReturnInt;
+ break;
+ case 6:
+ if (Priv->ReturnType != Alcatel_bool) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 6, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Private;
+ entry->Entries[i - j].Number = Priv->ReturnInt;
+ break;
+ case 7:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 7, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Work;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 8:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 8, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_General;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 9:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 9, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Fax;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 10:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 10, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Other;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 11:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 11, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Pager;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 12:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 12, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Mobile;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 13:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 13, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Number_Home;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 14:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 14, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Email;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 15:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 15, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Email2;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 16:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 16, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_StreetAddress;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 17:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 17, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_City;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 18:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 18, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_State;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 19:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 19, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Zip;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 20:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 20, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Country;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 21:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 21, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Custom1;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 22:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 22, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Custom2;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 23:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 23, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Custom3;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 24:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 24, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ entry->Entries[i - j].EntryType = PBK_Text_Custom4;
+ CopyUnicodeString(entry->Entries[i - j].Text, Priv->ReturnString);
+ break;
+ case 25:
+ if (Priv->ReturnType != Alcatel_int) {
+ smprintf(s,"WARNING: Received unexpected type %02X for field 25, ignoring\n", Priv->ReturnType);
+ entry->EntriesNum--;
+ j++;
+ break;
+ }
+ if (Priv->ReturnInt != 0) {
+ entry->Entries[i - j].EntryType = PBK_PictureID;
+ entry->Entries[i - j].Number = Priv->ReturnInt;
+ } else {
+ entry->EntriesNum--;
+ j++;
+ }
+ break;
+ default:
+ entry->EntriesNum--;
+ j++;
+ smprintf(s,"WARNING: Received unknown field %02X, ignoring. Type = %02X. Value = ", Priv->CurrentFields[i], Priv->ReturnType);
+ switch (Priv->ReturnType) {
+ case Alcatel_date:
+ smprintf(s, "%d.%d.%d", Priv->ReturnDateTime.Day, Priv->ReturnDateTime.Month, Priv->ReturnDateTime.Year);
+ break;
+ case Alcatel_time:
+ smprintf(s, "%d:%d:%d", Priv->ReturnDateTime.Hour, Priv->ReturnDateTime.Minute, Priv->ReturnDateTime.Second);
+ break;
+ case Alcatel_string:
+ case Alcatel_phone:
+ smprintf(s, "\"%s\"",DecodeUnicodeString(Priv->ReturnString));
+ break;
+ case Alcatel_enum:
+ case Alcatel_bool:
+ case Alcatel_int:
+ case Alcatel_byte:
+ smprintf(s, "%d", Priv->ReturnInt);
+ break;
+ }
+ smprintf(s,"\n");
+ }
+ }
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetMemory(s, entry);
+ }
+}
+
+static GSM_Error ALCATEL_GetNextMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry, bool start)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if (entry->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if (Priv->ContactsItemsCount == 0) return ERR_EMPTY;
+
+ if (start) entry->Location = 0;
+ if ((error = ALCATEL_GetNextId(s, &(entry->Location))) != ERR_NONE) return error;
+
+ return ALCATEL_GetMemory(s, entry);
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetNextMemory(s, entry, start);
+ }
+}
+
+static GSM_Error ALCATEL_AddMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int NamePosition = -1;
+ bool NameSet = false;
+ int i;
+
+
+ if (entry->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeContacts, 0))!= ERR_NONE) return error;
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case PBK_Number_General:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 8, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Mobile:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 12, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Work:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 7, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Fax:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 9, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Home:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 13, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Pager:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 11, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Other:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 10, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Note:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 4, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Email:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 14, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Email2:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 15, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_LastName:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 0, entry->Entries[i].Text)) != ERR_NONE) return error;
+ NameSet = true;
+ break;
+ case PBK_Text_FirstName:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 1, entry->Entries[i].Text)) != ERR_NONE) return error;
+ NameSet = true;
+ break;
+ case PBK_Text_Company:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 2, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_JobTitle:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 3, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Category:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 5, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case PBK_Private:
+ if ((error = ALCATEL_CreateField(s, Alcatel_bool, 6, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case PBK_Text_StreetAddress:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 16, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_City:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 17, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_State:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 18, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Zip:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 19, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Country:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 20, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom1:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 21, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom2:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 22, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom3:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 23, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom4:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 24, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_PictureID:
+ if (s->Phone.Data.Priv.ALCATEL.ProtocolVersion == V_1_1) {
+ if ((error = ALCATEL_CreateField(s, Alcatel_int, 25, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ } else {
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", entry->Entries[i].EntryType);
+ }
+ break;
+
+ case PBK_Text_Name: NamePosition = i; break;
+ /* Following fields are not supported: */
+ case PBK_Text_UserID:
+ case PBK_SMSListID:
+ case PBK_RingtoneFileSystemID:
+ case PBK_Date:
+ case PBK_Caller_Group:
+ case PBK_RingtoneID:
+ case PBK_Text_Postal:
+ case PBK_Text_URL:
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", entry->Entries[i].EntryType);
+ break;
+ }
+ }
+ if (NamePosition != -1) {
+ if (NameSet) {
+ smprintf(s,"WARNING: Ignoring name, not supported by phone\n");
+ } else {
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 1, entry->Entries[i].Text)) != ERR_NONE) return error;
+ }
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ entry->Location = Priv->CommitedRecord;
+ /* Refresh list */
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_AddMemory(s, entry);
+ }
+}
+
+static GSM_Error ALCATEL_SetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int NamePosition = -1;
+ bool NameSet = false;
+ int i;
+ bool UpdatedFields[26];
+
+ if (entry->Location == 0) return ERR_INVALIDLOCATION;
+
+ if (entry->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ /* Save modified entry */
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, entry->Location))!= ERR_NONE) {
+ /* Entry doesn't exist, we will create new one */
+ return ALCATEL_AddMemory(s, entry);
+ }
+ /* Get fields for current item */
+ if ((error = ALCATEL_GetFields(s, entry->Location))!= ERR_NONE) return error;
+
+ for (i = 0; i < 26; i++) { UpdatedFields[i] = false; }
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeContacts, entry->Location))!= ERR_NONE) return error;
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case PBK_Number_General:
+ UpdatedFields[8] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 8, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Mobile:
+ UpdatedFields[12] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 12, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Work:
+ UpdatedFields[7] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 7, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Fax:
+ UpdatedFields[9] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 9, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Home:
+ UpdatedFields[13] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 13, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Pager:
+ UpdatedFields[11] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 11, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Number_Other:
+ UpdatedFields[10] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, entry->Location, 10, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Note:
+ UpdatedFields[4] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 4, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Email:
+ UpdatedFields[14] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 14, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Email2:
+ UpdatedFields[15] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 15, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_LastName:
+ UpdatedFields[0] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 0, entry->Entries[i].Text)) != ERR_NONE) return error; NameSet = true;
+ break;
+ case PBK_Text_FirstName:
+ UpdatedFields[1] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 1, entry->Entries[i].Text)) != ERR_NONE) return error; NameSet = true;
+ break;
+ case PBK_Text_Company:
+ UpdatedFields[2] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 2, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_JobTitle:
+ UpdatedFields[3] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 3, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Category:
+ UpdatedFields[5] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, entry->Location, 5, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case PBK_Private:
+ UpdatedFields[6] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_bool, entry->Location, 6, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case PBK_Text_StreetAddress:
+ UpdatedFields[16] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 16, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_City:
+ UpdatedFields[17] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 17, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_State:
+ UpdatedFields[18] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 18, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Zip:
+ UpdatedFields[19] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 19, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Country:
+ UpdatedFields[20] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 20, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom1:
+ UpdatedFields[21] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 21, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom2:
+ UpdatedFields[22] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 22, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom3:
+ UpdatedFields[23] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 23, entry->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case PBK_Text_Custom4:
+ UpdatedFields[24] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 24, entry->Entries[i].Text)) != ERR_NONE) return error
+ ; break;
+ case PBK_PictureID:
+ if (s->Phone.Data.Priv.ALCATEL.ProtocolVersion == V_1_1) {
+ UpdatedFields[25] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_int, entry->Location, 25, &(entry->Entries[i].Number))) != ERR_NONE) return error;
+ } else {
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", entry->Entries[i].EntryType);
+ }
+ break;
+
+ case PBK_Text_Name: NamePosition = i; break;
+ /* Following fields are not supported: */
+ case PBK_SMSListID:
+ case PBK_Text_UserID:
+ case PBK_RingtoneFileSystemID:
+ case PBK_Date:
+ case PBK_Caller_Group:
+ case PBK_RingtoneID:
+ case PBK_Text_Postal:
+ case PBK_Text_URL:
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", entry->Entries[i].EntryType);
+ break;
+ }
+ }
+ if (NamePosition != -1) {
+ if (NameSet) {
+ smprintf(s,"WARNING: Ignoring name, not supported by phone\n");
+ } else {
+ UpdatedFields[1] = true; if ((error = ALCATEL_UpdateField(s, Alcatel_string, entry->Location, 1, entry->Entries[i].Text)) != ERR_NONE) return error;
+ }
+ }
+ /* If we didn't update some field, we have to delete it... */
+ for (i=0; i<Priv->CurrentFieldsCount; i++) {
+ if (!UpdatedFields[Priv->CurrentFields[i]]) if ((error = ALCATEL_DeleteField(s, entry->Location, Priv->CurrentFields[i])) != ERR_NONE) return error;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ entry->Location = Priv->CommitedRecord;
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetMemory(s, entry);
+ }
+}
+
+static GSM_Error ALCATEL_DeleteMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+
+ if (entry->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, entry->Location))!= ERR_NONE) {
+ /* Entry was empty => no error */
+ return ERR_NONE;
+ }
+ /* Do real delete */
+ error = ALCATEL_DeleteItem(s, entry->Location);
+ if (error != ERR_NONE) return error;
+
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_DeleteMemory(s, entry);
+ }
+}
+
+static GSM_Error ALCATEL_DeleteAllMemory(GSM_StateMachine *s, GSM_MemoryType type)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ if (type == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+
+ for (i=0; i<Priv->ContactsItemsCount; i++) {
+ error = ALCATEL_DeleteItem(s, Priv->ContactsItems[i]);
+ if (error != ERR_NONE) return error;
+ }
+
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_DeleteAllMemory(s, type);
+ }
+}
+
+static GSM_Error ALCATEL_GetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSMSC(s, smsc);
+}
+
+
+static GSM_Error ALCATEL_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_Error error;
+
+ if (Status->MemoryType == MEM_ME) {
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeContacts, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ Status->MemoryUsed = Priv->ContactsItemsCount;
+ Status->MemoryFree = ALCATEL_FREE_MEMORY;
+ return ERR_NONE;
+ } else {
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetMemoryStatus(s, Status);
+ }
+}
+
+static GSM_Error ALCATEL_GetSMS(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSMS(s, sms);
+}
+
+static GSM_Error ALCATEL_DeleteSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_DeleteSMS(s, sms);
+}
+
+static GSM_Error ALCATEL_AddSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_AddSMS(s, sms);
+}
+
+static GSM_Error ALCATEL_GetBatteryCharge(GSM_StateMachine *s, GSM_BatteryCharge *bat)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetBatteryCharge(s, bat);
+}
+
+static GSM_Error ALCATEL_GetSignalStrength(GSM_StateMachine *s, GSM_SignalQuality *sig)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSignalQuality(s, sig);
+}
+
+static GSM_Error ALCATEL_GetSMSFolders(GSM_StateMachine *s, GSM_SMSFolders *folders)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSMSFolders(s, folders);
+}
+
+static GSM_Error ALCATEL_GetNextSMS(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetNextSMS(s, sms, start);
+}
+
+static GSM_Error ALCATEL_GetSMSStatus(GSM_StateMachine *s, GSM_SMSMemoryStatus *status)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSMSStatus(s, status);
+}
+
+static GSM_Error ALCATEL_DialVoice(GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_DialVoice(s, number, ShowNumber);
+}
+
+static GSM_Error ALCATEL_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_AnswerCall(s,ID,all);
+}
+
+static GSM_Error ALCATEL_GetNetworkInfo(GSM_StateMachine *s, GSM_NetworkInfo *netinfo)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetNetworkInfo(s, netinfo);
+}
+
+static GSM_Error ALCATEL_GetDisplayStatus(GSM_StateMachine *s, GSM_DisplayFeatures *features)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetDisplayStatus(s, features);
+}
+
+static GSM_Error ALCATEL_SetAutoNetworkLogin(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetAutoNetworkLogin(s);
+}
+
+static GSM_Error ALCATEL_PressKey(GSM_StateMachine *s, GSM_KeyCode Key, bool Press)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_PressKey(s, Key, Press);
+}
+
+static GSM_Error ALCATEL_Reset(GSM_StateMachine *s, bool hard)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_Reset(s, hard);
+}
+
+static GSM_Error ALCATEL_CancelCall(GSM_StateMachine *s, int ID, bool all)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_CancelCall(s,ID,all);
+}
+
+static GSM_Error ALCATEL_SendSavedSMS(GSM_StateMachine *s, int Folder, int Location)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SendSavedSMS(s, Folder, Location);
+}
+
+static GSM_Error ALCATEL_SendSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SendSMS(s, sms);
+}
+
+static GSM_Error ALCATEL_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetDateTime(s, date_time);
+}
+
+static GSM_Error ALCATEL_SetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetSMSC(s, smsc);
+}
+
+static GSM_Error ALCATEL_EnterSecurityCode(GSM_StateMachine *s, GSM_SecurityCode Code)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_EnterSecurityCode(s, Code);
+}
+
+static GSM_Error ALCATEL_GetSecurityStatus(GSM_StateMachine *s, GSM_SecurityCodeType *Status)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSecurityStatus(s, Status);
+}
+
+static GSM_Error ALCATEL_ResetPhoneSettings(GSM_StateMachine *s, GSM_ResetSettingsType Type)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_ResetPhoneSettings(s, Type);
+}
+
+static GSM_Error ALCATEL_SendDTMF(GSM_StateMachine *s, char *sequence)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SendDTMF(s, sequence);
+}
+
+static GSM_Error ALCATEL_GetSIMIMSI(GSM_StateMachine *s, char *IMSI)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_GetSIMIMSI(s, IMSI);
+}
+
+static GSM_Error ALCATEL_GetCalendarStatus(GSM_StateMachine *s, GSM_CalendarStatus *status)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ status->Used = 0;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ status->Used = Priv->CalendarItemsCount;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Error error;
+ GSM_DateTime *dt = NULL;
+ GSM_DateTime evdate;
+ bool evdateused = true;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+ int j=0;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, Note->Location))!= ERR_NONE) {
+ Note->EntriesNum = 0;
+ return error;
+ }
+ if ((error = ALCATEL_GetFields(s, Note->Location))!= ERR_NONE) return error;
+
+ Note->EntriesNum = Priv->CurrentFieldsCount;
+
+ for (i=0; i < Priv->CurrentFieldsCount; i++) {
+ if ((error = ALCATEL_GetFieldValue(s, Note->Location, Priv->CurrentFields[i]))!= ERR_NONE) return error;
+ switch (Priv->CurrentFields[i]) {
+ case 0:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ j++;
+ Note->EntriesNum--;
+ evdate = Priv->ReturnDateTime;
+ evdateused = false;
+ break;
+ case 1:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_START_DATETIME;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ Note->Entries[i-j].Date.Day = evdate.Day;
+ Note->Entries[i-j].Date.Month = evdate.Month;
+ Note->Entries[i-j].Date.Year = evdate.Year;
+ Note->Entries[i-j].Date.Timezone = evdate.Timezone;
+ evdateused = true;
+ break;
+ case 2:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_END_DATETIME;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ Note->Entries[i-j].Date.Day = evdate.Day;
+ Note->Entries[i-j].Date.Month = evdate.Month;
+ Note->Entries[i-j].Date.Year = evdate.Year;
+ Note->Entries[i-j].Date.Timezone = evdate.Timezone;
+ evdateused = true;
+ break;
+ case 3:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (dt == NULL) {
+ Note->Entries[i-j].EntryType = CAL_ALARM_DATETIME;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ dt = &(Note->Entries[i-j].Date);
+ } else {
+ j++;
+ Note->EntriesNum--;
+ dt->Day = Priv->ReturnDateTime.Day;
+ dt->Month = Priv->ReturnDateTime.Month;
+ dt->Year = Priv->ReturnDateTime.Year;
+ dt->Timezone = Priv->ReturnDateTime.Timezone;
+ dt = NULL;
+ }
+ break;
+ case 4:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (dt == NULL) {
+ Note->Entries[i-j].EntryType = CAL_ALARM_DATETIME;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ dt = &(Note->Entries[i-j].Date);
+ } else {
+ j++;
+ Note->EntriesNum--;
+ dt->Hour = Priv->ReturnDateTime.Hour;
+ dt->Minute = Priv->ReturnDateTime.Minute;
+ dt->Second = Priv->ReturnDateTime.Second;
+ dt = NULL;
+ }
+ break;
+ case 5:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_TEXT;
+ CopyUnicodeString(Note->Entries[i-j].Text, Priv->ReturnString);
+ break;
+ case 6:
+ if (Priv->ReturnType != Alcatel_bool) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_PRIVATE;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 7:
+ if (Priv->ReturnType != Alcatel_enum) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ switch (Priv->ReturnInt) {
+ case 0:
+ Note->Type = GSM_CAL_MEETING;
+ break;
+ case 2:
+ Note->Type = GSM_CAL_BIRTHDAY;
+ break;
+ case 3:
+ Note->Type = GSM_CAL_CALL;
+ break;
+ case 4:
+ Note->Type = GSM_CAL_ALARM;
+ break;
+ case 5:
+ Note->Type = GSM_CAL_DAILY_ALARM;
+ break;
+ case 9:
+ /* I'd call this repeating event, but it makes no sense creating one more type ... */
+ Note->Type = GSM_CAL_MEETING;
+ break;
+ default:
+ smprintf(s,"WARNING: Received unknown event type %02X!\n", Priv->ReturnInt);
+ break;
+ }
+ j++;
+ Note->EntriesNum--;
+ break;
+ case 8:
+ if (Priv->ReturnType != Alcatel_int) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ /* 0xffffffff indicates that there is phone (BF5), 0 means none (BF5, BE5)*/
+ if (Priv->ReturnInt == 0xffffffff || Priv->ReturnInt == 0) {
+ j++;
+ Note->EntriesNum--;
+ } else {
+ Note->Entries[i-j].EntryType = CAL_CONTACTID;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ }
+ break;
+ case 9:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_PHONE;
+ CopyUnicodeString(Note->Entries[i-j].Text, Priv->ReturnString);
+ break;
+ case 10:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_DAYOFWEEK;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 11:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_DAY;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 12:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_WEEKOFMONTH;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 13:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_MONTH;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 17:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ /* In BF5 birthday has frequency = 1 */
+ if (Note->Type == GSM_CAL_BIRTHDAY) {
+ Note->EntriesNum--;
+ j++;
+ } else {
+ Note->Entries[i-j].EntryType = CAL_REPEAT_FREQUENCY;
+ Note->Entries[i-j].Number = Priv->ReturnInt;
+ }
+ break;
+ case 18:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_STARTDATE;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ break;
+ case 19:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ Note->Entries[i-j].EntryType = CAL_REPEAT_STOPDATE;
+ Note->Entries[i-j].Date = Priv->ReturnDateTime;
+ break;
+ case 20:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ /* This entry had always same value as the 3rd (alarm date) */
+ j++;
+ Note->EntriesNum--;
+ break;
+ case 21:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ Note->EntriesNum--;
+ j++;
+ break;
+ }
+ /* This entry had always same value as the 4th (alarm time) */
+ j++;
+ Note->EntriesNum--;
+ break;
+ default:
+ Note->EntriesNum--;
+ j++;
+ smprintf(s,"WARNING: Received unknown field %02X, ignoring. Type = %02X. Value = ", Priv->CurrentFields[i], Priv->ReturnType);
+ switch (Priv->ReturnType) {
+ case Alcatel_date:
+ smprintf(s, "%d.%d.%d", Priv->ReturnDateTime.Day, Priv->ReturnDateTime.Month, Priv->ReturnDateTime.Year);
+ break;
+ case Alcatel_time:
+ smprintf(s, "%d:%d:%d", Priv->ReturnDateTime.Hour, Priv->ReturnDateTime.Minute, Priv->ReturnDateTime.Second);
+ break;
+ case Alcatel_string:
+ case Alcatel_phone:
+ smprintf(s, "\"%s\"",DecodeUnicodeString(Priv->ReturnString));
+ break;
+ case Alcatel_enum:
+ case Alcatel_bool:
+ case Alcatel_int:
+ case Alcatel_byte:
+ smprintf(s, "%d", Priv->ReturnInt);
+ break;
+ }
+ smprintf(s,"\n");
+ }
+ }
+ /* The event didn't have start/stop time -> we need only date */
+ if (!evdateused) {
+ Note->EntriesNum++;
+ Note->Entries[i-j].EntryType = CAL_START_DATETIME;
+ Note->Entries[i-j].Date = evdate;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if (Priv->CalendarItemsCount == 0) return ERR_EMPTY;
+
+ if (start) Note->Location = 0;
+ if ((error = ALCATEL_GetNextId(s, &(Note->Location))) != ERR_NONE) return error;
+
+ return ALCATEL_GetCalendar(s, Note);
+}
+
+
+static GSM_Error ALCATEL_DeleteCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ /* Delete Calendar */
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, Note->Location))!= ERR_NONE) {
+ /* Entry was empty => no error */
+ return ERR_NONE;
+ }
+ error = ALCATEL_DeleteItem(s, Note->Location);
+ if (error != ERR_NONE) return error;
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+
+static GSM_Error ALCATEL_AddCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Error error;
+ unsigned int val;
+ bool contact_set = false;
+ bool phone_set = false;
+ bool date_set = false;
+ bool repeating = false;
+ int i;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeCalendar, 0))!= ERR_NONE) return error;
+
+ for (i = 0; i < Note->EntriesNum; i++) {
+ switch (Note->Entries[i].EntryType) {
+ case CAL_START_DATETIME:
+ if (!date_set) {
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 0, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ date_set = true;
+ }
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 1, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case CAL_END_DATETIME:
+ if (!date_set) {
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 0, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ date_set = true;
+ }
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 2, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case CAL_ALARM_DATETIME:
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 3, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 4, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ if (Note->Type == GSM_CAL_ALARM || Note->Type == GSM_CAL_DAILY_ALARM) {
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 20, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 21, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ }
+ break;
+ case CAL_TEXT:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 5, Note->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case CAL_PRIVATE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_bool, 6, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case CAL_CONTACTID:
+ if ((error = ALCATEL_CreateField(s, Alcatel_int, 8, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ contact_set = true;
+ break;
+ case CAL_PHONE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 9, Note->Entries[i].Text)) != ERR_NONE) return error;
+ phone_set = true;
+ break;
+ case CAL_REPEAT_DAYOFWEEK:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 10, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_DAY:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 11, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_WEEKOFMONTH:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 12, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_MONTH:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 13, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_FREQUENCY:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 17, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_STARTDATE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 18, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_STOPDATE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 19, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_SILENT_ALARM_DATETIME:
+ case CAL_RECURRANCE:
+ case CAL_LOCATION:
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", Note->Entries[i].EntryType);
+ break;
+ }
+ }
+
+ switch (Note->Type) {
+ case GSM_CAL_CALL:
+ val = 3;
+ break;
+ case GSM_CAL_BIRTHDAY:
+ val = 2;
+ break;
+ case GSM_CAL_ALARM:
+ val = 4;
+ break;
+ case GSM_CAL_DAILY_ALARM:
+ val = 5;
+ break;
+ default:
+ if (repeating) {
+ val = 9;
+ } else {
+ val = 0;
+ }
+ }
+ if ((error = ALCATEL_CreateField(s, Alcatel_enum, 7, &val)) != ERR_NONE) return error;
+
+ if (!contact_set) {
+ if (phone_set) {
+ val = 0xffffffff;
+ } else {
+ val = 0;
+ }
+ if ((error = ALCATEL_CreateField(s, Alcatel_int, 8, &val)) != ERR_NONE) return error;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ Note->Location = Priv->CommitedRecord;
+ /* Refresh list */
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_SetCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Error error;
+ unsigned int val;
+ bool contact_set = false;
+ bool phone_set = false;
+ bool date_set = false;
+ bool repeating = false;
+ int i;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ bool UpdatedFields[22];
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, Note->Location))!= ERR_NONE) {
+ /* Entry doesn't exist, we will create new one */
+ return ALCATEL_AddCalendar(s, Note);
+ }
+ /* Get fields for current item */
+ if ((error = ALCATEL_GetFields(s, Note->Location))!= ERR_NONE) return error;
+
+ for (i = 0; i < 22; i++) { UpdatedFields[i] = false; }
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeCalendar, Note->Location))!= ERR_NONE) return error;
+
+ for (i = 0; i < Note->EntriesNum; i++) {
+ switch (Note->Entries[i].EntryType) {
+ case CAL_START_DATETIME:
+ if (!date_set) {
+ UpdatedFields[0] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 0, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ date_set = true;
+ }
+ UpdatedFields[1] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_time, Note->Location, 1, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case CAL_END_DATETIME:
+ if (!date_set) {
+ UpdatedFields[0] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 0, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ date_set = true;
+ }
+ UpdatedFields[2] = true; if ((error = ALCATEL_UpdateField(s, Alcatel_time, Note->Location, 2, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case CAL_ALARM_DATETIME:
+ UpdatedFields[3] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 3, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[4] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_time, Note->Location, 4, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ if (Note->Type == GSM_CAL_ALARM || Note->Type == GSM_CAL_DAILY_ALARM) {
+ UpdatedFields[20] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 20, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[21] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_time, Note->Location, 21, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ }
+ break;
+ case CAL_TEXT:
+ UpdatedFields[5] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, Note->Location, 5, Note->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case CAL_PRIVATE:
+ UpdatedFields[6] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_bool, Note->Location, 6, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case CAL_CONTACTID:
+ UpdatedFields[8] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_int, Note->Location, 8, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ contact_set = true;
+ break;
+ case CAL_PHONE:
+ UpdatedFields[9] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, Note->Location, 9, Note->Entries[i].Text)) != ERR_NONE) return error;
+ phone_set = true;
+ break;
+ case CAL_REPEAT_DAYOFWEEK:
+ UpdatedFields[10] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, Note->Location, 10, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_DAY:
+ UpdatedFields[11] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, Note->Location, 11, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_WEEKOFMONTH:
+ UpdatedFields[12] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, Note->Location, 12, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_MONTH:
+ UpdatedFields[13] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, Note->Location, 13, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_FREQUENCY:
+ UpdatedFields[17] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, Note->Location, 17, &(Note->Entries[i].Number))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_STARTDATE:
+ UpdatedFields[18] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 18, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_REPEAT_STOPDATE:
+ UpdatedFields[19] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, Note->Location, 19, &(Note->Entries[i].Date))) != ERR_NONE) return error;
+ repeating = true;
+ break;
+ case CAL_SILENT_ALARM_DATETIME:
+ case CAL_RECURRANCE:
+ case CAL_LOCATION:
+ smprintf(s,"WARNING: Ignoring entry %d, not supported by phone\n", Note->Entries[i].EntryType);
+ break;
+ }
+ }
+
+ switch (Note->Type) {
+ case GSM_CAL_CALL:
+ val = 3;
+ break;
+ case GSM_CAL_BIRTHDAY:
+ val = 2;
+ break;
+ case GSM_CAL_ALARM:
+ val = 4;
+ break;
+ case GSM_CAL_DAILY_ALARM:
+ val = 5;
+ break;
+ default:
+ if (repeating) {
+ val = 9;
+ } else {
+ val = 0;
+ }
+ }
+ UpdatedFields[7] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_enum, Note->Location, 7, &val)) != ERR_NONE) return error;
+
+ if (!contact_set) {
+ if (phone_set) {
+ val = 0xffffffff;
+ } else {
+ val = 0;
+ }
+ UpdatedFields[8] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_int, Note->Location, 8, &val)) != ERR_NONE) return error;
+ }
+ /* If we didn't update some field, we have to delete it... */
+ for (i=0; i<Priv->CurrentFieldsCount; i++) {
+ if (!UpdatedFields[Priv->CurrentFields[i]]) if ((error = ALCATEL_DeleteField(s, Note->Location, Priv->CurrentFields[i])) != ERR_NONE) return error;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_DeleteAllCalendar (GSM_StateMachine *s)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+
+ for (i=0; i<Priv->CalendarItemsCount; i++) {
+ error = ALCATEL_DeleteItem(s, Priv->CalendarItems[i]);
+ if (error != ERR_NONE) return error;
+ }
+
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+
+static GSM_Error ALCATEL_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ GSM_Error error;
+ GSM_CalendarEntry Note;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+ bool Found = false;
+ bool DateSet = false;
+ int alarm_number = alarm->Location;
+ static GSM_DateTime nulldt = {0,0,0,0,0,0,0};
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+
+ for (i=0; i<Priv->CalendarItemsCount; i++) {
+ if ((error = ALCATEL_GetFieldValue(s, Priv->CalendarItems[i], 7))!= ERR_NONE) return error;
+ if (Priv->ReturnType != Alcatel_enum) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ continue;
+ }
+ if (Priv->ReturnInt == 4 || Priv->ReturnInt == 5) {
+ alarm_number--;
+ if (alarm_number == 0) {
+ Found = true;
+ break;
+ }
+ }
+ }
+
+ if (!Found) return ERR_EMPTY;
+
+ Note.Location = Priv->CalendarItems[i];
+
+ if ((error = ALCATEL_GetCalendar(s, &Note))!= ERR_NONE) return error;
+
+ if (Note.Type == GSM_CAL_ALARM) {
+ alarm->Repeating = false;
+ } else {
+ alarm->Repeating = true;
+ }
+
+ alarm->Text[0] = 0; alarm->Text[1] = 0;
+
+
+ for (i = 0; i < Note.EntriesNum; i++) {
+ if (Note.Entries[i].EntryType == CAL_TEXT) {
+ CopyUnicodeString(alarm->Text, Note.Entries[i].Text);
+ } else if (Note.Entries[i].EntryType == CAL_ALARM_DATETIME) {
+ alarm->DateTime = Note.Entries[i].Date;
+ DateSet = false;
+ }
+ }
+ if (!DateSet) {
+ alarm->DateTime = nulldt;
+ }
+
+ return ERR_NONE;
+}
+
+
+static GSM_Error ALCATEL_SetAlarm (GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ GSM_Error error;
+ GSM_CalendarEntry Note;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ GSM_DateTime dt;
+ int i;
+ bool Found = false;
+ int alarm_number = alarm->Location;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeCalendar, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+
+ for (i=0; i<Priv->CalendarItemsCount; i++) {
+ if ((error = ALCATEL_GetFieldValue(s, Priv->CalendarItems[i], 7))!= ERR_NONE) return error;
+ if (Priv->ReturnType != Alcatel_enum) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ continue;
+ }
+ if (Priv->ReturnInt == 4 || Priv->ReturnInt == 5) {
+ alarm_number--;
+ if (alarm_number == 0) {
+ Found = true;
+ break;
+ }
+ }
+ }
+
+ if (Found) {
+ Note.Location = Priv->CalendarItems[i];
+ }
+
+ Note.EntriesNum = 1;
+
+ Note.Entries[0].EntryType = CAL_ALARM_DATETIME;
+ Note.Entries[0].Date = alarm->DateTime;
+
+ if (alarm->Repeating) {
+ Note.Type = GSM_CAL_DAILY_ALARM;
+ GSM_GetCurrentDateTime(&dt);
+ Note.Entries[0].Date.Day = dt.Day;
+ Note.Entries[0].Date.Month = dt.Month;
+ Note.Entries[0].Date.Year = dt.Year;
+ } else {
+ Note.Type = GSM_CAL_ALARM;
+ }
+
+ if (alarm->Text[0] != 0 || alarm->Text[1] != 0) {
+ Note.EntriesNum++;
+ Note.Entries[1].EntryType = CAL_TEXT;
+ CopyUnicodeString(Note.Entries[1].Text, alarm->Text);
+ }
+
+ if (Found) {
+ return ALCATEL_SetCalendar(s, &Note);
+ } else {
+ return ALCATEL_AddCalendar(s, &Note);
+ }
+}
+
+
+static GSM_Error ALCATEL_GetToDoStatus(GSM_StateMachine *s, GSM_ToDoStatus *status)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ status->Used = 0;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ status->Used = Priv->ToDoItemsCount;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_Error error;
+ GSM_DateTime *dt = NULL;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+ int j=0;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, ToDo->Location))!= ERR_NONE) {
+ ToDo->EntriesNum = 0;
+ return error;
+ }
+ if ((error = ALCATEL_GetFields(s, ToDo->Location))!= ERR_NONE) return error;
+
+ ToDo->EntriesNum = Priv->CurrentFieldsCount;
+
+ for (i=0; i < Priv->CurrentFieldsCount; i++) {
+ if ((error = ALCATEL_GetFieldValue(s, ToDo->Location, Priv->CurrentFields[i]))!= ERR_NONE) return error;
+ switch (Priv->CurrentFields[i]) {
+ case 0:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ ToDo->Entries[i-j].EntryType = TODO_END_DATETIME;
+ ToDo->Entries[i-j].Date = Priv->ReturnDateTime;
+ break;
+ case 1:
+ if (Priv->ReturnType != Alcatel_bool) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ ToDo->Entries[i-j].EntryType = TODO_COMPLETED;
+ ToDo->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 2:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (dt == NULL) {
+ ToDo->Entries[i-j].EntryType = TODO_ALARM_DATETIME;
+ ToDo->Entries[i-j].Date = Priv->ReturnDateTime;
+ dt = &(ToDo->Entries[i-j].Date);
+ } else {
+ j++;
+ ToDo->EntriesNum--;
+ dt->Day = Priv->ReturnDateTime.Day;
+ dt->Month = Priv->ReturnDateTime.Month;
+ dt->Year = Priv->ReturnDateTime.Year;
+ dt->Timezone = Priv->ReturnDateTime.Timezone;
+ dt = NULL;
+ }
+ break;
+ case 3:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (dt == NULL) {
+ ToDo->Entries[i-j].EntryType = TODO_ALARM_DATETIME;
+ ToDo->Entries[i-j].Date = Priv->ReturnDateTime;
+ dt = &(ToDo->Entries[i-j].Date);
+ } else {
+ j++;
+ ToDo->EntriesNum--;
+ dt->Hour = Priv->ReturnDateTime.Hour;
+ dt->Minute = Priv->ReturnDateTime.Minute;
+ dt->Second = Priv->ReturnDateTime.Second;
+ dt = NULL;
+ }
+ break;
+ case 4:
+ if (Priv->ReturnType != Alcatel_string) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ ToDo->Entries[i-j].EntryType = TODO_TEXT;
+ CopyUnicodeString(ToDo->Entries[i-j].Text, Priv->ReturnString);
+ break;
+ case 5:
+ if (Priv->ReturnType != Alcatel_bool) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ ToDo->Entries[i-j].EntryType = TODO_PRIVATE;
+ ToDo->Entries[i-j].Number = Priv->ReturnInt;
+ break;
+ case 6:
+ if (Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (Priv->ReturnInt == 255) {
+ /* 255 means no category */
+ j++;
+ ToDo->EntriesNum--;
+ } else {
+ ToDo->Entries[i-j].EntryType = TODO_CATEGORY;
+ ToDo->Entries[i-j].Number = Priv->ReturnInt;
+ }
+ break;
+ case 7:
+ /* This one seems to be byte for BF5 and enum for BE5 */
+ if (Priv->ReturnType != Alcatel_enum && Priv->ReturnType != Alcatel_byte) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ switch (Priv->ReturnInt) {
+ case 0:
+ ToDo->Priority = GSM_Priority_High;
+ break;
+ case 1:
+ ToDo->Priority = GSM_Priority_Medium;
+ break;
+ case 2:
+ ToDo->Priority = GSM_Priority_Low;
+ break;
+ default:
+ ToDo->Priority = 0;
+ smprintf(s,"WARNING: Received unexpected priority %02X, ignoring\n", Priv->ReturnInt);
+ }
+ j++;
+ ToDo->EntriesNum--;
+ break;
+ case 8:
+ if (Priv->ReturnType != Alcatel_int) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ /* 0xffffffff indicates that there is phone, 0 means none */
+ if (Priv->ReturnInt == 0xffffffff || Priv->ReturnInt == 0) {
+ j++;
+ ToDo->EntriesNum--;
+ } else {
+ ToDo->Entries[i-j].EntryType = TODO_CONTACTID;
+ ToDo->Entries[i-j].Number = Priv->ReturnInt;
+ }
+ break;
+ case 9:
+ if (Priv->ReturnType != Alcatel_phone) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ ToDo->Entries[i-j].EntryType = TODO_PHONE;
+ CopyUnicodeString(ToDo->Entries[i-j].Text, Priv->ReturnString);
+ break;
+ case 10:
+ if (Priv->ReturnType != Alcatel_date) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckDate(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid date in phone, ignoring\n");
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ /* This entry had always same value as the 2nd (alarm date) */
+ j++;
+ ToDo->EntriesNum--;
+ break;
+ case 11:
+ if (Priv->ReturnType != Alcatel_time) {
+ smprintf(s,"WARNING: Received unexpected type %02X, ignoring\n", Priv->ReturnType);
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ if (!CheckTime(&(Priv->ReturnDateTime))) {
+ smprintf(s,"WARNING: Invalid time in phone, ignoring\n");
+ ToDo->EntriesNum--;
+ j++;
+ break;
+ }
+ /* This entry had always same value as the 3rd (alarm time) */
+ j++;
+ ToDo->EntriesNum--;
+ break;
+ default:
+ ToDo->EntriesNum--;
+ j++;
+ smprintf(s,"WARNING: Received unknown field %02X, ignoring. Type = %02X. Value = ", Priv->CurrentFields[i], Priv->ReturnType);
+ switch (Priv->ReturnType) {
+ case Alcatel_date:
+ smprintf(s, "%d.%d.%d", Priv->ReturnDateTime.Day, Priv->ReturnDateTime.Month, Priv->ReturnDateTime.Year);
+ break;
+ case Alcatel_time:
+ smprintf(s, "%d:%d:%d", Priv->ReturnDateTime.Hour, Priv->ReturnDateTime.Minute, Priv->ReturnDateTime.Second);
+ break;
+ case Alcatel_string:
+ case Alcatel_phone:
+ smprintf(s, "\"%s\"",DecodeUnicodeString(Priv->ReturnString));
+ break;
+ case Alcatel_enum:
+ case Alcatel_bool:
+ case Alcatel_int:
+ case Alcatel_byte:
+ smprintf(s, "%d", Priv->ReturnInt);
+ break;
+ }
+ smprintf(s,"\n");
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetNextToDo(GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool start)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if (Priv->ToDoItemsCount == 0) return ERR_EMPTY;
+
+ if (start) ToDo->Location = 0;
+ if ((error = ALCATEL_GetNextId(s, &(ToDo->Location))) != ERR_NONE) return error;
+
+ return ALCATEL_GetToDo(s, ToDo);
+}
+
+static GSM_Error ALCATEL_DeleteAllToDo (GSM_StateMachine *s)
+{
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+ int i;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+
+ for (i=0; i<Priv->ToDoItemsCount; i++) {
+ error = ALCATEL_DeleteItem(s, Priv->ToDoItems[i]);
+ if (error != ERR_NONE) return error;
+ }
+
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_AddToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_Error error;
+ unsigned int val;
+ bool contact_set = false;
+ bool phone_set = false;
+ int i;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeToDo, 0))!= ERR_NONE) return error;
+
+ switch (ToDo->Priority) {
+ case GSM_Priority_High:
+ val = 0;
+ break;
+ case GSM_Priority_Low:
+ val = 2;
+ break;
+ case GSM_Priority_Medium:
+ default:
+ val = 1;
+ break;
+ }
+ /* This one seems to be byte for BF5 and enum for BE5 */
+ if (s->Phone.Data.Priv.ALCATEL.ProtocolVersion == V_1_1) {
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 7, &val)) != ERR_NONE) return error;
+ } else {
+ if ((error = ALCATEL_CreateField(s, Alcatel_enum, 7, &val)) != ERR_NONE) return error;
+ }
+
+ for (i = 0; i < ToDo->EntriesNum; i++) {
+ switch (ToDo->Entries[i].EntryType) {
+ case TODO_END_DATETIME:
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 0, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case TODO_COMPLETED:
+ if ((error = ALCATEL_CreateField(s, Alcatel_bool, 1, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case TODO_ALARM_DATETIME:
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 2, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 3, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ if ((error = ALCATEL_CreateField(s, Alcatel_date, 10, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ if ((error = ALCATEL_CreateField(s, Alcatel_time, 11, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ break;
+ case TODO_TEXT:
+ if ((error = ALCATEL_CreateField(s, Alcatel_string, 4, ToDo->Entries[i].Text)) != ERR_NONE) return error;
+ break;
+ case TODO_PRIVATE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_bool, 5, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case TODO_CATEGORY:
+ if ((error = ALCATEL_CreateField(s, Alcatel_byte, 6, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ break;
+ case TODO_CONTACTID:
+ if ((error = ALCATEL_CreateField(s, Alcatel_int, 8, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ contact_set = true;
+ break;
+ case TODO_PHONE:
+ if ((error = ALCATEL_CreateField(s, Alcatel_phone, 9, ToDo->Entries[i].Text)) != ERR_NONE) return error;
+ phone_set = true;
+ break;
+ default:
+ break;
+ }
+ }
+ if (!contact_set) {
+ if (phone_set) {
+ val = 0xffffffff;
+ } else {
+ val = 0;
+ }
+ if ((error = ALCATEL_CreateField(s, Alcatel_int, 8, &val)) != ERR_NONE) return error;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ ToDo->Location = Priv->CommitedRecord;
+ /* Refresh list */
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_SetToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_Error error;
+ unsigned int val;
+ bool contact_set = false;
+ bool phone_set = false;
+ bool UpdatedFields[12];
+ int i;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ /* Save modified ToDo */
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, ToDo->Location))!= ERR_NONE) {
+ /* Entry doesn't exist, we will create new one */
+ return ALCATEL_AddToDo(s, ToDo);
+ }
+ /* Get fields for current item */
+ if ((error = ALCATEL_GetFields(s, ToDo->Location))!= ERR_NONE) return error;
+
+ for (i = 0; i < 12; i++) { UpdatedFields[i] = false; }
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateEdit, TypeToDo, ToDo->Location))!= ERR_NONE) return error;
+
+ switch (ToDo->Priority) {
+ case GSM_Priority_High:
+ val = 0;
+ break;
+ case GSM_Priority_Low:
+ val = 2;
+ break;
+ case GSM_Priority_Medium:
+ default:
+ val = 1;
+ break;
+ }
+ /* This one seems to be byte for BF5 and enum for BE5 */
+ if (s->Phone.Data.Priv.ALCATEL.ProtocolVersion == V_1_1) {
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, ToDo->Location, 7, &val)) != ERR_NONE) return error;
+ } else {
+ if ((error = ALCATEL_UpdateField(s, Alcatel_enum, ToDo->Location, 7, &val)) != ERR_NONE) return error;
+ }
+ UpdatedFields[7] = true;
+
+ for (i = 0; i < ToDo->EntriesNum; i++) {
+ switch (ToDo->Entries[i].EntryType) {
+ case TODO_END_DATETIME:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, ToDo->Location, 0, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[0] = true;
+ break;
+ case TODO_COMPLETED:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_bool, ToDo->Location, 1, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ UpdatedFields[1] = true;
+ break;
+ case TODO_ALARM_DATETIME:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, ToDo->Location, 2, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[2] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_time, ToDo->Location, 3, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[3] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_date, ToDo->Location, 10, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[10] = true;
+ if ((error = ALCATEL_UpdateField(s, Alcatel_time, ToDo->Location, 11, &(ToDo->Entries[i].Date))) != ERR_NONE) return error;
+ UpdatedFields[11] = true;
+ break;
+ case TODO_TEXT:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_string, ToDo->Location, 4, ToDo->Entries[i].Text)) != ERR_NONE) return error;
+ UpdatedFields[4] = true;
+ break;
+ case TODO_PRIVATE:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_bool, ToDo->Location, 5, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ UpdatedFields[5] = true;
+ break;
+ case TODO_CATEGORY:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_byte, ToDo->Location, 6, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ UpdatedFields[6] = true;
+ break;
+ case TODO_CONTACTID:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_int, ToDo->Location, 8, &(ToDo->Entries[i].Number))) != ERR_NONE) return error;
+ UpdatedFields[8] = true;
+ contact_set = true;
+ break;
+ case TODO_PHONE:
+ if ((error = ALCATEL_UpdateField(s, Alcatel_phone, ToDo->Location, 9, ToDo->Entries[i].Text)) != ERR_NONE) return error;
+ UpdatedFields[9] = true;
+ phone_set = true;
+ break;
+ default:
+ break;
+ }
+ }
+ if (!contact_set) {
+ if (phone_set) {
+ val = 0xffffffff;
+ } else {
+ val = 0;
+ }
+ if ((error = ALCATEL_UpdateField(s, Alcatel_int, ToDo->Location, 8, &val)) != ERR_NONE) return error;
+ UpdatedFields[8] = true;
+ }
+
+
+ /* If we didn't update some field, we have to delete it... */
+ for (i=0; i<Priv->CurrentFieldsCount; i++) {
+ if (!UpdatedFields[Priv->CurrentFields[i]]) if ((error = ALCATEL_DeleteField(s, ToDo->Location, Priv->CurrentFields[i])) != ERR_NONE) return error;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_DeleteToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ /* Delete ToDo */
+ if ((error = ALCATEL_GetAvailableIds(s, false))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsIdAvailable(s, ToDo->Location))!= ERR_NONE) {
+ /* Entry was empty => no error */
+ return ERR_NONE;
+ }
+ error = ALCATEL_DeleteItem(s, ToDo->Location);
+ if (error != ERR_NONE) return error;
+ /* Refresh list */
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, TypeToDo, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableIds(s, true))!= ERR_NONE) return error;
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetCategoryStatus(GSM_StateMachine *s, GSM_CategoryStatus *Status)
+{
+ GSM_Alcatel_BinaryType type;
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ switch (Status->Type) {
+ case Category_ToDo: type = TypeToDo; break;
+ case Category_Phonebook: type = TypeContacts; break;
+ default: return ERR_NOTSUPPORTED;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, type, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableCategoryIds(s))!= ERR_NONE) return error;
+
+ Status->Used = Priv->CurrentCategoriesCount;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetCategory(GSM_StateMachine *s, GSM_Category *Category)
+{
+ GSM_Alcatel_BinaryType type;
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ switch (Category->Type) {
+ case Category_ToDo: type = TypeToDo; break;
+ case Category_Phonebook: type = TypeContacts; break;
+ default: return ERR_NOTSUPPORTED;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, type, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetAvailableCategoryIds(s))!= ERR_NONE) return error;
+ if ((error = ALCATEL_IsCategoryIdAvailable(s, Category->Location))!= ERR_NONE) return error;
+ if ((error = ALCATEL_GetCategoryText(s, Category->Location))!= ERR_NONE) return error;
+
+ CopyUnicodeString(Category->Name, Priv->ReturnString);
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_AddCategory(GSM_StateMachine *s, GSM_Category *Category)
+{
+ GSM_Alcatel_BinaryType type;
+ GSM_Error error;
+ GSM_Phone_ALCATELData *Priv = &s->Phone.Data.Priv.ALCATEL;
+
+ switch (Category->Type) {
+ case Category_ToDo: type = TypeToDo; break;
+ case Category_Phonebook: type = TypeContacts; break;
+ default: return ERR_NOTSUPPORTED;
+ }
+ if ((error = ALCATEL_GoToBinaryState(s, StateSession, type, 0))!= ERR_NONE) return error;
+ if ((error = ALCATEL_AddCategoryText(s, Category->Name))!= ERR_NONE) return error;
+
+ Category->Location = Priv->ReturnInt;
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_GetProductCode(GSM_StateMachine *s, char *value)
+{
+ strcpy(value, s->Phone.Data.ModelInfo->model);
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_DispatchMessage(GSM_StateMachine *s)
+{
+ if (s->Phone.Data.Priv.ALCATEL.Mode == ModeBinary) {
+ return GSM_DispatchMessage(s);
+ } else {
+ return ATGEN_DispatchMessage(s);
+ }
+}
+
+static GSM_Error ALCATEL_ReplyGeneric(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ /* All error values are just VERY wild guesses, but these seems to work
+ * almost as expected ...
+ */
+ switch (msg.Buffer[8]) {
+ case 0x00: /* no error */
+ return ERR_NONE;
+ case 0x10: /* same thing opened in phone menus */
+ return ERR_INSIDEPHONEMENU;
+ case 0x13:
+ /* This appears in more cases:
+ * - phone needs PIN code
+ * - we want to close not opened session
+ * For normal users the second case shouldn't occur...
+ */
+ return ERR_SECURITYERROR;
+ case 0x14: /* Bad data */
+ case 0x2f: /* Closing session when not opened */
+ case 0x1f: /* Bad in/out counter in packet/ack */
+ case 0x0e: /* Openning session when not closed */
+ case 0x0C: /* Bad id (item/database) */
+ case 0x11: /* Bad list id */
+ case 0x2A: /* Nonexistant field/item id */
+ case 0x35: /* Too long text */
+ return ERR_BUG;
+ case 0x23: /* Session opened */
+ case 0x80: /* Transfer started */
+ return ERR_NONE;
+ case 0x82: /* Transfer canceled */
+ return ERR_CANCELED;
+ default:
+ smprintf(s, "WARNING: Packet seems to indicate some status by %02X, ignoring!\n", msg.Buffer[8]);
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error ALCATEL_ReplyCommit(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ s->Phone.Data.Priv.ALCATEL.CommitedRecord = msg.Buffer[12] + (msg.Buffer[11] << 8) + (msg.Buffer[10] << 16) + (msg.Buffer[9] << 24);
+ smprintf(s, "Created record %08x\n", s->Phone.Data.Priv.ALCATEL.CommitedRecord);
+ return ERR_NONE;
+}
+
+static GSM_Error ALCATEL_SetIncomingCB (GSM_StateMachine *s, bool enable)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetIncomingCB(s, enable);
+}
+
+static GSM_Error ALCATEL_SetIncomingSMS (GSM_StateMachine *s, bool enable)
+{
+ GSM_Error error;
+
+ if ((error = ALCATEL_SetATMode(s))!= ERR_NONE) return error;
+ return ATGEN_SetIncomingSMS(s, enable);
+}
+
+static GSM_Reply_Function ALCATELReplyFunctions[] = {
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelAttach },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelDetach },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelCommit },
+{ALCATEL_ReplyCommit, "\x02",0x00,0x00, ID_AlcatelCommit2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelEnd },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelClose },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelStart },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelSelect1 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelSelect2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelSelect3 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelBegin1 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelBegin2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelGetIds1 },
+{ALCATEL_ReplyGetIds, "\x02",0x00,0x00, ID_AlcatelGetIds2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelGetCategories1 },
+{ALCATEL_ReplyGetCategories, "\x02",0x00,0x00, ID_AlcatelGetCategories2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelGetCategoryText1 },
+{ALCATEL_ReplyGetCategoryText, "\x02",0x00,0x00, ID_AlcatelGetCategoryText2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelAddCategoryText1 },
+{ALCATEL_ReplyAddCategoryText, "\x02",0x00,0x00, ID_AlcatelAddCategoryText2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelGetFields1 },
+{ALCATEL_ReplyGetFields, "\x02",0x00,0x00, ID_AlcatelGetFields2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelGetFieldValue1 },
+{ALCATEL_ReplyGetFieldValue, "\x02",0x00,0x00, ID_AlcatelGetFieldValue2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelDeleteField },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelDeleteItem1 },
+{ALCATEL_ReplyDeleteItem, "\x02",0x00,0x00, ID_AlcatelDeleteItem2 },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelCreateField },
+{ALCATEL_ReplyGeneric, "\x02",0x00,0x00, ID_AlcatelUpdateField },
+{NULL, "\x00",0x00,0x00, ID_None }
+};
+
+GSM_Phone_Functions ALCATELPhone = {
+ /* AFAIK, any 50[0123] phone should work, but I'm not sure whether all
+ * they were ever really released, if yes add them here also.
+ */
+ "alcatel|OT501|OT701|OT715|OT535|OT735|BE5|BF5|BH4",
+ ALCATELReplyFunctions,
+ ALCATEL_Initialise,
+ ALCATEL_Terminate,
+ ALCATEL_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ ALCATEL_GetManufacturer,
+ ALCATEL_GetModel,
+ ALCATEL_GetFirmware,
+ ALCATEL_GetIMEI,
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ ALCATEL_GetProductCode,
+ NOTSUPPORTED, /* GetHardware */
+ NOTSUPPORTED, /* GetPPM */
+ ALCATEL_GetSIMIMSI,
+ ALCATEL_GetDateTime,
+ ALCATEL_SetDateTime,
+ ALCATEL_GetAlarm,
+ ALCATEL_SetAlarm,
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ ALCATEL_PressKey,
+ ALCATEL_Reset,
+ ALCATEL_ResetPhoneSettings,
+ ALCATEL_EnterSecurityCode,
+ ALCATEL_GetSecurityStatus,
+ ALCATEL_GetDisplayStatus,
+ ALCATEL_SetAutoNetworkLogin,
+ ALCATEL_GetBatteryCharge,
+ ALCATEL_GetSignalStrength,
+ ALCATEL_GetNetworkInfo,
+ ALCATEL_GetCategory,
+ ALCATEL_AddCategory,
+ ALCATEL_GetCategoryStatus,
+ ALCATEL_GetMemoryStatus,
+ ALCATEL_GetMemory,
+ ALCATEL_GetNextMemory,
+ ALCATEL_SetMemory,
+ ALCATEL_AddMemory,
+ ALCATEL_DeleteMemory,
+ ALCATEL_DeleteAllMemory,
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ ALCATEL_GetSMSC,
+ ALCATEL_SetSMSC,
+ ALCATEL_GetSMSStatus,
+ ALCATEL_GetSMS,
+ ALCATEL_GetNextSMS,
+ NOTSUPPORTED, /* SetSMS */
+ ALCATEL_AddSMS,
+ ALCATEL_DeleteSMS,
+ ALCATEL_SendSMS,
+ ALCATEL_SendSavedSMS,
+ ALCATEL_SetIncomingSMS,
+ ALCATEL_SetIncomingCB,
+ ALCATEL_GetSMSFolders,
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ ALCATEL_DialVoice,
+ ALCATEL_AnswerCall,
+ ALCATEL_CancelCall,
+ NOTSUPPORTED, /* HoldCall */
+ NOTSUPPORTED, /* UnholdCall */
+ NOTSUPPORTED, /* ConferenceCall */
+ NOTSUPPORTED, /* SplitCall */
+ NOTSUPPORTED, /* TransferCall */
+ NOTSUPPORTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NONEFUNCTION, /* SetIncomingCall */
+ NOTSUPPORTED, /* SetIncomingUSSD */
+ ALCATEL_SendDTMF,
+ NOTSUPPORTED, /* GetRingtone */
+ NOTSUPPORTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ NOTSUPPORTED, /* GetBitmap */
+ NOTSUPPORTED, /* SetBitmap */
+ ALCATEL_GetToDoStatus,
+ ALCATEL_GetToDo,
+ ALCATEL_GetNextToDo,
+ ALCATEL_SetToDo,
+ ALCATEL_AddToDo,
+ ALCATEL_DeleteToDo,
+ ALCATEL_DeleteAllToDo,
+ ALCATEL_GetCalendarStatus,
+ ALCATEL_GetCalendar,
+ ALCATEL_GetNextCalendar,
+ ALCATEL_SetCalendar,
+ ALCATEL_AddCalendar,
+ ALCATEL_DeleteCalendar,
+ ALCATEL_DeleteAllCalendar,
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFilePart */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/alcatel/alcatel.h b/gammu/emb/common/phone/alcatel/alcatel.h
new file mode 100644
index 0000000..718f436
--- a/dev/null
+++ b/gammu/emb/common/phone/alcatel/alcatel.h
@@ -0,0 +1,286 @@
+/* (c) 2002-2004 by Michal Cihar */
+
+/*
+ * High level functions for communication with Alcatel One Touch 501 and
+ * compatible mobile phone.
+ *
+ * This code implements functions to communicate with Alcatel phones,
+ * currently seem to work:
+ * - BE5 series (501/701)
+ * - BF5 series (715)
+ * - BH4 series (535/735)
+ * For some functions it uses normal AT mode (not implemented here, look at
+ * ../at/atgen.[ch]) for others it switches into binary mode and initialises
+ * underlaying protocol (see ../../protocol/alcatel/alcabus.[ch]) and
+ * communicates over it. Don't ask me why Alcatel uses such silly thing...
+ */
+
+#ifndef alcatel_h
+#define alcatel_h
+
+#include "../../gsmcomon.h"
+
+#ifndef GSM_USED_AT
+# define GSM_USED_AT
+#endif
+#ifndef GSM_USED_ALCABUS
+# define GSM_USED_ALCABUS
+#endif
+
+/**
+ * Determines which mode is phone currently using.
+ */
+typedef enum {
+ /**
+ * Phone accepts normal AT commands.
+ */
+ ModeAT = 1,
+ /**
+ * Binary mode using proprietary protocol.
+ */
+ ModeBinary
+} GSM_Alcatel_Mode;
+
+/**
+ * Protocol version being used for binary mode.
+ */
+typedef enum {
+ /**
+ * Version 1.0 used in BE5 phones (501, 701).
+ */
+ V_1_0 = 1,
+ /**
+ * Version 1.1 used in BF5 phones (715, 535, 735).
+ */
+ V_1_1
+} GSM_Alcatel_ProtocolVersion;
+
+/**
+ * State of binary mode.
+ */
+typedef enum {
+ /**
+ * Binary mode is active. No type selected.
+ */
+ StateAttached = 1,
+ /**
+ * Opened session of some type.
+ */
+ StateSession,
+ /**
+ * Some item is being edited.
+ */
+ StateEdit
+} GSM_Alcatel_BinaryState;
+
+/**
+ * Type of active binary session.
+ */
+typedef enum {
+ /**
+ * Calendar events.
+ */
+ TypeCalendar = 1,
+ /**
+ * Contacts.
+ */
+ TypeContacts,
+ /**
+ * To do items.
+ */
+ TypeToDo
+} GSM_Alcatel_BinaryType;
+
+/**
+ * Alcatel internal types.
+ */
+typedef enum {
+ /**
+ * Date, stored as @ref GSM_DateTime.
+ */
+ Alcatel_date,
+ /**
+ * Time, stored as @ref GSM_DateTime.
+ */
+ Alcatel_time,
+ /**
+ * String, strored as chars
+ */
+ Alcatel_string, /* char */
+ /**
+ * Phone number (can contain anything, but dialling it then may cause
+ * strange problems to phone) strored as chars.
+ */
+ Alcatel_phone,
+ /**
+ * Enumeration, usually used for user defined values (categories),
+ * stored as int.
+ */
+ Alcatel_enum,
+ /**
+ * Boolean, stored as int.
+ */
+ Alcatel_bool,
+ /**
+ * 32-bit unsigned integer, stored as int.
+ */
+ Alcatel_int,
+ /**
+ * 8-bit usigned integer, stored as int.
+ */
+ Alcatel_byte
+} GSM_Alcatel_FieldType;
+
+/**
+ * Return value for GetMemoryStatus. There is no way ho to determine free
+ * memory so we have to return some fixed value.
+ */
+#define ALCATEL_FREE_MEMORY 100
+/**
+ * Maximum location, that will driver attempt to read. Because phone can have
+ * up to 2^32 locations, we want to limit this a bit.
+ */
+/* There COULD be 0xffffffff on next line, but this is IMHO enough */
+#define ALCATEL_MAX_LOCATION 0xffff
+/**
+ * Maximum category count. Used for static cache size.
+ */
+#define ALCATEL_MAX_CATEGORIES 100
+
+/**
+ * Alcatel driver private data.
+ */
+typedef struct {
+ /***********************************/
+ /* Communication state information */
+ /***********************************/
+ /**
+ * Mode of current communication.
+ */
+ GSM_Alcatel_Mode Mode;
+ /**
+ * State of current communication if phone is in binary mode.
+ */
+ GSM_Alcatel_BinaryState BinaryState;
+ /**
+ * Type of current communication if phone is in session or edit state,
+ * zero otherwise.
+ */
+ GSM_Alcatel_BinaryType BinaryType;
+ /**
+ * Currently edited item when phone is in edit state, zero otherwise.
+ */
+ int BinaryItem;
+ /**
+ * Protocol version being used.
+ */
+ GSM_Alcatel_ProtocolVersion ProtocolVersion;
+
+ /*****************/
+ /* Return values */
+ /*****************/
+ /**
+ * Return value for commited record position.
+ */
+ int CommitedRecord;
+ /**
+ * Used for detecting end of large data data, that don't fit in one
+ * message.
+ */
+ bool TransferCompleted;
+ /**
+ * Type of currently received field.
+ */
+ GSM_Alcatel_FieldType ReturnType;
+ /**
+ * Value of currently returned field (if it can be saved in DateTime).
+ */
+ GSM_DateTime ReturnDateTime;
+ /**
+ * Value of currently returned field (if it can be saved in int).
+ */
+ unsigned int ReturnInt;
+ /**
+ * Value of currently returned field (if it can be saved in string).
+ */
+ unsigned char ReturnString[(GSM_PHONEBOOK_TEXT_LENGTH+1)*2];
+
+ /***********/
+ /* Caches: */
+ /***********/
+ /* Listings of available items for each type */
+ /**
+ * Pointer to list of active items.
+ */
+ int **CurrentList;
+ /**
+ * Pointer to currently count of active items.
+ */
+ int *CurrentCount;
+ /**
+ * Items locations in calendar.
+ */
+ int *CalendarItems;
+ /**
+ * Items locations in to do list.
+ */
+ int *ToDoItems;
+ /**
+ * Items locations in contacts.
+ */
+ int *ContactsItems;
+ /**
+ * Count of calendar items stored in @ref CalendarItems.
+ */
+ int CalendarItemsCount;
+ /**
+ * Count of todo items stored in @ref ToDoItems.
+ */
+ int ToDoItemsCount;
+ /**
+ * Count of contacts items stored in @ref ContactsItems.
+ */
+ int ContactsItemsCount;
+ /**
+ * Fields of currently active item.
+ */
+ int CurrentFields[GSM_PHONEBOOK_ENTRIES+1];
+ /**
+ * Count of fields listed in @ref CurrentFields.
+ */
+ int CurrentFieldsCount;
+ /**
+ * Location of current (eg. which identifies @ref CurrentFieldsCount
+ * and @ref CurrentFields) item.
+ */
+ int CurrentFieldsItem;
+ /**
+ * Type of current (eg. which identifies @ref CurrentFieldsCount
+ * and @ref CurrentFields) item.
+ */
+ GSM_Alcatel_BinaryType CurrentFieldsType;
+
+ /**
+ * Listing of categories of current type.
+ */
+ int CurrentCategories[ALCATEL_MAX_CATEGORIES+1];
+ /**
+ * Cache for category names of current type. The index here is not
+ * same as in @ref CurrentCategories, it is id of category here.
+ */
+ char CurrentCategoriesCache[ALCATEL_MAX_CATEGORIES+1][(GSM_MAX_CATEGORY_NAME_LENGTH + 1)*2];
+ /**
+ * Count of entries in @ref CurrentCategories.
+ */
+ int CurrentCategoriesCount;
+ /**
+ * Type of current categories in @ref CurrentCategories etc.
+ */
+ GSM_Alcatel_BinaryType CurrentCategoriesType;
+
+} GSM_Phone_ALCATELData;
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/at/atgen.c b/gammu/emb/common/phone/at/atgen.c
new file mode 100644
index 0000000..1834f15
--- a/dev/null
+++ b/gammu/emb/common/phone/at/atgen.c
@@ -0,0 +1,3669 @@
+/* (c) 2002-2004 by Marcin Wiacek and Michal Cihar */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_ATGEN
+
+#include <string.h>
+#include <time.h>
+#include <ctype.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+#include "../../service/sms/gsmsms.h"
+#include "../pfunc.h"
+#include "atgen.h"
+
+#ifdef GSM_ENABLE_ALCATEL
+extern GSM_Error ALCATEL_ProtocolVersionReply (GSM_Protocol_Message msg, GSM_StateMachine *s);
+#endif
+
+extern GSM_Error ATGEN_CMS35ReplyGetBitmap (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplySetBitmap (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplyGetRingtone (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplySetRingtone (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplyGetNextCal (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplySetCalendar (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_CMS35ReplyDeleteCalendar (GSM_Protocol_Message msg, GSM_StateMachine *s);
+extern GSM_Error ATGEN_SL45ReplyGetMemory (GSM_Protocol_Message msg, GSM_StateMachine *s);
+
+extern GSM_Error ATGEN_GetRingtone (GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone);
+extern GSM_Error ATGEN_SetRingtone (GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength);
+extern GSM_Error ATGEN_GetBitmap (GSM_StateMachine *s, GSM_Bitmap *Bitmap);
+extern GSM_Error ATGEN_SetBitmap (GSM_StateMachine *s, GSM_Bitmap *Bitmap);
+extern GSM_Error SIEMENS_GetNextCalendar (GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start);
+extern GSM_Error SIEMENS_AddCalendarNote (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+extern GSM_Error SIEMENS_DelCalendarNote (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+
+extern GSM_Error SONYERIC_GetNextCalendar (GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start);
+extern GSM_Error SONYERIC_GetNextToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool start);
+extern GSM_Error SONYERIC_GetToDoStatus (GSM_StateMachine *s, GSM_ToDoStatus *status);
+extern GSM_Error SONYERIC_AddCalendarNote (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+extern GSM_Error SONYERIC_AddToDo (GSM_StateMachine *s, GSM_ToDoEntry *ToDo);
+extern GSM_Error SONYERIC_DeleteAllToDo (GSM_StateMachine *s);
+extern GSM_Error SONYERIC_DelCalendarNote (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+extern GSM_Error SONYERIC_GetCalendarStatus (GSM_StateMachine *s, GSM_CalendarStatus *Status);
+
+typedef struct {
+ int Number;
+ char Text[60];
+} ATErrorCode;
+
+static ATErrorCode CMSErrorCodes[] = {
+ /*
+ * Error codes not specified here were either undefined or reserved in my
+ * copy of specifications, if you have newer one, please fill in the gaps.
+ */
+ /* 0...127 from GSM 04.11 Annex E-2 */
+ {1, "Unassigned (unallocated) number"},
+ {8, "Operator determined barring"},
+ {10, "Call barred"},
+ {21, "Short message transfer rejected"},
+ {27, "Destination out of service"},
+ {28, "Unidentified subscriber"},
+ {29, "Facility rejected"},
+ {30, "Unknown subscriber"},
+ {38, "Network out of order"},
+ {41, "Temporary failure"},
+ {42, "Congestion"},
+ {47, "Resources unavailable, unspecified"},
+ {50, "Requested facility not subscribed"},
+ {69, "Requested facility not implemented"},
+ {81, "Invalid short message transfer reference value"},
+ {95, "Invalid message, unspecified"},
+ {96, "Invalid mandatory information"},
+ {97, "Message type non-existent or not implemented"},
+ {98, "Message not compatible with short message protocol state"},
+ {99, "Information element non-existent or not implemented"},
+ {111, "Protocol error, unspecified"},
+ {127, "Interworking, unspecified"},
+ /* 128...255 from GSM 03.40 subclause 9.2.3.22 */
+ {0x80, "Telematic interworking not supported"},
+ {0x81, "Short message Type 0 not supported"},
+ {0x82, "Cannot replace short message"},
+ {0x8F, "Unspecified TP-PID error"},
+ {0x90, "Data coding scheme (alphabet) not supported"},
+ {0x91, "Message class not supported"},
+ {0x9F, "Unspecified TP-DCS error"},
+ {0xA0, "Command cannot be actioned"},
+ {0xA1, "Command unsupported"},
+ {0xAF, "Unspecified TP-Command error"},
+ {0xB0, "TPDU not supported"},
+ {0xC0, "SC busy"},
+ {0xC1, "No SC subscription"},
+ {0xC2, "SC system failure"},
+ {0xC3, "Invalid SME address"},
+ {0xC4, "Destination SME barred"},
+ {0xC5, "SM Rejected-Duplicate SM"},
+ {0xC6, "TP-VPF not supported"},
+ {0xC7, "TP-VP not supported"},
+ {0xD0, "SIM SMS storage full"},
+ {0xD1, "No SMS storage capability in SIM"},
+ {0xD2, "Error in MS"},
+ {0xD3, "Memory Capacity Exceede"},
+ {0xD4, "SIM Application Toolkit Busy"},
+ {0xFF, "Unspecified error cause"},
+ /* 300...511 from GSM 07.05 subclause 3.2.5 */
+ {300, "ME failure"},
+ {301, "SMS service of ME reserved"},
+ {302, "operation not allowed"},
+ {303, "operation not supported"},
+ {304, "invalid PDU mode parameter"},
+ {305, "invalid text mode parameter"},
+ {310, "SIM not inserted"},
+ {311, "SIM PIN required"},
+ {312, "PH-SIM PIN required"},
+ {313, "SIM failure"},
+ {314, "SIM busy"},
+ {315, "SIM wrong"},
+ {316, "SIM PUK required"},
+ {317, "SIM PIN2 required"},
+ {318, "SIM PUK2 required"},
+ {320, "memory failure"},
+ {321, "invalid memory index"},
+ {322, "memory full"},
+ {330, "SMSC address unknown"},
+ {331, "no network service"},
+ {332, "network timeout"},
+ {340, "no CNMA acknowledgement expected"},
+ {500, "unknown error"},
+ /* > 512 are manufacturer specific according to GSM 07.05 subclause 3.2.5 */
+ {-1, ""}
+};
+
+static ATErrorCode CMEErrorCodes[] = {
+ /* CME Error codes from GSM 07.07 section 9.2 */
+ {0, "phone failure"},
+ {1, "no connection to phone"},
+ {2, "phone-adaptor link reserved"},
+ {3, "operation not allowed"},
+ {4, "operation not supported"},
+ {5, "PH-SIM PIN required"},
+ {10, "SIM not inserted"},
+ {11, "SIM PIN required"},
+ {12, "SIM PUK required"},
+ {13, "SIM failure"},
+ {14, "SIM busy"},
+ {15, "SIM wrong"},
+ {16, "incorrect password"},
+ {17, "SIM PIN2 required"},
+ {18, "SIM PUK2 required"},
+ {20, "memory full"},
+ {21, "invalid index"},
+ {22, "not found"},
+ {23, "memory failure"},
+ {24, "text string too long"},
+ {25, "invalid characters in text string"},
+ {26, "dial string too long"},
+ {27, "invalid characters in dial string"},
+ {30, "no network service"},
+ {31, "network timeout"},
+ {100, "unknown"},
+};
+
+
+GSM_Error ATGEN_HandleCMEError(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->ErrorCode == 0) {
+ smprintf(s, "CME Error occured, but it's type not detected\n");
+ } else if (Priv->ErrorText == NULL) {
+ smprintf(s, "CME Error %i, no description available\n", Priv->ErrorCode);
+ } else {
+ smprintf(s, "CME Error %i: \"%s\"\n", Priv->ErrorCode, Priv->ErrorText);
+ }
+ /* For error codes descriptions see table a bit above */
+ switch (Priv->ErrorCode) {
+ case 3:
+ return ERR_PERMISSION;
+ case 4:
+ return ERR_NOTSUPPORTED;
+ case 5:
+ case 11:
+ case 12:
+ case 16:
+ case 17:
+ case 18:
+ return ERR_SECURITYERROR;
+ case 20:
+ return ERR_FULL;
+ case 21:
+ return ERR_INVALIDLOCATION;
+ case 22:
+ return ERR_EMPTY;
+ case 23:
+ return ERR_MEMORY;
+ case 24:
+ case 25:
+ case 26:
+ case 27:
+ return ERR_INVALIDDATA;
+ default:
+ return ERR_UNKNOWN;
+ }
+}
+
+GSM_Error ATGEN_HandleCMSError(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->ErrorCode == 0) {
+ smprintf(s, "CMS Error occured, but it's type not detected\n");
+ } else if (Priv->ErrorText == NULL) {
+ smprintf(s, "CMS Error %i, no description available\n", Priv->ErrorCode);
+ } else {
+ smprintf(s, "CMS Error %i: \"%s\"\n", Priv->ErrorCode, Priv->ErrorText);
+ }
+ /* For error codes descriptions see table a bit above */
+ switch (Priv->ErrorCode) {
+ case 304:
+ return ERR_NOTSUPPORTED;
+ case 305:
+ return ERR_BUG;
+ case 311:
+ case 312:
+ case 316:
+ case 317:
+ case 318:
+ return ERR_SECURITYERROR;
+ case 322:
+ return ERR_FULL;
+ case 321:
+ return ERR_INVALIDLOCATION;
+ default:
+ return ERR_UNKNOWN;
+ }
+}
+
+/* FIXME: Function doesn't respect quoting of parameters and thus +FOO:
+ * "ab","cd,ef" will consider as three arguments: "ab" >> "cd >> ef"
+ */
+int ATGEN_ExtractOneParameter(unsigned char *input, unsigned char *output)
+{
+ int position=0;
+
+ while (*input!=',' && *input!=0x0d && *input!=0x00) {
+ *output=*input;
+ input ++;
+ output ++;
+ position++;
+ }
+ *output=0;
+ position++;
+ return position;
+}
+
+void ATGEN_DecodeDateTime(GSM_DateTime *dt, unsigned char *input)
+{
+ dt->Year=2000+(*input-'0')*10; input++;
+ dt->Year=dt->Year+(*input-'0'); input++;
+
+ input++;
+ dt->Month=(*input-'0')*10; input++;
+ dt->Month=dt->Month+(*input-'0'); input++;
+
+ input++;
+ dt->Day=(*input-'0')*10; input++;
+ dt->Day=dt->Day+(*input-'0'); input++;
+
+ input++;
+ dt->Hour=(*input-'0')*10; input++;
+ dt->Hour=dt->Hour+(*input-'0'); input++;
+
+ input++;
+ dt->Minute=(*input-'0')*10; input++;
+ dt->Minute=dt->Minute+(*input-'0');input++;
+
+ input++;
+ dt->Second=(*input-'0')*10; input++;
+ dt->Second=dt->Second+(*input-'0');input++;
+
+ if (input!=NULL) {
+ input++;
+ dt->Timezone=(*input-'0')*10; input++;
+ dt->Timezone=dt->Timezone+(*input-'0');input++;
+ input=input-2;
+ if (*input=='-') dt->Timezone=-dt->Timezone;
+ }
+}
+
+GSM_Error ATGEN_DispatchMessage(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Protocol_Message *msg = s->Phone.Data.RequestMsg;
+ int i = 0, j, k;
+ char *err, *line;
+ ATErrorCode *ErrorCodes = NULL;
+
+ SplitLines(msg->Buffer, msg->Length, &Priv->Lines, "\x0D\x0A", 2, true);
+
+ /* Find number of lines */
+ while (Priv->Lines.numbers[i*2+1] != 0) {
+ /* FIXME: handle special chars correctly */
+ smprintf(s, "%i \"%s\"\n",i+1,GetLineString(msg->Buffer,Priv->Lines,i+1));
+ i++;
+ }
+
+ Priv->ReplyState = AT_Reply_Unknown;
+ Priv->ErrorText = NULL;
+ Priv->ErrorCode = 0;
+
+ line = GetLineString(msg->Buffer,Priv->Lines,i);
+ if (!strcmp(line,"OK")) Priv->ReplyState = AT_Reply_OK;
+ if (!strcmp(line,"> ")) Priv->ReplyState = AT_Reply_SMSEdit;
+ if (!strcmp(line,"CONNECT")) Priv->ReplyState = AT_Reply_Connect;
+ if (!strcmp(line,"ERROR" )) Priv->ReplyState = AT_Reply_Error;
+ if (!strncmp(line,"+CME ERROR:",11)) {
+ Priv->ReplyState = AT_Reply_CMEError;
+ ErrorCodes = CMEErrorCodes;
+ }
+ if (!strncmp(line,"+CMS ERROR:",11)) {
+ Priv->ReplyState = AT_Reply_CMSError;
+ ErrorCodes = CMSErrorCodes;
+ }
+ if (Priv->ReplyState == AT_Reply_CMEError || Priv->ReplyState == AT_Reply_CMSError) {
+ j = 0;
+ /* One char behind +CM[SE] ERROR */
+ err = line + 12;
+ while (err[j] && !isalnum(err[j])) j++;
+ if (isdigit(err[j])) {
+ Priv->ErrorCode = atoi(&(err[j]));
+ k = 0;
+ while (ErrorCodes[k].Number != -1) {
+ if (ErrorCodes[k].Number == Priv->ErrorCode) {
+ Priv->ErrorText = (char *)&(ErrorCodes[k].Text);
+ break;
+ }
+ k++;
+ }
+ } else if (isalpha(err[j])) {
+ k = 0;
+ while (ErrorCodes[k].Number != -1) {
+ if (!strncmp(err + j, ErrorCodes[k].Text, strlen(ErrorCodes[k].Text))) {
+ Priv->ErrorCode = ErrorCodes[k].Number;
+ Priv->ErrorText = (char *)&(ErrorCodes[k].Text);
+ break;
+ }
+ k++;
+ }
+ }
+ }
+ return GSM_DispatchMessage(s);
+}
+
+GSM_Error ATGEN_GenericReply(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ case AT_Reply_Connect:
+ return ERR_NONE;
+ case AT_Reply_Error:
+ return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ case AT_Reply_CMEError:
+ return ATGEN_HandleCMEError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_ReplyGetModel(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ if (s->Phone.Data.Priv.ATGEN.ReplyState != AT_Reply_OK) return ERR_NOTSUPPORTED;
+
+ if (strlen(GetLineString(msg.Buffer, Priv->Lines, 2)) <= MAX_MODEL_LENGTH) {
+ CopyLineString(Data->Model, msg.Buffer, Priv->Lines, 2);
+
+ /* Sometimes phone adds this before manufacturer (Sagem) */
+ if (strncmp("+CGMM: ", Data->Model, 7) == 0) {
+ memmove(Data->Model, Data->Model + 7, strlen(Data->Model + 7) + 1);
+ }
+
+ Data->ModelInfo = GetModelData(NULL,Data->Model,NULL);
+ if (Data->ModelInfo->number[0] == 0) Data->ModelInfo = GetModelData(NULL,NULL,Data->Model);
+ if (Data->ModelInfo->number[0] == 0) Data->ModelInfo = GetModelData(Data->Model,NULL,NULL);
+
+ if (Data->ModelInfo->number[0] != 0) strcpy(Data->Model,Data->ModelInfo->number);
+
+ if (strstr(msg.Buffer,"Nokia")) Priv->Manufacturer = AT_Nokia;
+ else if (strstr(msg.Buffer,"M20")) Priv->Manufacturer = AT_Siemens;
+ else if (strstr(msg.Buffer,"MC35")) Priv->Manufacturer = AT_Siemens;
+ else if (strstr(msg.Buffer, "iPAQ")) Priv->Manufacturer = AT_HP;
+
+ if (strstr(msg.Buffer,"M20")) strcpy(Data->Model,"M20");
+ else if (strstr(msg.Buffer,"MC35")) strcpy(Data->Model,"MC35");
+ else if (strstr(msg.Buffer, "iPAQ")) strcpy(Data->Model,"iPAQ");
+ } else {
+ smprintf(s, "WARNING: Model name too long, increase MAX_MODEL_LENGTH to at least %zd\n", strlen(GetLineString(msg.Buffer, Priv->Lines, 2)));
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_GetModel(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if (s->Phone.Data.Model[0] != 0) return ERR_NONE;
+
+ smprintf(s, "Getting model\n");
+ error=GSM_WaitFor (s, "AT+CGMM\r", 8, 0x00, 3, ID_GetModel);
+ if (error==ERR_NONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "[Connected model - \"%s\"]\n",s->Phone.Data.Model);
+ }
+ }
+ return error;
+}
+
+GSM_Error ATGEN_ReplyGetManufacturer(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Manufacturer info received\n");
+ Priv->Manufacturer = AT_Unknown;
+ if (strlen(GetLineString(msg.Buffer, Priv->Lines, 2)) <= MAX_MANUFACTURER_LENGTH) {
+ CopyLineString(s->Phone.Data.Manufacturer, msg.Buffer, Priv->Lines, 2);
+ } else {
+ smprintf(s, "WARNING: Manufacturer name too long, increase MAX_MANUFACTURER_LENGTH to at least %zd\n", strlen(GetLineString(msg.Buffer, Priv->Lines, 2)));
+ s->Phone.Data.Manufacturer[0] = 0;
+ }
+ /* Sometimes phone adds this before manufacturer (Sagem) */
+ if (strncmp("+CGMI: ", s->Phone.Data.Manufacturer, 7) == 0) {
+ memmove(s->Phone.Data.Manufacturer, s->Phone.Data.Manufacturer + 7, strlen(s->Phone.Data.Manufacturer + 7) + 1);
+ }
+ if (strstr(msg.Buffer,"Falcom")) {
+ smprintf(s, "Falcom\n");
+ strcpy(s->Phone.Data.Manufacturer,"Falcom");
+ Priv->Manufacturer = AT_Falcom;
+ if (strstr(msg.Buffer,"A2D")) {
+ strcpy(s->Phone.Data.Model,"A2D");
+ s->Phone.Data.ModelInfo = GetModelData(NULL,s->Phone.Data.Model,NULL);
+ smprintf(s, "Model A2D\n");
+ }
+ }
+ if (strstr(msg.Buffer,"Nokia")) {
+ smprintf(s, "Nokia\n");
+ strcpy(s->Phone.Data.Manufacturer,"Nokia");
+ Priv->Manufacturer = AT_Nokia;
+ }
+ if (strstr(msg.Buffer,"SIEMENS")) {
+ smprintf(s, "Siemens\n");
+ strcpy(s->Phone.Data.Manufacturer,"Siemens");
+ Priv->Manufacturer = AT_Siemens;
+ }
+ if (strstr(msg.Buffer,"ERICSSON")) {
+ smprintf(s, "Ericsson\n");
+ strcpy(s->Phone.Data.Manufacturer,"Ericsson");
+ Priv->Manufacturer = AT_Ericsson;
+ }
+ if (strstr(msg.Buffer,"iPAQ")) {
+ smprintf(s, "iPAQ\n");
+ strcpy(s->Phone.Data.Manufacturer,"HP");
+ Priv->Manufacturer = AT_HP;
+ }
+ if (strstr(msg.Buffer,"ALCATEL")) {
+ smprintf(s, "Alcatel\n");
+ strcpy(s->Phone.Data.Manufacturer,"Alcatel");
+ Priv->Manufacturer = AT_Alcatel;
+ }
+ if (strstr(msg.Buffer,"SAGEM")) {
+ smprintf(s, "Sagem\n");
+ strcpy(s->Phone.Data.Manufacturer,"Sagem");
+ Priv->Manufacturer = AT_Sagem;
+ }
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetManufacturer(GSM_StateMachine *s)
+{
+ if (s->Phone.Data.Manufacturer[0] != 0) return ERR_NONE;
+
+ return GSM_WaitFor (s, "AT+CGMI\r", 8, 0x00, 4, ID_GetManufacturer);
+}
+
+GSM_Error ATGEN_ReplyGetFirmwareCGMR(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ unsigned int i = 0;
+
+ strcpy(s->Phone.Data.Version,"unknown");
+ s->Phone.Data.VerNum = 0;
+ if (Priv->ReplyState == AT_Reply_OK) {
+ CopyLineString(s->Phone.Data.Version, msg.Buffer, Priv->Lines, 2);
+ /* Sometimes phone adds this before manufacturer (Sagem) */
+ if (strncmp("+CGMR: ", s->Phone.Data.Version, 7) == 0) {
+ memmove(s->Phone.Data.Version, s->Phone.Data.Version + 7, strlen(s->Phone.Data.Version + 7) + 1);
+ }
+ }
+ if (Priv->Manufacturer == AT_Ericsson) {
+ while (1) {
+ if (s->Phone.Data.Version[i] == 0x20) {
+ s->Phone.Data.Version[i] = 0x00;
+ break;
+ }
+ if (i == strlen(s->Phone.Data.Version)) break;
+ i++;
+ }
+ }
+ smprintf(s, "Received firmware version: \"%s\"\n",s->Phone.Data.Version);
+ GSM_CreateFirmwareNumber(s);
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyGetFirmwareATI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+// strcpy(Data->Version,"0.00");
+// *Data->VersionNum=0;
+// if (Data->Priv.ATGEN.ReplyState==AT_Reply_OK) {
+// CopyLineString(Data->Version, msg.Buffer, Priv->Lines, 2);
+// }
+// smprintf(s, "Received firmware version: \"%s\"\n",Data->Version);
+// GSM_CreateFirmwareNumber(Data);
+// return ERR_NONE;
+ case AT_Reply_Error:
+ return ERR_NOTSUPPORTED;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetFirmware(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if (s->Phone.Data.Version[0] != 0) return ERR_NONE;
+
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+
+// smprintf(s, "Getting firmware - method 1\n");
+// error=GSM_WaitFor (s, "ATI\r", 4, 0x00, 3, ID_GetFirmware);
+// if (error != ERR_NONE) {
+ smprintf(s, "Getting firmware - method 2\n");
+ error=GSM_WaitFor (s, "AT+CGMR\r", 8, 0x00, 3, ID_GetFirmware);
+// }
+ if (error==ERR_NONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "[Firmware version - \"%s\"]\n",s->Phone.Data.Version);
+ }
+ }
+ return error;
+}
+
+GSM_Error ATGEN_Initialise(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ char buff[2];
+
+ Priv->SMSMode = 0;
+ Priv->Manufacturer = 0;
+ Priv->PhoneSMSMemory = 0;
+ Priv->CanSaveSMS = false;
+ Priv->SIMSMSMemory = 0;
+ Priv->SMSMemory = 0;
+ Priv->PBKMemory = 0;
+ Priv->PBKSBNR = 0;
+ Priv->PBKCharset = 0;
+ Priv->UCS2CharsetFailed = false;
+ Priv->NonUCS2CharsetFailed = false;
+ Priv->PBKMemories[0] = 0;
+ Priv->FirstCalendarPos = 0;
+ Priv->NextMemoryEntry = 0;
+ Priv->FirstMemoryEntry = 0;
+ Priv->file.Used = 0;
+ Priv->file.Buffer = NULL;
+ Priv->OBEX = false;
+ Priv->MemorySize = 0;
+ Priv->TextLength = 0;
+ Priv->NumberLength = 0;
+
+ Priv->ErrorText = NULL;
+
+ if (s->ConnectionType != GCT_IRDAAT && s->ConnectionType != GCT_BLUEAT) {
+ /* We try to escape AT+CMGS mode, at least Siemens M20
+ * then needs to get some rest
+ */
+ smprintf(s, "Escaping SMS mode\n");
+ error = s->Protocol.Functions->WriteMessage(s, "\x1B\r", 2, 0x00);
+ if (error!=ERR_NONE) return error;
+
+ /* Grab any possible garbage */
+ while (s->Device.Functions->ReadDevice(s, buff, 2) > 0) my_sleep(10);
+ }
+
+ /* When some phones (Alcatel BE5) is first time connected, it needs extra
+ * time to react, sending just AT wakes up the phone and it then can react
+ * to ATE1. We don't need to check whether this fails as it is just to
+ * wake up the phone and does nothing.
+ */
+ smprintf(s, "Sending simple AT command to wake up some devices\n");
+ GSM_WaitFor (s, "AT\r", 3, 0x00, 2, ID_IncomingFrame);
+
+ smprintf(s, "Enabling echo\n");
+ error = GSM_WaitFor (s, "ATE1\r", 5, 0x00, 3, ID_EnableEcho);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Enabling CME errors\n");
+ /* Try numeric errors */
+ if (GSM_WaitFor (s, "AT+CMEE=1\r", 10, 0x00, 3, ID_EnableErrorInfo) != ERR_NONE) {
+ /* Try textual errors */
+ if (GSM_WaitFor (s, "AT+CMEE=2\r", 10, 0x00, 3, ID_EnableErrorInfo) != ERR_NONE) {
+ smprintf(s, "CME errors could not be enabled, some error types won't be detected.\n");
+ }
+ }
+
+ error = ATGEN_GetModel(s);
+ if (error != ERR_NONE) return error;
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_SLOWWRITE)) {
+ s->Protocol.Data.AT.FastWrite = true;
+ }
+
+ return error;
+}
+
+GSM_Error ATGEN_SetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ unsigned char req[50];
+
+ if (smsc->Location!=1) return ERR_NOTSUPPORTED;
+
+ sprintf(req, "AT+CSCA=\"%s\"\r",DecodeUnicodeString(smsc->Number));
+
+ smprintf(s, "Setting SMSC\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_SetSMSC);
+}
+
+GSM_Error ATGEN_ReplyGetSMSMemories(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ /* Reply here is:
+ * (memories for reading)[, (memories for writing)[, (memories for storing received messages)]]
+ * each memory is in quotes,
+ * Example: ("SM"), ("SM"), ("SM")
+ *
+ * We need to get from this supported memories. For this case
+ * we assume, that just appearence of memory makes it
+ * available for everything. Then we need to find out whether
+ * phone supports writing to memory. This is done by searching
+ * for "), (", which will appear between lists.
+ */
+ s->Phone.Data.Priv.ATGEN.CanSaveSMS = (strstr(msg.Buffer, "), (") != NULL);
+ if (strstr(msg.Buffer, "\"SM\"") != NULL) s->Phone.Data.Priv.ATGEN.SIMSMSMemory = AT_AVAILABLE;
+ else s->Phone.Data.Priv.ATGEN.SIMSMSMemory = AT_NOTAVAILABLE;
+ if (strstr(msg.Buffer, "\"ME\"") != NULL) s->Phone.Data.Priv.ATGEN.PhoneSMSMemory = AT_AVAILABLE;
+ else s->Phone.Data.Priv.ATGEN.PhoneSMSMemory = AT_NOTAVAILABLE;
+ smprintf(s, "Available SMS memories received, ME = %d, SM = %d\n", s->Phone.Data.Priv.ATGEN.PhoneSMSMemory, s->Phone.Data.Priv.ATGEN.SIMSMSMemory);
+ return ERR_NONE;
+ case AT_Reply_Error:
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error ATGEN_GetSMSMemories(GSM_StateMachine *s)
+{
+ smprintf(s, "Getting available SMS memories\n");
+ return GSM_WaitFor (s, "AT+CPMS=?\r", 10, 0x00, 4, ID_GetSMSMemories);
+}
+
+GSM_Error ATGEN_SetSMSMemory(GSM_StateMachine *s, bool SIM)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ char req[] = "AT+CPMS=\"XX\",\"XX\"\r";
+ int reqlen = 18;
+ GSM_Error error;
+
+ if ((SIM && Priv->SIMSMSMemory == 0) || (!SIM && Priv->PhoneSMSMemory == 0)) {
+ /* We silently ignore error here, because when this fails, we can try to setmemory anyway */
+ ATGEN_GetSMSMemories(s);
+ }
+
+ /* If phone can not save SMS, don't try to set memory for saving */
+ if (!Priv->CanSaveSMS) {
+ req[12] = '\r';
+ reqlen = 13;
+ }
+
+ if (SIM) {
+ if (Priv->SMSMemory == MEM_SM) return ERR_NONE;
+ if (Priv->SIMSMSMemory == AT_NOTAVAILABLE) return ERR_NOTSUPPORTED;
+
+ req[9] = 'S'; req[10] = 'M';
+ req[14] = 'S'; req[15] = 'M';
+
+ smprintf(s, "Setting SMS memory type to SM\n");
+ error=GSM_WaitFor (s, req, reqlen, 0x00, 3, ID_SetMemoryType);
+ if (Priv->SIMSMSMemory == 0 && error == ERR_NONE) {
+ Priv->SIMSMSMemory = AT_AVAILABLE;
+ }
+ if (error == ERR_NOTSUPPORTED) {
+ smprintf(s, "Can't access SIM card?\n");
+ return ERR_SECURITYERROR;
+ }
+ if (error != ERR_NONE) return error;
+ Priv->SMSMemory = MEM_SM;
+ } else {
+ if (Priv->SMSMemory == MEM_ME) return ERR_NONE;
+ if (Priv->PhoneSMSMemory == AT_NOTAVAILABLE) return ERR_NOTSUPPORTED;
+
+ req[9] = 'M'; req[10] = 'E';
+ req[14] = 'M'; req[15] = 'E';
+
+ smprintf(s, "Setting SMS memory type to ME\n");
+ error=GSM_WaitFor (s, req, reqlen, 0x00, 3, ID_SetMemoryType);
+ if (Priv->PhoneSMSMemory == 0 && error == ERR_NONE) {
+ Priv->PhoneSMSMemory = AT_AVAILABLE;
+ }
+ if (error == ERR_NONE) Priv->SMSMemory = MEM_ME;
+ }
+ return error;
+}
+
+GSM_Error ATGEN_GetSMSMode(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+
+ if (Priv->SMSMode != 0) return ERR_NONE;
+
+ smprintf(s, "Trying SMS PDU mode\n");
+ error=GSM_WaitFor (s, "AT+CMGF=0\r", 10, 0x00, 3, ID_GetSMSMode);
+ if (error==ERR_NONE) {
+ Priv->SMSMode = SMS_AT_PDU;
+ return ERR_NONE;
+ }
+
+ smprintf(s, "Trying SMS text mode\n");
+ error=GSM_WaitFor (s, "AT+CMGF=1\r", 10, 0x00, 3, ID_GetSMSMode);
+ if (error==ERR_NONE) {
+ smprintf(s, "Enabling displaying all parameters in text mode\n");
+ error=GSM_WaitFor (s, "AT+CSDH=1\r", 10, 0x00, 3, ID_GetSMSMode);
+ if (error == ERR_NONE) Priv->SMSMode = SMS_AT_TXT;
+ }
+
+ return error;
+}
+
+GSM_Error ATGEN_GetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char *folderid, int *location)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ int ifolderid, maxfolder;
+ GSM_Error error;
+
+ if (Priv->PhoneSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, false);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+ if (Priv->SIMSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, true);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+
+ if (Priv->SIMSMSMemory != AT_AVAILABLE && Priv->PhoneSMSMemory != AT_AVAILABLE) {
+ /* No SMS memory at all */
+ return ERR_NOTSUPPORTED;
+ }
+ if (Priv->SIMSMSMemory == AT_AVAILABLE && Priv->PhoneSMSMemory == AT_AVAILABLE) {
+ /* Both available */
+ maxfolder = 2;
+ } else {
+ /* One available */
+ maxfolder = 1;
+ }
+
+ /* simulate flat SMS memory */
+ if (sms->Folder == 0x00) {
+ ifolderid = sms->Location / PHONE_MAXSMSINFOLDER;
+ if (ifolderid + 1 > maxfolder) return ERR_NOTSUPPORTED;
+ *folderid = ifolderid + 1;
+ *location = sms->Location - ifolderid * PHONE_MAXSMSINFOLDER;
+ } else {
+ if (sms->Folder > 2 * maxfolder) return ERR_NOTSUPPORTED;
+ *folderid = sms->Folder <= 2 ? 1 : 2;
+ *location = sms->Location;
+ }
+ smprintf(s, "SMS folder %i & location %i -> ATGEN folder %i & location %i\n",
+ sms->Folder,sms->Location,*folderid,*location);
+
+ if (Priv->SIMSMSMemory == AT_AVAILABLE && *folderid == 1) {
+ return ATGEN_SetSMSMemory(s, true);
+ } else {
+ return ATGEN_SetSMSMemory(s, false);
+ }
+}
+
+void ATGEN_SetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char folderid, int location)
+{
+ sms->Folder = 0;
+ sms->Location = (folderid - 1) * PHONE_MAXSMSINFOLDER + location;
+ smprintf(s, "ATGEN folder %i & location %i -> SMS folder %i & location %i\n",
+ folderid,location,sms->Folder,sms->Location);
+}
+
+GSM_Error ATGEN_ReplyGetSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_SMSMessage *sms = &s->Phone.Data.GetSMSMessage->SMS[0];
+ int current = 0, current2, i;
+ unsigned char buffer[300],smsframe[800];
+ unsigned char firstbyte, TPDCS, TPUDL, TPStatus;
+ GSM_Error error;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ if (Priv->Lines.numbers[4] == 0x00) return ERR_EMPTY;
+ s->Phone.Data.GetSMSMessage->Number = 1;
+ s->Phone.Data.GetSMSMessage->SMS[0].Name[0] = 0;
+ s->Phone.Data.GetSMSMessage->SMS[0].Name[1] = 0;
+ switch (Priv->SMSMode) {
+ case SMS_AT_PDU:
+ CopyLineString(buffer, msg.Buffer, Priv->Lines, 2);
+ switch (buffer[7]) {
+ case '0': sms->State = SMS_UnRead; break;
+ case '1': sms->State = SMS_Read; break;
+ case '2': sms->State = SMS_UnSent; break;
+ default : sms->State = SMS_Sent; break;//case '3'
+ }
+ DecodeHexBin (buffer, GetLineString(msg.Buffer,Priv->Lines,3), strlen(GetLineString(msg.Buffer,Priv->Lines,3)));
+ /* Siemens MC35 (only ?) */
+ if (strstr(msg.Buffer,"+CMGR: 0,,0")!=NULL) return ERR_EMPTY;
+ /* Siemens M20 */
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_M20SMS)) {
+ if (buffer[1]!=NUMBER_UNKNOWN && buffer[1]!=NUMBER_INTERNATIONAL &&
+ buffer[1]!=NUMBER_ALPHANUMERIC) {
+ /* Seems to be Delivery Report */
+ smprintf(s, "SMS type - status report (M20 style)\n");
+ sms->PDU = SMS_Status_Report;
+ sms->Folder = 1; /*INBOX SIM*/
+ sms->InboxFolder = true;
+
+ smsframe[12]=buffer[current++];
+ smsframe[PHONE_SMSStatusReport.TPMR]=buffer[current++];
+ current2=((buffer[current])+1)/2+1;
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSStatusReport.Number+i]=buffer[current++];
+ for(i=0;i<7;i++) smsframe[PHONE_SMSStatusReport.DateTime+i]=buffer[current++];
+ smsframe[0] = 0;
+ for(i=0;i<7;i++) smsframe[PHONE_SMSStatusReport.SMSCTime+i]=buffer[current++];
+ smsframe[PHONE_SMSStatusReport.TPStatus]=buffer[current];
+ GSM_DecodeSMSFrame(sms,smsframe,PHONE_SMSStatusReport);
+ return ERR_NONE;
+ }
+ }
+ /* We use locations from SMS layouts like in ../phone2.c(h) */
+ for(i=0;i<buffer[0]+1;i++) smsframe[i]=buffer[current++];
+ smsframe[12]=buffer[current++];
+ /* See GSM 03.40 section 9.2.3.1 */
+ switch (smsframe[12] & 0x03) {
+ case 0x00:
+ smprintf(s, "SMS type - deliver\n");
+ sms->PDU = SMS_Deliver;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 1; /*INBOX SIM*/
+ } else {
+ sms->Folder = 3; /*INBOX ME*/
+ }
+ sms->InboxFolder = true;
+ current2=((buffer[current])+1)/2+1;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_M20SMS)) {
+ if (buffer[current+1]==NUMBER_ALPHANUMERIC) {
+ smprintf(s, "Trying to read alphanumeric number\n");
+ for(i=0;i<4;i++) smsframe[PHONE_SMSDeliver.Number+i]=buffer[current++];
+ current+=6;
+ for(i=0;i<current2-3;i++) smsframe[PHONE_SMSDeliver.Number+i+4]=buffer[current++];
+ } else {
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSDeliver.Number+i]=buffer[current++];
+ }
+ } else {
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSDeliver.Number+i]=buffer[current++];
+ }
+ smsframe[PHONE_SMSDeliver.TPPID] = buffer[current++];
+ smsframe[PHONE_SMSDeliver.TPDCS] = buffer[current++];
+ for(i=0;i<7;i++) smsframe[PHONE_SMSDeliver.DateTime+i]=buffer[current++];
+ smsframe[PHONE_SMSDeliver.TPUDL] = buffer[current++];
+ for(i=0;i<smsframe[PHONE_SMSDeliver.TPUDL];i++) smsframe[i+PHONE_SMSDeliver.Text]=buffer[current++];
+ GSM_DecodeSMSFrame(sms,smsframe,PHONE_SMSDeliver);
+ return ERR_NONE;
+ case 0x01:
+ smprintf(s, "SMS type - submit\n");
+ sms->PDU = SMS_Submit;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 2; /*OUTBOX SIM*/
+ smprintf(s, "Outbox SIM\n");
+ } else {
+ sms->Folder = 4; /*OUTBOX ME*/
+ }
+ sms->InboxFolder = false;
+ smsframe[PHONE_SMSSubmit.TPMR] = buffer[current++];
+ current2=((buffer[current])+1)/2+1;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_M20SMS)) {
+ if (buffer[current+1]==NUMBER_ALPHANUMERIC) {
+ smprintf(s, "Trying to read alphanumeric number\n");
+ for(i=0;i<4;i++) smsframe[PHONE_SMSSubmit.Number+i]=buffer[current++];
+ current+=6;
+ for(i=0;i<current2-3;i++) smsframe[PHONE_SMSSubmit.Number+i+4]=buffer[current++];
+ } else {
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSSubmit.Number+i]=buffer[current++];
+ }
+ } else {
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSSubmit.Number+i]=buffer[current++];
+ }
+ smsframe[PHONE_SMSSubmit.TPPID] = buffer[current++];
+ smsframe[PHONE_SMSSubmit.TPDCS] = buffer[current++];
+ /* See GSM 03.40 9.2.3.3 - TPVP can not exist in frame */
+ if ((smsframe[12] & 0x18)!=0) current++; //TPVP is ignored now
+ smsframe[PHONE_SMSSubmit.TPUDL] = buffer[current++];
+ for(i=0;i<smsframe[PHONE_SMSSubmit.TPUDL];i++) smsframe[i+PHONE_SMSSubmit.Text]=buffer[current++];
+ GSM_DecodeSMSFrame(sms,smsframe,PHONE_SMSSubmit);
+ return ERR_NONE;
+ case 0x02:
+ smprintf(s, "SMS type - status report\n");
+ sms->PDU = SMS_Status_Report;
+ sms->Folder = 1; /*INBOX SIM*/
+ sms->InboxFolder = true;
+ smprintf(s, "TPMR is %02x\n",buffer[current]);
+ smsframe[PHONE_SMSStatusReport.TPMR] = buffer[current++];
+ current2=((buffer[current])+1)/2+1;
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSStatusReport.Number+i]=buffer[current++];
+ for(i=0;i<7;i++) smsframe[PHONE_SMSStatusReport.DateTime+i]=buffer[current++];
+ for(i=0;i<7;i++) smsframe[PHONE_SMSStatusReport.SMSCTime+i]=buffer[current++];
+ smsframe[PHONE_SMSStatusReport.TPStatus]=buffer[current];
+ GSM_DecodeSMSFrame(sms,smsframe,PHONE_SMSStatusReport);
+ return ERR_NONE;
+ }
+ break;
+ case SMS_AT_TXT:
+ current = 0;
+ while (msg.Buffer[current]!='"') current++;
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ if (!strcmp(buffer,"\"0\"") || !strcmp(buffer,"\"REC UNREAD\"")) {
+ smprintf(s, "SMS type - deliver\n");
+ sms->State = SMS_UnRead;
+ sms->PDU = SMS_Deliver;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 1; /*INBOX SIM*/
+ } else {
+ sms->Folder = 3; /*INBOX ME*/
+ }
+ sms->InboxFolder = true;
+ } else if (!strcmp(buffer,"\"1\"") || !strcmp(buffer,"\"REC READ\"")) {
+ smprintf(s, "SMS type - deliver\n");
+ sms->State = SMS_Read;
+ sms->PDU = SMS_Deliver;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 1; /*INBOX SIM*/
+ } else {
+ sms->Folder = 3; /*INBOX ME*/
+ }
+ sms->InboxFolder = true;
+ } else if (!strcmp(buffer,"\"2\"") || !strcmp(buffer,"\"STO UNSENT\"")) {
+ smprintf(s, "SMS type - submit\n");
+ sms->State = SMS_UnSent;
+ sms->PDU = SMS_Submit;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 2; /*OUTBOX SIM*/
+ } else {
+ sms->Folder = 4; /*OUTBOX ME*/
+ }
+ sms->InboxFolder = false;
+ } else if (!strcmp(buffer,"\"3\"") || !strcmp(buffer,"\"STO SENT\"")) {
+ smprintf(s, "SMS type - submit\n");
+ sms->State = SMS_Sent;
+ sms->PDU = SMS_Submit;
+ if (Priv->SMSMemory == MEM_SM) {
+ sms->Folder = 2; /*OUTBOX SIM*/
+ } else {
+ sms->Folder = 4; /*OUTBOX ME*/
+ }
+ sms->InboxFolder = false;
+ }
+ current += ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* It's delivery report according to Nokia AT standards */
+ if (sms->Folder==1 && buffer[0]!=0 && buffer[0]!='"') {
+ /* ??? */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* format of sender number */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* Sender number */
+ /* FIXME: support for all formats */
+ EncodeUnicode(sms->Number,buffer+1,strlen(buffer)-2);
+ smprintf(s, "Sender \"%s\"\n",DecodeUnicodeString(sms->Number));
+ /* ??? */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* Sending datetime */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ i = strlen(buffer);
+ buffer[i] = ',';
+ i++;
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer+i);
+ smprintf(s, "\"%s\"\n",buffer);
+ ATGEN_DecodeDateTime(&sms->DateTime, buffer+1);
+ /* Date of SMSC response */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ i = strlen(buffer);
+ buffer[i] = ',';
+ i++;
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer+i);
+ smprintf(s, "\"%s\"\n",buffer);
+ ATGEN_DecodeDateTime(&sms->SMSCTime, buffer+1);
+ /* TPStatus */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ TPStatus=atoi(buffer);
+ buffer[PHONE_SMSStatusReport.TPStatus] = TPStatus;
+ error=GSM_DecodeSMSFrameStatusReportData(sms, buffer, PHONE_SMSStatusReport);
+ if (error!=ERR_NONE) return error;
+ /* NO SMSC number */
+ sms->SMSC.Number[0]=0;
+ sms->SMSC.Number[1]=0;
+ sms->PDU = SMS_Status_Report;
+ sms->ReplyViaSameSMSC=false;
+ } else {
+ /* Sender number */
+ /* FIXME: support for all formats */
+ EncodeUnicode(sms->Number,buffer+1,strlen(buffer)-2);
+ /* Sender number in alphanumeric format ? */
+ current += ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ if (strlen(buffer)!=0) EncodeUnicode(sms->Number,buffer+1,strlen(buffer)-2);
+ smprintf(s, "Sender \"%s\"\n",DecodeUnicodeString(sms->Number));
+ /* Sending datetime */
+ if (sms->Folder==1 || sms->Folder==3) {
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* FIXME: ATGEN_ExtractOneParameter() is broken as it doesn't respect
+ * quoting of parameters and thus +FOO: "ab","cd,ef" will consider
+ * as three arguments: "ab" >> "cd >> ef"
+ */
+ if (*buffer=='"') {
+ i = strlen(buffer);
+ buffer[i] = ',';
+ i++;
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer+i);
+ }
+ smprintf(s, "\"%s\"\n",buffer);
+ if (*buffer)
+ ATGEN_DecodeDateTime(&sms->DateTime, buffer+1);
+ else {
+ /* FIXME: What is the proper undefined GSM_DateTime ? */
+ memset(&sms->DateTime, 0, sizeof(sms->DateTime));
+ }
+ ATGEN_DecodeDateTime(&sms->DateTime, buffer+1);
+ }
+ /* Sender number format */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* First byte */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ firstbyte=atoi(buffer);
+ sms->ReplyViaSameSMSC=false;
+ /* GSM 03.40 section 9.2.3.17 (TP-Reply-Path) */
+ if ((firstbyte & 128)==128) sms->ReplyViaSameSMSC=true;
+ /* TP PID */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ sms->ReplaceMessage = 0;
+ if (atoi(buffer) > 0x40 && atoi(buffer) < 0x48) {
+ sms->ReplaceMessage = atoi(buffer) - 0x40;
+ }
+ smprintf(s, "TPPID: %02x %i\n",atoi(buffer),atoi(buffer));
+ /* TP DCS */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ TPDCS=atoi(buffer);
+ if (sms->Folder==2 || sms->Folder==4) {
+ /*TP VP */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ }
+ /* SMSC number */
+ /* FIXME: support for all formats */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ EncodeUnicode(sms->SMSC.Number,buffer+1,strlen(buffer)-2);
+ /* Format of SMSC number */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /* TPUDL */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ TPUDL=atoi(buffer);
+ current++;
+ sms->Coding = SMS_Coding_Default;
+ /* GSM 03.40 section 9.2.3.10 (TP-Data-Coding-Scheme)
+ * and GSM 03.38 section 4
+ */
+ if ((TPDCS & 0xf4) == 0xf4) sms->Coding=SMS_Coding_8bit;
+ if ((TPDCS & 0x08) == 0x08) sms->Coding=SMS_Coding_Unicode;
+ sms->Class = -1;
+ if ((TPDCS & 0xF3)==0xF0) sms->Class = 0;
+ if ((TPDCS & 0xF3)==0xF1) sms->Class = 1;
+ if ((TPDCS & 0xF3)==0xF2) sms->Class = 2;
+ if ((TPDCS & 0xF3)==0xF3) sms->Class = 3;
+ smprintf(s, "SMS class: %i\n",sms->Class);
+ switch (sms->Coding) {
+ case SMS_Coding_Default:
+ /* GSM 03.40 section 9.2.3.23 (TP-User-Data-Header-Indicator) */
+ /* If not SMS with UDH, it's coded normal */
+ /* If UDH available, treat it as Unicode or 8 bit */
+ if ((firstbyte & 0x40)!=0x40) {
+ sms->UDH.Type = UDH_NoUDH;
+ sms->Length = TPUDL;
+ EncodeUnicode(sms->Text,msg.Buffer+Priv->Lines.numbers[2*2],TPUDL);
+ break;
+ }
+ case SMS_Coding_Unicode:
+ case SMS_Coding_8bit:
+ DecodeHexBin(buffer+PHONE_SMSDeliver.Text, msg.Buffer+current, TPUDL*2);
+ buffer[PHONE_SMSDeliver.firstbyte] = firstbyte;
+ buffer[PHONE_SMSDeliver.TPDCS] = TPDCS;
+ buffer[PHONE_SMSDeliver.TPUDL] = TPUDL;
+ return GSM_DecodeSMSFrameText(sms, buffer, PHONE_SMSDeliver);
+ }
+ }
+ return ERR_NONE;
+ default:
+ break;
+ }
+ break;
+ case AT_Reply_CMSError:
+ if (Priv->ErrorCode == 320 || Priv->ErrorCode == 500) {
+ return ERR_EMPTY;
+ } else {
+ return ATGEN_HandleCMSError(s);
+ }
+ case AT_Reply_CMEError:
+ return ATGEN_HandleCMEError(s);
+ case AT_Reply_Error:
+ /* A2D returns Error with empty location */
+ return ERR_EMPTY;
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSMS(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ unsigned char req[20], folderid;
+ GSM_Error error;
+ int location, getfolder, add = 0;
+
+ error=ATGEN_GetSMSLocation(s,&sms->SMS[0], &folderid, &location);
+ if (error!=ERR_NONE) return error;
+ if (s->Phone.Data.Priv.ATGEN.SMSMemory == MEM_ME && IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_SMSME900)) add = 899;
+ sprintf(req, "AT+CMGR=%i\r", location + add);
+
+ error=ATGEN_GetSMSMode(s);
+ if (error != ERR_NONE) return error;
+
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.GetSMSMessage=sms;
+ smprintf(s, "Getting SMS\n");
+ error=GSM_WaitFor (s, req, strlen(req), 0x00, 5, ID_GetSMSMessage);
+ if (error==ERR_NONE) {
+ getfolder = sms->SMS[0].Folder;
+// if (getfolder != 0 && getfolder != sms->SMS[0].Folder) return ERR_EMPTY;
+ ATGEN_SetSMSLocation(s, &sms->SMS[0], folderid, location);
+ sms->SMS[0].Folder = getfolder;
+ sms->SMS[0].Memory = MEM_SM;
+ if (getfolder > 2) sms->SMS[0].Memory = MEM_ME;
+ }
+ return error;
+}
+
+GSM_Error ATGEN_GetNextSMS(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ int usedsms;
+
+ if (Priv->PhoneSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, false);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+ if (Priv->SIMSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, true);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+ if (Priv->SIMSMSMemory == AT_NOTAVAILABLE && Priv->PhoneSMSMemory == AT_NOTAVAILABLE) return ERR_NOTSUPPORTED;
+
+ if (start) {
+ error=s->Phone.Functions->GetSMSStatus(s,&Priv->LastSMSStatus);
+ if (error!=ERR_NONE) return error;
+ Priv->LastSMSRead = 0;
+ sms->SMS[0].Location = 0;
+ }
+ while (true) {
+ sms->SMS[0].Location++;
+ if (sms->SMS[0].Location < PHONE_MAXSMSINFOLDER) {
+ if (Priv->SIMSMSMemory == AT_AVAILABLE) {
+ usedsms = Priv->LastSMSStatus.SIMUsed;
+ } else {
+ usedsms = Priv->LastSMSStatus.PhoneUsed;
+ }
+
+ if (Priv->LastSMSRead >= usedsms) {
+ if (Priv->PhoneSMSMemory == AT_NOTAVAILABLE || Priv->LastSMSStatus.PhoneUsed==0) return ERR_EMPTY;
+ Priv->LastSMSRead = 0;
+ sms->SMS[0].Location = PHONE_MAXSMSINFOLDER + 1;
+ }
+ } else {
+ if (Priv->PhoneSMSMemory == AT_NOTAVAILABLE) return ERR_EMPTY;
+ if (Priv->LastSMSRead>=Priv->LastSMSStatus.PhoneUsed) return ERR_EMPTY;
+ }
+ sms->SMS[0].Folder = 0;
+ error=s->Phone.Functions->GetSMS(s, sms);
+ if (error==ERR_NONE) {
+ Priv->LastSMSRead++;
+ break;
+ }
+ if (error != ERR_EMPTY && error != ERR_INVALIDLOCATION) return error;
+ }
+ return error;
+}
+
+GSM_Error ATGEN_ReplyGetSMSStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_SMSMemoryStatus *SMSStatus = s->Phone.Data.SMSStatus;
+ char *start;
+ int current = 0;
+ unsigned char buffer[50];
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "SMS status received\n");
+ start = strstr(msg.Buffer, "+CPMS: ") + 7;
+ if (strstr(msg.Buffer,"ME")!=NULL) {
+ SMSStatus->PhoneUsed = atoi(start);
+ current+=ATGEN_ExtractOneParameter(start+current, buffer);
+ current+=ATGEN_ExtractOneParameter(start+current, buffer);
+ SMSStatus->PhoneSize = atoi(buffer);
+ smprintf(s, "Used : %i\n",SMSStatus->PhoneUsed);
+ smprintf(s, "Size : %i\n",SMSStatus->PhoneSize);
+ } else {
+ SMSStatus->SIMUsed = atoi(start);
+ current+=ATGEN_ExtractOneParameter(start+current, buffer);
+ current+=ATGEN_ExtractOneParameter(start+current, buffer);
+ SMSStatus->SIMSize = atoi(buffer);
+ smprintf(s, "Used : %i\n",SMSStatus->SIMUsed);
+ smprintf(s, "Size : %i\n",SMSStatus->SIMSize);
+ if (SMSStatus->SIMSize == 0) {
+ smprintf(s, "Can't access SIM card\n");
+ return ERR_SECURITYERROR;
+ }
+ }
+ return ERR_NONE;
+ case AT_Reply_Error:
+ if (strstr(msg.Buffer,"SM")!=NULL) {
+ smprintf(s, "Can't access SIM card\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_NOTSUPPORTED;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSMSStatus(GSM_StateMachine *s, GSM_SMSMemoryStatus *status)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+
+ /* No templates at all */
+ status->TemplatesUsed = 0;
+
+ status->SIMUsed = 0;
+ status->SIMUnRead = 0;
+ status->SIMSize = 0;
+
+ s->Phone.Data.SMSStatus=status;
+
+ if ((Priv->SIMSMSMemory == 0) || (Priv->PhoneSMSMemory == 0)) {
+ /* We silently ignore error here, because when this fails, we can try to setmemory anyway */
+ ATGEN_GetSMSMemories(s);
+ }
+
+ if (Priv->PhoneSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, false);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+ if (Priv->SIMSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, true);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+
+ if (Priv->SIMSMSMemory == AT_AVAILABLE) {
+ smprintf(s, "Getting SIM SMS status\n");
+ if (Priv->CanSaveSMS) {
+ error=GSM_WaitFor (s, "AT+CPMS=\"SM\",\"SM\"\r", 18, 0x00, 4, ID_GetSMSStatus);
+ } else {
+ error=GSM_WaitFor (s, "AT+CPMS=\"SM\"\r", 13, 0x00, 4, ID_GetSMSStatus);
+ }
+ if (error!=ERR_NONE) return error;
+ Priv->SMSMemory = MEM_SM;
+ }
+
+ status->PhoneUsed = 0;
+ status->PhoneUnRead = 0;
+ status->PhoneSize = 0;
+
+ if (Priv->PhoneSMSMemory == AT_AVAILABLE) {
+ smprintf(s, "Getting phone SMS status\n");
+ if (Priv->CanSaveSMS) {
+ error = GSM_WaitFor (s, "AT+CPMS=\"ME\",\"ME\"\r", 18, 0x00, 4, ID_GetSMSStatus);
+ } else {
+ error = GSM_WaitFor (s, "AT+CPMS=\"ME\"\r", 13, 0x00, 4, ID_GetSMSStatus);
+ }
+ if (error!=ERR_NONE) return error;
+ Priv->SMSMemory = MEM_ME;
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyGetIMEI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ CopyLineString(s->Phone.Data.IMEI, msg.Buffer, s->Phone.Data.Priv.ATGEN.Lines, 2);
+ smprintf(s, "Received IMEI %s\n",s->Phone.Data.IMEI);
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_GetIMEI (GSM_StateMachine *s)
+{
+ if (s->Phone.Data.IMEI[0] != 0) return ERR_NONE;
+ smprintf(s, "Getting IMEI\n");
+ return GSM_WaitFor (s, "AT+CGSN\r", 8, 0x00, 2, ID_GetIMEI);
+}
+
+GSM_Error ATGEN_ReplyAddSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ char *start;
+ int i;
+
+ if (s->Protocol.Data.AT.EditMode) {
+ if (s->Phone.Data.Priv.ATGEN.ReplyState != AT_Reply_SMSEdit) {
+ return ATGEN_HandleCMSError(s);
+ }
+ s->Protocol.Data.AT.EditMode = false;
+ return ERR_NONE;
+ }
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "SMS saved OK\n");
+ for(i=0;i<msg.Length;i++) {
+ if (msg.Buffer[i] == 0x00) msg.Buffer[i] = 0x20;
+ }
+ start = strstr(msg.Buffer, "+CMGW: ");
+ if (start == NULL) return ERR_UNKNOWN;
+ s->Phone.Data.SaveSMSMessage->Location = atoi(start+7);
+ smprintf(s, "Saved at location %i\n",s->Phone.Data.SaveSMSMessage->Location);
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Error\n");
+ return ERR_NOTSUPPORTED;
+ case AT_Reply_CMSError:
+ /* This error occurs in case that phone couldn't save SMS */
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_MakeSMSFrame(GSM_StateMachine *s, GSM_SMSMessage *message, unsigned char *hexreq, int *current, int *length2)
+{
+ GSM_Error error;
+ int i, length;
+ unsigned char req[1000], buffer[1000];
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_SMSC SMSC;
+
+ error=ATGEN_GetSMSMode(s);
+ if (error != ERR_NONE) return error;
+
+ length = 0;
+ *current = 0;
+ switch (Priv->SMSMode) {
+ case SMS_AT_PDU:
+ if (message->PDU == SMS_Deliver) {
+ smprintf(s, "SMS Deliver\n");
+ error=PHONE_EncodeSMSFrame(s,message,buffer,PHONE_SMSDeliver,&length,true);
+ if (error != ERR_NONE) return error;
+ length = length - PHONE_SMSDeliver.Text;
+ for (i=0;i<buffer[PHONE_SMSDeliver.SMSCNumber]+1;i++) {
+ req[(*current)++]=buffer[PHONE_SMSDeliver.SMSCNumber+i];
+ }
+ req[(*current)++]=buffer[PHONE_SMSDeliver.firstbyte];
+ for (i=0;i<((buffer[PHONE_SMSDeliver.Number]+1)/2+1)+1;i++) {
+ req[(*current)++]=buffer[PHONE_SMSDeliver.Number+i];
+ }
+ req[(*current)++]=buffer[PHONE_SMSDeliver.TPPID];
+ req[(*current)++]=buffer[PHONE_SMSDeliver.TPDCS];
+ for(i=0;i<7;i++) req[(*current)++]=buffer[PHONE_SMSDeliver.DateTime+i];
+ req[(*current)++]=buffer[PHONE_SMSDeliver.TPUDL];
+ for(i=0;i<length;i++) req[(*current)++]=buffer[PHONE_SMSDeliver.Text+i];
+ EncodeHexBin(hexreq, req, *current);
+ *length2 = *current * 2;
+ *current = *current - (req[PHONE_SMSDeliver.SMSCNumber]+1);
+ } else {
+ smprintf(s, "SMS Submit\n");
+ error=PHONE_EncodeSMSFrame(s,message,buffer,PHONE_SMSSubmit,&length,true);
+ if (error != ERR_NONE) return error;
+ length = length - PHONE_SMSSubmit.Text;
+ for (i=0;i<buffer[PHONE_SMSSubmit.SMSCNumber]+1;i++) {
+ req[(*current)++]=buffer[PHONE_SMSSubmit.SMSCNumber+i];
+ }
+ req[(*current)++]=buffer[PHONE_SMSSubmit.firstbyte];
+ req[(*current)++]=buffer[PHONE_SMSSubmit.TPMR];
+ for (i=0;i<((buffer[PHONE_SMSSubmit.Number]+1)/2+1)+1;i++) {
+ req[(*current)++]=buffer[PHONE_SMSSubmit.Number+i];
+ }
+ req[(*current)++]=buffer[PHONE_SMSSubmit.TPPID];
+ req[(*current)++]=buffer[PHONE_SMSSubmit.TPDCS];
+ req[(*current)++]=buffer[PHONE_SMSSubmit.TPVP];
+ req[(*current)++]=buffer[PHONE_SMSSubmit.TPUDL];
+ for(i=0;i<length;i++) req[(*current)++]=buffer[PHONE_SMSSubmit.Text+i];
+ EncodeHexBin(hexreq, req, *current);
+ *length2 = *current * 2;
+ *current = *current - (req[PHONE_SMSSubmit.SMSCNumber]+1);
+ }
+ break;
+ case SMS_AT_TXT:
+ if (Priv->Manufacturer == 0) {
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+ }
+ if (Priv->Manufacturer != AT_Nokia) {
+ if (message->Coding != SMS_Coding_Default) return ERR_NOTSUPPORTED;
+ }
+ error=PHONE_EncodeSMSFrame(s,message,req,PHONE_SMSDeliver,&i,true);
+ if (error != ERR_NONE) return error;
+ CopyUnicodeString(SMSC.Number,message->SMSC.Number);
+ SMSC.Location=1;
+ error=ATGEN_SetSMSC(s,&SMSC);
+ if (error!=ERR_NONE) return error;
+ sprintf(buffer, "AT+CSMP=%i,%i,%i,%i\r",
+ req[PHONE_SMSDeliver.firstbyte],
+ req[PHONE_SMSDeliver.TPVP],
+ req[PHONE_SMSDeliver.TPPID],
+ req[PHONE_SMSDeliver.TPDCS]);
+ error=GSM_WaitFor (s, buffer, strlen(buffer), 0x00, 4, ID_SetSMSParameters);
+ if (error==ERR_NOTSUPPORTED) {
+ /* Nokia Communicator 9000i doesn't support <vp> parameter */
+ sprintf(buffer, "AT+CSMP=%i,,%i,%i\r",
+ req[PHONE_SMSDeliver.firstbyte],
+ req[PHONE_SMSDeliver.TPPID],
+ req[PHONE_SMSDeliver.TPDCS]);
+ error=GSM_WaitFor (s, buffer, strlen(buffer), 0x00, 4, ID_SetSMSParameters);
+ }
+ if (error!=ERR_NONE) return error;
+ switch (message->Coding) {
+ case SMS_Coding_Default:
+ /* If not SMS with UDH, it's as normal text */
+ if (message->UDH.Type==UDH_NoUDH) {
+ strcpy(hexreq,DecodeUnicodeString(message->Text));
+ *length2 = UnicodeLength(message->Text);
+ break;
+ }
+ case SMS_Coding_Unicode:
+ case SMS_Coding_8bit:
+ error=PHONE_EncodeSMSFrame(s,message,buffer,PHONE_SMSDeliver,current,true);
+ if (error != ERR_NONE) return error;
+ EncodeHexBin (hexreq, buffer+PHONE_SMSDeliver.Text, buffer[PHONE_SMSDeliver.TPUDL]);
+ *length2 = buffer[PHONE_SMSDeliver.TPUDL] * 2;
+ break;
+ }
+ break;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_AddSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ GSM_Error error, error2;
+ int state,Replies,reply, current, current2;
+ unsigned char buffer[1000], hexreq[1000];
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+ unsigned char *statetxt;
+
+ /* This phone supports only sent/unsent messages on SIM */
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_SMSONLYSENT)) {
+ if (sms->Folder != 2) {
+ smprintf(s, "This phone supports only folder = 2!\n");
+ return ERR_NOTSUPPORTED;
+ }
+ }
+
+ sms->PDU = SMS_Submit;
+ switch (sms->Folder) {
+ case 1: sms->PDU = SMS_Deliver; /* Inbox SIM */
+ error=ATGEN_SetSMSMemory(s, true);
+ break;
+ case 2: error=ATGEN_SetSMSMemory(s, true); /* Outbox SIM */
+ break;
+ case 3: sms->PDU = SMS_Deliver;
+ error=ATGEN_SetSMSMemory(s, false); /* Inbox phone */
+ break;
+ case 4: error=ATGEN_SetSMSMemory(s, false); /* Outbox phone */
+ break;
+ default: return ERR_NOTSUPPORTED;
+ }
+ if (error!=ERR_NONE) return error;
+
+ error = ATGEN_MakeSMSFrame(s, sms, hexreq, &current, &current2);
+ if (error != ERR_NONE) return error;
+
+ switch (Phone->Priv.ATGEN.SMSMode) {
+ case SMS_AT_PDU:
+ if (sms->PDU == SMS_Deliver) {
+ state = 0;
+ if (sms->State == SMS_Read || sms->State == SMS_Sent) state = 1;
+ } else {
+ state = 2;
+ if (sms->State == SMS_Read || sms->State == SMS_Sent) state = 3;
+ }
+ /* Siemens M20 */
+ if (IsPhoneFeatureAvailable(Phone->ModelInfo, F_M20SMS)) {
+ /* No (good and 100% working) support for alphanumeric numbers */
+ if (sms->Number[1]!='+' && (sms->Number[1]<'0' || sms->Number[1]>'9')) {
+ EncodeUnicode(sms->Number,"123",3);
+ error = ATGEN_MakeSMSFrame(s, sms, hexreq, &current, &current2);
+ if (error != ERR_NONE) return error;
+ }
+ }
+ sprintf(buffer, "AT+CMGW=%i,%i\r",current,state);
+ break;
+ case SMS_AT_TXT:
+ if (sms->PDU == SMS_Deliver) {
+ statetxt = "REC UNREAD";
+ if (sms->State == SMS_Read || sms->State == SMS_Sent) statetxt = "REC READ";
+ } else {
+ statetxt = "STO UNSENT";
+ if (sms->State == SMS_Read || sms->State == SMS_Sent) statetxt = "STO SENT";
+ }
+ /* Siemens M20 */
+ if (IsPhoneFeatureAvailable(Phone->ModelInfo, F_M20SMS)) {
+ /* No (good and 100% working) support for alphanumeric numbers */
+ /* FIXME: Try to autodetect support for <stat> (statetxt) parameter although:
+ * Siemens M20 supports +CMGW <stat> specification but on my model it just
+ * reports ERROR (and <stat> is not respected).
+ * Fortunately it will write "+CMGW: <index>\n" before and the message gets written
+ */
+ if (sms->Number[1]!='+' && (sms->Number[1]<'0' || sms->Number[1]>'9')) {
+ sprintf(buffer, "AT+CMGW=\"123\",,\"%s\"\r",statetxt);
+ } else {
+ sprintf(buffer, "AT+CMGW=\"%s\",,\"%s\"\r",DecodeUnicodeString(sms->Number),statetxt);
+ }
+ } else {
+ sprintf(buffer, "AT+CMGW=\"%s\",,\"%s\"\r",DecodeUnicodeString(sms->Number),statetxt);
+ }
+ }
+
+ Phone->SaveSMSMessage = sms;
+
+ for (reply=0;reply<s->ReplyNum;reply++) {
+ if (reply!=0) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s, "[Retrying %i]\n", reply+1);
+ }
+ }
+ s->Protocol.Data.AT.EditMode = true;
+ Replies = s->ReplyNum;
+ s->ReplyNum = 1;
+ smprintf(s,"Waiting for modem prompt\n");
+ error=GSM_WaitFor (s, buffer, strlen(buffer), 0x00, 3, ID_SaveSMSMessage);
+ s->ReplyNum = Replies;
+ if (error == ERR_NONE) {
+ Phone->DispatchError = ERR_TIMEOUT;
+ Phone->RequestID = ID_SaveSMSMessage;
+ smprintf(s, "Saving SMS\n");
+ error = s->Protocol.Functions->WriteMessage(s, hexreq, current2, 0x00);
+ if (error!=ERR_NONE) return error;
+ my_sleep(500);
+ /* CTRL+Z ends entering */
+ error = s->Protocol.Functions->WriteMessage(s, "\x1A", 1, 0x00);
+ if (error!=ERR_NONE) return error;
+ error = GSM_WaitForOnce(s, NULL, 0x00, 0x00, 4);
+ if (error != ERR_TIMEOUT) return error;
+ } else {
+ smprintf(s, "Escaping SMS mode\n");
+ error2 = s->Protocol.Functions->WriteMessage(s, "\x1B\r", 2, 0x00);
+ if (error2 != ERR_NONE) return error2;
+ return error;
+ }
+ }
+
+ return Phone->DispatchError;
+}
+
+GSM_Error ATGEN_ReplySendSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ char *start;
+
+ if (s->Protocol.Data.AT.EditMode) {
+ if (s->Phone.Data.Priv.ATGEN.ReplyState != AT_Reply_SMSEdit) {
+ return ERR_UNKNOWN;
+ }
+ s->Protocol.Data.AT.EditMode = false;
+ return ERR_NONE;
+ }
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "SMS sent OK\n");
+ if (s->User.SendSMSStatus!=NULL) {
+ start = strstr(msg.Buffer, "+CMGW: ");
+ if (start != NULL) {
+ s->User.SendSMSStatus(s->CurrentConfig->Device,0,atoi(start+7));
+ } else {
+ s->User.SendSMSStatus(s->CurrentConfig->Device,0,0);
+ }
+ }
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ smprintf(s, "Error %i\n",Priv->ErrorCode);
+ if (s->User.SendSMSStatus!=NULL) s->User.SendSMSStatus(s->CurrentConfig->Device,Priv->ErrorCode,0);
+ return ATGEN_HandleCMSError(s);
+ case AT_Reply_Error:
+ return ERR_UNKNOWN;
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error ATGEN_SendSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ GSM_Error error,error2;
+ int current, current2, Replies;
+ unsigned char buffer[1000], hexreq[1000];
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+
+ if (sms->PDU == SMS_Deliver) sms->PDU = SMS_Submit;
+
+ error = ATGEN_MakeSMSFrame(s, sms, hexreq, &current, &current2);
+ if (error != ERR_NONE) return error;
+
+ switch (Phone->Priv.ATGEN.SMSMode) {
+ case SMS_AT_PDU:
+ sprintf(buffer, "AT+CMGS=%i\r",current);
+ break;
+ case SMS_AT_TXT:
+ sprintf(buffer, "AT+CMGS=\"%s\"\r",DecodeUnicodeString(sms->Number));
+ }
+
+ s->Protocol.Data.AT.EditMode = true;
+ Replies = s->ReplyNum;
+ s->ReplyNum = 1;
+ smprintf(s,"Waiting for modem prompt\n");
+ error=GSM_WaitFor (s, buffer, strlen(buffer), 0x00, 3, ID_IncomingFrame);
+ s->ReplyNum = Replies;
+ if (error == ERR_NONE) {
+ smprintf(s, "Sending SMS\n");
+ error = s->Protocol.Functions->WriteMessage(s, hexreq, current2, 0x00);
+ if (error!=ERR_NONE) return error;
+ my_sleep(500);
+ /* CTRL+Z ends entering */
+ error=s->Protocol.Functions->WriteMessage(s, "\x1A", 1, 0x00);
+ my_sleep(100);
+ return error;
+ } else {
+ smprintf(s, "Escaping SMS mode\n");
+ error2=s->Protocol.Functions->WriteMessage(s, "\x1B\r", 2, 0x00);
+ if (error2 != ERR_NONE) return error2;
+ }
+ return error;
+}
+
+GSM_Error ATGEN_SendSavedSMS(GSM_StateMachine *s, int Folder, int Location)
+{
+ GSM_Error error;
+ int location;
+ unsigned char smsfolder;
+ unsigned char req[100];
+ GSM_MultiSMSMessage msms;
+
+ msms.Number = 0;
+ msms.SMS[0].Folder = Folder;
+ msms.SMS[0].Location = Location;
+
+ /* By reading SMS we check if it is really inbox/outbox */
+ error = ATGEN_GetSMS(s, &msms);
+ if (error != ERR_NONE) return error;
+
+ /* Can not send from other folder that outbox */
+ if (msms.SMS[0].Folder != 2 && msms.SMS[0].Folder != 4) return ERR_NOTSUPPORTED;
+
+ error=ATGEN_GetSMSLocation(s, &msms.SMS[0], &smsfolder, &location);
+ if (error != ERR_NONE) return error;
+
+ sprintf(req, "AT+CMSS=%i\r",location);
+ return s->Protocol.Functions->WriteMessage(s, req, strlen(req), 0x00);
+}
+
+GSM_Error ATGEN_ReplyGetDateTime_Alarm(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int current = 19;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ if (msg.Buffer[current]==0x0d || msg.Buffer[current-1]==0x0d) {
+ smprintf(s, "Not set in phone\n");
+ return ERR_EMPTY;
+ } else {
+ if (Data->RequestID == ID_GetDateTime) {
+ ATGEN_DecodeDateTime(Data->DateTime, msg.Buffer+current);
+ } else {
+ ATGEN_DecodeDateTime(&(Data->Alarm->DateTime), msg.Buffer+current);
+ }
+ return ERR_NONE;
+ }
+ case AT_Reply_Error:
+ return ERR_NOTSUPPORTED;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ s->Phone.Data.DateTime=date_time;
+ smprintf(s, "Getting date & time\n");
+ return GSM_WaitFor (s, "AT+CCLK?\r", 9, 0x00, 4, ID_GetDateTime);
+}
+
+GSM_Error ATGEN_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ char req[128];
+
+ sprintf(req, "AT+CCLK=\"%02i/%02i/%02i,%02i:%02i:%02i+00\"\r",
+ date_time->Year-2000,date_time->Month,date_time->Day,
+ date_time->Hour,date_time->Minute,date_time->Second);
+
+ smprintf(s, "Setting date & time\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_SetDateTime);
+}
+
+GSM_Error ATGEN_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ if (alarm->Location != 1) return ERR_NOTSUPPORTED;
+
+ alarm->Repeating = true;
+ alarm->Text[0] = 0; alarm->Text[1] = 0;
+
+ s->Phone.Data.Alarm = alarm;
+ smprintf(s, "Getting alarm\n");
+ return GSM_WaitFor (s, "AT+CALA?\r", 9, 0x00, 4, ID_GetAlarm);
+}
+
+GSM_Error ATGEN_ReplyGetSMSC(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SMSC *SMSC = s->Phone.Data.SMSC;
+ int current;
+ int len;
+ unsigned char buffer[100];
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "SMSC info received\n");
+
+ current = 0;
+ while (msg.Buffer[current]!='"') current++;
+
+ /* SMSC number */
+ /* FIXME: support for all formats */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ /*
+ * Some phones return this as unicode encoded when they are
+ * switched to UCS2 mode, so we try to solve this correctly.
+ */
+ len = strlen(buffer + 1) - 1;
+ buffer[len + 1] = 0;
+ if ((len > 20) && (len % 4 == 0) && (strchr(buffer + 1, '+') == NULL)) {
+ /* This is probably unicode encoded number */
+ DecodeHexUnicode(SMSC->Number,buffer + 1,len);
+ } else {
+ EncodeUnicode(SMSC->Number,buffer + 1,len);
+ }
+ smprintf(s, "Number: \"%s\"\n",DecodeUnicodeString(SMSC->Number));
+
+ /* Format of SMSC number */
+ current+=ATGEN_ExtractOneParameter(msg.Buffer+current, buffer);
+ smprintf(s, "Format %s\n",buffer);
+ /* International number */
+ if (!strcmp(buffer,"145")) {
+ sprintf(buffer+1,"%s",DecodeUnicodeString(SMSC->Number));
+ if (strlen(buffer+1)!=0 && buffer[1] != '+') {
+ /* Sony Ericsson issue */
+ /* International number is without + */
+ buffer[0] = '+';
+ EncodeUnicode(SMSC->Number,buffer,strlen(buffer));
+ }
+ }
+
+ SMSC->Format = SMS_FORMAT_Text;
+ SMSC->Validity.Format = SMS_Validity_RelativeFormat;
+ SMSC->Validity.Relative = SMS_VALID_Max_Time;
+ SMSC->Name[0] = 0;
+ SMSC->Name[1] = 0;
+ SMSC->DefaultNumber[0] = 0;
+ SMSC->DefaultNumber[1] = 0;
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ if (smsc->Location==0x00 || smsc->Location!=0x01) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.SMSC=smsc;
+ smprintf(s, "Getting SMSC\n");
+ return GSM_WaitFor (s, "AT+CSCA?\r", 9, 0x00, 4, ID_GetSMSC);
+}
+
+GSM_Error ATGEN_ReplyGetNetworkLAC_CID(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_NetworkInfo *NetworkInfo = s->Phone.Data.NetworkInfo;
+ GSM_Lines Lines;
+ int i=0;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ char *answer;
+
+ if (s->Phone.Data.RequestID == ID_IncomingFrame) {
+ smprintf(s, "Incoming LAC & CID info\n");
+ return ERR_NONE;
+ }
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ break;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ SplitLines(GetLineString(msg.Buffer,Priv->Lines,2),
+ strlen(GetLineString(msg.Buffer,Priv->Lines,2)),
+ &Lines, ",", 1, true);
+
+ /* Find number of lines */
+ while (Lines.numbers[i*2+1] != 0) {
+ /* FIXME: handle special chars correctly */
+ smprintf(s, "%i \"%s\"\n",i+1,GetLineString(GetLineString(msg.Buffer,Priv->Lines,2),Lines,i+1));
+ i++;
+ }
+
+ smprintf(s, "Network LAC & CID & state received\n");
+ answer = GetLineString(GetLineString(msg.Buffer,Priv->Lines,2),Lines,2);
+ while (*answer == 0x20) answer++;
+#ifdef DEBUG
+ switch (answer[0]) {
+ case '0': smprintf(s, "Not registered into any network. Not searching for network\n"); break;
+ case '1': smprintf(s, "Home network\n"); break;
+ case '2': smprintf(s, "Not registered into any network. Searching for network\n"); break;
+ case '3': smprintf(s, "Registration denied\n"); break;
+ case '4': smprintf(s, "Unknown\n"); break;
+ case '5': smprintf(s, "Registered in roaming network\n"); break;
+ default : smprintf(s, "Unknown\n");
+ }
+#endif
+ switch (answer[0]) {
+ case '0': NetworkInfo->State = GSM_NoNetwork; break;
+ case '1': NetworkInfo->State = GSM_HomeNetwork; break;
+ case '2': NetworkInfo->State = GSM_RequestingNetwork; break;
+ case '3': NetworkInfo->State = GSM_RegistrationDenied; break;
+ case '4': NetworkInfo->State = GSM_NetworkStatusUnknown;break;
+ case '5': NetworkInfo->State = GSM_RoamingNetwork; break;
+ default : NetworkInfo->State = GSM_NetworkStatusUnknown;break;
+ }
+ if (NetworkInfo->State == GSM_HomeNetwork ||
+ NetworkInfo->State == GSM_RoamingNetwork) {
+ memset(NetworkInfo->CID,0,4);
+ memset(NetworkInfo->LAC,0,4);
+
+ if (Lines.numbers[3*2+1]==0) return ERR_NONE;
+
+ answer = GetLineString(GetLineString(msg.Buffer,Priv->Lines,2),Lines,3);
+ while (*answer == 0x20) answer++;
+ sprintf(NetworkInfo->CID, "%c%c%c%c", answer[1], answer[2], answer[3], answer[4]);
+
+ answer = GetLineString(GetLineString(msg.Buffer,Priv->Lines,2),Lines,4);
+ while (*answer == 0x20) answer++;
+ sprintf(NetworkInfo->LAC, "%c%c%c%c", answer[1], answer[2], answer[3], answer[4]);
+
+ smprintf(s, "CID : %s\n",NetworkInfo->CID);
+ smprintf(s, "LAC : %s\n",NetworkInfo->LAC);
+ }
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyGetNetworkCode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_NetworkInfo *NetworkInfo = s->Phone.Data.NetworkInfo;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Network code received\n");
+ if (Priv->Manufacturer == AT_Falcom) {
+ NetworkInfo->NetworkCode[0] = msg.Buffer[22];
+ NetworkInfo->NetworkCode[1] = msg.Buffer[23];
+ NetworkInfo->NetworkCode[2] = msg.Buffer[24];
+ NetworkInfo->NetworkCode[3] = ' ';
+ NetworkInfo->NetworkCode[4] = msg.Buffer[25];
+ NetworkInfo->NetworkCode[5] = msg.Buffer[26];
+ } else {
+ NetworkInfo->NetworkCode[0] = msg.Buffer[23];
+ NetworkInfo->NetworkCode[1] = msg.Buffer[24];
+ NetworkInfo->NetworkCode[2] = msg.Buffer[25];
+ NetworkInfo->NetworkCode[3] = ' ';
+ NetworkInfo->NetworkCode[4] = msg.Buffer[26];
+ NetworkInfo->NetworkCode[5] = msg.Buffer[27];
+ }
+ NetworkInfo->NetworkCode[6] = 0;
+ smprintf(s, " Network code : %s\n", NetworkInfo->NetworkCode);
+ smprintf(s, " Network name for Gammu : %s ",
+ DecodeUnicodeString(GSM_GetNetworkName(NetworkInfo->NetworkCode)));
+ smprintf(s, "(%s)\n",DecodeUnicodeString(GSM_GetCountryName(NetworkInfo->NetworkCode)));
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetNetworkInfo(GSM_StateMachine *s, GSM_NetworkInfo *netinfo)
+{
+ GSM_Error error;
+
+ s->Phone.Data.NetworkInfo=netinfo;
+
+ netinfo->NetworkName[0] = 0;
+ netinfo->NetworkName[1] = 0;
+ netinfo->NetworkCode[0] = 0;
+
+ smprintf(s, "Enable full network info\n");
+ error=GSM_WaitFor(s, "AT+CREG=2\r", 10, 0x00, 4, ID_GetNetworkInfo);
+ if ((error != ERR_NONE) &&
+ (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) &&
+ (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Ericsson)) return error;
+
+ smprintf(s, "Getting network LAC and CID and state\n");
+ error=GSM_WaitFor(s, "AT+CREG?\r", 9, 0x00, 4, ID_GetNetworkInfo);
+ if (error != ERR_NONE) return error;
+
+ if (netinfo->State == GSM_HomeNetwork || netinfo->State == GSM_RoamingNetwork) {
+ smprintf(s, "Setting short network name format\n");
+ error=GSM_WaitFor(s, "AT+COPS=3,2\r", 12, 0x00, 4, ID_GetNetworkInfo);
+
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting network code\n");
+ error=GSM_WaitFor(s, "AT+COPS?\r", 9, 0x00, 4, ID_GetNetworkInfo);
+ }
+ return error;
+}
+
+GSM_Error ATGEN_ReplyGetPBKMemories(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "PBK memories received\n");
+ if (strlen(msg.Buffer) > AT_PBK_MAX_MEMORIES) {
+ smprintf(s, "ERROR: Too long phonebook memories information received! (Recevided %d, AT_PBK_MAX_MEMORIES is %d\n", strlen(msg.Buffer), AT_PBK_MAX_MEMORIES);
+ return ERR_MOREMEMORY;
+ }
+ memcpy(s->Phone.Data.Priv.ATGEN.PBKMemories,msg.Buffer,strlen(msg.Buffer));
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_SetPBKMemory(GSM_StateMachine *s, GSM_MemoryType MemType)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ char req[] = "AT+CPBS=\"XX\"\r";
+ GSM_Error error;
+
+ if (Priv->PBKMemory == MemType) return ERR_NONE;
+
+ /* Zero values that are for actual memory */
+ Priv->MemorySize = 0;
+ Priv->FirstMemoryEntry = 0;
+ Priv->NextMemoryEntry = 0;
+ Priv->TextLength = 0;
+ Priv->NumberLength = 0;
+
+ if (Priv->PBKMemories[0] == 0) {
+ error=GSM_WaitFor (s, "AT+CPBS=?\r", 10, 0x00, 3, ID_SetMemoryType);
+ if (error != ERR_NONE) return error;
+ }
+
+ switch (MemType) {
+ case MEM_SM:
+ req[9] = 'S'; req[10] = 'M';
+ break;
+ case MEM_ME:
+ if (strstr(Priv->PBKMemories,"ME")==NULL) return ERR_NOTSUPPORTED;
+ req[9] = 'M'; req[10] = 'E';
+ break;
+ case MEM_RC:
+ if (strstr(Priv->PBKMemories,"RC")==NULL) return ERR_NOTSUPPORTED;
+ req[9] = 'R'; req[10] = 'C';
+ break;
+ case MEM_MC:
+ if (strstr(Priv->PBKMemories,"MC")==NULL) return ERR_NOTSUPPORTED;
+ req[9] = 'M'; req[10] = 'C';
+ break;
+ case MEM_ON:
+ if (strstr(Priv->PBKMemories,"ON")==NULL) return ERR_NOTSUPPORTED;
+ req[9] = 'O'; req[10] = 'N';
+ break;
+ case MEM_FD:
+ if (strstr(Priv->PBKMemories,"FD")==NULL) return ERR_NOTSUPPORTED;
+ req[9] = 'F'; req[10] = 'D';
+ break;
+ case MEM_DC:
+ if (strstr(Priv->PBKMemories,"DC")!=NULL) {
+ req[9] = 'D'; req[10] = 'C';
+ break;
+ }
+ if (strstr(Priv->PBKMemories,"LD")!=NULL) {
+ req[9] = 'L'; req[10] = 'D';
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+ break;
+ default:
+ return ERR_NOTSUPPORTED;
+ }
+
+ smprintf(s, "Setting memory type\n");
+ error=GSM_WaitFor (s, req, 13, 0x00, 3, ID_SetMemoryType);
+ if (error == ERR_NONE) Priv->PBKMemory = MemType;
+ return error;
+}
+
+GSM_Error ATGEN_ReplyGetCPBSMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_MemoryStatus *MemoryStatus = s->Phone.Data.MemoryStatus;
+ char *start;
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Memory status received\n");
+ MemoryStatus->MemoryUsed = 0;
+ MemoryStatus->MemoryFree = 0;
+ start = strchr(msg.Buffer, ',');
+ if (start) {
+ start++;
+ MemoryStatus->MemoryUsed = atoi(start);
+ start = strchr(start, ',');
+ if (start) {
+ start++;
+ MemoryStatus->MemoryFree = atoi(start) - MemoryStatus->MemoryUsed;
+ return ERR_NONE;
+ } else return ERR_UNKNOWN;
+ } else return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_ReplyGetCPBRMemoryInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ char *pos;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Memory info received\n");
+ /* Parse +CPBR: (first-last),max_number_len,max_name_len */
+
+ /* Parse first location */
+ pos = strchr(msg.Buffer, '(');
+ if (!pos) return ERR_UNKNOWN;
+ pos++;
+ Priv->FirstMemoryEntry = atoi(pos);
+
+ /* Parse last location*/
+ pos = strchr(pos, '-');
+ if (!pos) return ERR_UNKNOWN;
+ pos++;
+ Priv->MemorySize = atoi(pos) + 1 - Priv->FirstMemoryEntry;
+
+ /* Parse number length*/
+ pos = strchr(pos, ',');
+ if (!pos) return ERR_UNKNOWN;
+ pos++;
+ Priv->NumberLength = atoi(pos);
+
+ /* Parse text length*/
+ pos = strchr(pos, ',');
+ if (!pos) return ERR_UNKNOWN;
+ pos++;
+ Priv->TextLength = atoi(pos);
+
+ return ERR_NONE;
+ case AT_Reply_Error:
+ return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error ATGEN_ReplyGetCPBRMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_MemoryStatus *MemoryStatus = s->Phone.Data.MemoryStatus;
+ int line=0;
+ char *str;
+ int cur;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Memory entries received\n");
+ /* Walk through lines with +CPBR: */
+ while (Priv->Lines.numbers[line*2+1]!=0) {
+ str = GetLineString(msg.Buffer,Priv->Lines,line+1);
+ if (strncmp(str, "+CPBR: ", 7) == 0) {
+ MemoryStatus->MemoryUsed++;
+ if (sscanf(str, "+CPBR: %d,", &cur) == 1) {
+ cur -= Priv->FirstMemoryEntry - 1;
+ if (cur == Priv->NextMemoryEntry || Priv->NextMemoryEntry == 0)
+ Priv->NextMemoryEntry = cur + 1;
+ }
+ }
+ line++;
+ }
+ return ERR_NONE;
+ case AT_Reply_Error:
+ return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error ATGEN_GetMemoryInfo(GSM_StateMachine *s, GSM_MemoryStatus *Status, GSM_AT_NeededMemoryInfo NeededInfo)
+{
+ GSM_Error error;
+ char req[20];
+ int start;
+ int end;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ smprintf(s, "Getting memory information\n");
+
+ Priv->MemorySize = 0;
+ Priv->TextLength = 0;
+ Priv->NumberLength = 0;
+
+ error = GSM_WaitFor (s, "AT+CPBR=?\r", 10, 0x00, 4, ID_GetMemoryStatus);
+ if (error != ERR_NONE) return error;
+ if (NeededInfo == AT_Total || NeededInfo == AT_Sizes || NeededInfo == AT_First) return ERR_NONE;
+
+ smprintf(s, "Getting memory status by reading values\n");
+
+ s->Phone.Data.MemoryStatus = Status;
+ Status->MemoryUsed = 0;
+ Status->MemoryFree = 0;
+ start = Priv->FirstMemoryEntry;
+ Priv->NextMemoryEntry = 0;
+ while (1) {
+ end = start + 20;
+ if (end > Priv->MemorySize) end = Priv->MemorySize;
+ sprintf(req, "AT+CPBR=%i,%i\r", start, end);
+ error = GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetMemoryStatus);
+ if (error != ERR_NONE) return error;
+ if (NeededInfo == AT_NextEmpty && Priv->NextMemoryEntry != 0 && Priv->NextMemoryEntry != end + 1) return ERR_NONE;
+ if (end == Priv->MemorySize) {
+ Status->MemoryFree = Priv->MemorySize - Status->MemoryUsed;
+ return ERR_NONE;
+ }
+ start = end + 1;
+ }
+}
+
+GSM_Error ATGEN_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ GSM_Error error;
+
+ error = ATGEN_SetPBKMemory(s, Status->MemoryType);
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.MemoryStatus=Status;
+
+ /* in some phones doesn't work or doesn't return memory status inside */
+ /* Some workaround for buggy mobile, that hangs after "AT+CPBS?" for other
+ * memory than SM.
+ */
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_BROKENCPBS) || (Status->MemoryType == MEM_SM)) {
+ smprintf(s, "Getting memory status\n");
+ error=GSM_WaitFor (s, "AT+CPBS?\r", 9, 0x00, 4, ID_GetMemoryStatus);
+ if (error == ERR_NONE) return ERR_NONE;
+ }
+
+ return ATGEN_GetMemoryInfo(s, Status, AT_Status);
+}
+
+GSM_Error ATGEN_SetPBKCharset(GSM_StateMachine *s, bool PreferUnicode)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+
+ /* Have we already selected something? */
+ if (Priv->PBKCharset!=0) {
+ /* If we want unicode charset and we have it already or setting of it
+ * failed, we have nothing to do. */
+ if (PreferUnicode && (Priv->PBKCharset==AT_PBK_UCS2 || Priv->UCS2CharsetFailed)) return ERR_NONE;
+
+ /* If we don't need unicode charset and we have some (or have unicode
+ * charset when other failed), we have nothing to do. */
+ if (!PreferUnicode && (Priv->PBKCharset!=AT_PBK_UCS2 || Priv->NonUCS2CharsetFailed)) return ERR_NONE;
+ }
+
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+
+ if (PreferUnicode && !Priv->UCS2CharsetFailed) {
+ smprintf(s, "Setting charset to UCS2\n");
+ error=GSM_WaitFor (s, "AT+CSCS=\"UCS2\"\r", 15, 0x00, 3, ID_SetMemoryCharset);
+ if (error == ERR_NONE) {
+ Priv->PBKCharset = AT_PBK_UCS2;
+ return ERR_NONE;
+ } else {
+ Priv->UCS2CharsetFailed = true;
+ }
+ }
+
+ smprintf(s, "Setting charset to HEX\n");
+ error=GSM_WaitFor (s, "AT+CSCS=\"HEX\"\r", 14, 0x00, 3, ID_SetMemoryCharset);
+ /* Falcom replies OK for HEX mode and send everything
+ * in normal format */
+ if (error == ERR_NONE && Priv->Manufacturer != AT_Falcom) {
+ Priv->PBKCharset = AT_PBK_HEX;
+ return ERR_NONE;
+ }
+
+ smprintf(s, "Setting charset to GSM\n");
+ error=GSM_WaitFor (s, "AT+CSCS=\"GSM\"\r", 14, 0x00, 3, ID_SetMemoryCharset);
+ if (error == ERR_NONE) {
+ Priv->PBKCharset = AT_PBK_GSM;
+ return ERR_NONE;
+ }
+
+ if (!Priv->UCS2CharsetFailed) {
+ Priv->NonUCS2CharsetFailed = true;
+ smprintf(s, "Setting charset to UCS2\n");
+ error=GSM_WaitFor (s, "AT+CSCS=\"UCS2\"\r", 15, 0x00, 3, ID_SetMemoryCharset);
+ if (error == ERR_NONE) {
+ Priv->PBKCharset = AT_PBK_UCS2;
+ return ERR_NONE;
+ } else {
+ Priv->UCS2CharsetFailed = true;
+ }
+ }
+
+ return error;
+}
+
+GSM_Error ATGEN_ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_MemoryEntry *Memory = s->Phone.Data.Memory;
+ char *pos;
+ unsigned char buffer[500],buffer2[500];
+ int len;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Phonebook entry received\n");
+ Memory->EntriesNum = 0;
+ if (Priv->Lines.numbers[4]==0) return ERR_EMPTY;
+ pos = strstr(msg.Buffer, "+CPBR:");
+ if (pos == NULL) return ERR_UNKNOWN;
+ /* Go after +CPBR: */
+ pos += 6;
+
+ /* Location */
+ while (*pos && !isdigit(*pos)) pos++;
+ Memory->Location = atoi(pos) + 1 - Priv->FirstMemoryEntry;
+ smprintf(s, "Location: %d\n", Memory->Location);
+
+ /* Number */
+ while (*pos != '"') pos++;
+ pos += ATGEN_ExtractOneParameter(pos, buffer);
+ smprintf(s, "Number: %s\n",buffer);
+ Memory->EntriesNum++;
+ Memory->Entries[0].EntryType = PBK_Number_General;
+ Memory->Entries[0].VoiceTag = 0;
+ Memory->Entries[0].SMSList[0] = 0;
+
+ len = strlen(buffer + 1) - 1;
+ if (Priv->PBKCharset == AT_PBK_HEX && (len > 10) && (len % 2 == 0) && (strchr(buffer + 1, '+') == NULL)) {
+ /* This is probably hex encoded number */
+ DecodeHexBin(buffer2, buffer+1, len);
+ DecodeDefault(Memory->Entries[0].Text ,buffer2, strlen(buffer2), false, NULL);
+ } else if (Priv->PBKCharset == AT_PBK_UCS2 && (len > 20) && (len % 4 == 0) && (strchr(buffer + 1, '+') == NULL)) {
+ /* This is probably unicode encoded number */
+ DecodeHexUnicode(Memory->Entries[0].Text, buffer + 1,len);
+ } else {
+ EncodeUnicode(Memory->Entries[0].Text, buffer + 1, len);
+ }
+
+ /* Number format */
+ pos += ATGEN_ExtractOneParameter(pos, buffer);
+ smprintf(s, "Number format: %s\n",buffer);
+
+ /* International number */
+ if (!strcmp(buffer,"145")) {
+ sprintf(buffer+1,"%s",DecodeUnicodeString(Memory->Entries[0].Text));
+ if (strlen(buffer+1)!=0 && buffer[1] != '+') {
+ /* Sony Ericsson issue */
+ /* International number is without + */
+ buffer[0] = '+';
+ EncodeUnicode(Memory->Entries[0].Text,buffer,strlen(buffer));
+ }
+ }
+
+ /* Name */
+ pos += ATGEN_ExtractOneParameter(pos, buffer);
+ smprintf(s, "Name text: %s\n",buffer);
+ Memory->EntriesNum++;
+ Memory->Entries[1].EntryType=PBK_Text_Name;
+ switch (Priv->PBKCharset) {
+ case AT_PBK_HEX:
+ DecodeHexBin(buffer2,buffer+1,strlen(buffer)-2);
+ DecodeDefault(Memory->Entries[1].Text,buffer2,strlen(buffer2),false,NULL);
+ break;
+ case AT_PBK_GSM:
+ DecodeDefault(Memory->Entries[1].Text,buffer+1,strlen(buffer)-2,false,NULL);
+ break;
+ case AT_PBK_UCS2:
+ DecodeHexUnicode(Memory->Entries[1].Text,buffer+1,strlen(buffer+1) - 1);
+ break;
+ }
+ return ERR_NONE;
+ case AT_Reply_CMEError:
+ return ATGEN_HandleCMEError(s);
+ case AT_Reply_Error:
+ smprintf(s, "Error - too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_PrivGetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry, int endlocation)
+{
+ GSM_Error error;
+ unsigned char req[20];
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (entry->Location==0x00) return ERR_INVALIDLOCATION;
+
+ if (entry->MemoryType == MEM_ME) {
+ if (Priv->PBKSBNR == 0) {
+ sprintf(req, "AT^SBNR=?\r");
+ smprintf(s, "Checking availablity of SBNR\n");
+ error=GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetMemory);
+ switch (error) {
+ case ERR_NONE:
+ Priv->PBKSBNR = AT_SBNR_AVAILABLE;
+ break;
+ case ERR_UNKNOWN:
+ case ERR_NOTSUPPORTED:
+ Priv->PBKSBNR = AT_SBNR_NOTAVAILABLE;
+ break;
+ default:
+ return error;
+ }
+ }
+ if (Priv->PBKSBNR == AT_SBNR_AVAILABLE) {
+ sprintf(req, "AT^SBNR=vcf,%i\r",entry->Location-1);
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetMemory);
+ }
+ }
+
+ error=ATGEN_GetManufacturer(s);
+ if (error != ERR_NONE) return error;
+
+ error=ATGEN_SetPBKMemory(s, entry->MemoryType);
+ if (error != ERR_NONE) return error;
+
+ if (Priv->FirstMemoryEntry == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_First);
+ if (error != ERR_NONE) return error;
+ }
+
+
+ error=ATGEN_SetPBKCharset(s, true); /* For reading we prefer unicode */
+ if (error != ERR_NONE) return error;
+
+ if (endlocation == 0) {
+ sprintf(req, "AT+CPBR=%i\r", entry->Location + Priv->FirstMemoryEntry - 1);
+ } else {
+ sprintf(req, "AT+CPBR=%i,%i\r", entry->Location + Priv->FirstMemoryEntry - 1, endlocation + Priv->FirstMemoryEntry - 1);
+ }
+
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetMemory);
+}
+
+GSM_Error ATGEN_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ return ATGEN_PrivGetMemory(s, entry, 0);
+}
+
+GSM_Error ATGEN_GetNextMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry, bool start)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ int step = 0;
+
+ if (Priv->MemorySize == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_Total);
+ if (error != ERR_NONE) return error;
+ }
+
+ if (start) {
+ entry->Location = 1;
+ } else {
+ entry->Location++;
+ }
+ while ((error = ATGEN_PrivGetMemory(s, entry, step == 0 ? 0 : MIN(Priv->MemorySize, entry->Location + step))) == ERR_EMPTY) {
+ entry->Location += step + 1;
+ if (entry->Location > Priv->MemorySize) break;
+ /* SNBR works only for one location */
+ if (entry->MemoryType != MEM_ME || Priv->PBKSBNR != AT_SBNR_AVAILABLE) step = MIN(step + 2, 20);
+ }
+ if (error == ERR_INVALIDLOCATION) return ERR_EMPTY;
+ return error;
+}
+
+GSM_Error ATGEN_DeleteAllMemory(GSM_StateMachine *s, GSM_MemoryType type)
+{
+ GSM_Error error;
+ unsigned char req[100];
+ int i;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ error = ATGEN_SetPBKMemory(s, type);
+ if (error != ERR_NONE) return error;
+
+ if (Priv->MemorySize == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_Total);
+ if (error != ERR_NONE) return error;
+ }
+
+ if (Priv->FirstMemoryEntry == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_First);
+ if (error != ERR_NONE) return error;
+ }
+
+
+ smprintf(s, "Deleting all phonebook entries\n");
+ for (i = Priv->FirstMemoryEntry; i < Priv->FirstMemoryEntry + Priv->MemorySize; i++) {
+ sprintf(req, "AT+CPBW=%d\r",i);
+ error = GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_SetMemory);
+ if (error != ERR_NONE) return error;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyDialVoice(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Dial voice OK\n");
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Dial voice error\n");
+ return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_DialVoice(GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber)
+{
+ char req[39] = "ATDT";
+
+ if (ShowNumber != GSM_CALL_DefaultNumberPresence) return ERR_NOTSUPPORTED;
+ if (strlen(number) > 32) return (ERR_UNKNOWN);
+
+ strcat(req, number);
+ strcat(req, ";\r");
+
+ smprintf(s, "Making voice call\n");
+ return GSM_WaitFor (s, req, 4+2+strlen(number), 0x00, 5, ID_DialVoice);
+}
+
+GSM_Error ATGEN_ReplyEnterSecurityCode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Security code was OK\n");
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Incorrect security code\n");
+ return ERR_SECURITYERROR;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_EnterSecurityCode(GSM_StateMachine *s, GSM_SecurityCode Code)
+{
+ unsigned char req[50];
+
+ switch (Code.Type) {
+ case SEC_Pin :
+ sprintf(req, "AT+CPIN=\"%s\"\r" , Code.Code);
+ break;
+ case SEC_Pin2 :
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer == AT_Siemens) {
+ sprintf(req, "AT+CPIN2=\"%s\"\r", Code.Code);
+ } else {
+ sprintf(req, "AT+CPIN=\"%s\"\r" , Code.Code);
+ }
+ break;
+ default : return ERR_NOTIMPLEMENTED;
+ }
+
+ smprintf(s, "Entering security code\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 6, ID_EnterSecurityCode);
+}
+
+GSM_Error ATGEN_ReplyGetSecurityStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SecurityCodeType *Status = s->Phone.Data.SecurityStatus;
+
+ smprintf(s, "Security status received - ");
+ if (strstr(msg.Buffer,"READY")) {
+ *Status = SEC_None;
+ smprintf(s, "nothing to enter\n");
+ return ERR_NONE;
+ }
+ if (strstr(msg.Buffer,"PH_SIM PIN")) {
+ smprintf(s, "no SIM inside or other error\n");
+ return ERR_UNKNOWN;
+ }
+ if (strstr(msg.Buffer,"SIM PIN2")) {
+ *Status = SEC_Pin2;
+ smprintf(s, "waiting for PIN2\n");
+ return ERR_NONE;
+ }
+ if (strstr(msg.Buffer,"SIM PUK2")) {
+ *Status = SEC_Puk2;
+ smprintf(s, "waiting for PUK2\n");
+ return ERR_NONE;
+ }
+ if (strstr(msg.Buffer,"SIM PIN")) {
+ *Status = SEC_Pin;
+ smprintf(s, "waiting for PIN\n");
+ return ERR_NONE;
+ }
+ if (strstr(msg.Buffer,"SIM PUK")) {
+ *Status = SEC_Puk;
+ smprintf(s, "waiting for PUK\n");
+ return ERR_NONE;
+ }
+ smprintf(s, "unknown\n");
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSecurityStatus(GSM_StateMachine *s, GSM_SecurityCodeType *Status)
+{
+ s->Phone.Data.SecurityStatus=Status;
+
+ smprintf(s, "Getting security code status\n");
+ /* Please note, that A2D doesn't return OK on the end.
+ * Because of it ReplyGetSecurityStatus is called after receiving line
+ * with +CPIN:
+ */
+ return GSM_WaitFor (s, "AT+CPIN?\r", 9, 0x00, 4, ID_GetSecurityStatus);
+}
+
+GSM_Error ATGEN_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (all) {
+ smprintf(s, "Answering all calls\n");
+ return GSM_WaitFor (s, "ATA\r", 4, 0x00, 4, ID_AnswerCall);
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error ATGEN_ReplyCancelCall(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Call call;
+
+ switch(s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Calls canceled\n");
+ call.CallIDAvailable = false;
+ call.Status = GSM_CALL_CallLocalEnd;
+ if (s->User.IncomingCall) s->User.IncomingCall(s->CurrentConfig->Device, call);
+
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ return ERR_UNKNOWN;
+ }
+}
+
+GSM_Error ATGEN_CancelCall(GSM_StateMachine *s, int ID, bool all)
+{
+ GSM_Error error;
+
+ if (all) {
+ smprintf(s, "Dropping all calls\n");
+ error = GSM_WaitFor (s, "ATH\r", 4, 0x00, 4, ID_CancelCall);
+ if (error == ERR_UNKNOWN) {
+ return GSM_WaitFor (s, "AT+CHUP\r", 8, 0x00, 4, ID_CancelCall);
+ }
+ return error;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error ATGEN_ReplyReset(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Reset done\n");
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_Reset(GSM_StateMachine *s, bool hard)
+{
+ GSM_Error error;
+
+ if (!hard) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Resetting device\n");
+ /* Siemens 35 */
+ error=GSM_WaitFor (s, "AT+CFUN=1,1\r", 12, 0x00, 8, ID_Reset);
+ if (error != ERR_NONE) {
+ /* Siemens M20 */
+ error=GSM_WaitFor (s, "AT^SRESET\r", 10, 0x00, 8, ID_Reset);
+ }
+ return error;
+}
+
+GSM_Error ATGEN_ReplyResetPhoneSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Reset done\n");
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ResetPhoneSettings(GSM_StateMachine *s, GSM_ResetSettingsType Type)
+{
+ smprintf(s, "Resetting settings to default\n");
+ return GSM_WaitFor (s, "AT&F\r", 5, 0x00, 4, ID_ResetPhoneSettings);
+}
+
+GSM_Error ATGEN_SetAutoNetworkLogin(GSM_StateMachine *s)
+{
+ smprintf(s, "Enabling automatic network login\n");
+ return GSM_WaitFor (s, "AT+COPS=0\r", 10, 0x00, 4, ID_SetAutoNetworkLogin);
+}
+
+GSM_Error ATGEN_SendDTMF(GSM_StateMachine *s, char *sequence)
+{
+ unsigned char req[80] = "AT+VTS=";
+ int n;
+
+ for (n = 0; n < 32; n++) {
+ if (sequence[n] == '\0') break;
+ if (n != 0) req[6 + 2 * n] = ',';
+ req[7 + 2 * n] = sequence[n];
+ }
+
+ strcat(req, ";\r");
+
+ smprintf(s, "Sending DTMF\n");
+ return GSM_WaitFor (s, req, 7+2+2*strlen(sequence), 0x00, 4, ID_SendDTMF);
+}
+
+GSM_Error ATGEN_ReplyDeleteSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "SMS deleted OK\n");
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_DeleteSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ unsigned char req[20], folderid;
+ GSM_Error error;
+ int location;
+ GSM_MultiSMSMessage msms;
+
+ msms.Number = 0;
+ msms.SMS[0] = *sms;
+
+ /* By reading SMS we check if it is really inbox/outbox */
+ error = ATGEN_GetSMS(s, &msms);
+ if (error != ERR_NONE) return error;
+
+ error = ATGEN_GetSMSLocation(s, sms, &folderid, &location);
+ if (error != ERR_NONE) return error;
+
+ sprintf(req, "AT+CMGD=%i\r",location);
+
+ smprintf(s, "Deleting SMS\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 5, ID_DeleteSMSMessage);
+}
+
+GSM_Error ATGEN_GetSMSFolders(GSM_StateMachine *s, GSM_SMSFolders *folders)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ int used = 0;
+
+ if (Priv->PhoneSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, false);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+ if (Priv->SIMSMSMemory == 0) {
+ error = ATGEN_SetSMSMemory(s, true);
+ if (error != ERR_NONE && error != ERR_NOTSUPPORTED) return error;
+ }
+
+ folders->Number = 0;
+ if (Priv->PhoneSMSMemory == AT_NOTAVAILABLE && Priv->SIMSMSMemory == AT_NOTAVAILABLE) {
+ return ERR_NONE;
+ }
+
+ PHONE_GetSMSFolders(s,folders);
+
+ if (Priv->SIMSMSMemory == AT_AVAILABLE) {
+ used = 2;
+ }
+
+ if (Priv->PhoneSMSMemory == AT_AVAILABLE) {
+ if (used != 0) {
+ CopyUnicodeString(folders->Folder[used ].Name,folders->Folder[0].Name);
+ CopyUnicodeString(folders->Folder[used + 1].Name,folders->Folder[1].Name);
+ folders->Folder[used ].InboxFolder = true;
+ folders->Folder[used + 1].InboxFolder = false;
+ }
+ folders->Folder[used ].Memory = MEM_ME;
+ folders->Folder[used + 1].Memory = MEM_ME;
+ folders->Number += 2;
+ used += 2;
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplySetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Phonebook entry written OK\n");
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ case AT_Reply_CMEError:
+ return ATGEN_HandleCMEError(s);
+ case AT_Reply_Error:
+ return ERR_INVALIDDATA;
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error ATGEN_DeleteMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ unsigned char req[100];
+
+ if (entry->Location < 1) return ERR_INVALIDLOCATION;
+
+ error = ATGEN_SetPBKMemory(s, entry->MemoryType);
+ if (error != ERR_NONE) return error;
+
+ if (Priv->FirstMemoryEntry == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_First);
+ if (error != ERR_NONE) return error;
+ }
+
+ sprintf(req, "AT+CPBW=%d\r",entry->Location + Priv->FirstMemoryEntry - 1);
+
+ smprintf(s, "Deleting phonebook entry\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_SetMemory);
+}
+
+GSM_Error ATGEN_PrivSetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ /* REQUEST_SIZE should be big enough to handle all possibl cases
+ * correctly, especially with unicode entries */
+#define REQUEST_SIZE ((4 * GSM_PHONEBOOK_TEXT_LENGTH) + 30)
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ int Group, Name, Number,NumberType=0, len;
+ GSM_Error error;
+ unsigned char req[REQUEST_SIZE + 1];
+ unsigned char name[2*(GSM_PHONEBOOK_TEXT_LENGTH + 1)];
+ unsigned char uname[2*(GSM_PHONEBOOK_TEXT_LENGTH + 1)];
+ unsigned char number[GSM_PHONEBOOK_TEXT_LENGTH + 1];
+ int reqlen;
+ bool PreferUnicode = false;
+
+ if (entry->Location == 0) return ERR_INVALIDLOCATION;
+
+ error = ATGEN_SetPBKMemory(s, entry->MemoryType);
+ if (error != ERR_NONE) return error;
+
+ GSM_PhonebookFindDefaultNameNumberGroup(entry, &Name, &Number, &Group);
+
+ name[0] = 0;
+ if (Name != -1) {
+ len = UnicodeLength(entry->Entries[Name].Text);
+
+ /* Compare if we would loose some information when not using
+ * unicode */
+ EncodeDefault(name, entry->Entries[Name].Text, &len, true, NULL);
+ DecodeDefault(uname, name, len, true, NULL);
+ if (!mywstrncmp(uname, entry->Entries[Name].Text, len)) {
+ /* Get maximal text length */
+ if (Priv->TextLength == 0) {
+ ATGEN_GetMemoryInfo(s, NULL, AT_Sizes);
+ }
+
+ /* I char stored in GSM alphabet takes 7 bits, one
+ * unicode 16, if storing in unicode would truncate
+ * text, do not use it, otherwise we will use it */
+ if ((Priv->TextLength != 0) && ((Priv->TextLength * 7 / 16) <= len)) {
+ PreferUnicode = false;
+ } else {
+ PreferUnicode = true;
+ }
+ }
+
+ error = ATGEN_SetPBKCharset(s, PreferUnicode);
+ if (error != ERR_NONE) return error;
+
+ switch (Priv->PBKCharset) {
+ case AT_PBK_HEX:
+ EncodeHexBin(name, DecodeUnicodeString(entry->Entries[Name].Text), UnicodeLength(entry->Entries[Name].Text));
+ len = strlen(name);
+ break;
+ case AT_PBK_GSM:
+ smprintf(s, "str: %s\n", DecodeUnicodeString(entry->Entries[Name].Text));
+ len = UnicodeLength(entry->Entries[Name].Text);
+ EncodeDefault(name, entry->Entries[Name].Text, &len, true, NULL);
+ break;
+ case AT_PBK_UCS2:
+ EncodeHexUnicode(name, entry->Entries[Name].Text, UnicodeLength(entry->Entries[Name].Text));
+ len = strlen(name);
+ break;
+ }
+ } else {
+ smprintf(s, "WARNING: No usable name found!\n");
+ len = 0;
+ }
+
+ if (Number != -1) {
+ GSM_PackSemiOctetNumber(entry->Entries[Number].Text, number, false);
+ NumberType = number[0];
+ sprintf(number,"%s",DecodeUnicodeString(entry->Entries[Number].Text));
+ } else {
+ smprintf(s, "WARNING: No usable number found!\n");
+ number[0] = 0;
+ }
+
+ if (Priv->FirstMemoryEntry == 0) {
+ error = ATGEN_GetMemoryInfo(s, NULL, AT_First);
+ if (error != ERR_NONE) return error;
+ }
+
+ /* We can't use here:
+ * sprintf(req, "AT+CPBW=%d, \"%s\", %i, \"%s\"\r",
+ * entry->Location, number, NumberType, name);
+ * because name can contain 0 when using GSM alphabet.
+ */
+ sprintf(req, "AT+CPBW=%d, \"%s\", %i, \"", entry->Location + Priv->FirstMemoryEntry - 1, number, NumberType);
+ reqlen = strlen(req);
+ if (reqlen + len > REQUEST_SIZE - 2) {
+ smprintf(s, "WARNING: Text truncated to fit in buffer!\n");
+ len = REQUEST_SIZE - 2 - reqlen;
+ }
+ memcpy(req + reqlen, name, len);
+ reqlen += len;
+ memcpy(req + reqlen, "\"\r", 2);
+ reqlen += 2;
+
+ smprintf(s, "Writing phonebook entry\n");
+ return GSM_WaitFor (s, req, reqlen, 0x00, 4, ID_SetMemory);
+#undef REQUEST_SIZE
+}
+
+GSM_Error ATGEN_SetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ if (entry->Location == 0) return ERR_INVALIDLOCATION;
+ return ATGEN_PrivSetMemory(s, entry);
+}
+
+GSM_Error ATGEN_AddMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_Error error;
+ GSM_MemoryStatus Status;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ /* Find out empty location */
+ error = ATGEN_GetMemoryInfo(s, &Status, AT_NextEmpty);
+ if (error != ERR_NONE) return error;
+ if (Priv->NextMemoryEntry == 0) return ERR_FULL;
+ entry->Location = Priv->NextMemoryEntry;
+
+ return ATGEN_PrivSetMemory(s, entry);
+}
+
+/* Use ATGEN_ExtractOneParameter ?? */
+void Extract_CLIP_number(char *dest, char *buf)
+{
+ char *start, *stop;
+ int i = 0;
+
+ stop = strstr(buf, ",");
+ if (stop != NULL) {
+ start = strstr(buf, ":");
+ if (start != NULL) {
+ for (start = start + 2; start + i < stop; i++)
+ dest[i] = start[i];
+ }
+ }
+ dest[i] = 0; /* end the number */
+
+ return;
+}
+
+GSM_Error ATGEN_ReplyIncomingCallInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ char num[128];
+ GSM_Call call;
+
+ smprintf(s, "Incoming call info\n");
+ if (s->Phone.Data.EnableIncomingCall && s->User.IncomingCall!=NULL) {
+ call.CallIDAvailable = false;
+ num[0] = 0;
+ if (strstr(msg.Buffer, "RING")) {
+ call.Status = GSM_CALL_IncomingCall;
+ Extract_CLIP_number(num, msg.Buffer);
+ } else if (strstr(msg.Buffer, "NO CARRIER")) {
+ call.Status = GSM_CALL_CallEnd;
+ } else if (strstr(msg.Buffer, "COLP:")) {
+ call.Status = GSM_CALL_CallStart;
+ Extract_CLIP_number(num, msg.Buffer);
+ } else {
+ smprintf(s, "CLIP: error\n");
+ return ERR_NONE;
+ }
+ EncodeUnicode(call.PhoneNumber, num, strlen(num));
+
+ s->User.IncomingCall(s->CurrentConfig->Device, call);
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_IncomingGPRS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ /* "+CGREG: 1,1" */
+ smprintf(s, "GPRS change\n");
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_IncomingBattery(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int level = 0;
+ char *p;
+
+ /* "_OBS: 92,1" */
+ p = strstr(msg.Buffer, "_OBS:");
+ if (p) level = atoi(p + 5);
+ smprintf(s, "Battery level changed to %d\n", level);
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_IncomingNetworkLevel(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int level = 0;
+ char *p;
+
+ /* "_OSIGQ: 12,0" */
+ p = strstr(msg.Buffer, "_OSIGQ: ");
+ if (p) level = atoi(p + 7);
+ smprintf(s, "Network level changed to %d\n", level);
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyGetSIMIMSI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ char *c;
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ CopyLineString(Data->PhoneString, msg.Buffer, Priv->Lines, 2);
+
+ /* Read just IMSI also on phones that prepend it by "<IMSI>:" (Alcatel BE5) */
+ c = strstr(Data->PhoneString, "<IMSI>:");
+ if (c != NULL) {
+ c += 7;
+ memmove(Data->PhoneString, c, strlen(c) + 1);
+ }
+
+ smprintf(s, "Received IMSI %s\n",Data->PhoneString);
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "No access to SIM card or not supported by device\n");
+ return ERR_SECURITYERROR;
+ case AT_Reply_CMEError:
+ return ATGEN_HandleCMEError(s);
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSIMIMSI(GSM_StateMachine *s, char *IMSI)
+{
+ s->Phone.Data.PhoneString = IMSI;
+ smprintf(s, "Getting SIM IMSI\n");
+ return GSM_WaitFor (s, "AT+CIMI\r", 8, 0x00, 4, ID_GetSIMIMSI);
+}
+
+GSM_Error ATGEN_GetDisplayStatus(GSM_StateMachine *s, GSM_DisplayFeatures *features)
+{
+ return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.DisplayFeatures = features;
+ smprintf(s, "Getting display status\n");
+ return GSM_WaitFor (s, "AT+CIND?\r",9, 0x00, 4, ID_GetDisplayStatus);
+}
+
+GSM_Error ATGEN_IncomingSMSCInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_ReplyGetBatteryCharge(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ int i;
+
+ Data->BatteryCharge->BatteryPercent = -1;
+ Data->BatteryCharge->ChargeState = 0;
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Battery level received\n");
+ Data->BatteryCharge->BatteryPercent = atoi(msg.Buffer+17);
+ i = atoi(msg.Buffer+14);
+ if (i >= 0 && i <= 3) {
+ Data->BatteryCharge->ChargeState = i + 1;
+ }
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Can't get battery level\n");
+ return ERR_UNKNOWN;
+ case AT_Reply_CMSError:
+ smprintf(s, "Can't get battery level\n");
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetBatteryCharge(GSM_StateMachine *s, GSM_BatteryCharge *bat)
+{
+ s->Phone.Data.BatteryCharge = bat;
+ smprintf(s, "Getting battery charge\n");
+ return GSM_WaitFor (s, "AT+CBC\r", 7, 0x00, 4, ID_GetBatteryCharge);
+}
+
+GSM_Error ATGEN_ReplyGetSignalQuality(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SignalQuality *Signal = s->Phone.Data.SignalQuality;
+ int i;
+ char *pos;
+
+ Signal->SignalStrength = -1;
+ Signal->SignalPercent = -1;
+ Signal->BitErrorRate = -1;
+
+ switch (s->Phone.Data.Priv.ATGEN.ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Signal quality info received\n");
+ i = atoi(msg.Buffer+15);
+ if (i != 99) {
+ /* from GSM 07.07 section 8.5 */
+ Signal->SignalStrength = 2 * i - 113;
+
+ /* FIXME: this is wild guess and probably will be phone dependant */
+ Signal->SignalPercent = 15 * i;
+ if (Signal->SignalPercent > 100) Signal->SignalPercent = 100;
+ }
+ pos = strchr(msg.Buffer + 15, ',');
+ if (pos != NULL) {
+ i = atoi(pos + 1);
+ /* from GSM 05.08 section 8.2.4 */
+ switch (i) {
+ case 0: Signal->BitErrorRate = 0; break; /* 0.14 */
+ case 1: Signal->BitErrorRate = 0; break; /* 0.28 */
+ case 2: Signal->BitErrorRate = 1; break; /* 0.57 */
+ case 3: Signal->BitErrorRate = 1; break; /* 1.13 */
+ case 4: Signal->BitErrorRate = 2; break; /* 2.26 */
+ case 5: Signal->BitErrorRate = 5; break; /* 4.53 */
+ case 6: Signal->BitErrorRate = 9; break; /* 9.05 */
+ case 7: Signal->BitErrorRate = 18; break; /* 18.10 */
+ }
+ }
+ return ERR_NONE;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error ATGEN_GetSignalQuality(GSM_StateMachine *s, GSM_SignalQuality *sig)
+{
+ s->Phone.Data.SignalQuality = sig;
+ smprintf(s, "Getting signal quality info\n");
+ return GSM_WaitFor (s, "AT+CSQ\r", 7, 0x00, 4, ID_GetSignalQuality);
+}
+
+/* When use AT+CPIN?, A2D returns it without OK and because of it Gammu
+ parses answer without it.
+ MC35 and other return OK after answer for AT+CPIN?. Here we handle it.
+ Any better idea ?
+ */
+GSM_Error ATGEN_ReplyOK(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+static GSM_Error ATGEN_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer==AT_Siemens ) return SIEMENS_GetNextCalendar(s,Note,start);
+ if (Priv->Manufacturer==AT_Ericsson) return SONYERIC_GetNextCalendar(s,Note,start);
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error ATGEN_Terminate(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ free(Priv->file.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_AddCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer==AT_Siemens) return SIEMENS_AddCalendarNote(s, Note);
+ if (Priv->Manufacturer==AT_Ericsson) return SONYERIC_AddCalendarNote(s, Note);
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error ATGEN_DelCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer==AT_Siemens) return SIEMENS_DelCalendarNote(s, Note);
+ if (Priv->Manufacturer==AT_Ericsson) return SONYERIC_DelCalendarNote(s, Note);
+ return ERR_NOTSUPPORTED;
+}
+
+GSM_Error ATGEN_PressKey(GSM_StateMachine *s, GSM_KeyCode Key, bool Press)
+{
+ GSM_Error error;
+ unsigned char Frame[] = "AT+CKPD=\"?\"\r";
+
+ if (Press) {
+ switch (Key) {
+ case GSM_KEY_1 : Frame[9] = '1'; break;
+ case GSM_KEY_2 : Frame[9] = '2'; break;
+ case GSM_KEY_3 : Frame[9] = '3'; break;
+ case GSM_KEY_4 : Frame[9] = '4'; break;
+ case GSM_KEY_5 : Frame[9] = '5'; break;
+ case GSM_KEY_6 : Frame[9] = '6'; break;
+ case GSM_KEY_7 : Frame[9] = '7'; break;
+ case GSM_KEY_8 : Frame[9] = '8'; break;
+ case GSM_KEY_9 : Frame[9] = '9'; break;
+ case GSM_KEY_0 : Frame[9] = '0'; break;
+ case GSM_KEY_HASH : Frame[9] = '#'; break;
+ case GSM_KEY_ASTERISK : Frame[9] = '*'; break;
+ case GSM_KEY_POWER : return ERR_NOTSUPPORTED;
+ case GSM_KEY_GREEN : Frame[9] = 'S'; break;
+ case GSM_KEY_RED : Frame[9] = 'E'; break;
+ case GSM_KEY_INCREASEVOLUME : Frame[9] = 'U'; break;
+ case GSM_KEY_DECREASEVOLUME : Frame[9] = 'D'; break;
+ case GSM_KEY_UP : Frame[9] = '^'; break;
+ case GSM_KEY_DOWN : Frame[9] = 'V'; break;
+ case GSM_KEY_MENU : Frame[9] = 'F'; break;
+ case GSM_KEY_NAMES : Frame[9] = 'C'; break;
+ default : return ERR_NOTSUPPORTED;
+ }
+ smprintf(s, "Pressing key\n");
+ error = GSM_WaitFor (s, Frame, 12, 0x00, 4, ID_PressKey);
+ if (error != ERR_NONE) return error;
+
+ /* Strange. My T310 needs it */
+ return GSM_WaitFor (s, "ATE1\r", 5, 0x00, 4, ID_EnableEcho);
+ } else {
+ return ERR_NONE;
+ }
+}
+
+#ifdef GSM_ENABLE_CELLBROADCAST
+
+GSM_Error ATGEN_ReplyIncomingCB(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_CBMessage CB;
+ int i,j;
+ char Buffer[300],Buffer2[300];
+
+ smprintf(s, "CB received\n");
+ return ERR_NONE;
+
+ DecodeHexBin (Buffer,msg.Buffer+6,msg.Length-6);
+ DumpMessage(stdout, di.dl ,Buffer,msg.Length-6);
+
+ CB.Channel = Buffer[4];
+
+ for (j=0;j<msg.Length;j++) {
+ smprintf(s, "j=%i\n",j);
+ i=GSM_UnpackEightBitsToSeven(0, msg.Buffer[6], msg.Buffer[6], msg.Buffer+j, Buffer2);
+// i = msg.Buffer[6] - 1;
+// while (i!=0) {
+// if (Buffer[i] == 13) i = i - 1; else break;
+// }
+ DecodeDefault(CB.Text, Buffer2, msg.Buffer[6], false, NULL);
+ smprintf(s, "Channel %i, text \"%s\"\n",CB.Channel,DecodeUnicodeString(CB.Text));
+ }
+ if (s->Phone.Data.EnableIncomingCB && s->User.IncomingCB!=NULL) {
+ s->User.IncomingCB(s->CurrentConfig->Device,CB);
+ }
+ return ERR_NONE;
+}
+
+#endif
+
+GSM_Error ATGEN_SetIncomingCB(GSM_StateMachine *s, bool enable)
+{
+#ifdef GSM_ENABLE_CELLBROADCAST
+ if (s->Phone.Data.EnableIncomingCB!=enable) {
+ s->Phone.Data.EnableIncomingCB = enable;
+ if (enable) {
+ smprintf(s, "Enabling incoming CB\n");
+ return GSM_WaitFor(s, "AT+CNMI=3,,2\r", 13, 0x00, 4, ID_SetIncomingCB);
+ } else {
+ smprintf(s, "Disabling incoming CB\n");
+ return GSM_WaitFor(s, "AT+CNMI=3,,0\r", 13, 0x00, 4, ID_SetIncomingCB);
+ }
+ }
+ return ERR_NONE;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error ATGEN_IncomingSMSInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Incoming SMS\n");
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_IncomingSMSDeliver(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ GSM_SMSMessage sms;
+ int current = 0, current2, i=0;
+ unsigned char buffer[300],smsframe[800];
+
+ smprintf(s, "Incoming SMS received (Deliver)\n");
+ if (Data->EnableIncomingSMS && s->User.IncomingSMS!=NULL) {
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ sms.PDU = SMS_Deliver;
+
+ /* T310 with larger SMS goes crazy and mix this incoming
+ * frame with normal answers. PDU is always last frame
+ * We find its' number and parse it */
+ while (Data->Priv.ATGEN.Lines.numbers[i*2+1] != 0) {
+ /* FIXME: handle special chars correctly */
+ i++;
+ }
+ DecodeHexBin (buffer,
+ GetLineString(msg.Buffer,Data->Priv.ATGEN.Lines,i),
+ strlen(GetLineString(msg.Buffer,Data->Priv.ATGEN.Lines,i)));
+
+ /* We use locations from SMS layouts like in ../phone2.c(h) */
+ for(i=0;i<buffer[0]+1;i++) smsframe[i]=buffer[current++];
+ smsframe[12]=buffer[current++];
+
+ current2=((buffer[current])+1)/2+1;
+ for(i=0;i<current2+1;i++) smsframe[PHONE_SMSDeliver.Number+i]=buffer[current++];
+ smsframe[PHONE_SMSDeliver.TPPID] = buffer[current++];
+ smsframe[PHONE_SMSDeliver.TPDCS] = buffer[current++];
+ for(i=0;i<7;i++) smsframe[PHONE_SMSDeliver.DateTime+i]=buffer[current++];
+ smsframe[PHONE_SMSDeliver.TPUDL] = buffer[current++];
+ for(i=0;i<smsframe[PHONE_SMSDeliver.TPUDL];i++) smsframe[i+PHONE_SMSDeliver.Text]=buffer[current++];
+ GSM_DecodeSMSFrame(&sms,smsframe,PHONE_SMSDeliver);
+
+ s->User.IncomingSMS(s->CurrentConfig->Device,sms);
+ }
+ return ERR_NONE;
+}
+
+/* I don't have phone able to do it and can't fill it */
+GSM_Error ATGEN_IncomingSMSReport(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Incoming SMS received (Report)\n");
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_SetIncomingSMS(GSM_StateMachine *s, bool enable)
+{
+ /* Nokia returns OK, but doesn't return anything */
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer == AT_Nokia) return ERR_NOTSUPPORTED;
+
+ if (s->Phone.Data.EnableIncomingSMS!=enable) {
+ s->Phone.Data.EnableIncomingSMS = enable;
+ if (enable) {
+ smprintf(s, "Enabling incoming SMS\n");
+
+ /* Delivery reports */
+ GSM_WaitFor(s, "AT+CNMI=3,,,1\r", 14, 0x00, 4, ID_SetIncomingSMS);
+
+ /* SMS deliver */
+ return GSM_WaitFor(s, "AT+CNMI=3,3\r", 12, 0x00, 4, ID_SetIncomingSMS);
+ } else {
+ smprintf(s, "Disabling incoming SMS\n");
+ return GSM_WaitFor(s, "AT+CNMI=3,0\r", 12, 0x00, 4, ID_SetIncomingSMS);
+ }
+ }
+ return ERR_NONE;
+}
+
+GSM_Reply_Function ATGENReplyFunctions[] = {
+{ATGEN_GenericReply, "AT\r" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_GenericReply, "ATE1" ,0x00,0x00,ID_EnableEcho },
+{ATGEN_GenericReply, "AT+CMEE=" ,0x00,0x00,ID_EnableErrorInfo },
+{ATGEN_GenericReply, "AT+CKPD=" ,0x00,0x00,ID_PressKey },
+{ATGEN_ReplyGetSIMIMSI, "AT+CIMI" ,0x00,0x00,ID_GetSIMIMSI },
+{ATGEN_GenericReply, "AT*EOBEX" ,0x00,0x00,ID_SetOBEX },
+
+#ifdef GSM_ENABLE_CELLBROADCAST
+{ATGEN_ReplyIncomingCB, "+CBM:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_GenericReply, "AT+CNMI" ,0x00,0x00,ID_SetIncomingCB },
+#endif
+
+{ATGEN_IncomingBattery, "_OBS:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyGetBatteryCharge, "AT+CBC" ,0x00,0x00,ID_GetBatteryCharge },
+
+{ATGEN_ReplyGetModel, "AT+CGMM" ,0x00,0x00,ID_GetModel },
+{ATGEN_ReplyGetManufacturer, "AT+CGMI" ,0x00,0x00,ID_GetManufacturer },
+{ATGEN_ReplyGetFirmwareCGMR, "AT+CGMR" ,0x00,0x00,ID_GetFirmware },
+{ATGEN_ReplyGetFirmwareATI, "ATI" ,0x00,0x00,ID_GetFirmware },
+{ATGEN_ReplyGetIMEI, "AT+CGSN" ,0x00,0x00,ID_GetIMEI },
+
+{ATGEN_ReplySendSMS, "AT+CMGS" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplySendSMS, "AT+CMSS" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_GenericReply, "AT+CNMI" ,0x00,0x00,ID_SetIncomingSMS },
+{ATGEN_GenericReply, "AT+CMGF" ,0x00,0x00,ID_GetSMSMode },
+{ATGEN_GenericReply, "AT+CSDH" ,0x00,0x00,ID_GetSMSMode },
+{ATGEN_ReplyGetSMSMessage, "AT+CMGR" ,0x00,0x00,ID_GetSMSMessage },
+{ATGEN_GenericReply, "AT+CPMS" ,0x00,0x00,ID_SetMemoryType },
+{ATGEN_ReplyGetSMSStatus, "AT+CPMS" ,0x00,0x00,ID_GetSMSStatus },
+{ATGEN_ReplyGetSMSMemories, "AT+CPMS=?" ,0x00,0x00,ID_GetSMSMemories },
+{ATGEN_ReplyAddSMSMessage, "AT+CMGW" ,0x00,0x00,ID_SaveSMSMessage },
+{ATGEN_GenericReply, "AT+CSMP" ,0x00,0x00,ID_SetSMSParameters },
+{ATGEN_GenericReply, "AT+CSCA" ,0x00,0x00,ID_SetSMSC },
+{ATGEN_ReplyGetSMSC, "AT+CSCA?" ,0x00,0x00,ID_GetSMSC },
+{ATGEN_ReplyDeleteSMSMessage, "AT+CMGD" ,0x00,0x00,ID_DeleteSMSMessage },
+{ATGEN_GenericReply, "ATE1" ,0x00,0x00,ID_SetSMSParameters },
+{ATGEN_GenericReply, "\x1b\x0D" ,0x00,0x00,ID_SetSMSParameters },
+{ATGEN_IncomingSMSInfo, "+CMTI:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_IncomingSMSDeliver, "+CMT:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_IncomingSMSReport, "+CDS:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_IncomingSMSCInfo, "^SCN:" ,0x00,0x00,ID_IncomingFrame },
+
+{ATGEN_ReplyGetDateTime_Alarm, "AT+CCLK?" ,0x00,0x00,ID_GetDateTime },
+{ATGEN_GenericReply, "AT+CCLK=" ,0x00,0x00,ID_SetDateTime },
+{ATGEN_ReplyGetDateTime_Alarm, "AT+CALA?" ,0x00,0x00,ID_GetAlarm },
+
+{ATGEN_ReplyGetNetworkLAC_CID, "AT+CREG?" ,0x00,0x00,ID_GetNetworkInfo },
+{ATGEN_GenericReply, "AT+CREG=2" ,0x00,0x00,ID_GetNetworkInfo },
+{ATGEN_GenericReply, "AT+COPS=" ,0x00,0x00,ID_GetNetworkInfo },
+{ATGEN_GenericReply, "AT+COPS=" ,0x00,0x00,ID_SetAutoNetworkLogin},
+{ATGEN_ReplyGetNetworkCode, "AT+COPS" ,0x00,0x00,ID_GetNetworkInfo },
+{ATGEN_ReplyGetSignalQuality, "AT+CSQ" ,0x00,0x00,ID_GetSignalQuality },
+{ATGEN_IncomingNetworkLevel, "_OSIGQ:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_IncomingGPRS, "+CGREG:" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyGetNetworkLAC_CID, "+CREG:" ,0x00,0x00,ID_IncomingFrame },
+
+{ATGEN_ReplyGetPBKMemories, "AT+CPBS=?" ,0x00,0x00,ID_SetMemoryType },
+{ATGEN_GenericReply, "AT+CPBS=" ,0x00,0x00,ID_SetMemoryType },
+{ATGEN_ReplyGetCPBSMemoryStatus,"AT+CPBS?" ,0x00,0x00,ID_GetMemoryStatus },
+{ATGEN_ReplyGetCPBRMemoryInfo, "AT+CPBR=?" ,0x00,0x00,ID_GetMemoryStatus },
+{ATGEN_ReplyGetCPBRMemoryStatus,"AT+CPBR=" ,0x00,0x00,ID_GetMemoryStatus },
+{ATGEN_GenericReply, "AT+CSCS=" ,0x00,0x00,ID_SetMemoryCharset },
+{ATGEN_ReplyGetMemory, "AT+CPBR=" ,0x00,0x00,ID_GetMemory },
+{ATGEN_GenericReply, "AT^SBNR=?" ,0x00,0x00,ID_GetMemory },
+{ATGEN_SL45ReplyGetMemory, "AT^SBNR" ,0x00,0x00,ID_GetMemory },
+{ATGEN_ReplySetMemory, "AT+CPBW" ,0x00,0x00,ID_SetMemory },
+
+{ATGEN_CMS35ReplyGetBitmap, "AT^SBNR=\"bmp\"" ,0x00,0x00,ID_GetBitmap },
+{ATGEN_CMS35ReplySetBitmap, "AT^SBNW=\"bmp\"" ,0x00,0x00,ID_SetBitmap },
+
+{ATGEN_CMS35ReplyGetRingtone, "AT^SBNR=\"mid\"" ,0x00,0x00,ID_GetRingtone },
+{ATGEN_CMS35ReplySetRingtone, "AT^SBNW=\"mid\"" ,0x00,0x00,ID_SetRingtone },
+
+{ATGEN_CMS35ReplyGetNextCal, "AT^SBNR=\"vcs\"" ,0x00,0x00,ID_GetCalendarNote },
+{ATGEN_CMS35ReplySetCalendar, "AT^SBNW=\"vcs\"" ,0x00,0x00,ID_SetCalendarNote },
+{ATGEN_CMS35ReplyDeleteCalendar,"AT^SBNW=\"vcs\"" ,0x00,0x00,ID_DeleteCalendarNote },
+
+{ATGEN_ReplyEnterSecurityCode, "AT+CPIN=" ,0x00,0x00,ID_EnterSecurityCode },
+{ATGEN_ReplyEnterSecurityCode, "AT+CPIN2=" ,0x00,0x00,ID_EnterSecurityCode },
+{ATGEN_ReplyGetSecurityStatus, "AT+CPIN?" ,0x00,0x00,ID_GetSecurityStatus },
+{ATGEN_ReplyOK, "OK" ,0x00,0x00,ID_IncomingFrame },
+
+{ATGEN_GenericReply, "AT+VTS" ,0x00,0x00,ID_SendDTMF },
+{ATGEN_ReplyCancelCall, "AT+CHUP" ,0x00,0x00,ID_CancelCall },
+{ATGEN_ReplyDialVoice, "ATDT" ,0x00,0x00,ID_DialVoice },
+{ATGEN_ReplyCancelCall, "ATH" ,0x00,0x00,ID_CancelCall },
+{ATGEN_GenericReply, "AT+CLIP=1" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyIncomingCallInfo, "+CLIP" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyIncomingCallInfo, "+COLP" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyIncomingCallInfo, "RING" ,0x00,0x00,ID_IncomingFrame },
+{ATGEN_ReplyIncomingCallInfo, "NO CARRIER" ,0x00,0x00,ID_IncomingFrame },
+
+{ATGEN_ReplyReset, "AT^SRESET" ,0x00,0x00,ID_Reset },
+{ATGEN_ReplyReset, "AT+CFUN=1,1" ,0x00,0x00,ID_Reset },
+{ATGEN_ReplyResetPhoneSettings, "AT&F" ,0x00,0x00,ID_ResetPhoneSettings },
+
+#ifdef GSM_ENABLE_ALCATEL
+/* Why do I give Alcatel specific things here? It's simple, Alcatel needs
+ * some AT commands to start it's binary mode, so this needs to be in AT
+ * related stuff.
+ *
+ * XXX: AT+IFC could later move outside this ifdef, because it is not Alcatel
+ * specific and it's part of ETSI specifications
+ */
+{ATGEN_GenericReply, "AT+IFC" ,0x00,0x00,ID_SetFlowControl },
+{ALCATEL_ProtocolVersionReply, "AT+CPROT=?" ,0x00,0x00,ID_AlcatelProtocol },
+{ATGEN_GenericReply, "AT+CPROT" ,0x00,0x00,ID_AlcatelConnect },
+#endif
+
+{NULL, "\x00" ,0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions ATGENPhone = {
+ "A2D|iPAQ|at|M20|S25|MC35|C35i|5110|5130|5190|5210|6110|6130|6150|6190|6210|6250|6310|6310i|6510|7110|8210|8250|8290|8310|8390|8850|8855|8890|8910|9110|9210",
+ ATGENReplyFunctions,
+ ATGEN_Initialise,
+ ATGEN_Terminate,
+ ATGEN_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ ATGEN_GetManufacturer,
+ ATGEN_GetModel,
+ ATGEN_GetFirmware,
+ ATGEN_GetIMEI,
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ NOTSUPPORTED, /* GetProductCode */
+ NOTSUPPORTED, /* GetHardware */
+ NOTSUPPORTED, /* GetPPM */
+ ATGEN_GetSIMIMSI,
+ ATGEN_GetDateTime,
+ ATGEN_SetDateTime,
+ ATGEN_GetAlarm,
+ NOTIMPLEMENTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ ATGEN_PressKey,
+ ATGEN_Reset,
+ ATGEN_ResetPhoneSettings,
+ ATGEN_EnterSecurityCode,
+ ATGEN_GetSecurityStatus,
+ ATGEN_GetDisplayStatus,
+ ATGEN_SetAutoNetworkLogin,
+ ATGEN_GetBatteryCharge,
+ ATGEN_GetSignalQuality,
+ ATGEN_GetNetworkInfo,
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ ATGEN_GetMemoryStatus,
+ ATGEN_GetMemory,
+ ATGEN_GetNextMemory,
+ ATGEN_SetMemory,
+ ATGEN_AddMemory,
+ ATGEN_DeleteMemory,
+ ATGEN_DeleteAllMemory,
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ ATGEN_GetSMSC,
+ ATGEN_SetSMSC,
+ ATGEN_GetSMSStatus,
+ ATGEN_GetSMS,
+ ATGEN_GetNextSMS,
+ NOTSUPPORTED, /* SetSMS */
+ ATGEN_AddSMS,
+ ATGEN_DeleteSMS,
+ ATGEN_SendSMS,
+ ATGEN_SendSavedSMS,
+ ATGEN_SetIncomingSMS,
+ ATGEN_SetIncomingCB,
+ ATGEN_GetSMSFolders,
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ ATGEN_DialVoice,
+ ATGEN_AnswerCall,
+ ATGEN_CancelCall,
+ NOTSUPPORTED, /* HoldCall */
+ NOTSUPPORTED, /* UnholdCall */
+ NOTSUPPORTED, /* ConferenceCall */
+ NOTSUPPORTED, /* SplitCall */
+ NOTSUPPORTED, /* TransferCall */
+ NOTSUPPORTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NONEFUNCTION, /* SetIncomingCall */
+ NOTSUPPORTED, /* SetIncomingUSSD */
+ ATGEN_SendDTMF,
+ ATGEN_GetRingtone,
+ ATGEN_SetRingtone,
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ ATGEN_GetBitmap, /* GetBitmap */
+ ATGEN_SetBitmap, /* SetBitmap */
+ SONYERIC_GetToDoStatus,
+ NOTSUPPORTED, /* GetToDo */
+ SONYERIC_GetNextToDo,
+ NOTSUPPORTED, /* SetToDo */
+ SONYERIC_AddToDo,
+ NOTSUPPORTED, /* DeleteToDo */
+ SONYERIC_DeleteAllToDo,
+ SONYERIC_GetCalendarStatus,
+ NOTIMPLEMENTED, /* GetCalendar */
+ ATGEN_GetNextCalendar,
+ NOTIMPLEMENTED, /* SetCalendar */
+ ATGEN_AddCalendarNote,
+ ATGEN_DelCalendarNote,
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFile */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/at/atgen.h b/gammu/emb/common/phone/at/atgen.h
new file mode 100644
index 0000000..0e08ee4
--- a/dev/null
+++ b/gammu/emb/common/phone/at/atgen.h
@@ -0,0 +1,110 @@
+/* (c) 2002-2004 by Marcin Wiacek and Michal Cihar */
+
+#ifndef atgen_h
+#define atgen_h
+
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "../../service/sms/gsmsms.h"
+
+#ifndef GSM_USED_AT
+# define GSM_USED_AT
+#endif
+#ifndef GSM_USED_BLUEAT
+# define GSM_USED_BLUEAT
+#endif
+#ifndef GSM_USED_IRDAAT
+# define GSM_USED_IRDAAT
+#endif
+
+#define MAX_VCALENDAR_LOCATION 50
+
+typedef enum {
+ SMS_AT_PDU = 1,
+ SMS_AT_TXT
+} GSM_AT_SMS_Modes;
+
+typedef enum {
+ AT_Reply_OK = 1,
+ AT_Reply_Connect,
+ AT_Reply_Error,
+ AT_Reply_Unknown,
+ AT_Reply_CMSError,
+ AT_Reply_CMEError,
+ AT_Reply_SMSEdit
+} GSM_AT_Reply_State;
+
+typedef enum {
+ AT_Nokia = 1,
+ AT_Alcatel,
+ AT_Siemens,
+ AT_HP,
+ AT_Falcom,
+ AT_Ericsson,
+ AT_Sagem,
+ AT_Unknown
+} GSM_AT_Manufacturer;
+
+typedef enum {
+ AT_PBK_HEX = 1,
+ AT_PBK_GSM,
+ AT_PBK_UCS2
+} GSM_AT_PBK_Charset;
+
+typedef enum {
+ AT_AVAILABLE = 1,
+ AT_NOTAVAILABLE
+} GSM_AT_SMSMemory;
+
+typedef enum {
+ AT_SBNR_AVAILABLE = 1,
+ AT_SBNR_NOTAVAILABLE
+} GSM_AT_SBNR;
+
+typedef enum {
+ AT_Status,
+ AT_NextEmpty,
+ AT_Total,
+ AT_First,
+ AT_Sizes
+} GSM_AT_NeededMemoryInfo;
+
+#define AT_PBK_MAX_MEMORIES 200
+
+typedef struct {
+ GSM_AT_Manufacturer Manufacturer; /* Who is manufacturer */
+ GSM_Lines Lines; /* Allow to simply get each line in response */
+ GSM_AT_Reply_State ReplyState; /* What response type - error, OK, etc. */
+ int ErrorCode; /* Error codes from responses */
+ char *ErrorText; /* Error description */
+
+ GSM_MemoryType PBKMemory; /* Last read PBK memory */
+ char PBKMemories[AT_PBK_MAX_MEMORIES + 1]; /* Supported by phone PBK memories */
+ int NextMemoryEntry; /* Next empty memory entry */
+ int FirstMemoryEntry; /* First memory entry to be read */
+ GSM_AT_PBK_Charset PBKCharset; /* Last read PBK charset */
+ bool UCS2CharsetFailed; /* Whether setting of UCS2 charset has already failed */
+ bool NonUCS2CharsetFailed; /* Whether setting of non-UCS2 charset has already failed */
+ GSM_AT_SBNR PBKSBNR;
+ int NumberLength;
+ int TextLength;
+ int MemorySize;
+
+ GSM_SMSMemoryStatus LastSMSStatus;
+ int LastSMSRead;
+ int FirstCalendarPos;
+ bool CanSaveSMS;
+ GSM_AT_SMSMemory PhoneSMSMemory; /* Is phone SMS memory available ? */
+ GSM_AT_SMSMemory SIMSMSMemory; /* Is SIM SMS memory available ? */
+ GSM_MemoryType SMSMemory; /* Last read SMS memory */
+ GSM_AT_SMS_Modes SMSMode; /* PDU or TEXT mode for SMS ? */
+
+ bool OBEX;
+ GSM_File file;
+} GSM_Phone_ATGENData;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/at/siemens.c b/gammu/emb/common/phone/at/siemens.c
new file mode 100644
index 0000000..ab7dd2c
--- a/dev/null
+++ b/gammu/emb/common/phone/at/siemens.c
@@ -0,0 +1,320 @@
+/* (c) 2002-2003 by Walek */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_ATGEN
+
+#include <string.h>
+#include <time.h>
+#include <ctype.h>
+
+#include "../../misc/coding/coding.h"
+#include "../../gsmcomon.h"
+#include "../../service/sms/gsmsms.h"
+#include "../pfunc.h"
+
+extern GSM_Error ATGEN_HandleCMSError(GSM_StateMachine *s);
+
+GSM_Error ATGEN_CMS35ReplySetFunction (GSM_Protocol_Message msg, GSM_StateMachine *s,char *function)
+{
+ if (s->Protocol.Data.AT.EditMode) {
+ s->Protocol.Data.AT.EditMode = false;
+ return ERR_NONE;
+ }
+ dbgprintf ("Written %s",function);
+ if (s->Phone.Data.Priv.ATGEN.ReplyState == AT_Reply_OK){
+ dbgprintf (" - OK\n");
+ return ERR_NONE;
+ } else {
+ dbgprintf (" - error\n");
+ return ERR_UNKNOWN;
+ }
+}
+
+GSM_Error GetSiemensFrame(GSM_Protocol_Message msg, GSM_StateMachine *s, char *templ,
+ unsigned char *buffer, int *len)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ int i=2, pos=0, length=0;
+ unsigned char buf[512];
+
+ if (strstr(GetLineString(msg.Buffer,Priv->Lines,2),"OK")) return ERR_EMPTY;
+ if (!strstr(GetLineString(msg.Buffer,Priv->Lines,2),templ)) return ERR_UNKNOWN;
+
+ while (1) {
+ if (Priv->Lines.numbers[i*2+1]==0) break;
+ if ((!strstr(GetLineString(msg.Buffer,Priv->Lines,i+1),templ)) &&
+ (strstr(GetLineString(msg.Buffer,Priv->Lines,i),templ))){
+ length = strlen(GetLineString(msg.Buffer,Priv->Lines,i+1));
+ DecodeHexBin(buf, GetLineString(msg.Buffer,Priv->Lines,i+1),length);
+ length = length/2;
+ memcpy (buffer+pos,buf,length);
+ pos+=length;
+ }
+ i++;
+ }
+ *len = pos;
+ return ERR_NONE;
+}
+
+GSM_Error SetSiemensFrame (GSM_StateMachine *s, unsigned char *buff, char *templ,
+ int Location, GSM_Phone_RequestID RequestID, int len)
+{
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+ GSM_Error error;
+ unsigned char req[20],req1[512],hexreq[2096];
+ int MaxFrame,CurrentFrame,size,sz,pos=0;
+
+ EncodeHexBin(hexreq,buff,len);
+ size = len * 2;
+ MaxFrame = size / 352;
+ if (size % 352) MaxFrame++;
+
+ for (CurrentFrame=0;CurrentFrame<MaxFrame;CurrentFrame++) {
+ pos=CurrentFrame*352;
+ if (pos+352 < size) sz = 352; else sz = size - pos;
+ sprintf(req, "AT^SBNW=\"%s\",%i,%i,%i\r",templ,Location,CurrentFrame+1,MaxFrame);
+ s->Protocol.Data.AT.EditMode = true;
+ error = GSM_WaitFor (s, req, strlen(req), 0x00, 3, RequestID);
+ s->Phone.Data.DispatchError=ERR_TIMEOUT;
+ s->Phone.Data.RequestID=RequestID;
+ if (error!=ERR_NONE) return error;
+ memcpy (req1,hexreq+pos,sz);
+ error = s->Protocol.Functions->WriteMessage(s, req1, sz, 0x00);
+ if (error!=ERR_NONE) return error;
+ error = s->Protocol.Functions->WriteMessage(s,"\x1A", 1, 0x00);
+ if (error!=ERR_NONE) return error;
+ error = GSM_WaitForOnce(s, NULL, 0x00, 0x00, 4);
+ if (error == ERR_TIMEOUT) return error;
+ }
+ return Phone->DispatchError;
+}
+
+GSM_Error ATGEN_CMS35ReplyGetBitmap(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char buffer[4096];
+ int length;
+ GSM_Error error;
+
+ error = GetSiemensFrame(msg,s,"bmp",buffer,&length);
+ if (error!=ERR_NONE) return error;
+ dbgprintf ("Operator logo received lenght=%i\n",length);
+ error = BMP2Bitmap (buffer,NULL,s->Phone.Data.Bitmap);
+ if (error==ERR_NONE) return error;
+ else return ERR_UNKNOWN;
+}
+
+GSM_Error ATGEN_CMS35ReplySetBitmap(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ATGEN_CMS35ReplySetFunction (msg, s, "Operator Logo");
+}
+
+GSM_Error ATGEN_GetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char req[32];
+
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+ if (Bitmap->Type!=GSM_OperatorLogo) return ERR_NOTSUPPORTED;
+ if (Bitmap->Location-1 < 0) Bitmap->Location++;
+ s->Phone.Data.Bitmap=Bitmap;
+ sprintf(req, "AT^SBNR=\"bmp\",%i\r", Bitmap->Location-1);
+ smprintf(s, "Getting Bitmap\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetBitmap);
+}
+
+GSM_Error ATGEN_SetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char buffer[4096];
+ int length;
+ GSM_Error error;
+
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+ if (Bitmap->Type!=GSM_OperatorLogo) return ERR_NOTSUPPORTED;
+
+ error = Bitmap2BMP (buffer,NULL,Bitmap);
+ if (error!=ERR_NONE) return error;
+ length = 0x100 * buffer[3] + buffer[2];
+ buffer[58]=0xff; buffer[59]=0xff; buffer[60]=0xff;
+ if (Bitmap->Location-1 < 0) Bitmap->Location++;
+ s->Phone.Data.Bitmap=Bitmap;
+ return SetSiemensFrame(s, buffer,"bmp",Bitmap->Location-1,
+ ID_SetBitmap,length);
+}
+
+GSM_Error ATGEN_CMS35ReplyGetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char buffer[32];
+ int length;
+ GSM_Error error;
+
+ error = GetSiemensFrame(msg,s,"mid",s->Phone.Data.Ringtone->NokiaBinary.Frame,&length);
+ if (error!=ERR_NONE) return error;
+ dbgprintf ("Midi ringtone received\n");
+
+ s->Phone.Data.Ringtone->Format = RING_MIDI;
+ s->Phone.Data.Ringtone->NokiaBinary.Length = length;
+ sprintf(buffer,"Individual");
+ EncodeUnicode (s->Phone.Data.Ringtone->Name,buffer,strlen(buffer));
+ return ERR_NONE;
+}
+
+GSM_Error ATGEN_GetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone)
+{
+ unsigned char req[32];
+
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.Ringtone=Ringtone;
+ sprintf(req, "AT^SBNR=\"mid\",%i\r", Ringtone->Location-1);
+ smprintf(s, "Getting RingTone\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetRingtone);
+}
+
+GSM_Error ATGEN_CMS35ReplySetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ATGEN_CMS35ReplySetFunction (msg, s, "Ringtone");
+}
+
+GSM_Error ATGEN_SetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength)
+{
+ GSM_Phone_Data *Phone = &s->Phone.Data;
+
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+
+ if (Ringtone->Location==255) Ringtone->Location=1;
+ if (Ringtone->Location-1 > 1) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.Ringtone = Ringtone;
+ Phone->Ringtone = Ringtone;
+ return SetSiemensFrame(s, Ringtone->NokiaBinary.Frame,"mid",Ringtone->Location-1,
+ ID_SetRingtone,Ringtone->NokiaBinary.Length);
+}
+
+GSM_Error ATGEN_CMS35ReplyGetNextCal(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ GSM_CalendarEntry *Calendar = Data->Cal;
+ GSM_ToDoEntry ToDo;
+ GSM_Error error;
+ unsigned char buffer[354];
+ int len, pos=0;
+
+ if (Data->Priv.ATGEN.ReplyState != AT_Reply_OK) return ERR_UNKNOWN;
+
+ error = GetSiemensFrame(msg,s,"vcs",buffer,&len);
+ if (error!=ERR_NONE) return error;
+ error=GSM_DecodeVCALENDAR_VTODO(buffer,&pos,Calendar,&ToDo,Siemens_VCalendar,0);
+
+ return error;
+}
+
+GSM_Error SIEMENS_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ unsigned char req[32];
+ int Location;
+
+ if (Priv->Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+
+ if (start) Note->Location=Priv->FirstCalendarPos;
+ s->Phone.Data.Cal = Note;
+ Note->EntriesNum = 0;
+ smprintf(s, "Getting VCALENDAR\n");
+ Location = Note->Location;
+ while (1){
+ Location++;
+ sprintf(req, "AT^SBNR=\"vcs\",%i\r",Location);
+ error = GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_GetCalendarNote);
+ if ((error!=ERR_NONE) && (error!=ERR_EMPTY)) return ERR_INVALIDLOCATION;
+ Note->Location = Location;
+ Priv->FirstCalendarPos = Location;
+ if (Location > MAX_VCALENDAR_LOCATION) return ERR_EMPTY;
+ if (error==ERR_NONE) return error;
+ }
+ return error;
+}
+
+GSM_Error ATGEN_CMS35ReplySetCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ATGEN_CMS35ReplySetFunction (msg, s, "Calendar Note");
+}
+
+GSM_Error ATGEN_CMS35ReplyDeleteCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ if (Data->Cal->Location > MAX_VCALENDAR_LOCATION) return ERR_UNKNOWN;
+
+ if (Data->Priv.ATGEN.ReplyState== AT_Reply_OK) {
+ smprintf(s, "Calendar note deleted\n");
+ return ERR_NONE;
+ } else {
+ smprintf(s, "Can't delete calendar note\n");
+ return ERR_UNKNOWN;
+ }
+}
+
+GSM_Error SIEMENS_DelCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ unsigned char req[32];
+
+ if (s->Phone.Data.Priv.ATGEN.Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+ s->Phone.Data.Cal = Note;
+ sprintf(req, "AT^SBNW=\"vcs\",%i,0\r",Note->Location);
+ smprintf(s, "Deleting calendar note\n");
+ return GSM_WaitFor (s, req, strlen(req), 0x00, 4, ID_DeleteCalendarNote);
+}
+
+GSM_Error SIEMENS_AddCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+ unsigned char req[500];
+ int size=0;
+
+ if (Priv->Manufacturer!=AT_Siemens) return ERR_NOTSUPPORTED;
+// if (Note->Location==0x00) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.Cal = Note;
+ error=GSM_EncodeVCALENDAR(req,&size,Note,true,Siemens_VCalendar);
+
+ return SetSiemensFrame (s,req,"vcs",Note->Location,ID_SetCalendarNote,size);
+}
+
+/* (c) by Timo Teras */
+GSM_Error ATGEN_SL45ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+#ifndef ENABLE_LGPL
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_MemoryEntry *Memory = s->Phone.Data.Memory;
+ unsigned char buffer[500],buffer2[500];
+
+ switch (Priv->ReplyState) {
+ case AT_Reply_OK:
+ smprintf(s, "Phonebook entry received\n");
+ CopyLineString(buffer, msg.Buffer, Priv->Lines, 3);
+ DecodeHexBin(buffer2,buffer,strlen(buffer));
+ Memory->EntriesNum = 0;
+ DecodeVCARD21Text(buffer2, Memory);
+ if (Memory->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+ case AT_Reply_Error:
+ smprintf(s, "Error - too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ case AT_Reply_CMSError:
+ return ATGEN_HandleCMSError(s);
+ default:
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+#else
+ return ERR_NOTIMPLEMENTED;
+#endif
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/at/sonyeric.c b/gammu/emb/common/phone/at/sonyeric.c
new file mode 100644
index 0000000..4b2670a
--- a/dev/null
+++ b/gammu/emb/common/phone/at/sonyeric.c
@@ -0,0 +1,411 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_ATGEN
+
+#include <string.h>
+#include <time.h>
+#include <ctype.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+
+extern GSM_Reply_Function ATGENReplyFunctions[];
+extern GSM_Error ATGEN_DispatchMessage (GSM_StateMachine *s);
+
+#ifdef GSM_ENABLE_OBEXGEN
+
+extern GSM_Reply_Function OBEXGENReplyFunctions[];
+extern GSM_Error OBEXGEN_GetFilePart (GSM_StateMachine *s, GSM_File *File);
+extern GSM_Error OBEXGEN_AddFilePart (GSM_StateMachine *s, GSM_File *File, int *Pos);
+extern GSM_Error OBEXGEN_Disconnect (GSM_StateMachine *s);
+
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+
+static GSM_Error SONYERIC_SetOBEXMode(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+
+ if (Priv->OBEX) return ERR_NONE;
+
+ dbgprintf ("Changing to OBEX\n");
+
+ error=GSM_WaitFor (s, "AT*EOBEX\r", 9, 0x00, 4, ID_SetOBEX);
+ if (error != ERR_NONE) return error;
+
+ error = s->Protocol.Functions->Terminate(s);
+ if (error != ERR_NONE) return error;
+
+ s->Protocol.Functions = &OBEXProtocol;
+ error = s->Protocol.Functions->Initialise(s);
+ if (error != ERR_NONE) {
+ s->Protocol.Functions = &ATProtocol;
+ return error;
+ }
+ strcpy(s->CurrentConfig->Model,"seobex");
+ s->Phone.Data.Priv.OBEXGEN.Service = 0;
+ s->Phone.Functions->DispatchMessage = GSM_DispatchMessage;
+ s->Phone.Functions->ReplyFunctions = OBEXGENReplyFunctions;
+ Priv->OBEX = true;
+ return ERR_NONE;
+}
+
+static GSM_Error SONYERIC_SetATMode(GSM_StateMachine *s)
+{
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ GSM_Error error;
+
+ if (!Priv->OBEX) return ERR_NONE;
+
+ dbgprintf ("Changing to AT\n");
+
+ error = OBEXGEN_Disconnect(s);
+ if (error != ERR_NONE) return error;
+
+ error = s->Protocol.Functions->Terminate(s);
+ if (error != ERR_NONE) return error;
+
+ s->Protocol.Functions = &ATProtocol;
+ error = s->Protocol.Functions->Initialise(s);
+ if (error != ERR_NONE) {
+ s->Protocol.Functions = &OBEXProtocol;
+ return error;
+ }
+ strcpy(s->CurrentConfig->Model,"at");
+ s->Phone.Functions->DispatchMessage = ATGEN_DispatchMessage;
+ s->Phone.Functions->ReplyFunctions = ATGENReplyFunctions;
+ Priv->OBEX = false;
+ return ERR_NONE;
+}
+
+static GSM_Error SONYERIC_GetFile(GSM_StateMachine *s, GSM_File *File, unsigned char *FileName)
+{
+ GSM_Error error;
+
+ strcpy(File->ID_FullName,FileName);
+ File->Used = 0;
+ if (File->Buffer != NULL) free(File->Buffer);
+ File->Buffer = NULL;
+
+ error = SONYERIC_SetOBEXMode(s);
+ if (error != ERR_NONE) return error;
+
+ error = ERR_NONE;
+ while (error == ERR_NONE) error = OBEXGEN_GetFilePart(s,File);
+ if (error != ERR_EMPTY) return error;
+
+ return SONYERIC_SetATMode(s);
+}
+
+static GSM_Error SONYERIC_SetFile(GSM_StateMachine *s, unsigned char *FileName, unsigned char *Buffer, int Length)
+{
+ GSM_Error error;
+ GSM_File File;
+ int Pos = 0;
+
+ error = SONYERIC_SetOBEXMode(s);
+ if (error != ERR_NONE) return error;
+
+ strcpy(File.ID_FullName,FileName);
+ EncodeUnicode(File.Name,FileName,strlen(FileName));
+ File.Used = Length;
+ File.Buffer = malloc(Length);
+ memcpy(File.Buffer,Buffer,Length);
+
+ error = ERR_NONE;
+ while (error == ERR_NONE) error = OBEXGEN_AddFilePart(s,&File,&Pos);
+ free(File.Buffer);
+ if (error != ERR_EMPTY) return error;
+
+ return SONYERIC_SetATMode(s);
+}
+
+#endif
+
+GSM_Error SONYERIC_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ GSM_ToDoEntry ToDo;
+ int Pos, num, Loc;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (start) {
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ Note->Location = 1;
+ } else {
+ Note->Location++;
+ }
+ smprintf(s, "Getting calendar note %i\n",Note->Location);
+
+ Loc = Note->Location;
+ Pos = 0;
+ num = 0;
+ while (1) {
+ error = GSM_DecodeVCALENDAR_VTODO(Priv->file.Buffer, &Pos, Note, &ToDo, SonyEricsson_VCalendar, SonyEricsson_VToDo);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (Note->EntriesNum != 0) {
+ num++;
+ if (num == Loc) return ERR_NONE;
+ }
+ }
+ return ERR_EMPTY;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_GetNextToDo(GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool start)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ GSM_CalendarEntry Calendar;
+ int Pos, num, Loc;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer!=AT_Ericsson) return ERR_NOTSUPPORTED;
+
+ if (start) {
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ ToDo->Location = 1;
+ } else {
+ ToDo->Location++;
+ }
+ smprintf(s,"Getting ToDo %i\n",ToDo->Location);
+
+ Loc = ToDo->Location;
+ Pos = 0;
+ num = 0;
+ while (1) {
+ error = GSM_DecodeVCALENDAR_VTODO(Priv->file.Buffer, &Pos, &Calendar, ToDo, SonyEricsson_VCalendar, SonyEricsson_VToDo);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (ToDo->EntriesNum != 0) {
+ num++;
+ if (num == Loc) return ERR_NONE;
+ }
+ }
+
+ return ERR_EMPTY;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_GetToDoStatus(GSM_StateMachine *s, GSM_ToDoStatus *status)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ GSM_ToDoEntry ToDo;
+ GSM_CalendarEntry Calendar;
+ int Pos;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer!=AT_Ericsson) return ERR_NOTSUPPORTED;
+
+ smprintf(s,"Getting ToDo status\n");
+
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ status->Used = 0;
+ Pos = 0;
+ while (1) {
+ error = GSM_DecodeVCALENDAR_VTODO(Priv->file.Buffer, &Pos, &Calendar, &ToDo, SonyEricsson_VCalendar, SonyEricsson_VToDo);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (ToDo.EntriesNum != 0) status->Used++;
+ }
+
+ return ERR_NONE;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_AddCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ unsigned char req[5000];
+ int size=0;
+
+ smprintf(s,"Adding calendar note\n");
+
+ GSM_EncodeVCALENDAR(req,&size,Note,true,SonyEricsson_VCalendar);
+
+ return SONYERIC_SetFile(s, "telecom/cal/luid/.vcs", req, size);
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_AddToDo(GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ unsigned char req[5000];
+ int size=0;
+
+ if (Priv->Manufacturer!=AT_Ericsson) return ERR_NOTSUPPORTED;
+
+ smprintf(s,"Adding ToDo\n");
+
+ GSM_EncodeVTODO(req,&size,ToDo,true,SonyEricsson_VToDo);
+
+ return SONYERIC_SetFile(s, "telecom/cal/luid/.vcs", req, size);
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_DeleteAllToDo(GSM_StateMachine *s)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ int Pos,Level = 0,Used;
+ unsigned char *Buf;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ unsigned char Line[2000];
+
+ if (Priv->Manufacturer!=AT_Ericsson) return ERR_NOTSUPPORTED;
+
+ smprintf(s,"Deleting all ToDo\n");
+
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ Buf = NULL;
+ Used = 0;
+ while (1) {
+ MyGetLine(Priv->file.Buffer, &Pos, Line, Priv->file.Used);
+ if (strlen(Line) == 0) break;
+ dbgprintf("Line is %s,%i,%i\n",Line,Priv->file.Used,Pos);
+ switch (Level) {
+ case 0:
+ if (strstr(Line,"BEGIN:VTODO")) {
+ Level = 2;
+ break;
+ }
+ Buf=(unsigned char *)realloc(Buf,Used+strlen(Line)+3);
+ strcpy(Buf+Used,Line);
+ Used=Used+strlen(Line)+3;
+ Buf[Used-3] = 13;
+ Buf[Used-2] = 10;
+ Buf[Used-1] = 0x00;
+ break;
+ case 2: /* ToDo note */
+ if (strstr(Line,"END:VTODO")) {
+ Level = 0;
+ }
+ break;
+ }
+ }
+
+ error = SONYERIC_SetFile(s, "telecom/cal.vcs", Buf, Used);
+// if (Buf != NULL) free(Buf);
+ return error;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_DelCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ int Pos,Level = 0,Loc=0,Used;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+ unsigned char Line[2000];
+ unsigned char *Buf;
+
+ smprintf(s, "Deleting calendar note %i\n",Note->Location);
+
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ Buf = NULL;
+ Used = 0;
+ while (1) {
+ MyGetLine(Priv->file.Buffer, &Pos, Line, Priv->file.Used);
+ if (strlen(Line) == 0) break;
+ dbgprintf("Line is %s,%i,%i\n",Line,Priv->file.Used,Pos);
+ switch (Level) {
+ case 0:
+ if (strstr(Line,"BEGIN:VEVENT")) {
+ Loc++;
+ if (Loc == Note->Location) {
+ Level = 1;
+ break;
+ }
+ }
+ Buf=(unsigned char *)realloc(Buf,Used+strlen(Line)+3);
+ strcpy(Buf+Used,Line);
+ Used=Used+strlen(Line)+3;
+ Buf[Used-3] = 13;
+ Buf[Used-2] = 10;
+ Buf[Used-1] = 0x00;
+ break;
+ case 1: /* Calendar note */
+ if (strstr(Line,"END:VEVENT")) {
+ Level = 0;
+ }
+ break;
+ }
+ }
+
+ DumpMessage(s->di.df, s->di.dl, Buf, Used);
+
+ error = SONYERIC_SetFile(s, "telecom/cal.vcs", Buf, Used);
+ if (Buf != NULL) free(Buf);
+ return error;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error SONYERIC_GetCalendarStatus(GSM_StateMachine *s, GSM_CalendarStatus *Status)
+{
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+ GSM_Error error;
+ GSM_ToDoEntry ToDo;
+ GSM_CalendarEntry Calendar;
+ int Pos;
+ GSM_Phone_ATGENData *Priv = &s->Phone.Data.Priv.ATGEN;
+
+ if (Priv->Manufacturer!=AT_Ericsson) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Getting calendar status\n");
+
+ error = SONYERIC_GetFile(s, &Priv->file, "telecom/cal.vcs");
+ if (error != ERR_NONE) return error;
+
+ Status->Used = 0;
+ Pos = 0;
+ while (1) {
+ error = GSM_DecodeVCALENDAR_VTODO(Priv->file.Buffer, &Pos, &Calendar, &ToDo, SonyEricsson_VCalendar, SonyEricsson_VToDo);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (Calendar.EntriesNum != 0) Status->Used++;
+ }
+
+ return ERR_NONE;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/dct3comm.h b/gammu/emb/common/phone/nokia/dct3/dct3comm.h
new file mode 100644
index 0000000..0b3e42e
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/dct3comm.h
@@ -0,0 +1,16 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef phone_nokia_dct3common_h
+#define phone_nokia_dct3common_h
+
+typedef struct {
+ int Locations[4];
+ int CurrentLocation;
+ int ID;
+} DCT3_WAPSettings_Locations;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/dct3func.c b/gammu/emb/common/phone/nokia/dct3/dct3func.c
new file mode 100644
index 0000000..beef33c
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/dct3func.c
@@ -0,0 +1,1535 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Markus Plail, Pavel Janik, others and Gnokii */
+/* resetting DCT4 phones settings (c) by Walek */
+
+#include <string.h> /* memcpy only */
+#include <stdio.h>
+#include <ctype.h>
+
+#include "../../../gsmstate.h"
+#include "../../../misc/coding/coding.h"
+#include "../../../service/sms/gsmsms.h"
+#include "../../pfunc.h"
+#include "../nfunc.h"
+#include "dct3func.h"
+
+#ifdef GSM_ENABLE_NOKIA_DCT3
+
+GSM_Error DCT3_DeleteWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+
+ /* We have to enable WAP frames in phone */
+ error=DCT3DCT4_EnableWAPFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ return DCT3DCT4_DeleteWAPBookmarkPart(s,bookmark);
+}
+
+GSM_Error DCT3_GetWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+
+ /* We have to enable WAP frames in phone */
+ error=DCT3DCT4_EnableWAPFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ return DCT3DCT4_GetWAPBookmarkPart(s,bookmark);
+}
+
+GSM_Error DCT3_ReplyPressKey(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[2]) {
+ case 0x46:
+ smprintf(s, "Pressing key OK\n");
+ if (Data->PressKey) return ERR_NONE;
+ break;
+ case 0x47:
+ smprintf(s, "Releasing key OK\n");
+ if (!Data->PressKey) return ERR_NONE;
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3_PressKey(GSM_StateMachine *s, GSM_KeyCode Key, bool Press)
+{
+ unsigned char PressReq[] = {0x00, 0x01, 0x46, 0x00, 0x01,
+ 0x0a}; /* Key code */
+ unsigned char ReleaseReq[] = {0x00, 0x01, 0x47, 0x00, 0x01, 0x0c};
+
+ if (Press) {
+ PressReq[5] = Key;
+ s->Phone.Data.PressKey = true;
+ smprintf(s, "Pressing key\n");
+ return GSM_WaitFor (s, PressReq, 6, 0xd1, 4, ID_PressKey);
+ } else {
+ s->Phone.Data.PressKey = false;
+ smprintf(s, "Releasing key\n");
+ return GSM_WaitFor (s, ReleaseReq, 6, 0xd1, 4, ID_PressKey);
+ }
+}
+
+GSM_Error DCT3_ReplyPlayTone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Tone played\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_PlayTone(GSM_StateMachine *s, int Herz, unsigned char Volume, bool start)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00,0x01,0x8f,
+ 0x00, /* Volume */
+ 0x00, /* HerzLo */
+ 0x00}; /* HerzHi */
+
+ if (start) {
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+ }
+
+ /* For Herz==255*255 we have silent */
+ if (Herz!=255*255) {
+ req[3]=Volume;
+ req[5]=Herz%256;
+ req[4]=Herz/256;
+ } else {
+ req[3]=0;
+ req[5]=0;
+ req[4]=0;
+ }
+
+ return GSM_WaitFor (s, req, 6, 0x40, 4, ID_PlayTone);
+}
+
+#ifdef GSM_ENABLE_CELLBROADCAST
+
+GSM_Error DCT3_ReplyIncomingCB(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_CBMessage CB;
+ int i;
+ char Buffer[300];
+
+ smprintf(s, "CB received\n");
+ CB.Channel = msg.Buffer[7];
+ i = GSM_UnpackEightBitsToSeven(0, msg.Buffer[9], msg.Buffer[9], msg.Buffer+10, Buffer);
+ i = msg.Buffer[9] - 1;
+ while (i!=0) {
+ if (Buffer[i] == 13) i = i - 1; else break;
+ }
+ DecodeDefault(CB.Text, Buffer, i + 1, false, NULL);
+ smprintf(s, "Channel %i, text \"%s\"\n",CB.Channel,DecodeUnicodeString(CB.Text));
+ if (s->Phone.Data.EnableIncomingCB && s->User.IncomingCB!=NULL) {
+ s->User.IncomingCB(s->CurrentConfig->Device,CB);
+ }
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_ReplySetIncomingCB(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x21:
+ smprintf(s, "CB set\n");
+ return ERR_NONE;
+ case 0x22:
+ smprintf(s, "CB not set\n");
+ return ERR_UNKNOWN;
+ case 0xCA:
+ smprintf(s, "No network and no CB\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+#endif
+
+GSM_Error DCT3_SetIncomingCB(GSM_StateMachine *s, bool enable)
+{
+#ifdef GSM_ENABLE_CELLBROADCAST
+ unsigned char reqOn[] = {N6110_FRAME_HEADER, 0x20, 0x01,
+ 0x01, 0x00, 0x00, 0x01, 0x01};
+ unsigned char reqOff[] = {N6110_FRAME_HEADER, 0x20, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ if (s->Phone.Data.EnableIncomingCB!=enable) {
+ s->Phone.Data.EnableIncomingCB = enable;
+ if (enable) {
+ smprintf(s, "Enabling incoming CB\n");
+ return GSM_WaitFor(s, reqOn, 10, 0x02, 4, ID_SetIncomingCB);
+ } else {
+ smprintf(s, "Disabling incoming CB\n");
+ return GSM_WaitFor(s, reqOff, 10, 0x02, 4, ID_SetIncomingCB);
+ }
+ }
+ return ERR_NONE;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+GSM_Error DCT3_ReplySetSMSC(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "SMSC set\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_SetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ unsigned char req[100] = {N6110_FRAME_HEADER, 0x30, 0x64};
+
+ memset(req+6,100-6,0);
+
+ /* SMSC location */
+ req[5] = smsc->Location;
+
+ /* SMSC format */
+ switch (smsc->Format) {
+ case SMS_FORMAT_Text : req[7] = 0x00; break;
+ case SMS_FORMAT_Fax : req[7] = 0x22; break;
+ case SMS_FORMAT_Pager : req[7] = 0x26; break;
+ case SMS_FORMAT_Email : req[7] = 0x32; break;
+ }
+
+ /* SMS validity */
+ req[9] = smsc->Validity.Relative;
+
+ /* Default number for SMS messages */
+ req[10] = GSM_PackSemiOctetNumber(smsc->DefaultNumber, req+11, true);
+
+ /* SMSC number */
+ req[22] = GSM_PackSemiOctetNumber(smsc->Number, req+23, false);
+
+ /* SMSC name */
+ memcpy(req + 34, DecodeUnicodeString(smsc->Name),UnicodeLength(smsc->Name));
+
+ smprintf(s, "Setting SMSC\n");
+ return GSM_WaitFor (s, req, 35+UnicodeLength(smsc->Name), 0x02, 4, ID_SetSMSC);
+}
+
+GSM_Error DCT3_ReplyEnableSecurity(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "State of security commands set\n");
+ return ERR_NONE;
+}
+
+/* If you set make some things (for example, change Security Code from
+ * phone's menu, disable and enable phone), it won't answer for 0x40 frames
+ * and you won't be able to play tones, get netmonitor, etc.
+ * This function do thing called "Enabling extended security commands"
+ * and it enables 0x40 frame functions.
+ * This frame can also some other things - see below
+ */
+GSM_Error DCT3_EnableSecurity (GSM_StateMachine *s, unsigned char status)
+{
+ unsigned char req[] = {0x00, 0x01, 0x64,
+ 0x01}; /* 0x00/0x01 - off/on,
+ * 0x03/0x04 - soft/hard reset,
+ * 0x06 - CONTACT SERVICE
+ */
+
+ /* 0x06 MAKES CONTACT SERVICE! BE CAREFULL! */
+ /* When use 0x03 and had during session changed time & date
+ * some phones (like 6150 or 6210) can ask for time & date after reset
+ * or disable clock on the screen
+ */
+ if (status!=0x06) req[3] = status;
+ smprintf(s, "Setting state of security commands\n");
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_EnableSecurity);
+}
+
+GSM_Error DCT3_ReplyGetIMEI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ memcpy(s->Phone.Data.IMEI,msg.Buffer + 4, 16);
+ smprintf(s, "Received IMEI %s\n",s->Phone.Data.IMEI);
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_GetIMEI (GSM_StateMachine *s)
+{
+ unsigned char req[] = {0x00, 0x01, 0x66, 0x00};
+ GSM_Error error;
+
+ if (strlen(s->Phone.Data.IMEI)!=0) return ERR_NONE;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+
+ smprintf(s, "Getting IMEI\n");
+ return GSM_WaitFor (s, req, 4, 0x40, 2, ID_GetIMEI);
+}
+
+GSM_Error DCT3_ReplySIMLogin(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Login for SIM card\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_ReplySIMLogout(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Logout for SIM card\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_ReplyGetDateTime(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Date & time received\n");
+ if (msg.Buffer[4]==0x01) {
+ NOKIA_DecodeDateTime(s, msg.Buffer+8, s->Phone.Data.DateTime);
+ return ERR_NONE;
+ }
+ smprintf(s, "Not set in phone\n");
+ return ERR_EMPTY;
+}
+
+GSM_Error DCT3_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time, unsigned char msgtype)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x62};
+
+ s->Phone.Data.DateTime=date_time;
+ smprintf(s, "Getting date & time\n");
+ return GSM_WaitFor (s, req, 4, msgtype, 4, ID_GetDateTime);
+}
+
+GSM_Error DCT3_ReplyGetAlarm(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Alarm: ");
+ if (msg.Buffer[8]==0x02) {
+ smprintf(s, "set to %02i:%02i\n", msg.Buffer[9], msg.Buffer[10]);
+ Data->Alarm->Repeating = true;
+ Data->Alarm->Text[0] = 0;
+ Data->Alarm->Text[1] = 0;
+ Data->Alarm->DateTime.Hour = msg.Buffer[9];
+ Data->Alarm->DateTime.Minute = msg.Buffer[10];
+ Data->Alarm->DateTime.Second = 0;
+ return ERR_NONE;
+ }
+ smprintf(s, "not set\n");
+ return ERR_EMPTY;
+}
+
+GSM_Error DCT3_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm, unsigned char msgtype)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x6d};
+
+ if (alarm->Location!=1) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.Alarm=alarm;
+ smprintf(s, "Getting alarm\n");
+ return GSM_WaitFor (s, req, 4, msgtype, 4, ID_GetAlarm);
+}
+
+GSM_Error DCT3_ReplySetDateTime(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Date & time: ");
+ if (msg.Buffer[4]==0x01) {
+ smprintf(s, "set OK\n");
+ return ERR_NONE;
+ }
+ smprintf(s, "error setting\n");
+ return ERR_UNKNOWN;
+}
+
+GSM_Error DCT3_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time, unsigned char msgtype)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x60, 0x01, 0x01, 0x07,
+ 0x00, 0x00, /* Year */
+ 0x00, /* Month */
+ 0x00, /* Day */
+ 0x00, /* Hour */
+ 0x00, /* Minute */
+ 0x00}; /* Unknown. Not seconds */
+
+ NOKIA_EncodeDateTime(s, req+7, date_time);
+ smprintf(s, "Setting date & time\n");
+ return GSM_WaitFor (s, req, 14, msgtype, 4, ID_SetDateTime);
+}
+
+GSM_Error DCT3_ReplySetAlarm(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Alarm: ");
+ if (msg.Buffer[4]==0x01) {
+ smprintf(s, "set OK\n");
+ return ERR_NONE;
+ }
+ smprintf(s, "error setting\n");
+ return ERR_UNKNOWN;
+}
+
+GSM_Error DCT3_SetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm, unsigned char msgtype)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x6b, 0x01, 0x20, 0x03,
+ 0x02, /* Unknown. Not for enabling/disabling */
+ 0x00, /* Hour */
+ 0x00, /* Minute */
+ 0x00}; /* Unknown. Not seconds */
+
+ if (alarm->Location != 1) return ERR_NOTSUPPORTED;
+
+ req[8] = alarm->DateTime.Hour;
+ req[9] = alarm->DateTime.Minute;
+
+ smprintf(s, "Setting alarm\n");
+ return GSM_WaitFor (s, req, 11, msgtype, 4, ID_SetAlarm);
+}
+
+GSM_Error DCT3_ReplyGetSMSC(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x34:
+ smprintf(s, "SMSC received\n");
+ Data->SMSC->Format = SMS_FORMAT_Text;
+ switch (msg.Buffer[6]) {
+ case 0x00: Data->SMSC->Format = SMS_FORMAT_Text; break;
+ case 0x22: Data->SMSC->Format = SMS_FORMAT_Fax; break;
+ case 0x26: Data->SMSC->Format = SMS_FORMAT_Pager; break;
+ case 0x32: Data->SMSC->Format = SMS_FORMAT_Email; break;
+ }
+ Data->SMSC->Validity.Format = SMS_Validity_RelativeFormat;
+ Data->SMSC->Validity.Relative = msg.Buffer[8];
+
+ i=33;
+ while (msg.Buffer[i]!=0) {i++;}
+ i=i-33;
+ if (i>GSM_MAX_SMSC_NAME_LENGTH) {
+ smprintf(s, "Too long name\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ EncodeUnicode(Data->SMSC->Name,msg.Buffer+33,i);
+ smprintf(s, "Name \"%s\"\n", DecodeUnicodeString(Data->SMSC->Name));
+
+ GSM_UnpackSemiOctetNumber(Data->SMSC->DefaultNumber,msg.Buffer+9,true);
+ smprintf(s, "Default number \"%s\"\n", DecodeUnicodeString(Data->SMSC->DefaultNumber));
+
+ GSM_UnpackSemiOctetNumber(Data->SMSC->Number,msg.Buffer+21,false);
+ smprintf(s, "Number \"%s\"\n", DecodeUnicodeString(Data->SMSC->Number));
+
+ return ERR_NONE;
+ case 0x35:
+ smprintf(s, "Getting SMSC failed\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3_GetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x33, 0x64,
+ 0x00}; /* Location */
+
+ if (smsc->Location==0x00) return ERR_INVALIDLOCATION;
+
+ req[5]=smsc->Location;
+
+ s->Phone.Data.SMSC=smsc;
+ smprintf(s, "Getting SMSC\n");
+ return GSM_WaitFor (s, req, 6, 0x02, 4, ID_GetSMSC);
+}
+
+GSM_Error DCT3_ReplyGetNetworkInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int count;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+#ifdef DEBUG
+ GSM_NetworkInfo NetInfo;
+ char name[100];
+
+ smprintf(s, "Network info received\n");
+ smprintf(s, " Status : ");
+ switch (msg.Buffer[8]) {
+ case 0x01: smprintf(s, "home network"); break;
+ case 0x02: smprintf(s, "roaming network"); break;
+ case 0x03: smprintf(s, "requesting network"); break;
+ case 0x04: smprintf(s, "not registered in the network"); break;
+ default : smprintf(s, "unknown");
+ }
+ smprintf(s, "\n");
+ smprintf(s, "Network selection : %s\n", msg.Buffer[9]==1?"manual":"automatic");
+ if (msg.Buffer[8]<0x03) {
+ sprintf(NetInfo.CID, "%02x%02x", msg.Buffer[10], msg.Buffer[11]);
+ smprintf(s, "CID : %s\n", NetInfo.CID);
+
+ sprintf(NetInfo.LAC, "%02x%02x", msg.Buffer[12], msg.Buffer[13]);
+ smprintf(s, "LAC : %s\n", NetInfo.LAC);
+
+ smprintf(s, "Network code : %s\n", NetInfo.NetworkCode);
+ NOKIA_DecodeNetworkCode(msg.Buffer+14,NetInfo.NetworkCode);
+ smprintf(s, "Network name for Gammu : %s ",
+ DecodeUnicodeString(GSM_GetNetworkName(NetInfo.NetworkCode)));
+ smprintf(s, "(%s)\n",DecodeUnicodeString(GSM_GetCountryName(NetInfo.NetworkCode)));
+
+ if (msg.Length>18) {
+ if (msg.Buffer[18]==0x00) {
+ /* In 6210 name is in "normal" Unicode */
+ memcpy(name,msg.Buffer+18,msg.Buffer[17]*2);
+ name[msg.Buffer[17]*2] =0x00;
+ name[msg.Buffer[17]*2+1]=0x00;
+ smprintf(s, "Network name for phone : %s\n",DecodeUnicodeString(name));
+ } else {
+ /* In 9210 first 0x00 is cut from Unicode string */
+ name[0] = 0;
+ memcpy(name+1,msg.Buffer+18,msg.Buffer[17]*2);
+ name[msg.Buffer[17]*2+1]=0x00;
+ name[msg.Buffer[17]*2+2]=0x00;
+ smprintf(s, "Network name for phone : %s\n",DecodeUnicodeString(name));
+ }
+ }
+ }
+#endif
+ if (Data->RequestID==ID_GetNetworkInfo) {
+ Data->NetworkInfo->NetworkName[0] = 0x00;
+ Data->NetworkInfo->NetworkName[1] = 0x00;
+ Data->NetworkInfo->State = 0;
+ switch (msg.Buffer[8]) {
+ case 0x01: Data->NetworkInfo->State = GSM_HomeNetwork; break;
+ case 0x02: Data->NetworkInfo->State = GSM_RoamingNetwork; break;
+ case 0x03: Data->NetworkInfo->State = GSM_RequestingNetwork; break;
+ case 0x04: Data->NetworkInfo->State = GSM_NoNetwork; break;
+ }
+ if (Data->NetworkInfo->State == GSM_HomeNetwork || Data->NetworkInfo->State == GSM_RoamingNetwork) {
+ if (msg.Buffer[18]==0x00) {
+ /* In 6210 name is in "normal" Unicode */
+ memcpy(Data->NetworkInfo->NetworkName,msg.Buffer+18,msg.Buffer[17]*2);
+ Data->NetworkInfo->NetworkName[msg.Buffer[17]*2] = 0x00;
+ Data->NetworkInfo->NetworkName[msg.Buffer[17]*2+1] = 0x00;
+ } else {
+ /* In 9210 first 0x00 is cut from Unicode string */
+ Data->NetworkInfo->NetworkName[0] = 0;
+ memcpy(Data->NetworkInfo->NetworkName+1,msg.Buffer+18,msg.Buffer[17]*2);
+ Data->NetworkInfo->NetworkName[msg.Buffer[17]*2+1]=0x00;
+ Data->NetworkInfo->NetworkName[msg.Buffer[17]*2+2]=0x00;
+ }
+ NOKIA_DecodeNetworkCode(msg.Buffer+14,Data->NetworkInfo->NetworkCode);
+ sprintf(Data->NetworkInfo->CID, "%02x%02x", msg.Buffer[10], msg.Buffer[11]);
+ sprintf(Data->NetworkInfo->LAC, "%02x%02x", msg.Buffer[12], msg.Buffer[13]);
+ }
+ }
+ /* 6210/6250/7110 */
+ if (Data->RequestID==ID_GetBitmap) {
+ if (msg.Buffer[4]==0x02) {
+ smprintf(s, "Operator logo available\n");
+ count = 7;
+ /* skip network info */
+ count += msg.Buffer[count];
+ count ++;
+ Data->Bitmap->BitmapWidth = msg.Buffer[count++];
+ Data->Bitmap->BitmapHeight = msg.Buffer[count++];
+ count+=4;
+ PHONE_DecodeBitmap(GSM_Nokia7110OperatorLogo,msg.Buffer+count,Data->Bitmap);
+ NOKIA_DecodeNetworkCode(msg.Buffer+14,Data->Bitmap->NetworkCode);
+ } else {
+ Data->Bitmap->BitmapWidth = 78;
+ Data->Bitmap->BitmapHeight = 21;
+ GSM_ClearBitmap(Data->Bitmap);
+ strcpy(Data->Bitmap->NetworkCode,"000 00");
+ }
+ }
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_GetNetworkInfo(GSM_StateMachine *s, GSM_NetworkInfo *netinfo)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x70};
+
+ s->Phone.Data.NetworkInfo=netinfo;
+ smprintf(s, "Getting network info\n");
+ return GSM_WaitFor (s, req, 4, 0x0a, 4, ID_GetNetworkInfo);
+}
+
+GSM_Error DCT3_ReplyDialCommand(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Answer for call commands\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_DialVoice(GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber)
+{
+ unsigned int i = 0;
+ GSM_Error error;
+ unsigned char req[100] = {0x00, 0x01, 0x7c,
+ 0x01}; /* call command */
+
+ if (ShowNumber != GSM_CALL_DefaultNumberPresence) return ERR_NOTSUPPORTED;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+
+ for (i=0; i < strlen(number); i++) req[4+i]=number[i];
+ req[4+i+1]=0;
+
+ smprintf(s, "Making voice call\n");
+ return GSM_WaitFor (s, req, 4+strlen(number)+1, 0x40, 4, ID_DialVoice);
+}
+
+static GSM_Error DCT3_CancelAllCalls(GSM_StateMachine *s)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00, 0x01, 0x7c,
+ 0x03}; /* call command */
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+
+ smprintf(s, "Canceling calls\n");
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_CancelCall);
+}
+
+GSM_Error DCT3_CancelCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (!all) return DCT3DCT4_CancelCall(s,ID);
+ return DCT3_CancelAllCalls(s);
+}
+
+GSM_Error DCT3_AnswerAllCalls(GSM_StateMachine *s)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00, 0x01, 0x7c,
+ 0x02}; /* call command */
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+
+ smprintf(s, "Answering calls\n");
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_AnswerCall);
+}
+
+GSM_Error DCT3_Reset(GSM_StateMachine *s, bool hard)
+{
+ GSM_Error error;
+
+ if (hard) {
+ error=DCT3_EnableSecurity(s, 0x04);
+ } else {
+ error=DCT3_EnableSecurity(s, 0x03);
+ }
+ if (error == ERR_NONE) {
+ s->Phone.Data.EnableIncomingSMS = false;
+ s->Phone.Data.EnableIncomingCB = false;
+ }
+ return error;
+}
+
+GSM_Error DCT3_ReplyGetWAPBookmark(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return DCT3DCT4_ReplyGetWAPBookmark (msg,s,false);
+}
+
+GSM_Error DCT3_SetWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+ int count = 4, location;
+ unsigned char req[600] = {N6110_FRAME_HEADER, 0x09};
+
+ /* We have to enable WAP frames in phone */
+ error=DCT3DCT4_EnableWAPFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ location = bookmark->Location - 1;
+ if (bookmark->Location == 0) location = 0xffff;
+ req[count++] = (location & 0xff00) >> 8;
+ req[count++] = location & 0x00ff;
+
+ count += NOKIA_SetUnicodeString(s, req+count, bookmark->Title, false);
+ count += NOKIA_SetUnicodeString(s, req+count, bookmark->Address, false);
+
+ /* unknown */
+ req[count++] = 0x01; req[count++] = 0x80; req[count++] = 0x00;
+ req[count++] = 0x00; req[count++] = 0x00; req[count++] = 0x00;
+ req[count++] = 0x00; req[count++] = 0x00; req[count++] = 0x00;
+
+ smprintf(s, "Setting WAP bookmark\n");
+ error = GSM_WaitFor (s, req, count, 0x3f, 4, ID_SetWAPBookmark);
+ if (error != ERR_NONE) {
+ if (error == ERR_INSIDEPHONEMENU || error == ERR_EMPTY || error == ERR_FULL) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error DCT3_ReplyGetWAPSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int tmp,Number;
+// int tmp2;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+#ifdef GSM_ENABLE_NOKIA6110
+ GSM_Phone_N6110Data *Priv6110 = &s->Phone.Data.Priv.N6110;
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ GSM_Phone_N7110Data *Priv7110 = &s->Phone.Data.Priv.N7110;
+#endif
+
+ switch(msg.Buffer[3]) {
+ case 0x16:
+ smprintf(s, "WAP settings part 1 received OK\n");
+
+ tmp = 4;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].Title,false);
+ smprintf(s, "Title: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].Title));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].HomePage,false);
+ smprintf(s, "Homepage: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].HomePage));
+#ifdef DEBUG
+ smprintf(s, "Connection type: ");
+ switch (msg.Buffer[tmp]) {
+ case 0x00: smprintf(s, "temporary\n"); break;
+ case 0x01: smprintf(s, "continuous\n"); break;
+ default: smprintf(s, "unknown\n");
+ }
+ smprintf(s, "Connection security: ");
+ switch (msg.Buffer[tmp+13]) {
+ case 0x00: smprintf(s, "off\n"); break;
+ case 0x01: smprintf(s, "on\n"); break;
+ default: smprintf(s, "unknown\n");
+ }
+#endif
+ Data->WAPSettings->Settings[0].IsContinuous = false;
+ if (msg.Buffer[tmp] == 0x01) Data->WAPSettings->Settings[0].IsContinuous = true;
+ Data->WAPSettings->Settings[0].IsSecurity = false;
+ if (msg.Buffer[tmp+13] == 0x01) Data->WAPSettings->Settings[0].IsSecurity = true;
+
+ /* I'm not sure here. Experimental values from 6210 5.56 */
+// tmp2 = DecodeUnicodeLength(Data->WAPSettings->Settings[0].Title);
+// if (tmp2 != 0) tmp2 --;
+// tmp2 += tmp;
+ if (!(UnicodeLength(Data->WAPSettings->Settings[0].Title)) % 2) tmp++;
+ if (UnicodeLength(Data->WAPSettings->Settings[0].HomePage)!=0) tmp++;
+
+ smprintf(s, "ID for writing %i\n",msg.Buffer[tmp+5]);
+
+ smprintf(s, "Current set location in phone %i\n",msg.Buffer[tmp+6]);
+
+ smprintf(s, "1 location %i\n",msg.Buffer[tmp+8]);
+ smprintf(s, "2 location %i\n",msg.Buffer[tmp+9]);
+ smprintf(s, "3 location %i\n",msg.Buffer[tmp+10]);
+ smprintf(s, "4 location %i\n",msg.Buffer[tmp+11]);
+#ifdef GSM_ENABLE_NOKIA7110
+ if (strstr(N7110Phone.models, Data->ModelInfo->model) != NULL) {
+ Priv7110->WAPLocations.ID = msg.Buffer[tmp+5];
+ Priv7110->WAPLocations.CurrentLocation = msg.Buffer[tmp+6];
+ Priv7110->WAPLocations.Locations[0] = msg.Buffer[tmp+8];
+ Priv7110->WAPLocations.Locations[1] = msg.Buffer[tmp+9];
+ Priv7110->WAPLocations.Locations[2] = msg.Buffer[tmp+10];
+ Priv7110->WAPLocations.Locations[3] = msg.Buffer[tmp+11];
+
+// Priv7110->WAPLocations.CurrentLocation = msg.Buffer[tmp2+1];
+// Priv7110->WAPLocations.Locations[0] = msg.Buffer[tmp2+3];
+// Priv7110->WAPLocations.Locations[1] = msg.Buffer[tmp2+4];
+// Priv7110->WAPLocations.Locations[2] = msg.Buffer[tmp2+5];
+// Priv7110->WAPLocations.Locations[3] = msg.Buffer[tmp2+6];
+ }
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ if (strstr(N6110Phone.models, Data->ModelInfo->model) != NULL) {
+ Priv6110->WAPLocations.ID = msg.Buffer[tmp+5];
+ Priv6110->WAPLocations.CurrentLocation = msg.Buffer[tmp+6];
+ Priv6110->WAPLocations.Locations[0] = msg.Buffer[tmp+8];
+ Priv6110->WAPLocations.Locations[1] = msg.Buffer[tmp+9];
+ Priv6110->WAPLocations.Locations[2] = msg.Buffer[tmp+10];
+ Priv6110->WAPLocations.Locations[3] = msg.Buffer[tmp+11];
+ }
+#endif
+ return ERR_NONE;
+ case 0x17:
+ smprintf(s, "WAP settings part 1 receiving error\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside WAP settings menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x02:
+ smprintf(s, "Invalid or empty\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ break;
+ case 0x1c:
+ smprintf(s, "WAP settings part 2 received OK\n");
+ Number = Data->WAPSettings->Number;
+ switch (msg.Buffer[5]) {
+ case 0x00:
+ Data->WAPSettings->Settings[Number].Bearer = WAPSETTINGS_BEARER_SMS;
+ smprintf(s, "Settings for SMS bearer:\n");
+ tmp = 6;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].Service,false);
+ smprintf(s, "Service number: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Service));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].Server,false);
+ smprintf(s, "Server number: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Server));
+ break;
+ case 0x01:
+ Data->WAPSettings->Settings[Number].Bearer = WAPSETTINGS_BEARER_DATA;
+ smprintf(s, "Settings for data bearer:\n");
+ Data->WAPSettings->Settings[Number].ManualLogin = false;
+ tmp = 10;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].IPAddress,false);
+ smprintf(s, "IP address: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].IPAddress));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].DialUp,false);
+ smprintf(s, "Dial-up number: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].DialUp));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].User,false);
+ smprintf(s, "User name: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].User));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].Password,false);
+ smprintf(s, "Password: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Password));
+#ifdef DEBUG
+ smprintf(s, "Authentication type: ");
+ switch (msg.Buffer[6]) {
+ case 0x00: smprintf(s, "normal\n"); break;
+ case 0x01: smprintf(s, "secure\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Data call type: ");
+ switch (msg.Buffer[7]) {
+ case 0x00: smprintf(s, "analogue\n"); break;
+ case 0x01: smprintf(s, "ISDN\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Data call speed: ");
+ switch (msg.Buffer[9]) {
+ case 0x01: smprintf(s, "9600\n"); break;
+ case 0x02: smprintf(s, "14400\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+#endif
+ Data->WAPSettings->Settings[Number].IsNormalAuthentication=true;
+ if (msg.Buffer[6]==0x01) Data->WAPSettings->Settings[Number].IsNormalAuthentication=false;
+ Data->WAPSettings->Settings[Number].IsISDNCall=false;
+ if (msg.Buffer[7]==0x01) Data->WAPSettings->Settings[Number].IsISDNCall=true;
+ Data->WAPSettings->Settings[Number].Speed = WAPSETTINGS_SPEED_9600;
+ if (msg.Buffer[9]==0x02) Data->WAPSettings->Settings[Number].Speed = WAPSETTINGS_SPEED_14400;
+ break;
+ case 0x02:
+ Data->WAPSettings->Settings[Number].Bearer=WAPSETTINGS_BEARER_USSD;
+ smprintf(s, "Settings for USSD bearer:\n");
+ tmp = 7;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].Service,false);
+#ifdef DEBUG
+ if (msg.Buffer[6]==0x01)
+ smprintf(s, "Service number: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Service));
+ else
+ smprintf(s, "IP address: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Service));
+#endif
+ Data->WAPSettings->Settings[Number].IsIP=true;
+ if (msg.Buffer[6]==0x01) Data->WAPSettings->Settings[Number].IsIP=false;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[Number].Code,false);
+ smprintf(s, "Service code: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[Number].Code));
+ }
+ Data->WAPSettings->Number++;
+ return ERR_NONE;
+ case 0x1d:
+ smprintf(s, "Incorrect WAP settings location\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3_GetWAPSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+#ifdef GSM_ENABLE_NOKIA6110
+ GSM_Phone_N6110Data *Priv6110 = &s->Phone.Data.Priv.N6110;
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ GSM_Phone_N7110Data *Priv7110 = &s->Phone.Data.Priv.N7110;
+#endif
+ GSM_Error error;
+ int i;
+ unsigned char req[] = {N6110_FRAME_HEADER,0x15,
+ 0x00}; /* Location */
+ unsigned char req2[] = {N6110_FRAME_HEADER,0x1b,
+ 0x00}; /* Location */
+
+ /* We have to enable WAP frames in phone */
+ error=DCT3DCT4_EnableWAPFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ s->Phone.Data.WAPSettings = settings;
+ settings->Number = 0;
+ settings->ReadOnly = false;
+
+ req[4] = settings->Location-1;
+ smprintf(s, "Getting WAP settings part 1\n");
+ error = GSM_WaitFor (s, req, 5, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) return error;
+
+#ifdef GSM_ENABLE_NOKIA7110
+ if (strstr(N7110Phone.models, s->Phone.Data.ModelInfo->model) != NULL) {
+ for (i=0;i<4;i++) {
+ req2[4] = Priv7110->WAPLocations.Locations[i];
+ smprintf(s, "Getting WAP settings part 2\n");
+ error=GSM_WaitFor (s, req2, 5, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) return error;
+ if (Priv7110->WAPLocations.Locations[i] == Priv7110->WAPLocations.CurrentLocation) {
+ settings->ActiveBearer = settings->Settings[settings->Number-1].Bearer;
+ }
+ }
+ }
+#endif
+#ifdef GSM_ENABLE_NOKIA6110
+ if (strstr(N6110Phone.models, s->Phone.Data.ModelInfo->model) != NULL) {
+ for (i=0;i<4;i++) {
+ req2[4] = Priv6110->WAPLocations.Locations[i];
+ smprintf(s, "Getting WAP settings part 2\n");
+ error=GSM_WaitFor (s, req2, 5, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) return error;
+ if (Priv6110->WAPLocations.Locations[i] == Priv6110->WAPLocations.CurrentLocation) {
+ settings->ActiveBearer = settings->Settings[settings->Number-1].Bearer;
+ }
+ }
+ }
+#endif
+ if (error == ERR_NONE) {
+ for (i=1;i<3;i++) {
+ CopyUnicodeString(settings->Settings[i].Title,settings->Settings[0].Title);
+ CopyUnicodeString(settings->Settings[i].HomePage,settings->Settings[0].HomePage);
+ settings->Settings[i].IsContinuous = settings->Settings[0].IsContinuous;
+ settings->Settings[i].IsSecurity = settings->Settings[0].IsSecurity;
+
+ settings->Settings[i].IsContinuous = settings->Settings[0].IsContinuous;
+ settings->Settings[i].IsSecurity = settings->Settings[0].IsSecurity;
+ }
+ error = DCT3DCT4_GetActiveConnectSet(s);
+ }
+ if (error != ERR_NONE) return error;
+
+ settings->Proxy[0] = 0x00;
+ settings->Proxy[1] = 0x00;
+ settings->ProxyPort = 8080;
+
+ settings->Proxy2[0] = 0x00;
+ settings->Proxy2[1] = 0x00;
+ settings->Proxy2Port = 8080;
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error DCT3_ReplySetWAPSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch(msg.Buffer[3]) {
+ case 0x19:
+ smprintf(s, "WAP settings part 1 set OK\n");
+ return ERR_NONE;
+ case 0x1a:
+ smprintf(s, "WAP settings part 1 setting error\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside WAP settings menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x02:
+ smprintf(s, "Incorrect data\n");
+ return ERR_UNKNOWN;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ case 0x1F:
+ smprintf(s, "WAP settings part 2 set OK\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3_SetWAPSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+#ifdef GSM_ENABLE_NOKIA6110
+ GSM_Phone_N6110Data *Priv6110 = &s->Phone.Data.Priv.N6110;
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ GSM_Phone_N7110Data *Priv7110 = &s->Phone.Data.Priv.N7110;
+#endif
+ GSM_Error error;
+ GSM_MultiWAPSettings settings2;
+ int i,pos,phone1=-1,phone2=-1,phone3=-1;
+ int ID=0,locations[4],loc1=-1,loc2=-1,loc3=-1;
+ unsigned char req[] = {N6110_FRAME_HEADER,0x15,
+ 0x00}; /* Location */
+ unsigned char req2[] = {N6110_FRAME_HEADER,0x1b,
+ 0x00}; /* Location */
+ unsigned char SetReq[200] = {N7110_FRAME_HEADER, 0x18,
+ 0x00}; /* Location */
+ unsigned char SetReq2[200] = {N7110_FRAME_HEADER, 0x1e,
+ 0x00}; /* Location */
+
+ /* We have to enable WAP frames in phone */
+ error=DCT3DCT4_EnableWAPFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ s->Phone.Data.WAPSettings = &settings2;
+ settings2.Number = 0;
+
+ req[4] = settings->Location-1;
+ smprintf(s, "Getting WAP settings part 1\n");
+ error = GSM_WaitFor (s, req, 6, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) return error;
+
+#ifdef GSM_ENABLE_NOKIA6110
+ if (strstr(N6110Phone.models, s->Phone.Data.ModelInfo->model) != NULL) {
+ for (i=0;i<4;i++) locations[i] = Priv6110->WAPLocations.Locations[i];
+ ID = Priv6110->WAPLocations.ID;
+ }
+#endif
+#ifdef GSM_ENABLE_NOKIA7110
+ if (strstr(N7110Phone.models, s->Phone.Data.ModelInfo->model) != NULL) {
+ for (i=0;i<4;i++) locations[i] = Priv7110->WAPLocations.Locations[i];
+ ID = Priv7110->WAPLocations.ID;
+ }
+#endif
+
+ /* Now we get info about supported types by phone and their locations */
+ for (i=0;i<4;i++) {
+ settings2.Number = 0;
+ settings2.Settings[0].Bearer = 0;
+ req2[4] = locations[i];
+ smprintf(s, "Getting WAP settings part 2\n");
+ error=GSM_WaitFor (s, req2, 6, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) return error;
+ switch (settings2.Settings[0].Bearer) {
+ case WAPSETTINGS_BEARER_DATA: phone1 = locations[i]; break;
+ case WAPSETTINGS_BEARER_SMS : phone2 = locations[i]; break;
+ case WAPSETTINGS_BEARER_USSD: phone3 = locations[i]; break;
+ default : break;
+ }
+ if (error != ERR_NONE) return error;
+ }
+
+ /* We have some phone locations and some data to set. We try to
+ * find info about locations in phone used to write concrete bearers
+ */
+ for (i=0;i<settings->Number;i++) {
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_DATA) {
+ if (phone1 != -1) loc1=i;
+ }
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_SMS) {
+ if (phone2 != -1) loc2=i;
+ }
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_USSD) {
+ if (phone3 != -1) loc3=i;
+ }
+ }
+
+ pos = 5;
+ memset(SetReq + pos, 0, 200 - pos);
+ SetReq[4] = settings->Location - 1;
+ if (loc1 != -1) {
+ /* Name */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc1].Title, false);
+ /* HomePage */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc1].HomePage, false);
+ if (settings->Settings[loc1].IsContinuous) SetReq[pos] = 0x01;
+ pos++;
+ SetReq[pos++] = ID;
+
+ SetReq[pos] = phone1; /* bearer */
+ switch (settings->ActiveBearer) {
+ case WAPSETTINGS_BEARER_DATA:
+ if (loc1 != -1) SetReq[pos] = phone1;
+ break;
+ case WAPSETTINGS_BEARER_SMS:
+ if (loc2 != -1) SetReq[pos] = phone2;
+ break;
+ case WAPSETTINGS_BEARER_USSD:
+ if (loc3 != -1) SetReq[pos] = phone3;
+ break;
+ default: break;
+ }
+ pos++;
+
+ if (settings->Settings[loc1].IsSecurity) SetReq[pos] = 0x01;
+ pos++;
+ } else if (loc2 != -1) {
+ /* Name */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc2].Title, false);
+ /* HomePage */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc2].HomePage, false);
+ if (settings->Settings[loc2].IsContinuous) SetReq[pos] = 0x01;
+ pos++;
+ SetReq[pos++] = ID;
+
+ SetReq[pos] = phone2; /* bearer */
+ switch (settings->ActiveBearer) {
+ case WAPSETTINGS_BEARER_DATA:
+ if (loc1 != -1) SetReq[pos] = phone1;
+ break;
+ case WAPSETTINGS_BEARER_SMS:
+ if (loc2 != -1) SetReq[pos] = phone2;
+ break;
+ case WAPSETTINGS_BEARER_USSD:
+ if (loc3 != -1) SetReq[pos] = phone3;
+ break;
+ default: break;
+ }
+ pos++;
+
+ if (settings->Settings[loc2].IsSecurity) SetReq[pos] = 0x01;
+ pos++;
+ } else if (loc3 != -1) {
+ /* Name */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc3].Title, false);
+ /* HomePage */
+ pos += NOKIA_SetUnicodeString(s, SetReq + pos, settings->Settings[loc3].HomePage, false);
+ if (settings->Settings[loc3].IsContinuous) SetReq[pos] = 0x01;
+ pos++;
+ SetReq[pos++] = ID;
+
+ SetReq[pos] = phone3; /* bearer */
+ switch (settings->ActiveBearer) {
+ case WAPSETTINGS_BEARER_DATA:
+ if (loc1 != -1) SetReq[pos] = phone1;
+ break;
+ case WAPSETTINGS_BEARER_SMS:
+ if (loc2 != -1) SetReq[pos] = phone2;
+ break;
+ case WAPSETTINGS_BEARER_USSD:
+ if (loc3 != -1) SetReq[pos] = phone3;
+ break;
+ default: break;
+ }
+ pos++;
+
+ if (settings->Settings[loc3].IsSecurity) SetReq[pos] = 0x01;
+ pos++;
+ } else {
+ return ERR_UNKNOWN; /* We have to have write something known */
+ }
+ memcpy(SetReq + pos, "\x01\x80\x00\x00\x00\x00\x00\x00\x00", 9);
+ pos += 9;
+
+ smprintf(s, "Writing WAP settings part 1\n");
+ error=GSM_WaitFor (s, SetReq, pos, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+
+ /* Data */
+ if (phone1 != -1) {
+ pos = 4;
+ memset(SetReq2 + pos, 0, 200 - pos);
+ SetReq2[pos++] = phone1;
+ SetReq2[pos++] = 0x02;
+ SetReq2[pos++] = 0x01; /* GSMdata */
+ if (loc1 != -1) {
+ if (!settings->Settings[loc1].IsNormalAuthentication) SetReq2[pos] = 0x01;
+ }
+ pos++;
+ if (loc1 != -1) {
+ if (settings->Settings[loc1].IsISDNCall) SetReq2[pos] = 0x01;
+ }
+ pos++;
+ if (loc1 != -1) {
+ switch (settings->Settings[loc1].Speed) {
+ case WAPSETTINGS_SPEED_9600 : SetReq2[pos++] = 0x01; break;
+ case WAPSETTINGS_SPEED_14400 : SetReq2[pos++] = 0x02; break;
+ default : SetReq2[pos++] = 0x02; break;
+ }
+ switch (settings->Settings[loc1].Speed) {
+ case WAPSETTINGS_SPEED_9600 : SetReq2[pos++] = 0x01; break;
+ case WAPSETTINGS_SPEED_14400 : SetReq2[pos++] = 0x02; break;
+ default : SetReq2[pos++] = 0x02; break;
+ }
+ } else pos+=2;
+ if (loc1 != -1) {
+ /* IP */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc1].IPAddress, false);
+ /* Number */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc1].DialUp, false);
+ /* Username */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc1].User, false);
+ /* Password */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc1].Password, false);
+ } else pos+=5;
+ memcpy(SetReq2 + pos, "\x80\x00\x00\x00\x00\x00\x00\x00", 8);
+ pos += 8;
+ smprintf(s, "Writing WAP settings part 2 (Data bearer)\n");
+ error=GSM_WaitFor (s, SetReq2, pos, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+ }
+ /* SMS */
+ if (phone2 != -1) {
+ pos = 4;
+ memset(SetReq2 + pos, 0, 200 - pos);
+ SetReq2[pos++] = phone2;
+ SetReq2[pos++] = 0x02;
+ SetReq2[pos++] = 0x00; /* SMS */
+ if (loc2 != -1) {
+ /* Service number */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc2].Service, false);
+ /* Server number */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc2].Server, false);
+ } else pos += 2;
+ memcpy(SetReq2 + pos, "\x80\x00\x00\x00\x00\x00\x00\x00", 8);
+ pos += 8;
+ smprintf(s, "Writing WAP settings part 2 (SMS bearer)\n");
+ error=GSM_WaitFor (s, SetReq2, pos, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+ }
+ /* USSD */
+ if (phone3 != -1) {
+ pos = 4;
+ memset(SetReq2 + pos, 0, 200 - pos);
+ SetReq2[pos++] = phone3;
+ SetReq2[pos++] = 0x02;
+ SetReq2[pos++] = 0x02; /* USSD */
+ if (loc3 != -1) {
+ if (!settings->Settings[loc3].IsIP) SetReq2[pos] = 0x01;
+ }
+ pos++;
+ if (loc3 != -1) {
+ /* Service number or IP address */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc3].Service, false);
+ /* Code number */
+ pos += NOKIA_SetUnicodeString(s, SetReq2 + pos, settings->Settings[loc3].Code, false);
+ } else pos+=2;
+ memcpy(SetReq2 + pos, "\x80\x00\x00\x00\x00\x00\x00\x00", 8);
+ pos += 8;
+ smprintf(s, "Writing WAP settings part 2 (USSD bearer)\n");
+ error=GSM_WaitFor (s, SetReq2, pos, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+ }
+ error = DCT3DCT4_SetActiveConnectSet(s, settings);
+ if (error != ERR_NONE) return error;
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error DCT3_ReplySendSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s, "SMS sent OK\n");
+ if (s->User.SendSMSStatus!=NULL) s->User.SendSMSStatus(s->CurrentConfig->Device,0,0);
+ return ERR_NONE;
+ case 0x03:
+ smprintf(s, "Error %i\n",msg.Buffer[6]);
+ if (s->User.SendSMSStatus!=NULL) s->User.SendSMSStatus(s->CurrentConfig->Device,msg.Buffer[6],0);
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3_SendSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int length;
+ GSM_Error error;
+ unsigned char req[256] = {N6110_FRAME_HEADER, 0x01, 0x02, 0x00};
+
+ error=PHONE_EncodeSMSFrame(s,sms,req+6,PHONE_SMSSubmit,&length, true);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Sending sms\n");
+ return s->Protocol.Functions->WriteMessage(s, req, 6+length, 0x02);
+}
+
+GSM_Error DCT3_ReplyNetmonitor(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x00:
+ smprintf(s, "Netmonitor correctly set\n");
+ break;
+ default:
+ smprintf(s, "Menu %i\n",msg.Buffer[3]);
+ smprintf(s, "%s\n",msg.Buffer+4);
+ strcpy(s->Phone.Data.Netmonitor,msg.Buffer+4);
+ break;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error DCT3_Netmonitor(GSM_StateMachine *s, int testnumber, char *value)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00, 0x01, 0x7e,
+ 0x00}; /* Test number */
+
+ value[0] = 0;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+
+ req[3] = testnumber;
+
+ smprintf(s, "Getting netmonitor test\n");
+ s->Phone.Data.Netmonitor = value;
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_Netmonitor);
+}
+
+GSM_Error DCT3_GetManufactureMonth(GSM_StateMachine *s, char *value)
+{
+ GSM_Error error;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+ return NOKIA_GetPhoneString(s,"\x00\x01\xCC\x02",4,0x40,value,ID_GetManufactureMonth,5);
+}
+
+GSM_Error DCT3_GetProductCode(GSM_StateMachine *s, char *value)
+{
+ GSM_Error error;
+
+ if (strlen(s->Phone.Data.ProductCodeCache)!=0) {
+ strcpy(value,s->Phone.Data.ProductCodeCache);
+ return ERR_NONE;
+ }
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+ return NOKIA_GetPhoneString(s,"\x00\x01\xCA\x01",4,0x40,value,ID_GetProductCode,5);
+}
+
+GSM_Error DCT3_GetOriginalIMEI(GSM_StateMachine *s, char *value)
+{
+ GSM_Error error;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+ return NOKIA_GetPhoneString(s,"\x00\x01\xCC\x01",4,0x40,value,ID_GetOriginalIMEI,5);
+}
+
+GSM_Error DCT3_GetHardware(GSM_StateMachine *s, char *value)
+{
+ GSM_Error error;
+
+ if (strlen(s->Phone.Data.HardwareCache)!=0) {
+ strcpy(value,s->Phone.Data.HardwareCache);
+ return ERR_NONE;
+ }
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+ return NOKIA_GetPhoneString(s,"\x00\x01\xC8\x05",4,0x40,value,ID_GetHardware,5);
+}
+
+GSM_Error DCT3_GetPPM(GSM_StateMachine *s, char *value)
+{
+ GSM_Error error;
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+ return NOKIA_GetPhoneString(s,"\x00\x01\xC8\x12",4,0x40,value,ID_GetPPM,5);
+}
+
+GSM_Error DCT3_GetSMSStatus(GSM_StateMachine *s, GSM_SMSMemoryStatus *status)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x36, 0x64};
+
+ s->Phone.Data.SMSStatus=status;
+ smprintf(s, "Getting SMS status\n");
+ return GSM_WaitFor (s, req, 5, 0x14, 2, ID_GetSMSStatus);
+
+ /* 6210 family doesn't show in frame with SMS status info
+ * about Templates. We get separately info about this SMS folder.
+ */
+}
+
+GSM_Error DCT3_ReplyDeleteSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch(msg.Buffer[3]) {
+ case 0x0b:
+ smprintf(s, "SMS deleted\n");
+ return ERR_NONE;
+ case 0x0c:
+ smprintf(s, "Error deleting SMS\n");
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ /* Not tested on 6210 */
+ smprintf(s, "Unknown meaning, SMS seems to be deleted\n");
+ return ERR_NONE;
+ case 0x02:
+ /* Not tested on 6210 */
+ smprintf(s, "Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ case 0x06:
+ /* Not tested on 6210 */
+ smprintf(s, "Phone is OFF\n");
+ return ERR_PHONEOFF;
+ default:
+ smprintf(s, "Unknown error: %02x\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error N71_92_ReplyGetSignalQuality(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Network level received: %i\n",msg.Buffer[4]);
+ Data->SignalQuality->SignalStrength = -1;
+ Data->SignalQuality->SignalPercent = ((int)msg.Buffer[4]);
+ Data->SignalQuality->BitErrorRate = -1;
+ return ERR_NONE;
+}
+
+GSM_Error N71_92_GetSignalQuality(GSM_StateMachine *s, GSM_SignalQuality *sig)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x81};
+
+ s->Phone.Data.SignalQuality = sig;
+ smprintf(s, "Getting network level\n");
+ return GSM_WaitFor (s, req, 4, 0x0a, 4, ID_GetSignalQuality);
+}
+
+GSM_Error N71_92_ReplyGetBatteryCharge(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Battery level received: %i\n",msg.Buffer[5]);
+ Data->BatteryCharge->BatteryPercent = ((int)msg.Buffer[5]);
+ Data->BatteryCharge->ChargeState = 0;
+ return ERR_NONE;
+}
+
+GSM_Error N71_92_GetBatteryCharge(GSM_StateMachine *s, GSM_BatteryCharge *bat)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x02};
+
+ s->Phone.Data.BatteryCharge = bat;
+ smprintf(s, "Getting battery level\n");
+ return GSM_WaitFor (s, req, 4, 0x17, 4, ID_GetBatteryCharge);
+}
+
+GSM_Error N71_92_ReplyPhoneSetting(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Bitmap_Types BmpType;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[4]) {
+ case 0x02:
+ if (Data->RequestID == ID_GetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Welcome note text received\n");
+ CopyUnicodeString(Data->Bitmap->Text,msg.Buffer+6);
+ smprintf(s, "Text is \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ return ERR_NONE;
+ }
+ if (Data->RequestID == ID_SetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Startup text set\n");
+ return ERR_NONE;
+ }
+ case 0x15:
+ if (Data->RequestID == ID_GetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Startup logo received\n");
+ BmpType=GSM_Nokia7110StartupLogo;
+ if (msg.Buffer[17]==0x60) BmpType=GSM_Nokia6210StartupLogo;
+ if (msg.Buffer[17]==0xc0) BmpType=GSM_NokiaStartupLogo;
+ PHONE_DecodeBitmap(BmpType, msg.Buffer+22, Data->Bitmap);
+ return ERR_NONE;
+ }
+ if (Data->RequestID == ID_SetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Startup logo set\n");
+ return ERR_NONE;
+ }
+ case 0x17:
+ if (Data->RequestID == ID_GetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Dealer note text received\n");
+ CopyUnicodeString(Data->Bitmap->Text,msg.Buffer+6);
+ smprintf(s, "Text is \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ return ERR_NONE;
+ }
+ if (Data->RequestID == ID_SetBitmap || Data->RequestID == ID_EachFrame) {
+ smprintf(s, "Dealer text set\n");
+ return ERR_NONE;
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error N71_92_GetPhoneSetting(GSM_StateMachine *s, int Request, int Setting)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0xee,
+ 0x1c}; /* Setting */
+
+ req[4]=Setting;
+ return GSM_WaitFor (s, req, 5, 0x7a, 4, Request);
+}
+
+GSM_Error N71_92_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ return DCT3_GetDateTime(s, date_time, 0x19);
+}
+
+GSM_Error N71_92_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ return DCT3_SetDateTime(s, date_time, 0x19);
+}
+
+GSM_Error DCT3_DecodeSMSFrame(GSM_StateMachine *s, GSM_SMSMessage *SMS, unsigned char *buffer)
+{
+ switch (buffer[12] & 0x03) {
+ case 0x00:
+ smprintf(s, "SMS type - deliver\n");
+ SMS->PDU = SMS_Deliver;
+ return GSM_DecodeSMSFrame(SMS,buffer,PHONE_SMSDeliver);
+ case 0x01:
+ smprintf(s, "SMS type - submit\n");
+ SMS->PDU = SMS_Submit;
+ return GSM_DecodeSMSFrame(SMS,buffer,PHONE_SMSSubmit);
+ case 0x02:
+ smprintf(s, "SMS type - delivery report\n");
+ SMS->PDU = SMS_Status_Report;
+ return GSM_DecodeSMSFrame(SMS,buffer,PHONE_SMSStatusReport);
+ }
+ return ERR_UNKNOWN;
+}
+
+GSM_Error N61_91_ReplySetOpLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x31:
+ smprintf(s, "Operator logo set OK\n");
+ return ERR_NONE;
+ case 0x32:
+ smprintf(s, "Error setting operator logo\n");
+ switch (msg.Buffer[4]) {
+ case 0x7d:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error N61_71_ReplyResetPhoneSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phone settings cleaned OK\n");
+ return ERR_NONE;
+}
+
+GSM_Error N61_71_ResetPhoneSettings(GSM_StateMachine *s, GSM_ResetSettingsType Type)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00, 0x01, 0x65,
+ 0x01}; /* Reset type */
+
+ switch (Type) {
+ case GSM_RESET_PHONESETTINGS : req[3] = 0x01; break;
+ case GSM_RESET_DEVICE : req[3] = 0x02; break;
+ case GSM_RESET_USERINTERFACE : req[3] = 0x08; break;
+ case GSM_RESET_USERINTERFACE_PHONESETTINGS : req[3] = 0x38; break;
+ case GSM_RESET_FULLFACTORY : req[3] = 0xff; break;
+ }
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error != ERR_NONE) return error;
+
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_ResetPhoneSettings);
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/dct3func.h b/gammu/emb/common/phone/nokia/dct3/dct3func.h
new file mode 100644
index 0000000..66b67ec
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/dct3func.h
@@ -0,0 +1,78 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef phone_nokia_dct3_h
+#define phone_nokia_dct3_h
+
+GSM_Error DCT3_ReplyPressKey (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyPlayTone (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyEnableSecurity (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetIMEI (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetSMSC (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySIMLogin (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySIMLogout (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetDateTime (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetAlarm (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySetDateTime (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySetAlarm (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyDialCommand (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetWAPBookmark (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetNetworkInfo (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySendSMSMessage (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySetSMSC (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyGetWAPSettings (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplySetWAPSettings (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyNetmonitor (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyDeleteSMSMessage (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_92_ReplyGetSignalQuality (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_92_ReplyGetBatteryCharge (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_92_ReplyPhoneSetting (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N61_71_ReplyResetPhoneSettings(GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N61_91_ReplySetOpLogo (GSM_Protocol_Message msg, GSM_StateMachine *s);
+#ifdef GSM_ENABLE_CELLBROADCAST
+GSM_Error DCT3_ReplySetIncomingCB (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3_ReplyIncomingCB (GSM_Protocol_Message msg, GSM_StateMachine *s);
+#endif
+
+GSM_Error DCT3_DeleteWAPBookmark (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+GSM_Error DCT3_GetWAPBookmark (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+GSM_Error DCT3_PressKey (GSM_StateMachine *s, GSM_KeyCode Key, bool Press);
+GSM_Error DCT3_PlayTone (GSM_StateMachine *s, int Herz, unsigned char Volume, bool start);
+GSM_Error DCT3_EnableSecurity (GSM_StateMachine *s, unsigned char status );
+GSM_Error DCT3_GetIMEI (GSM_StateMachine *s);
+GSM_Error DCT3_GetSMSC (GSM_StateMachine *s, GSM_SMSC *smsc );
+GSM_Error DCT3_GetNetworkInfo (GSM_StateMachine *s, GSM_NetworkInfo *netinfo );
+GSM_Error DCT3_DialVoice (GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber);
+GSM_Error DCT3_Reset (GSM_StateMachine *s, bool hard );
+GSM_Error DCT3_CancelCall (GSM_StateMachine *s, int ID, bool all);
+GSM_Error DCT3_AnswerAllCalls (GSM_StateMachine *s);
+GSM_Error DCT3_SendSMSMessage (GSM_StateMachine *s, GSM_SMSMessage *sms );
+GSM_Error DCT3_GetAlarm (GSM_StateMachine *s, GSM_Alarm *alarm, unsigned char msgtype);
+GSM_Error DCT3_GetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time, unsigned char msgtype );
+GSM_Error DCT3_SetAlarm (GSM_StateMachine *s, GSM_Alarm *alarm, unsigned char msgtype);
+GSM_Error DCT3_SetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time, unsigned char msgtype );
+GSM_Error DCT3_SetIncomingCB (GSM_StateMachine *s, bool enable );
+GSM_Error DCT3_GetSMSStatus (GSM_StateMachine *s, GSM_SMSMemoryStatus *status );
+GSM_Error DCT3_SetSMSC (GSM_StateMachine *s, GSM_SMSC *smsc );
+GSM_Error DCT3_GetWAPSettings (GSM_StateMachine *s, GSM_MultiWAPSettings *settings );
+GSM_Error DCT3_SetWAPSettings (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+GSM_Error DCT3_SetWAPBookmark (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+GSM_Error DCT3_Netmonitor (GSM_StateMachine *s, int testnumber, char *value );
+GSM_Error DCT3_GetManufactureMonth (GSM_StateMachine *s, char *value );
+GSM_Error DCT3_GetProductCode (GSM_StateMachine *s, char *value);
+GSM_Error DCT3_GetOriginalIMEI (GSM_StateMachine *s, char *value);
+GSM_Error DCT3_GetHardware (GSM_StateMachine *s, char *value);
+GSM_Error DCT3_GetPPM (GSM_StateMachine *s, char *value);
+GSM_Error N61_71_ResetPhoneSettings (GSM_StateMachine *s, GSM_ResetSettingsType Type);
+GSM_Error N71_92_GetBatteryCharge (GSM_StateMachine *s, GSM_BatteryCharge *bat);
+GSM_Error N71_92_GetSignalQuality (GSM_StateMachine *s, GSM_SignalQuality *sig);
+GSM_Error N71_92_GetPhoneSetting (GSM_StateMachine *s, int Request, int Setting);
+GSM_Error N71_92_GetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time );
+GSM_Error N71_92_SetDateTime (GSM_StateMachine *s, GSM_DateTime *date_time );
+
+GSM_Error DCT3_DecodeSMSFrame (GSM_StateMachine *s, GSM_SMSMessage *SMS, unsigned char *buffer);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n6110.c b/gammu/emb/common/phone/nokia/dct3/n6110.c
new file mode 100644
index 0000000..263d12b
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n6110.c
@@ -0,0 +1,2884 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Markus Plail and Gnokii */
+/* Authentication function (c) 1999 or earlier by Pavel Janik */
+/* 5210 calendar IDs by Frederick Ros */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA6110
+
+#include <string.h>
+
+#include "../../../../cfg/config.h"
+#include "../../../misc/coding/coding.h"
+#include "../../../service/sms/gsmsms.h"
+#include "../../../gsmcomon.h"
+#include "../../pfunc.h"
+#include "../nfunc.h"
+#include "n6110.h"
+#include "dct3func.h"
+
+static unsigned char N6110_MEMORY_TYPES[] = {
+ MEM_ME, 0x02,
+ MEM_SM, 0x03,
+ MEM_ON, 0x05,
+ MEM_DC, 0x07,
+ MEM_RC, 0x08,
+ MEM_MC, 0x09,
+ MEM_VM, 0x0b,
+ 0x00, 0x00
+};
+
+static GSM_Error N6110_ReplyGetPhoneLanguage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ N6110_Language lang = N6110_Auto;
+
+ if (msg.Buffer[3] == 0x15) return ERR_NONE;
+
+ smprintf(s, "Phone language is %02x\n",msg.Buffer[6]);
+ switch (msg.Buffer[6]) {
+ case 0x21: lang = N6110_Europe; break; //Polish
+ }
+ s->Phone.Data.Priv.N6110.PhoneLanguage = lang;
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_GetPhoneLanguage(GSM_StateMachine *s)
+{
+ unsigned char feat_req[] = {N6110_FRAME_HEADER, 0x13, 0x01,
+ 0x00, /* Profile location */
+ 0x00}; /* Feature number */
+
+ s->Phone.Data.Priv.N6110.PhoneLanguage = N6110_Auto;
+
+ feat_req[5] = 0;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ feat_req[6] = 0x1E;
+ } else {
+ feat_req[6] = 0x21;
+ }
+ smprintf(s, "Getting profile feature\n");
+ return GSM_WaitFor (s, feat_req, 7, 0x05, 4, ID_GetLanguage);
+}
+
+struct N6110_Lang_Char {
+ N6110_Language Lang;
+ unsigned char Phone;
+ unsigned char Unicode1;
+ unsigned char Unicode2;
+};
+
+static struct N6110_Lang_Char N6110_Lang_Table[] = {
+{N6110_Europe,0x13,0x01,0x04},//Latin capital letter a with ogonek
+{N6110_Europe,0x14,0x01,0x05},//Latin small letter a with ogonek
+{N6110_Europe,0x15,0x01,0x06},//Latin capital letter c with acute
+{N6110_Europe,0x17,0x01,0x07},//Latin small letter c with acute
+{N6110_Europe,0x1D,0x01,0x18},//Latin capital letter e with ogonek
+{N6110_Europe,0x1E,0x01,0x19},//Latin small letter e with ogonek
+{N6110_Europe,0x83,0x00,0xD3},//Latin capital letter o with acute
+{N6110_Europe,0x8E,0x01,0x41},//Latin capital letter l with stroke
+{N6110_Europe,0x90,0x01,0x42},//Latin small letter l with stroke
+{N6110_Europe,0x92,0x01,0x43},//Latin capital letter n with acute
+{N6110_Europe,0x93,0x01,0x44},//Latin small letter n with acute
+{N6110_Europe,0x9A,0x00,0xF3},//Latin small letter o with acute
+{N6110_Europe,0xB2,0x20,0xAC},//euro
+{N6110_Europe,0xB5,0x01,0x5A},//Latin capital letter s with acute
+{N6110_Europe,0xB6,0x01,0x5B},//Latin small letter s with acute
+{N6110_Europe,0xE7,0x01,0x79},//Latin capital letter z with acute
+{N6110_Europe,0xEE,0x01,0x7A},//Latin small letter z with acute
+{N6110_Europe,0xF4,0x01,0x7C},//Latin small letter z with dot above
+{N6110_Europe,0xF0,0x01,0x7B},//Latin capital letter z with dot above
+{0,0,0,0}
+};
+
+static void N6110_EncodeUnicode(GSM_StateMachine *s, unsigned char *dest, const unsigned char *src, int len)
+{
+ int i_len = 0, o_len, i;
+ wchar_t wc;
+ GSM_Phone_N6110Data *Priv = &s->Phone.Data.Priv.N6110;
+ bool found;
+
+ for (o_len = 0; i_len < len; o_len++) {
+ found = false;
+ if (Priv->PhoneLanguage != N6110_Auto) {
+ i = 0;
+ while(1) {
+ if (N6110_Lang_Table[i].Lang == 0) break;
+ if (N6110_Lang_Table[i].Lang == Priv->PhoneLanguage &&
+ N6110_Lang_Table[i].Phone == src[i_len]) {
+ dest[o_len*2] = N6110_Lang_Table[i].Unicode1;
+ dest[(o_len*2)+1] = N6110_Lang_Table[i].Unicode2;
+ i_len++;
+ found = true;
+ break;
+ }
+ i++;
+ }
+ }
+ if (!found) {
+ i_len += EncodeWithUnicodeAlphabet(&src[i_len], &wc);
+ dest[o_len*2] = (wc >> 8) & 0xff;
+ dest[(o_len*2)+1] = wc & 0xff;
+ }
+ }
+ dest[o_len*2] = 0;
+ dest[(o_len*2)+1] = 0;
+}
+
+#ifndef ENABLE_LGPL
+
+/* This function provides Nokia authentication protocol.
+ * Nokia authentication protocol is used in the communication between Nokia
+ * mobile phones (e.g. Nokia 6110) and Nokia Cellular Data Suite software,
+ * commercially sold by Nokia Corp.
+ * The authentication scheme is based on the token send by the phone to the
+ * software. The software does it's magic (see the function
+ * N6110_GetNokiaAuthentication) and returns the result back to the phone.
+ * If the result is correct the phone responds with the message "Accessory
+ * connected!" displayed on the LCD. Otherwise it will display "Accessory not
+ * supported" and some functions will not be available for use (?).
+ * The specification of the protocol is not publicly available, no comment.
+ */
+static void N6110_GetNokiaAuthentication(unsigned char *Imei, unsigned char *MagicBytes, unsigned char *MagicResponse)
+{
+ int i, j, CRC=0;
+ unsigned char Temp[16]; /* This is our temporary working area. */
+
+ /* Here we put FAC (Final Assembly Code) and serial number into our area. */
+ Temp[0] = Imei[6]; Temp[1] = Imei[7];
+ Temp[2] = Imei[8]; Temp[3] = Imei[9];
+ Temp[4] = Imei[10]; Temp[5] = Imei[11];
+ Temp[6] = Imei[12]; Temp[7] = Imei[13];
+
+ /* And now the TAC (Type Approval Code). */
+ Temp[8] = Imei[2]; Temp[9] = Imei[3];
+ Temp[10] = Imei[4]; Temp[11] = Imei[5];
+
+ /* And now we pack magic bytes from the phone. */
+ Temp[12] = MagicBytes[0]; Temp[13] = MagicBytes[1];
+ Temp[14] = MagicBytes[2]; Temp[15] = MagicBytes[3];
+
+ for (i=0; i<=11; i++) if (Temp[i + 1]& 1) Temp[i]<<=1;
+ switch (Temp[15] & 0x03) {
+ case 1:
+ case 2: j = Temp[13] & 0x07;
+ for (i=0; i<=3; i++) Temp[i+j] ^= Temp[i+12];
+ break;
+ default: j = Temp[14] & 0x07;
+ for (i=0; i<=3; i++) Temp[i + j] |= Temp[i + 12];
+ }
+ for (i=0; i<=15; i++) CRC ^= Temp[i];
+ for (i=0; i<=15; i++) {
+ switch (Temp[15 - i] & 0x06) {
+ case 0: j = Temp[i] | CRC; break;
+ case 2:
+ case 4: j = Temp[i] ^ CRC; break;
+ case 6: j = Temp[i] & CRC; break;
+ }
+ if (j == CRC) j = 0x2c;
+ if (Temp[i] == 0) j = 0;
+ MagicResponse[i] = j;
+ }
+}
+
+static GSM_Error N6110_ReplyGetMagicBytes(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_N6110Data *Priv = &s->Phone.Data.Priv.N6110;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ sprintf(Data->IMEI, "%s", msg.Buffer+9);
+ sprintf(Data->HardwareCache, "%s", msg.Buffer+39);
+ sprintf(Data->ProductCodeCache, "%s", msg.Buffer+31);
+
+ smprintf(s, "Message: Mobile phone identification received:\n");
+ smprintf(s, "IMEI : %s\n", msg.Buffer+9);
+ smprintf(s, "Model : %s\n", msg.Buffer+25);
+ smprintf(s, "Production Code : %s\n", msg.Buffer+31);
+ smprintf(s, "HW : %s\n", msg.Buffer+39);
+ smprintf(s, "Firmware : %s\n", msg.Buffer+44);
+
+ /* These bytes are probably the source of the "Accessory not connected"
+ * messages on the phone when trying to emulate NCDS... I hope....
+ * UPDATE: of course, now we have the authentication algorithm.
+ */
+ smprintf(s, " Magic bytes : %02x %02x %02x %02x\n", msg.Buffer[50], msg.Buffer[51], msg.Buffer[52], msg.Buffer[53]);
+
+ Priv->MagicBytes[0]=msg.Buffer[50];
+ Priv->MagicBytes[1]=msg.Buffer[51];
+ Priv->MagicBytes[2]=msg.Buffer[52];
+ Priv->MagicBytes[3]=msg.Buffer[53];
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_MakeAuthentication(GSM_StateMachine *s)
+{
+ GSM_Phone_N6110Data *Priv = &s->Phone.Data.Priv.N6110;
+ GSM_Error error;
+ unsigned char connect4[] = {N6110_FRAME_HEADER, 0x10};
+ unsigned char magic_connect[] = {
+ N6110_FRAME_HEADER, 0x12,
+ /* The real magic goes here ... These bytes are filled in
+ * with the function N6110_GetNokiaAuthentication. */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ /* NOKIA&GNOKII Accessory */
+ 'N', 'O', 'K', 'I', 'A', '&', 'N', 'O', 'K', 'I', 'A',
+ 'a', 'c', 'c', 'e', 's', 's', 'o', 'r', 'y',
+ 0x00, 0x00, 0x00, 0x00};
+
+ smprintf(s, "Getting magic bytes for authentication\n");
+ error=GSM_WaitFor (s, connect4, 4, 0x64, 4, ID_MakeAuthentication);
+ if (error!=ERR_NONE) return error;
+
+ N6110_GetNokiaAuthentication(s->Phone.Data.IMEI, Priv->MagicBytes, magic_connect+4);
+ smprintf(s, "Sending authentication bytes\n");
+ return s->Protocol.Functions->WriteMessage(s, magic_connect, 45, 0x64);
+}
+
+#endif
+
+static GSM_Error N6110_ShowStartInfo(GSM_StateMachine *s, bool enable)
+{
+#ifdef ENABLE_LGPL
+ return ERR_NONE;
+#else
+ GSM_Error error=ERR_NONE;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_MAGICBYTES)) {
+ if (s->ConnectionType == GCT_FBUS2 ||
+ s->ConnectionType == GCT_FBUS2IRDA) {
+ error=N6110_MakeAuthentication(s);
+ }
+ }
+ return error;
+#endif
+}
+
+static GSM_Error N6110_Initialise (GSM_StateMachine *s)
+{
+#ifdef DEBUG
+ DCT3_SetIncomingCB(s,true);
+#endif
+ N6110_GetPhoneLanguage(s);
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ return DCT3_GetDateTime(s, date_time, 0x11);
+}
+
+static GSM_Error N6110_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ return DCT3_GetAlarm(s, alarm, 0x11);
+}
+
+static GSM_Error N6110_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ return DCT3_SetDateTime(s, date_time, 0x11);
+}
+
+static GSM_Error N6110_SetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ return DCT3_SetAlarm(s, alarm, 0x11);
+}
+
+static GSM_Error N6110_ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int count;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Phonebook entry received\n");
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ Data->Memory->EntriesNum = 0;
+ count=5;
+ /* If name is not empty */
+ if (msg.Buffer[count]!=0x00) {
+ if (msg.Buffer[count]>GSM_PHONEBOOK_TEXT_LENGTH) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ Data->Memory->Entries[Data->Memory->EntriesNum].EntryType=PBK_Text_Name;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOPBKUNICODE)) {
+ if (Data->Memory->MemoryType==MEM_DC ||
+ Data->Memory->MemoryType==MEM_RC ||
+ Data->Memory->MemoryType==MEM_MC ||
+ Data->Memory->MemoryType==MEM_ME) {
+ N6110_EncodeUnicode(s,Data->Memory->Entries[Data->Memory->EntriesNum].Text,
+ msg.Buffer+count+1,msg.Buffer[count]);
+ } else {
+ EncodeUnicode(Data->Memory->Entries[Data->Memory->EntriesNum].Text,
+ msg.Buffer+count+1,msg.Buffer[count]);
+ }
+ } else {
+ memcpy(Data->Memory->Entries[Data->Memory->EntriesNum].Text,
+ msg.Buffer+count+1,msg.Buffer[count]);
+ Data->Memory->Entries[Data->Memory->EntriesNum].Text[msg.Buffer[count]]=0x00;
+ Data->Memory->Entries[Data->Memory->EntriesNum].Text[msg.Buffer[count]+1]=0x00;
+ }
+ smprintf(s, "Name \"%s\"\n",
+ DecodeUnicodeString(Data->Memory->Entries[Data->Memory->EntriesNum].Text));
+ Data->Memory->EntriesNum++;
+ }
+ count=count+msg.Buffer[count]+1;
+
+ /* If number is empty */
+ if (msg.Buffer[count]==0x00) return ERR_EMPTY;
+
+ if (msg.Buffer[count]>GSM_PHONEBOOK_TEXT_LENGTH) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ Data->Memory->Entries[Data->Memory->EntriesNum].EntryType = PBK_Number_General;
+ Data->Memory->Entries[Data->Memory->EntriesNum].VoiceTag = 0;
+ Data->Memory->Entries[Data->Memory->EntriesNum].SMSList[0] = 0;
+ EncodeUnicode(Data->Memory->Entries[Data->Memory->EntriesNum].Text,
+ msg.Buffer+count+1,msg.Buffer[count]);
+ smprintf(s, "Number \"%s\"\n",
+ DecodeUnicodeString(Data->Memory->Entries[Data->Memory->EntriesNum].Text));
+ Data->Memory->EntriesNum++;
+ count=count+msg.Buffer[count]+1;
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOCALLER)) {
+ if (msg.Buffer[count]<5) {
+ Data->Memory->Entries[Data->Memory->EntriesNum].EntryType=PBK_Caller_Group;
+ smprintf(s, "Caller group \"%i\"\n",msg.Buffer[count]);
+ Data->Memory->Entries[Data->Memory->EntriesNum].Number=msg.Buffer[count]+1;
+ Data->Memory->EntriesNum++;
+ }
+ }
+ count++;
+
+ if (Data->Memory->MemoryType==MEM_DC ||
+ Data->Memory->MemoryType==MEM_RC ||
+ Data->Memory->MemoryType==MEM_MC) {
+ NOKIA_DecodeDateTime(s, msg.Buffer+count+1,&Data->Memory->Entries[Data->Memory->EntriesNum].Date);
+ Data->Memory->Entries[Data->Memory->EntriesNum].EntryType=PBK_Date;
+
+ /* These values are set, when date and time unavailable in phone.
+ * Values from 3310 - in other can be different */
+ if (Data->Memory->Entries[2].Date.Day !=20 ||
+ Data->Memory->Entries[2].Date.Month !=1 ||
+ Data->Memory->Entries[2].Date.Year !=2118||
+ Data->Memory->Entries[2].Date.Hour !=3 ||
+ Data->Memory->Entries[2].Date.Minute!=14 ||
+ Data->Memory->Entries[2].Date.Second!=7)
+ Data->Memory->EntriesNum++;
+ }
+
+ return ERR_NONE;
+ default:
+ switch (msg.Buffer[4]) {
+ case 0x6f:
+ smprintf(s, "Phone is OFF\n");
+ return ERR_PHONEOFF;
+ case 0x74:
+ /* TODO: check if not too high */
+ smprintf(s, "ERROR: Empty ????\n");
+ Data->Memory->EntriesNum = 0;
+ return ERR_EMPTY;
+ case 0x7d:
+ smprintf(s, "ERROR: Invalid memory type\n");
+ return ERR_NOTSUPPORTED;
+ case 0x8d:
+ smprintf(s, "ERROR: no PIN\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x01,
+ 0x00, /* memory type */
+ 0x00, /* location */
+ 0x00};
+
+ req[4] = NOKIA_GetMemoryType(s, entry->MemoryType,N6110_MEMORY_TYPES);
+ if (req[4]==0xff) return ERR_NOTSUPPORTED;
+
+ req[5] = entry->Location;
+ if (entry->MemoryType==MEM_DC || entry->MemoryType==MEM_RC || entry->MemoryType==MEM_MC) req[5]--;
+
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, 7, 0x03, 4, ID_GetMemory);
+}
+
+static GSM_Error N6110_ReplyGetMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Memory status received\n");
+ switch (msg.Buffer[3]) {
+ case 0x08:
+ smprintf(s, "Memory type: %i\n",msg.Buffer[4]);
+
+ smprintf(s, "Free : %i\n",msg.Buffer[5]);
+ Data->MemoryStatus->MemoryFree=msg.Buffer[5];
+
+ smprintf(s, "Used : %i\n",msg.Buffer[6]);
+ Data->MemoryStatus->MemoryUsed=msg.Buffer[6];
+
+ return ERR_NONE;
+ break;
+ case 0x09:
+ switch (msg.Buffer[4]) {
+ case 0x6f:
+ smprintf(s, "Phone is probably powered off.\n");
+ return ERR_TIMEOUT;
+ case 0x7d:
+ smprintf(s, "Memory type not supported by phone model.\n");
+ return ERR_NOTSUPPORTED;
+ case 0x8d:
+ smprintf(s, "Waiting for security code.\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+static GSM_Error N6110_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07,
+ 0x00}; /* memory type */
+
+ req[4] = NOKIA_GetMemoryType(s, Status->MemoryType,N6110_MEMORY_TYPES);
+ if (req[4]==0xff) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.MemoryStatus=Status;
+ smprintf(s, "Getting memory status\n");
+ return GSM_WaitFor (s, req, 5, 0x03, 4, ID_GetMemoryStatus);
+}
+
+static GSM_Error N6110_ReplyGetSMSStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "SMS status received\n");
+ switch (msg.Buffer[3]) {
+ case 0x37:
+ smprintf(s, "SIM size : %i\n",msg.Buffer[7]);
+ smprintf(s, "Used in SIM : %i\n",msg.Buffer[10]);
+ smprintf(s, "Unread in SIM : %i\n",msg.Buffer[11]);
+ Data->SMSStatus->SIMUsed = msg.Buffer[10];
+ Data->SMSStatus->SIMUnRead = msg.Buffer[11];
+ Data->SMSStatus->SIMSize = msg.Buffer[7];
+ Data->SMSStatus->PhoneUsed = 0;
+ Data->SMSStatus->PhoneUnRead = 0;
+ Data->SMSStatus->PhoneSize = 0;
+ Data->SMSStatus->TemplatesUsed = 0;
+ return ERR_NONE;
+ case 0x38:
+ smprintf(s, "Error. No PIN ?\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_ReplyGetSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "SMS Message received\n");
+ switch(msg.Buffer[3]) {
+ case 0x08:
+ Data->GetSMSMessage->Number = 1;
+ Data->GetSMSMessage->SMS[0].Name[0] = 0;
+ Data->GetSMSMessage->SMS[0].Name[1] = 0;
+ Data->GetSMSMessage->SMS[0].Memory = MEM_SM;
+ NOKIA_DecodeSMSState(s, msg.Buffer[4], &Data->GetSMSMessage->SMS[0]);
+ switch (msg.Buffer[7]) {
+ case 0x00: case 0x01: /* Report or SMS_Deliver */
+ Data->GetSMSMessage->SMS[0].Folder = 0x01;
+ Data->GetSMSMessage->SMS[0].InboxFolder = true;
+ break;
+ case 0x02: /* SMS_Submit */
+ Data->GetSMSMessage->SMS[0].Folder = 0x02;
+ Data->GetSMSMessage->SMS[0].InboxFolder = false;
+ break;
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+ DCT3_DecodeSMSFrame(s, &Data->GetSMSMessage->SMS[0],msg.Buffer+8);
+ return ERR_NONE;
+ case 0x09:
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "Unknown. Probably phone too busy\n");
+ return ERR_UNKNOWN;
+ case 0x02:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ case 0x06:
+ smprintf(s, "Phone is OFF\n");
+ return ERR_PHONEOFF;
+ case 0x07:
+ smprintf(s, "Empty\n");
+ return ERR_EMPTY;
+ case 0x0c:
+ smprintf(s, "Access error. No PIN ?\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07, 0x02,
+ 0x00, /* Location */
+ 0x01, 0x64};
+
+ if (sms->SMS[0].Folder!=0x00) return ERR_NOTSUPPORTED;
+
+ req[5] = sms->SMS[0].Location;
+
+ s->Phone.Data.GetSMSMessage=sms;
+ smprintf(s, "Getting sms\n");
+ return GSM_WaitFor (s, req, 8, 0x02, 4, ID_GetSMSMessage);
+}
+
+static GSM_Error N6110_GetNextSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start)
+{
+ GSM_Phone_N6110Data *Priv = &s->Phone.Data.Priv.N6110;
+ GSM_Error error;
+
+ if (start) {
+ error=s->Phone.Functions->GetSMSStatus(s,&Priv->LastSMSStatus);
+ if (error!=ERR_NONE) return error;
+ Priv->LastSMSRead=0;
+ sms->SMS[0].Location=0;
+ }
+ while (true) {
+ sms->SMS[0].Location++;
+ if (Priv->LastSMSRead>=(Priv->LastSMSStatus.SIMUsed+Priv->LastSMSStatus.PhoneUsed+Priv->LastSMSStatus.TemplatesUsed)) return ERR_EMPTY;
+ error=s->Phone.Functions->GetSMS(s, sms);
+ if (error==ERR_NONE) {
+ Priv->LastSMSRead++;
+ break;
+ }
+ if (error != ERR_EMPTY) return error;
+ }
+ return error;
+}
+
+static GSM_Error N6110_ReplyGetStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+#ifdef DEBUG
+ smprintf(s, "Phone status received :\n");
+ smprintf(s, "Mode : ");
+ switch (msg.Buffer[4]) {
+ case 0x01: smprintf(s, "registered within the network\n"); break;
+ case 0x02: smprintf(s, "call in progress\n"); break; /* ringing or already answered call */
+ case 0x03: smprintf(s, "waiting for security code\n"); break;
+ case 0x04: smprintf(s, "powered off\n"); break;
+ default : smprintf(s, "unknown\n");
+ }
+ smprintf(s, "Power source : ");
+ switch (msg.Buffer[7]) {
+ case 0x01: smprintf(s, "AC/DC\n"); break;
+ case 0x02: smprintf(s, "battery\n"); break;
+ default : smprintf(s, "unknown\n");
+ }
+ smprintf(s, "Battery Level : %d\n", msg.Buffer[8]);
+ smprintf(s, "Signal strength : %d\n", msg.Buffer[5]);
+#endif
+
+ switch (Data->RequestID) {
+ case ID_GetBatteryCharge:
+ Data->BatteryCharge->BatteryPercent = ((int)msg.Buffer[8])*25;
+ switch (msg.Buffer[7]) {
+ case 0x01: Data->BatteryCharge->ChargeState = GSM_BatteryConnected; break;
+ case 0x02: Data->BatteryCharge->ChargeState = GSM_BatteryPowered; break;
+ default : Data->BatteryCharge->ChargeState = 0;
+ }
+ return ERR_NONE;
+ case ID_GetSignalQuality:
+ Data->SignalQuality->SignalPercent = ((int)msg.Buffer[5])*25;
+ return ERR_NONE;
+ default:
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+static GSM_Error N6110_GetStatus(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x01};
+
+ return GSM_WaitFor (s, req, 4, 0x04, 4, ID);
+}
+
+static GSM_Error N6110_GetSignalQuality(GSM_StateMachine *s, GSM_SignalQuality *sig)
+{
+ char value[100];
+ GSM_Error error;
+
+ sig->BitErrorRate = -1;
+ sig->SignalStrength = -1; /* TODO for netmon */
+
+ smprintf(s, "Getting network level\n");
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_POWER_BATT)) {
+ error = DCT3_Netmonitor(s, 1, value);
+ if (error!=ERR_NONE) return error;
+ sig->SignalPercent = 100;
+ if (value[4]!='-') {
+ if (value[5]=='9' && value[6]>'4') sig->SignalPercent = 25;
+ if (value[5]=='9' && value[6]<'5') sig->SignalPercent = 50;
+ if (value[5]=='8' && value[6]>'4') sig->SignalPercent = 75;
+ } else sig->SignalPercent = 0;
+ return ERR_NONE;
+ } else {
+ s->Phone.Data.SignalQuality = sig;
+ return N6110_GetStatus(s, ID_GetSignalQuality);
+ }
+}
+
+static GSM_Error N6110_GetBatteryCharge(GSM_StateMachine *s, GSM_BatteryCharge *bat)
+{
+ char value[100];
+ GSM_Error error;
+
+ smprintf(s, "Getting battery level\n");
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_POWER_BATT)) {
+ error = DCT3_Netmonitor(s, 23, value);
+ if (error!=ERR_NONE) return error;
+ bat->BatteryPercent = 100;
+ bat->ChargeState = 0;
+ if (value[29]=='7') bat->BatteryPercent = 75;
+ if (value[29]=='5') bat->BatteryPercent = 50;
+ if (value[29]=='2') bat->BatteryPercent = 25;
+ return ERR_NONE;
+ } else {
+ s->Phone.Data.BatteryCharge = bat;
+ return N6110_GetStatus(s, ID_GetBatteryCharge);
+ }
+}
+
+static GSM_Error N6110_ReplySaveSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "SMS message saving status\n");
+ switch (msg.Buffer[3]) {
+ case 0x05:
+ smprintf(s, "Saved at location %i\n",msg.Buffer[5]);
+ Data->SaveSMSMessage->Location=msg.Buffer[5];
+ return ERR_NONE;
+ case 0x06:
+ switch (msg.Buffer[4]) {
+ case 0x02:
+ smprintf(s, "All locations busy\n");
+ return ERR_FULL;
+ case 0x03:
+ smprintf(s, "Too high ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_PrivSetSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int length;
+ GSM_Error error;
+ unsigned char req[256] = {N6110_FRAME_HEADER, 0x04,
+ 0x00, /* SMS status */
+ 0x02,
+ 0x00, /* SMS location */
+ 0x02}; /* SMS type */
+
+ req[6] = sms->Location;
+ if (sms->Folder==1) { /* Inbox */
+ req[4] = 1; /* SMS status - GSM_Read */
+ req[7] = 0x00; /* SMS type */
+ sms->PDU = SMS_Deliver;
+ error=PHONE_EncodeSMSFrame(s,sms,req+8,PHONE_SMSDeliver,&length,true);
+ } else { /* Outbox */
+ req[4] = 5; /* SMS status - GSM_Sent */
+ req[7] = 0x02; /* SMS type */
+ sms->PDU = SMS_Submit;
+ error=PHONE_EncodeSMSFrame(s,sms,req+8,PHONE_SMSSubmit,&length,true);
+ }
+ if (error != ERR_NONE) return error;
+
+ /* SMS State - GSM_Read -> GSM_Unread and GSM_Sent -> GSM_UnSent */
+ if (sms->State == SMS_UnSent || sms->State == SMS_UnRead) req[4] |= 0x02;
+
+ s->Phone.Data.SaveSMSMessage=sms;
+ smprintf(s, "Saving sms\n");
+ return GSM_WaitFor (s, req, 8+length, 0x14, 4, ID_SaveSMSMessage);
+}
+
+static GSM_Error N6110_SetSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ if (sms->Location == 0) return ERR_INVALIDLOCATION;
+ return N6110_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N6110_AddSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ sms->Location = 0;
+ return N6110_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N6110_ReplySetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x37:
+ smprintf(s, "Ringtone set OK\n");
+ return ERR_NONE;
+ break;
+ case 0x38:
+ smprintf(s, "Error setting ringtone\n");
+ switch (msg.Buffer[4]) {
+ case 0x7d:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_ReplySetBinRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "Set at location %i\n",msg.Buffer[3]+1);
+ return ERR_NONE;
+ default:
+ smprintf(s, "Invalid location. Too high ?\n");
+ return ERR_INVALIDLOCATION;
+ }
+}
+
+static GSM_Error N6110_SetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength)
+{
+ GSM_NetworkInfo NetInfo;
+ GSM_Error error;
+ int size=200,current=8;
+ GSM_UDHHeader UDHHeader;
+ unsigned char req[1000] = {N6110_FRAME_HEADER, 0x36,
+ 0x00, /* Location */
+ 0x00,0x78};
+ unsigned char reqBin[1000] = {0x00,0x01,0xa0,0x00,0x00,0x0c,0x01,0x2c};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NORING)) return ERR_NOTSUPPORTED;
+ if (Ringtone->Location == 0) return ERR_INVALIDLOCATION;
+
+ switch (Ringtone->Format) {
+ case RING_NOTETONE:
+ if (Ringtone->Location==255) {
+ /* Only 6110, 6130 and 6150 support it */
+ if (strcmp(s->Phone.Data.Model,"NSE-3") == 0 || strcmp(s->Phone.Data.Model,"NSK-3") == 0 ||
+ strcmp(s->Phone.Data.Model,"NSM-1") == 0) {
+ req[0] = 0x0c;
+ req[1] = 0x01;
+ UDHHeader.Type = UDH_NokiaRingtone;
+ GSM_EncodeUDHHeader(&UDHHeader);
+ /* We copy UDH now */
+ memcpy(req+2,UDHHeader.Text,UDHHeader.Length);
+ *maxlength=GSM_EncodeNokiaRTTLRingtone(*Ringtone, req+2+UDHHeader.Length, &size);
+ error = s->Protocol.Functions->WriteMessage(s, req, 2+UDHHeader.Length+size, 0x12);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+ /* We have to make something (not important, what) now */
+ /* no answer from phone*/
+ return DCT3_GetNetworkInfo(s,&NetInfo);
+ } else {
+ return ERR_NOTSUPPORTED;
+ }
+ }
+ *maxlength=GSM_EncodeNokiaRTTLRingtone(*Ringtone, req+7, &size);
+ req[4] = Ringtone->Location - 1;
+ smprintf(s, "Setting ringtone\n");
+ return GSM_WaitFor (s, req, 7 + size, 0x05, 4, ID_SetRingtone);
+ case RING_NOKIABINARY:
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+ memcpy(reqBin+current,DecodeUnicodeString(Ringtone->Name),UnicodeLength(Ringtone->Name));
+ current += UnicodeLength(Ringtone->Name);
+ reqBin[current++] = 0x00;
+ reqBin[current++] = 0x00;
+ reqBin[current++] = 0x00;/*xxx*/
+ memcpy(reqBin+current,Ringtone->NokiaBinary.Frame,Ringtone->NokiaBinary.Length);
+ current=current+Ringtone->NokiaBinary.Length;
+ reqBin[3]=Ringtone->Location-1;
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"3210")) reqBin[5]=0x10;
+ smprintf(s, "Setting binary ringtone\n");
+ return GSM_WaitFor (s, reqBin, current, 0x40, 4, ID_SetRingtone);
+ case RING_MIDI:
+ return ERR_NOTSUPPORTED;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6110_ReplyGetOpLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int count=5;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Operator logo received\n");
+ NOKIA_DecodeNetworkCode(msg.Buffer+count,Data->Bitmap->NetworkCode);
+ count = count + 3;
+ smprintf(s, "Network code : %s\n", Data->Bitmap->NetworkCode);
+ smprintf(s, "Network name for Gammu : %s ",
+ DecodeUnicodeString(GSM_GetNetworkName(Data->Bitmap->NetworkCode)));
+ smprintf(s, "(%s)\n",DecodeUnicodeString(GSM_GetCountryName(Data->Bitmap->NetworkCode)));
+
+ count = count + 3; /* We ignore size */
+ Data->Bitmap->BitmapWidth = msg.Buffer[count++];
+ Data->Bitmap->BitmapHeight = msg.Buffer[count++];
+ count++;
+ PHONE_DecodeBitmap(GSM_NokiaOperatorLogo,msg.Buffer+count,Data->Bitmap);
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_ReplyGetStartup(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i, count = 5;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Startup logo & notes received\n");
+ for (i=0;i<msg.Buffer[4];i++) {
+ switch (msg.Buffer[count++]) {
+ case 0x01:
+ smprintf(s, "Startup logo\n");
+ if (Data->Bitmap->Type == GSM_StartupLogo) {
+ Data->Bitmap->BitmapHeight = msg.Buffer[count++];
+ Data->Bitmap->BitmapWidth = msg.Buffer[count++];
+ PHONE_DecodeBitmap(GSM_NokiaStartupLogo, msg.Buffer + count, Data->Bitmap);
+ } else {
+ count = count + 2;
+ }
+ count = count + PHONE_GetBitmapSize(GSM_NokiaStartupLogo,0,0);
+ break;
+ case 0x02:
+ smprintf(s, "Welcome note\n");
+ if (Data->Bitmap->Type == GSM_WelcomeNote_Text) {
+ EncodeUnicode(Data->Bitmap->Text,msg.Buffer+count, msg.Buffer[count]);
+ smprintf(s, "Text is \"%s\"\n",Data->Bitmap->Text);
+ }
+ count = count + msg.Buffer[count] + 1;
+ break;
+ case 0x03:
+ smprintf(s, "Dealer welcome note\n");
+ if (Data->Bitmap->Type == GSM_DealerNote_Text) {
+ EncodeUnicode(Data->Bitmap->Text,msg.Buffer+count, msg.Buffer[count]);
+ smprintf(s, "Text is \"%s\"\n",Data->Bitmap->Text);
+ }
+ count = count + msg.Buffer[count] + 1;
+ break;
+ default:
+ smprintf(s, "Unknown block\n");
+ return ERR_UNKNOWNRESPONSE;
+ break;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_ReplyGetCallerLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int count;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x11:
+ smprintf(s, "Caller group info received\n");
+ EncodeUnicode(Data->Bitmap->Text,msg.Buffer+6,msg.Buffer[5]);
+ smprintf(s, "Name : \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ Data->Bitmap->DefaultName = false;
+ if (msg.Buffer[5] == 0x00) Data->Bitmap->DefaultName = true;
+ count = msg.Buffer[5] + 6;
+ Data->Bitmap->RingtoneID = msg.Buffer[count++];
+ Data->Bitmap->DefaultRingtone = false;
+ Data->Bitmap->FileSystemRingtone = false;
+ if (Data->Bitmap->RingtoneID == 16) Data->Bitmap->DefaultRingtone = true;
+ smprintf(s, "Ringtone ID: %02x\n",Data->Bitmap->RingtoneID);
+ Data->Bitmap->BitmapEnabled=(msg.Buffer[count++]==1);
+#ifdef DEBUG
+ smprintf(s, "Caller group logo ");
+ if (Data->Bitmap->BitmapEnabled) {
+ smprintf(s, "enabled\n");
+ } else {
+ smprintf(s, "disabled\n");
+ }
+#endif
+ count = count + 3; /* We ignore size */
+ Data->Bitmap->BitmapWidth = msg.Buffer[count++];
+ Data->Bitmap->BitmapHeight = msg.Buffer[count++];
+ count++;
+ PHONE_DecodeBitmap(GSM_NokiaCallerLogo,msg.Buffer+count,Data->Bitmap);
+ Data->Bitmap->DefaultBitmap = false;
+ return ERR_NONE;
+ case 0x12:
+ smprintf(s, "Error getting caller group info\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_ReplyGetSetPicture(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int count = 5, i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s, "Picture Image received\n");
+ if (msg.Buffer[count]!=0) {
+ GSM_UnpackSemiOctetNumber(Data->Bitmap->Sender, msg.Buffer + 5, true);
+ /* Convert number of semioctets to number of chars */
+ i = msg.Buffer[5];
+ if (i % 2) i++;
+ i=i / 2 + 1;
+ count = count + i + 1;
+ } else {
+ Data->Bitmap->Sender[0] = 0x00;
+ Data->Bitmap->Sender[1] = 0x00;
+ count+=2;
+ }
+ smprintf(s, "Sender : \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Sender));
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOPICTUREUNI) ||
+ (!strcmp(Data->Model,"NHM-5") && Data->VerNum < 5.79)) {
+ count++;
+ EncodeUnicode(Data->Bitmap->Text,msg.Buffer+count+1,msg.Buffer[count]);
+ count += UnicodeLength(Data->Bitmap->Text) + 1;
+ } else {
+ if (!strcmp(Data->Model,"NHM-5")) {
+ i = msg.Buffer[count] * 256 + msg.Buffer[count+1];
+ } else {
+ /* 3410 4.26 */
+ i = msg.Buffer[count] * 256 + msg.Buffer[count+1] - 2;
+ count += 2;
+ }
+ memcpy(Data->Bitmap->Text,msg.Buffer+count+2,i);
+ Data->Bitmap->Text[i] = 0;
+ Data->Bitmap->Text[i+1] = 0;
+ count += i + 2;
+ }
+ smprintf(s, "Text : \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ Data->Bitmap->BitmapWidth = msg.Buffer[count++];
+ Data->Bitmap->BitmapHeight = msg.Buffer[count++];
+ PHONE_DecodeBitmap(GSM_NokiaPictureImage, msg.Buffer + count + 2, Data->Bitmap);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,Data->Bitmap);
+#endif
+ return ERR_NONE;
+ break;
+ case 0x04:
+ smprintf(s, "Picture Image set OK\n");
+ return ERR_NONE;
+ case 0x05:
+ smprintf(s, "Can't set Picture Image - invalid location ?\n");
+ return ERR_INVALIDLOCATION;
+ break;
+ case 0x06:
+ smprintf(s, "Can't get Picture Image - invalid location ?\n");
+ return ERR_INVALIDLOCATION;
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ GSM_Error error;
+ unsigned char req[10] = {N6110_FRAME_HEADER};
+
+ s->Phone.Data.Bitmap=Bitmap;
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ case GSM_WelcomeNote_Text:
+ case GSM_DealerNote_Text:
+ if (Bitmap->Type == GSM_StartupLogo && IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOSTARTUP)) return ERR_NOTSUPPORTED;
+ req[3] = 0x16;
+ return GSM_WaitFor (s, req, 4, 0x05, 4, ID_GetBitmap);
+ case GSM_CallerGroupLogo:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOCALLER)) return ERR_NOTSUPPORTED;
+ req[3] = 0x10;
+ req[4] = Bitmap->Location - 1;
+ error = GSM_WaitFor (s, req, 5, 0x03, 4, ID_GetBitmap);
+ if (error==ERR_NONE) NOKIA_GetDefaultCallerGroupName(s,Bitmap);
+ return error;
+ case GSM_OperatorLogo:
+ req[3] = 0x33;
+ req[4] = 0x01;
+ return GSM_WaitFor (s, req, 5, 0x05, 4, ID_GetBitmap);
+ case GSM_PictureImage:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOPICTURE)) return ERR_NOTSUPPORTED;
+ req[3] = 0x01;
+ req[4] = Bitmap->Location - 1;
+ return GSM_WaitFor (s, req, 5, 0x47, 4, ID_GetBitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6110_ReplySetProfileFeature(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x11:
+ smprintf(s, "Feature of profile set\n");
+ return ERR_NONE;
+ case 0x12:
+ smprintf(s, "Error setting profile feature\n");
+ return ERR_NOTSUPPORTED;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_SetProfileFeature(GSM_StateMachine *s, unsigned char profile, unsigned char feature, unsigned char value)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x10, 0x01,
+ 0x00, /* Profile */
+ 0x00, /* Feature */
+ 0x00}; /* Value */
+
+ req[5]=profile;
+ req[6]=feature;
+ req[7]=value;
+ smprintf(s, "Setting profile feature\n");
+ return GSM_WaitFor (s, req, 8, 0x05, 4, ID_SetProfile);
+}
+
+static GSM_Error N6110_ReplySetStartup(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Startup logo set OK\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_ReplySetCallerLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x14:
+ smprintf(s, "Caller group set OK\n");
+ return ERR_NONE;
+ case 0x15:
+ smprintf(s, "Error setting caller group\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_SetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char reqPreview[1000] = {0x0c,0x01};
+ unsigned char req[600] = {N6110_FRAME_HEADER};
+ GSM_UDH UDHType = UDH_NokiaOperatorLogo;
+ int count = 0, textlen, Width, Height;
+ GSM_UDHHeader UDHHeader;
+ GSM_NetworkInfo NetInfo;
+ GSM_Error error;
+
+ switch (Bitmap->Type) {
+ case GSM_CallerGroupLogo:
+ case GSM_OperatorLogo:
+ if (Bitmap->Location == 255) {
+ /* Only 6110, 6130 and 6150 support it */
+ if (strcmp(s->Phone.Data.Model,"NSE-3") == 0 || strcmp(s->Phone.Data.Model,"NSK-3") == 0 ||
+ strcmp(s->Phone.Data.Model,"NSM-1") == 0) {
+ if (Bitmap->Type==GSM_CallerGroupLogo) UDHType = UDH_NokiaCallerLogo;
+ UDHHeader.Type = UDHType;
+ GSM_EncodeUDHHeader(&UDHHeader);
+ /* We copy UDH now */
+ memcpy(reqPreview+2,UDHHeader.Text,UDHHeader.Length);
+ count = count + UDHHeader.Length;
+ if (Bitmap->Type == GSM_OperatorLogo) {
+ NOKIA_EncodeNetworkCode(reqPreview+count,Bitmap->NetworkCode);
+ count = count + 3;
+ } else {
+ if (Bitmap->DefaultBitmap) {
+ Bitmap->BitmapWidth = 72;
+ Bitmap->BitmapHeight = 14;
+ GSM_ClearBitmap(Bitmap);
+ }
+ }
+ NOKIA_CopyBitmap(GSM_NokiaOperatorLogo,Bitmap,reqPreview, &count);
+ reqPreview[count]=0x00;
+ error = s->Protocol.Functions->WriteMessage(s, reqPreview, count + 1, 0x12);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+ /* We have to make something (not important, what) now */
+ /* no answer from phone*/
+ return DCT3_GetNetworkInfo(s,&NetInfo);
+ } else {
+ smprintf(s, "%s\n",s->Phone.Data.Model);
+ return ERR_NOTSUPPORTED;
+ }
+ }
+ break;
+ default:
+ break;
+ }
+
+ count = 3;
+
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOSTARTUP)) return ERR_NOTSUPPORTED;
+ if (Bitmap->Location != 1) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOSTARTANI)) return ERR_NOTSUPPORTED;
+ }
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOSTARTANI)) {
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"3210")) {
+ error = N6110_SetProfileFeature(s,0,0x2e,((unsigned char)(Bitmap->Location-1)));
+ } else {
+ error = N6110_SetProfileFeature(s,0,0x29,((unsigned char)(Bitmap->Location-1)));
+ }
+ if (error == ERR_NOTSUPPORTED) error = ERR_SECURITYERROR;
+ if (error != ERR_NONE) return error;
+ if (Bitmap->Location != 1) return ERR_NONE;
+ }
+ req[count++] = 0x18;
+ req[count++] = 0x01; /* One block */
+ req[count++] = 0x01;
+ PHONE_GetBitmapWidthHeight(GSM_NokiaStartupLogo, &Width, &Height);
+ req[count++] = Height;
+ req[count++] = Width;
+ PHONE_EncodeBitmap(GSM_NokiaStartupLogo, req + count, Bitmap);
+ count = count + PHONE_GetBitmapSize(GSM_NokiaStartupLogo,0,0);
+ return GSM_WaitFor (s, req, count, 0x05, 4, ID_SetBitmap);
+ case GSM_WelcomeNote_Text:
+ case GSM_DealerNote_Text:
+ req[count++] = 0x18;
+ req[count++] = 0x01; /* One block */
+ if (Bitmap->Type == GSM_WelcomeNote_Text) {
+ req[count++] = 0x02;
+ } else {
+ req[count++] = 0x03;
+ }
+ textlen = UnicodeLength(Bitmap->Text);
+ req[count++] = textlen;
+ memcpy(req + count,DecodeUnicodeString(Bitmap->Text),textlen);
+ count += textlen;
+ return GSM_WaitFor (s, req, count, 0x05, 4, ID_SetBitmap);
+ case GSM_CallerGroupLogo:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOCALLER)) return ERR_NOTSUPPORTED;
+ req[count++] = 0x13;
+ req[count++] = Bitmap->Location - 1;
+ if (Bitmap->DefaultName) {
+ req[count++] = 0;
+ } else {
+ textlen = UnicodeLength(Bitmap->Text);
+ req[count++] = textlen;
+ memcpy(req+count,DecodeUnicodeString(Bitmap->Text),textlen);
+ count += textlen;
+ }
+ if (Bitmap->DefaultRingtone) {
+ req[count++] = 16;
+ } else {
+ req[count++] = Bitmap->RingtoneID;
+ }
+ /* Value here is number of phone menu connected
+ * with caller logo in Nokia 61x0: 0x00 = Off, 0x01 = On,
+ * 0x02 = View Graphics, 0x03 = Send Graphics,
+ * 0x04 = Send via IR. For higher menu option connected with
+ * caller logo is not displayed
+ */
+ if (Bitmap->DefaultBitmap) {
+ Bitmap->BitmapWidth = 72;
+ Bitmap->BitmapHeight = 14;
+ GSM_ClearBitmap(Bitmap);
+ req[count++] = 0;
+ } else {
+ if (Bitmap->BitmapEnabled) req[count++] = 0x01; else req[count++] = 0x00;
+ }
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 4) / 256;
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 4) % 256;
+ NOKIA_CopyBitmap(GSM_NokiaCallerLogo, Bitmap, req, &count);
+ return GSM_WaitFor (s, req, count, 0x03, 4, ID_SetBitmap);
+ case GSM_OperatorLogo:
+ req[count++] = 0x30;
+ req[count++] = 0x01;
+ NOKIA_EncodeNetworkCode(req+count, Bitmap->NetworkCode);
+ count = count + 3;
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaOperatorLogo,0,0) + 4) / 256;
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaOperatorLogo,0,0) + 4) % 256;
+ NOKIA_CopyBitmap(GSM_NokiaOperatorLogo, Bitmap, req, &count);
+ return GSM_WaitFor (s, req, count, 0x05, 4, ID_SetBitmap);
+ case GSM_PictureImage:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOPICTURE)) return ERR_NOTSUPPORTED;
+ req[count++] = 0x03;
+ req[count++] = Bitmap->Location - 1;
+ if (Bitmap->Sender[0]!=0 || Bitmap->Sender[1]!=0) {
+ req[count]=GSM_PackSemiOctetNumber(Bitmap->Sender, req+count+1,true);
+ /* Convert number of semioctets to number of chars and add count */
+ textlen = req[count];
+ if (textlen % 2) textlen++;
+ count += textlen / 2 + 1;
+ count++;
+ } else {
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ }
+ req[count++] = 0x00;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOPICTUREUNI) ||
+ (!strcmp(s->Phone.Data.Model,"NHM-5") && s->Phone.Data.VerNum < 5.79)) {
+ textlen = UnicodeLength(Bitmap->Text);
+ req[count++] = textlen;
+ memcpy(req+count,DecodeUnicodeString(Bitmap->Text),textlen);
+ count += textlen;
+ } else {
+ textlen = UnicodeLength(Bitmap->Text)*2;
+ if (!strcmp(s->Phone.Data.Model,"NHM-5")) {
+ req[count++] = textlen;
+ } else {
+ /* 3410 4.26 */
+ req[count++] = textlen+2;
+ req[count++] = 0x00;
+ req[count++] = 0x1e;
+ }
+ memcpy(req+count,Bitmap->Text,textlen);
+ count += textlen;
+ }
+ NOKIA_CopyBitmap(GSM_NokiaPictureImage, Bitmap, req, &count);
+ return GSM_WaitFor (s, req, count, 0x47, 4, ID_SetBitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6110_ReplyCallInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ int tmp, count;
+ GSM_Call call;
+
+ call.CallIDAvailable = true;
+ call.Status = 0;
+ smprintf(s, "Call info, ");
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s, "Call established, waiting for answer\n");
+ call.Status = GSM_CALL_CallEstablished;
+ break;
+ case 0x03:
+ smprintf(s, "Call started\n");
+ /* no phone number in frame */
+ call.Status = GSM_CALL_CallStart;
+ break;
+ case 0x04:
+ smprintf(s, "Remote end hang up\n");
+ smprintf(s, "CC : %i\n",msg.Buffer[6]);
+ call.Status = GSM_CALL_CallRemoteEnd;
+ call.StatusCode = msg.Buffer[6];
+ break;
+ case 0x05:
+ smprintf(s, "Incoming call\n");
+ smprintf(s, "Number : \"");
+ count=msg.Buffer[6];
+ for (tmp=0; tmp <count; tmp++) smprintf(s, "%c", msg.Buffer[7+tmp]);
+ smprintf(s, "\"\nName : \"");
+ for (tmp=0; tmp<msg.Buffer[7+count]; tmp++) smprintf(s, "%c", msg.Buffer[8+count+tmp]);
+ smprintf(s, "\"\n");
+
+ call.Status = GSM_CALL_IncomingCall;
+ EncodeUnicode(call.PhoneNumber, msg.Buffer+7, msg.Buffer[6]);
+ break;
+ case 0x07:
+ smprintf(s, "Call answer initiated\n");
+ break;
+ case 0x09:
+ smprintf(s, "Call released\n");
+ call.Status = GSM_CALL_CallLocalEnd;
+ break;
+ case 0x0a:
+ smprintf(s, "Call is being released\n");
+ break;
+ case 0x23:
+ smprintf(s, "Call held\n");
+ call.Status = GSM_CALL_CallHeld;
+ break;
+ case 0x25:
+ smprintf(s, "Call resumed\n");
+ call.Status = GSM_CALL_CallResumed;
+ break;
+ case 0x27:
+ smprintf(s, "Call switched\n");
+ /* incorrect call id in frame - 6150 5.22 */
+ call.CallIDAvailable = false;
+ call.Status = GSM_CALL_CallSwitched;
+ break;
+ case 0x29:
+ smprintf(s, "Joining call to the conference (conference)\n");
+ break;
+ case 0x2A:
+ smprintf(s, "Removing call from the conference (split)\n");
+ break;
+ }
+ if (call.CallIDAvailable) smprintf(s, "Call ID : %d\n",msg.Buffer[4]);
+ if (Data->EnableIncomingCall && s->User.IncomingCall!=NULL && call.Status != 0) {
+ if (call.CallIDAvailable) call.CallID = msg.Buffer[4];
+ s->User.IncomingCall(s->CurrentConfig->Device, call);
+ }
+ if (s->Phone.Data.RequestID == ID_CancelCall) {
+ if (msg.Buffer[3] == 0x09) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ /* when we canceled call and see frame about other
+ * call releasing, we don't give ERR_NONE for "our"
+ * call release command
+ */
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_AnswerCall) {
+ if (msg.Buffer[3] == 0x07) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_UnholdCall) {
+ if (msg.Buffer[3] == 0x25) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_HoldCall) {
+ if (msg.Buffer[3] == 0x23) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_ConferenceCall) {
+ if (msg.Buffer[3] == 0x29) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_SplitCall) {
+ if (msg.Buffer[3] == 0x2B) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_DeleteSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0a, 0x02,
+ 0x00}; /* Location */
+
+ if (sms->Folder!=0x00) return ERR_NOTSUPPORTED;
+
+ req[5]=sms->Location;
+
+ smprintf(s, "Deleting sms\n");
+ return GSM_WaitFor (s, req, 6, 0x14, 4, ID_DeleteSMSMessage);
+}
+
+static GSM_Error N6110_ReplySetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Reply for writing memory\n");
+ switch (msg.Buffer[3]) {
+ case 0x05:
+ smprintf(s, "Done OK\n");
+ return ERR_NONE;
+ case 0x06:
+ smprintf(s, "Error\n");
+ switch (msg.Buffer[4]) {
+ case 0x7d:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ case 0x90:
+ smprintf(s, "Too long name...or other error\n");
+ return ERR_NOTSUPPORTED;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_SetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ int current, Group, Name, Number;
+ unsigned char req[128] = {N6110_FRAME_HEADER, 0x04,
+ 0x00, /* memory type */
+ 0x00}; /* location */
+
+ if (entry->Location == 0) return ERR_NOTSUPPORTED;
+
+ GSM_PhonebookFindDefaultNameNumberGroup(entry, &Name, &Number, &Group);
+
+ req[4] = NOKIA_GetMemoryType(s, entry->MemoryType,N6110_MEMORY_TYPES);
+ req[5] = entry->Location;
+
+ current = 7;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOPBKUNICODE)) {
+ if (Name != -1) {
+ req[6] = UnicodeLength(entry->Entries[Name].Text);
+ memcpy(req+current,DecodeUnicodeString(entry->Entries[Name].Text),UnicodeLength(entry->Entries[Name].Text));
+ current += UnicodeLength(entry->Entries[Name].Text);
+ } else req[6] = 0;
+ } else {
+ if (Name != -1) {
+ req[6] = UnicodeLength(entry->Entries[Name].Text)*2+2;
+ memcpy(req+current,entry->Entries[Name].Text,UnicodeLength(entry->Entries[Name].Text)*2);
+ current += UnicodeLength(entry->Entries[Name].Text)*2;
+ } else req[6] = 0;
+ req[current++]=0x00;
+ req[current++]=0x00;
+ }
+
+ if (Number != -1) {
+ req[current++]=UnicodeLength(entry->Entries[Number].Text);
+ memcpy(req+current,DecodeUnicodeString(entry->Entries[Number].Text),UnicodeLength(entry->Entries[Number].Text));
+ current += UnicodeLength(entry->Entries[Number].Text);
+ } else req[current++] = 0;
+
+ /* This allow to save 14 characters name into SIM memory, when
+ * no caller group is selected. */
+ if (Group == -1) {
+ req[current++] = 0xff;
+ } else {
+ req[current++] = entry->Entries[Group].Number-1;
+ }
+
+ smprintf(s, "Writing phonebook entry\n");
+ return GSM_WaitFor (s, req, current, 0x03, 4, ID_SetMemory);
+}
+
+static GSM_Error N6110_DeleteMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ GSM_MemoryEntry dwa;
+
+ dwa.Location = entry->Location;
+ dwa.MemoryType = entry->MemoryType;
+ dwa.EntriesNum = 0;
+
+ return N6110_SetMemory(s, &dwa);
+}
+
+static GSM_Error N6110_ReplyGetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ char buffer[2000];
+ GSM_Error error;
+ int i,end,start;
+
+ smprintf(s, "Ringtone received\n");
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ switch (Data->Ringtone->Format) {
+ case RING_NOTETONE:
+ memcpy(buffer,msg.Buffer,msg.Length);
+ i=7;
+ if (buffer[9]==0x4a && buffer[10]==0x3a) i=8;
+ buffer[i]=0x02;
+ error=GSM_DecodeNokiaRTTLRingtone(Data->Ringtone, buffer+i, msg.Length-i);
+ if (error!=ERR_NONE) return ERR_EMPTY;
+ return ERR_NONE;
+ case RING_NOKIABINARY:
+ i=8;
+ while (msg.Buffer[i]!=0) {
+ i++;
+ if (i>msg.Length) return ERR_EMPTY;
+ }
+ EncodeUnicode(Data->Ringtone->Name,msg.Buffer+8,i-8);
+ smprintf(s, "Name \"%s\"\n",DecodeUnicodeString(Data->Ringtone->Name));
+ /* Looking for start && end */
+ end=0;start=0;i=0;
+ while (true) {
+ if (start!=0) {
+ if (msg.Buffer[i]==0x07 && msg.Buffer[i+1]==0x0b) {
+ end=i+2; break;
+ }
+ if (msg.Buffer[i]==0x0e && msg.Buffer[i+1]==0x0b) {
+ end=i+2; break;
+ }
+ } else {
+ if (msg.Buffer[i]==0x02 && msg.Buffer[i+1]==0xfc && msg.Buffer[i+2]==0x09) {
+ start = i;
+ }
+ }
+ i++;
+ if (i==msg.Length-3) return ERR_EMPTY;
+ }
+ /* Copying frame */
+ memcpy(Data->Ringtone->NokiaBinary.Frame,msg.Buffer+start,end-start);
+ Data->Ringtone->NokiaBinary.Length=end-start;
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, Data->Ringtone->NokiaBinary.Frame, Data->Ringtone->NokiaBinary.Length);
+#endif
+ return ERR_NONE;
+ case RING_MIDI:
+ return ERR_NOTSUPPORTED;
+ }
+ smprintf(s, "Ringtone format is %i\n",Data->Ringtone->Format);
+ break;
+ default:
+ smprintf(s, "Invalid location. Too high ?\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone)
+{
+ GSM_Error error;
+ unsigned char req[] = {0x00, 0x01, 0x9e,
+ 0x00}; /* location */
+
+ if (PhoneRingtone) return ERR_NOTSUPPORTED;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NORING)) return ERR_NOTSUPPORTED;
+ if (Ringtone->Location == 0) return ERR_INVALIDLOCATION;
+
+ if (Ringtone->Format == 0x00) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_RING_SM)) {
+ Ringtone->Format = RING_NOTETONE;
+ } else {
+ Ringtone->Format = RING_NOKIABINARY;
+ }
+ }
+
+ switch (Ringtone->Format) {
+ case RING_NOTETONE:
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_RING_SM)) return ERR_NOTSUPPORTED;
+ break;
+ case RING_NOKIABINARY:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_RING_SM)) return ERR_NOTSUPPORTED;
+ break;
+ case RING_MIDI:
+ return ERR_NOTSUPPORTED;
+ }
+
+ error=DCT3_EnableSecurity (s, 0x01);
+ if (error!=ERR_NONE) return error;
+
+ req[3]=Ringtone->Location-1;
+ s->Phone.Data.Ringtone=Ringtone;
+ smprintf(s, "Getting (binary) ringtone\n");
+ return GSM_WaitFor (s, req, 4, 0x40, 4, ID_GetRingtone);
+}
+
+static GSM_Error N6110_ReplyGetSecurityStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ *s->Phone.Data.SecurityStatus = msg.Buffer[4];
+
+#ifdef DEBUG
+ smprintf(s, "Security code status\n");
+ switch(msg.Buffer[4]) {
+ case SEC_SecurityCode: smprintf(s, "waiting for Security Code.\n"); break;
+ case SEC_Pin : smprintf(s, "waiting for PIN.\n"); break;
+ case SEC_Pin2 : smprintf(s, "waiting for PIN2.\n"); break;
+ case SEC_Puk : smprintf(s, "waiting for PUK.\n"); break;
+ case SEC_Puk2 : smprintf(s, "waiting for PUK2.\n"); break;
+ case SEC_None : smprintf(s, "nothing to enter.\n"); break;
+ default : smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+#endif
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_GetSecurityStatus(GSM_StateMachine *s, GSM_SecurityCodeType *Status)
+{
+ unsigned char req[4] = {N6110_FRAME_HEADER, 0x07};
+
+ s->Phone.Data.SecurityStatus=Status;
+ smprintf(s, "Getting security code status\n");
+ return GSM_WaitFor (s, req, 4, 0x08, 2, ID_GetSecurityStatus);
+}
+
+static GSM_Error N6110_ReplyEnterSecurityCode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x0b:
+ smprintf(s, "Security code OK\n");
+ return ERR_NONE;
+ case 0x0c:
+ switch (msg.Buffer[4]) {
+ case 0x88:
+ smprintf(s, "Wrong code\n");
+ return ERR_SECURITYERROR;
+ case 0x8b:
+ smprintf(s, "Not required\n");
+ return ERR_NONE;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_EnterSecurityCode(GSM_StateMachine *s, GSM_SecurityCode Code)
+{
+ int len = 0;
+ unsigned char req[15] = {N6110_FRAME_HEADER, 0x0a,
+ 0x00}; /* Type of code to enter */
+
+ req[4]=Code.Type;
+
+ len = strlen(Code.Code);
+ memcpy(req+5,Code.Code,len);
+ req[5+len]=0x00;
+ req[6+len]=0x00;
+
+ smprintf(s, "Entering security code\n");
+ return GSM_WaitFor (s, req, 7+len, 0x08, 4, ID_EnterSecurityCode);
+}
+
+static GSM_Error N6110_ReplyGetSpeedDial(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x17:
+ smprintf(s, "Speed dial received\n");
+ switch (msg.Buffer[4]) {
+ case 0x02:
+ Data->SpeedDial->MemoryType = MEM_ME;
+ smprintf(s, "ME ");
+ break;
+ case 0x03:
+ Data->SpeedDial->MemoryType = MEM_SM;
+ smprintf(s, "SIM ");
+ break;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ Data->SpeedDial->MemoryLocation = msg.Buffer[5];
+ if (msg.Buffer[5] == 0x00) Data->SpeedDial->MemoryLocation = Data->SpeedDial->Location;
+ Data->SpeedDial->MemoryNumberID = 2;
+ smprintf(s, "location %i\n",Data->SpeedDial->MemoryLocation);
+ return ERR_NONE;
+ case 0x18:
+ smprintf(s, "Error getting speed dial. Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetSpeedDial(GSM_StateMachine *s, GSM_SpeedDial *SpeedDial)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x16,
+ 0x01}; /* location */
+
+ req[4] = SpeedDial->Location;
+
+ s->Phone.Data.SpeedDial=SpeedDial;
+ smprintf(s, "Getting speed dial\n");
+ return GSM_WaitFor (s, req, 5, 0x03, 4, ID_GetSpeedDial);
+}
+
+static GSM_Error N6110_ReplySendDTMF(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x40:
+ smprintf(s, "During sending DTMF\n");
+ return ERR_NONE;
+ case 0x51:
+ smprintf(s, "DTMF sent OK\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_ReplyGetDisplayStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Display status received\n");
+ if (Data->RequestID == ID_GetDisplayStatus) Data->DisplayFeatures->Number=0;
+ for (i=0;i<msg.Buffer[4];i++) {
+ if (msg.Buffer[2*i+6] == 0x02) {
+#ifdef DEBUG
+ switch (msg.Buffer[2*i+5]) {
+ case 0x01: smprintf(s, "Call in progress\n"); break;
+ case 0x02: smprintf(s, "Unknown\n"); break;
+ case 0x03: smprintf(s, "Unread SMS\n"); break;
+ case 0x04: smprintf(s, "Voice call\n"); break;
+ case 0x05: smprintf(s, "Fax call active\n"); break;
+ case 0x06: smprintf(s, "Data call active\n"); break;
+ case 0x07: smprintf(s, "Keyboard lock\n"); break;
+ case 0x08: smprintf(s, "SMS storage full\n"); break;
+ }
+#endif
+ if (Data->RequestID == ID_GetDisplayStatus) {
+ switch (msg.Buffer[2*i+5]) {
+ case 0x01: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_CallActive;
+ break;
+ case 0x03: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_UnreadSMS;
+ break;
+ case 0x04: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_VoiceCall;
+ break;
+ case 0x05: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_FaxCall;
+ break;
+ case 0x06: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_DataCall;
+ break;
+ case 0x07: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_KeypadLocked;
+ break;
+ case 0x08: Data->DisplayFeatures->Feature[Data->DisplayFeatures->Number] = GSM_SMSMemoryFull;
+ break;
+ }
+ if (msg.Buffer[2*i+5]!=0x02) Data->DisplayFeatures->Number++;
+ }
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_GetDisplayStatus(GSM_StateMachine *s, GSM_DisplayFeatures *features)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x51};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_DISPSTATUS)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.DisplayFeatures = features;
+ smprintf(s, "Getting display status\n");
+ return GSM_WaitFor (s, req, 4, 0x0d, 4, ID_GetDisplayStatus);
+}
+
+static GSM_Profile_PhoneTableValue Profile6110[] = {
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL1, 0x00,0x00},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL2, 0x00,0x01},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL3, 0x00,0x02},
+ {Profile_KeypadTone, PROFILE_KEYPAD_OFF, 0x00,0xff},
+ {Profile_Lights, PROFILE_LIGHTS_OFF, 0x01,0x00},
+ {Profile_Lights, PROFILE_LIGHTS_AUTO, 0x01,0x01},
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGING, 0x02,0x01},
+ {Profile_CallAlert, PROFILE_CALLALERT_BEEPONCE, 0x02,0x02},
+ {Profile_CallAlert, PROFILE_CALLALERT_OFF, 0x02,0x04},
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGONCE, 0x02,0x05},
+ {Profile_CallAlert, PROFILE_CALLALERT_ASCENDING, 0x02,0x06},
+ {Profile_CallAlert, PROFILE_CALLALERT_CALLERGROUPS,0x02,0x07},
+ /* Ringtone ID */
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL1, 0x04,0x06},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL2, 0x04,0x07},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL3, 0x04,0x08},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL4, 0x04,0x09},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL5, 0x04,0x0a},
+ {Profile_MessageTone, PROFILE_MESSAGE_NOTONE, 0x05,0x00},
+ {Profile_MessageTone, PROFILE_MESSAGE_STANDARD, 0x05,0x01},
+ {Profile_MessageTone, PROFILE_MESSAGE_SPECIAL, 0x05,0x02},
+ {Profile_MessageTone, PROFILE_MESSAGE_BEEPONCE, 0x05,0x03},
+ {Profile_MessageTone, PROFILE_MESSAGE_ASCENDING, 0x05,0x04},
+ {Profile_Vibration, PROFILE_VIBRATION_OFF, 0x06,0x00},
+ {Profile_Vibration, PROFILE_VIBRATION_ON, 0x06,0x01},
+ {Profile_WarningTone, PROFILE_WARNING_OFF, 0x07,0xff},
+ {Profile_WarningTone, PROFILE_WARNING_ON, 0x07,0x04},
+ /* Caller groups */
+ {Profile_AutoAnswer, PROFILE_AUTOANSWER_OFF, 0x09,0x00},
+ {Profile_AutoAnswer, PROFILE_AUTOANSWER_ON, 0x09,0x01},
+ {0x00, 0x00, 0x00,0x00}
+};
+
+static GSM_Profile_PhoneTableValue Profile3310[] = {
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL1, 0x00,0x00},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL2, 0x00,0x01},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL3, 0x00,0x02},
+ {Profile_KeypadTone, PROFILE_KEYPAD_OFF, 0x00,0xff},
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGING, 0x01,0x01},
+ {Profile_CallAlert, PROFILE_CALLALERT_BEEPONCE, 0x01,0x02},
+ {Profile_CallAlert, PROFILE_CALLALERT_OFF, 0x01,0x04},
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGONCE, 0x01,0x05},
+ {Profile_CallAlert, PROFILE_CALLALERT_ASCENDING, 0x01,0x06},
+ /* Ringtone ID */
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL1, 0x03,0x06},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL2, 0x03,0x07},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL3, 0x03,0x08},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL4, 0x03,0x09},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL5, 0x03,0x0a},
+ {Profile_MessageTone, PROFILE_MESSAGE_NOTONE, 0x04,0x00},
+ {Profile_MessageTone, PROFILE_MESSAGE_STANDARD, 0x04,0x01},
+ {Profile_MessageTone, PROFILE_MESSAGE_SPECIAL, 0x04,0x02},
+ {Profile_MessageTone, PROFILE_MESSAGE_BEEPONCE, 0x04,0x03},
+ {Profile_MessageTone, PROFILE_MESSAGE_ASCENDING, 0x04,0x04},
+ {Profile_MessageTone, PROFILE_MESSAGE_PERSONAL, 0x04,0x05},
+ {Profile_Vibration, PROFILE_VIBRATION_OFF, 0x05,0x00},
+ {Profile_Vibration, PROFILE_VIBRATION_ON, 0x05,0x01},
+ {Profile_Vibration, PROFILE_VIBRATION_FIRST, 0x05,0x02},
+ {Profile_WarningTone, PROFILE_WARNING_OFF, 0x06,0xff},
+ {Profile_WarningTone, PROFILE_WARNING_ON, 0x06,0x04},
+ {Profile_ScreenSaver, PROFILE_SAVER_OFF, 0x07,0x00},
+ {Profile_ScreenSaver, PROFILE_SAVER_ON, 0x07,0x01},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_5SEC, 0x08,0x00},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_20SEC, 0x08,0x01},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_1MIN, 0x08,0x02},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_2MIN, 0x08,0x03},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_5MIN, 0x08,0x04},
+ {Profile_ScreenSaverTime,PROFILE_SAVER_TIMEOUT_10MIN, 0x08,0x05},
+ {0x00, 0x00, 0x00,0x00}
+};
+
+static GSM_Error N6110_ReplyGetProfileFeature(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x14:
+ smprintf(s, "Profile feature %02x with value %02x\n",msg.Buffer[6],msg.Buffer[8]);
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ switch (msg.Buffer[6]) {
+ case 0x02:
+ smprintf(s, "Ringtone ID\n");
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_RingtoneID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = msg.Buffer[8];
+ Data->Profile->FeaturesNumber++;
+ break;
+ case 0x09 :
+ smprintf(s, "screen saver number\n");
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_ScreenSaverNumber;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = msg.Buffer[8] + 1;
+ Data->Profile->FeaturesNumber++;
+ break;
+ case 0x24:
+ smprintf(s, "selected profile\n");
+ if (msg.Buffer[8] + 1 == Data->Profile->Location) Data->Profile->Active = true;
+ break;
+ default:
+ NOKIA_FindFeatureValue(s, Profile3310,msg.Buffer[6],msg.Buffer[8],Data,false);
+ }
+ return ERR_NONE;
+ }
+ switch (msg.Buffer[6]) {
+ case 0x01: /* Lights */
+ if (Data->Profile->CarKitProfile) {
+ NOKIA_FindFeatureValue(s, Profile6110,msg.Buffer[6],msg.Buffer[8],Data,false);
+ }
+ break;
+ case 0x03:
+ smprintf(s, "Ringtone ID\n");
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_RingtoneID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = msg.Buffer[8];
+ Data->Profile->FeaturesNumber++;
+ break;
+ case 0x08: /* Caller groups */
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES51)) {
+ NOKIA_FindFeatureValue(s, Profile6110,msg.Buffer[6],msg.Buffer[8],Data,true);
+ }
+ break;
+ case 0x09: /* Autoanswer */
+ if (Data->Profile->CarKitProfile || Data->Profile->HeadSetProfile) {
+ NOKIA_FindFeatureValue(s, Profile6110,msg.Buffer[6],msg.Buffer[8],Data,false);
+ }
+ break;
+ case 0x2A:
+ smprintf(s, "selected profile\n");
+ if (msg.Buffer[8] + 1 == Data->Profile->Location) Data->Profile->Active = true;
+ break;
+ default:
+ NOKIA_FindFeatureValue(s, Profile6110,msg.Buffer[6],msg.Buffer[8],Data,false);
+ }
+ return ERR_NONE;
+ case 0x15:
+ smprintf(s, "Invalid profile location\n");
+ return ERR_INVALIDLOCATION;
+ case 0x1b:
+ Data->Profile->Name[0] = 0;
+ Data->Profile->Name[1] = 0;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ EncodeUnicode(Data->Profile->Name,msg.Buffer+10,msg.Buffer[9]);
+ } else {
+ if (msg.Length > 0x0A) {
+ CopyUnicodeString(Data->Profile->Name,msg.Buffer+10);
+ }
+ }
+ smprintf(s, "Profile name: \"%s\"\n",Data->Profile->Name);
+ Data->Profile->DefaultName = false;
+ if (msg.Buffer[9]==0x00) Data->Profile->DefaultName = true;
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ GSM_Error error;
+ int i,j;
+ unsigned char name_req[] = {N6110_FRAME_HEADER, 0x1a, 0x00};
+ unsigned char feat_req[] = {N6110_FRAME_HEADER, 0x13, 0x01,
+ 0x00, /* Profile location */
+ 0x00}; /* Feature number */
+
+ s->Phone.Data.Profile=Profile;
+
+ smprintf(s, "Getting profile name\n");
+ error = GSM_WaitFor (s, name_req, 5, 0x05, 4, ID_GetProfile);
+ if (error!=ERR_NONE) return error;
+ if (Profile->DefaultName) {
+ NOKIA_GetDefaultProfileName(s, Profile);
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES51)) {
+ switch(Profile->Location) {
+ case 1: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Personal"),strlen(GetMsg(s->msg,"Personal")));
+ break;
+ case 2: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Car"),strlen(GetMsg(s->msg,"Car")));
+ break;
+ case 3: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Headset"),strlen(GetMsg(s->msg,"Headset")));
+ break;
+ }
+ }
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ switch(Profile->Location) {
+ case 1: EncodeUnicode(Profile->Name,GetMsg(s->msg,"General"),strlen(GetMsg(s->msg,"General")));
+ break;
+ case 2: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Silent"),strlen(GetMsg(s->msg,"Silent")));
+ break;
+ case 3: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Discreet"),strlen(GetMsg(s->msg,"Discreet")));
+ break;
+ case 4: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Loud"),strlen(GetMsg(s->msg,"Loud")));
+ break;
+ case 5: EncodeUnicode(Profile->Name,GetMsg(s->msg,"My style"),strlen(GetMsg(s->msg,"My style")));
+ break;
+ case 6: Profile->Name[0] = 0; Profile->Name[1] = 0;
+ break;
+ }
+ }
+ }
+
+ Profile->FeaturesNumber = 0;
+
+ Profile->CarKitProfile = false;
+ Profile->HeadSetProfile = false;
+ if (Profile->Location == 6) Profile->CarKitProfile = true;
+ if (Profile->Location == 7) Profile->HeadSetProfile = true;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES51)) {
+ if (Profile->Location == 2) Profile->CarKitProfile = true;
+ if (Profile->Location == 3) Profile->HeadSetProfile = true;
+ }
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ Profile->HeadSetProfile = false; //fixme
+ Profile->CarKitProfile = false;
+ }
+
+ for (i = 0x00; i <= 0x09; i++) {
+ feat_req[5] = Profile->Location - 1;
+ feat_req[6] = i;
+ smprintf(s, "Getting profile feature\n");
+ error = GSM_WaitFor (s, feat_req, 7, 0x05, 4, ID_GetProfile);
+ if (error!=ERR_NONE) return error;
+ }
+
+ for (i=0;i<Profile->FeaturesNumber;i++) {
+ if (Profile->FeatureID[i] == Profile_CallAlert &&
+ Profile->FeatureValue[i] != PROFILE_CALLALERT_CALLERGROUPS) {
+ for (j=0;j<5;j++) Profile->CallerGroups[j] = true;
+ }
+ }
+
+ Profile->Active = false;
+ feat_req[5] = 0;
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) {
+ feat_req[6] = 0x24;
+ } else {
+ feat_req[6] = 0x2A;
+ }
+ smprintf(s, "Getting profile feature\n");
+ error = GSM_WaitFor (s, feat_req, 7, 0x05, 4, ID_GetProfile);
+
+ return error;
+}
+
+static GSM_Error N6110_SetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ int i;
+ bool found;
+ unsigned char ID,Value;
+ GSM_Error error;
+ GSM_Profile_PhoneTableValue *ProfilePhone = Profile6110;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PROFILES33)) ProfilePhone = Profile3310;
+
+ for (i=0;i<Profile->FeaturesNumber;i++) {
+ found = false;
+ if (ProfilePhone == Profile3310) {
+ switch (Profile->FeatureID[i]) {
+ case Profile_RingtoneID:
+ ID = 0x02;
+ Value = Profile->FeatureValue[i];
+ found = true;
+ break;
+ case Profile_ScreenSaverNumber:
+ ID = 0x09;
+ Value = Profile->FeatureValue[i];
+ found = true;
+ break;
+ default:
+ found=NOKIA_FindPhoneFeatureValue(
+ s,
+ ProfilePhone,
+ Profile->FeatureID[i],Profile->FeatureValue[i],
+ &ID,&Value);
+ }
+ }
+ if (ProfilePhone == Profile6110) {
+ switch (Profile->FeatureID[i]) {
+ case Profile_RingtoneID:
+ ID = 0x03;
+ Value = Profile->FeatureValue[i];
+ found = true;
+ break;
+ default:
+ found=NOKIA_FindPhoneFeatureValue(
+ s,
+ ProfilePhone,
+ Profile->FeatureID[i],Profile->FeatureValue[i],
+ &ID,&Value);
+ }
+ }
+ if (found) {
+ error=N6110_SetProfileFeature (s,((unsigned char)(Profile->Location-1)),ID,Value);
+ if (error!=ERR_NONE) return error;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_ReplyIncomingSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ GSM_SMSMessage sms;
+
+#ifdef DEBUG
+ smprintf(s, "SMS message received\n");
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+7);
+#endif
+ if (Data->EnableIncomingSMS && s->User.IncomingSMS!=NULL) {
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+7);
+
+ s->User.IncomingSMS(s->CurrentConfig->Device,sms);
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6110_ReplyAddCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Writting calendar note: ");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "OK\n");
+ return ERR_NONE;
+ case 0x73:
+ case 0x7d:
+ smprintf(s, "error\n");
+ return ERR_UNKNOWN;
+ case 0x81:
+ smprintf(s,"during editing notes in phone menu\n");
+ return ERR_INSIDEPHONEMENU;
+ default:
+ smprintf(s, "unknown ERROR %i\n",msg.Buffer[4]);
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_AddCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ bool Reminder3310 = false;
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location, i, current;
+ unsigned char mychar1,mychar2;
+ unsigned char req[200] = {N6110_FRAME_HEADER, 0x64, 0x01, 0x10,
+ 0x00, /* Length of the rest of the frame */
+ 0x00, /* Calendar note type */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x01, 0x00, 0x66, 0x01};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOCALENDAR)) return ERR_NOTSUPPORTED;
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(Note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL52)) {
+ switch(Note->Type) {
+ case GSM_CAL_REMINDER: req[7]=0x01; break;
+ case GSM_CAL_CALL : req[7]=0x02; break;
+ case GSM_CAL_MEETING : req[7]=0x03; break;
+ case GSM_CAL_BIRTHDAY: req[7]=0x04; break;
+ case GSM_CAL_T_ATHL : req[7]=0x05; break;
+ case GSM_CAL_T_BALL : req[7]=0x06; break;
+ case GSM_CAL_T_CYCL : req[7]=0x07; break;
+ case GSM_CAL_T_BUDO : req[7]=0x08; break;
+ case GSM_CAL_T_DANC : req[7]=0x09; break;
+ case GSM_CAL_T_EXTR : req[7]=0x0a; break;
+ case GSM_CAL_T_FOOT : req[7]=0x0b; break;
+ case GSM_CAL_T_GOLF : req[7]=0x0c; break;
+ case GSM_CAL_T_GYM : req[7]=0x0d; break;
+ case GSM_CAL_T_HORS : req[7]=0x0e; break;
+ case GSM_CAL_T_HOCK : req[7]=0x0f; break;
+ case GSM_CAL_T_RACE : req[7]=0x10; break;
+ case GSM_CAL_T_RUGB : req[7]=0x11; break;
+ case GSM_CAL_T_SAIL : req[7]=0x12; break;
+ case GSM_CAL_T_STRE : req[7]=0x13; break;
+ case GSM_CAL_T_SWIM : req[7]=0x14; break;
+ case GSM_CAL_T_TENN : req[7]=0x15; break;
+ case GSM_CAL_T_TRAV : req[7]=0x16; break;
+ case GSM_CAL_T_WINT : req[7]=0x17; break;
+ default : req[7]=0x01; break;
+ }
+ } else {
+ switch(Note->Type) {
+ case GSM_CAL_CALL : req[7]=0x02; break;
+ case GSM_CAL_MEETING : req[7]=0x03; break;
+ case GSM_CAL_BIRTHDAY: req[7]=0x04; break;
+ case GSM_CAL_REMINDER:
+ default : req[7]=0x01; break;
+ }
+ }
+
+ if (Time == -1) return ERR_UNKNOWN;
+ NOKIA_EncodeDateTime(s, req+8, &Note->Entries[Time].Date);
+ req[14] = Note->Entries[Time].Date.Second;
+
+ if (Alarm != -1) {
+ NOKIA_EncodeDateTime(s, req+15, &Note->Entries[Alarm].Date);
+ req[21] = Note->Entries[Alarm].Date.Second;
+ }
+
+ current = 23;
+
+ if (Text != -1) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL52) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL82)) {
+ req[22] = UnicodeLength(Note->Entries[Text].Text)*2;
+ memcpy(req+current,Note->Entries[Text].Text,UnicodeLength(Note->Entries[Text].Text)*2);
+ current += UnicodeLength(Note->Entries[Text].Text)*2;
+ } else {
+ req[22] = UnicodeLength(Note->Entries[Text].Text);
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL33)) {
+ Reminder3310 = true;
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"3310") && s->Phone.Data.VerNum<5.11) {
+ if (Note->Type!=GSM_CAL_REMINDER) Reminder3310 = false;
+ }
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"3330") && s->Phone.Data.VerNum<=4.50) {
+ if (Note->Type!=GSM_CAL_REMINDER) Reminder3310 = false;
+ }
+ if (Reminder3310) {
+ req[22]++; /* one additional char */
+ req[current++] = 0x01; /* we use now subset 1 */
+ for (i=0;i<((int)UnicodeLength(Note->Entries[Text].Text));i++) {
+ /* Euro char */
+ if (Note->Entries[Text].Text[i*2]==0x20 && Note->Entries[Text].Text[i*2+1]==0xAC) {
+ req[current++] = 0xe2;
+ req[current++] = 0x82;
+ req[current++] = 0xac;
+ req[23] = 0x03; /* use subset 3 */
+ req[22]+=2; /* two additional chars */
+ } else if (EncodeWithUTF8Alphabet(Note->Entries[Text].Text[i*2],Note->Entries[Text].Text[i*2+1],&mychar1,&mychar2)) {
+ req[current++] = mychar1;
+ req[current++] = mychar2;
+ req[23] = 0x03; /* use subset 3 */
+ req[22]++; /* one additional char */
+ } else {
+ current+=DecodeWithUnicodeAlphabet(((wchar_t)(Note->Entries[Text].Text[i*2]*256+Note->Entries[Text].Text[i*2+1])),req+current);
+ }
+ }
+ }
+ }
+ if (!Reminder3310) {
+ memcpy(req+current,DecodeUnicodeString(Note->Entries[Text].Text),UnicodeLength(Note->Entries[Text].Text));
+ current += UnicodeLength(Note->Entries[Text].Text);
+ }
+ }
+ } else req[22] = 0x00;
+
+ if (Note->Type == GSM_CAL_CALL) {
+ if (Phone != -1) {
+ req[current++] = UnicodeLength(Note->Entries[Phone].Text);
+ memcpy(req+current,DecodeUnicodeString(Note->Entries[Phone].Text),UnicodeLength(Note->Entries[Phone].Text));
+ current += UnicodeLength(Note->Entries[Phone].Text);
+ } else req[current++] = 0x00;
+ }
+
+ req[6] = current - 8;
+
+ smprintf(s, "Writing calendar note\n");
+ return GSM_WaitFor (s, req, current, 0x13, 4, ID_SetCalendarNote);
+}
+
+static GSM_Error N6110_ReplyDeleteCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Deleting calendar note: ");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "done OK\n");
+ return ERR_NONE;
+ case 0x81:
+ smprintf(s,"during editing notes in phone menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x93:
+ smprintf(s, "Can't be done - too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "unknown ERROR %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+static GSM_Error N6110_DeleteCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x68,
+ 0x00}; /* Location */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOCALENDAR)) return ERR_NOTSUPPORTED;
+
+ req[4] = Note->Location;
+
+ smprintf(s, "Deleting calendar note\n");
+ return GSM_WaitFor (s, req, 5, 0x13, 5, ID_DeleteCalendarNote);
+}
+
+/* for example: "Euro_char" text */
+static void Decode3310Subset3(int j, GSM_Protocol_Message msg, GSM_Phone_Data *Data)
+{
+ wchar_t wc;
+ int len = 0;
+ int i;
+ bool charfound;
+ GSM_CalendarEntry *Entry = Data->Cal;
+
+ i = j;
+ while (i!=msg.Buffer[23]) {
+ EncodeWithUnicodeAlphabet(msg.Buffer+24+i,&wc);
+ charfound = false;
+ if (i!=msg.Buffer[23]-2) {
+ if (msg.Buffer[24+i] ==0xe2 && msg.Buffer[24+i+1]==0x82 &&
+ msg.Buffer[24+i+2]==0xac) {
+ wc = 0x20 * 256 + 0xac;
+ i+=2;
+ charfound = true;
+ }
+ }
+ if (i!=msg.Buffer[23]-1 && !charfound) {
+ if (msg.Buffer[24+i]>=0xc2) {
+ wc = DecodeWithUTF8Alphabet(msg.Buffer[24+i],msg.Buffer[24+i+1]);
+ i++;
+ }
+ }
+ Entry->Entries[Entry->EntriesNum].Text[len++] = (wc >> 8) & 0xff;
+ Entry->Entries[Entry->EntriesNum].Text[len++] = wc & 0xff;
+ i++;
+ }
+ Entry->Entries[Entry->EntriesNum].Text[len++] = 0;
+ Entry->Entries[Entry->EntriesNum].Text[len++] = 0;
+}
+
+/* For example: "a with : above" char */
+static void Decode3310Subset2(int j, GSM_Protocol_Message msg, GSM_Phone_Data *Data)
+{
+ int len = 0;
+ int i;
+ GSM_CalendarEntry *Entry = Data->Cal;
+
+ i = j;
+ while (i!=msg.Buffer[23]) {
+ Entry->Entries[Entry->EntriesNum].Text[len++] = 0x00;
+ Entry->Entries[Entry->EntriesNum].Text[len++] = msg.Buffer[24+i];
+ i++;
+ }
+ Entry->Entries[Entry->EntriesNum].Text[len++] = 0;
+ Entry->Entries[Entry->EntriesNum].Text[len++] = 0;
+}
+
+static GSM_Error N6110_ReplyGetNextCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i = 0;
+ bool SpecialSubSet = false;
+ GSM_CalendarEntry *Entry = s->Phone.Data.Cal;
+
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Calendar note received\n");
+ switch (msg.Buffer[8]) {
+ case 0x01: Entry->Type = GSM_CAL_REMINDER; break;
+ case 0x02: Entry->Type = GSM_CAL_CALL; break;
+ case 0x03: Entry->Type = GSM_CAL_MEETING; break;
+ case 0x04: Entry->Type = GSM_CAL_BIRTHDAY; break;
+ case 0x05: Entry->Type = GSM_CAL_T_ATHL; break;
+ case 0x06: Entry->Type = GSM_CAL_T_BALL; break;
+ case 0x07: Entry->Type = GSM_CAL_T_CYCL; break;
+ case 0x08: Entry->Type = GSM_CAL_T_BUDO; break;
+ case 0x09: Entry->Type = GSM_CAL_T_DANC; break;
+ case 0x0a: Entry->Type = GSM_CAL_T_EXTR; break;
+ case 0x0b: Entry->Type = GSM_CAL_T_FOOT; break;
+ case 0x0c: Entry->Type = GSM_CAL_T_GOLF; break;
+ case 0x0d: Entry->Type = GSM_CAL_T_GYM; break;
+ case 0x0e: Entry->Type = GSM_CAL_T_HORS; break;
+ case 0x0f: Entry->Type = GSM_CAL_T_HOCK; break;
+ case 0x10: Entry->Type = GSM_CAL_T_RACE; break;
+ case 0x11: Entry->Type = GSM_CAL_T_RUGB; break;
+ case 0x12: Entry->Type = GSM_CAL_T_SAIL; break;
+ case 0x13: Entry->Type = GSM_CAL_T_STRE; break;
+ case 0x14: Entry->Type = GSM_CAL_T_SWIM; break;
+ case 0x15: Entry->Type = GSM_CAL_T_TENN; break;
+ case 0x16: Entry->Type = GSM_CAL_T_TRAV; break;
+ case 0x17: Entry->Type = GSM_CAL_T_WINT; break;
+ default :
+ smprintf(s, "Unknown note type %i\n",msg.Buffer[8]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+#ifdef DEBUG
+ switch (msg.Buffer[8]) {
+ case 0x01: smprintf(s, "Reminder\n"); break;
+ case 0x02: smprintf(s, "Call\n"); break;
+ case 0x03: smprintf(s, "Meeting\n"); break;
+ case 0x04: smprintf(s, "Birthday\n"); break;
+ }
+#endif
+ Entry->EntriesNum = 0;
+
+ NOKIA_DecodeDateTime(s, msg.Buffer+9, &Entry->Entries[0].Date);
+ smprintf(s, "Time : %02i-%02i-%04i %02i:%02i:%02i\n",
+ Entry->Entries[0].Date.Day,Entry->Entries[0].Date.Month,Entry->Entries[0].Date.Year,
+ Entry->Entries[0].Date.Hour,Entry->Entries[0].Date.Minute,Entry->Entries[0].Date.Second);
+ Entry->Entries[0].EntryType = CAL_START_DATETIME;
+ Entry->EntriesNum++;
+
+ NOKIA_DecodeDateTime(s, msg.Buffer+16, &Entry->Entries[1].Date);
+ if (Entry->Entries[1].Date.Year!=0) {
+ smprintf(s, "Alarm : %02i-%02i-%04i %02i:%02i:%02i\n",
+ Entry->Entries[1].Date.Day,Entry->Entries[1].Date.Month,Entry->Entries[1].Date.Year,
+ Entry->Entries[1].Date.Hour,Entry->Entries[1].Date.Minute,Entry->Entries[1].Date.Second);
+ Entry->Entries[1].EntryType = CAL_ALARM_DATETIME;
+ Entry->EntriesNum++;
+ } else {
+ smprintf(s, "No alarm\n");
+ }
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL52) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL82)) {
+ memcpy(Entry->Entries[Entry->EntriesNum].Text,msg.Buffer+24,msg.Buffer[23]);
+ Entry->Entries[Entry->EntriesNum].Text[msg.Buffer[23] ]=0;
+ Entry->Entries[Entry->EntriesNum].Text[msg.Buffer[23]+1]=0;
+ } else {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_CAL33)) {
+ /* first char is subset for 33xx and reminders */
+ if (Entry->Type == GSM_CAL_REMINDER) {
+ i=1;
+ smprintf(s, "Subset %i in reminder note !\n",msg.Buffer[24]);
+ }
+ SpecialSubSet = true;
+ switch (msg.Buffer[24]) {
+ case 2 : Decode3310Subset2(i,msg,&s->Phone.Data); break;
+ case 3 : Decode3310Subset3(i,msg,&s->Phone.Data); break;
+ default : SpecialSubSet = false; break;
+ }
+ }
+ if (!SpecialSubSet) {
+ N6110_EncodeUnicode(s,Entry->Entries[Entry->EntriesNum].Text,msg.Buffer+24+i,msg.Buffer[23]-i);
+ }
+ }
+ smprintf(s, "Text \"%s\"\n",DecodeUnicodeString(Entry->Entries[Entry->EntriesNum].Text));
+ if (msg.Buffer[23] != 0x00) {
+ Entry->Entries[Entry->EntriesNum].EntryType = CAL_TEXT;
+ Entry->EntriesNum++;
+ }
+
+ if (Entry->Type == GSM_CAL_CALL) {
+ EncodeUnicode(Entry->Entries[Entry->EntriesNum].Text,msg.Buffer+24+msg.Buffer[23]+1,msg.Buffer[24+msg.Buffer[23]]);
+ smprintf(s, "Phone : \"%s\"\n",DecodeUnicodeString(Entry->Entries[Entry->EntriesNum].Text));
+ if (msg.Buffer[24+msg.Buffer[23]] != 0x00) {
+ Entry->Entries[Entry->EntriesNum].EntryType = CAL_PHONE;
+ Entry->EntriesNum++;
+ }
+ }
+ return ERR_NONE;
+ case 0x93:
+ smprintf(s, "Can't get calendar note - too high location?\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6110_GetNextCalendarNote(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location;
+ GSM_Error error;
+ GSM_DateTime date_time;
+ GSM_Phone_N6110Data *Priv = &s->Phone.Data.Priv.N6110;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x66,
+ 0x00}; /* Location */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOCALENDAR)) return ERR_NOTSUPPORTED;
+
+ if (start) {
+ Priv->LastCalendarPos = 1;
+ } else {
+ Priv->LastCalendarPos++;
+ }
+
+ Note->Location = Priv->LastCalendarPos;
+ req[4] = Priv->LastCalendarPos;
+
+ s->Phone.Data.Cal=Note;
+ smprintf(s, "Getting calendar note\n");
+ error=GSM_WaitFor (s, req, 5, 0x13, 4, ID_GetCalendarNote);
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(Note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+ /* 2090 year is set for example in 3310 */
+ if (error == ERR_NONE && Note->Entries[Time].Date.Year == 2090) {
+ error=N6110_GetDateTime(s, &date_time);
+ if (error == ERR_NONE) Note->Entries[Time].Date.Year = date_time.Year;
+ }
+ return error;
+}
+
+GSM_Error N6110_ReplyUSSDInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char buffer[2000],buffer2[4000];
+ int tmp;
+
+ tmp=GSM_UnpackEightBitsToSeven(0, 82, 82, msg.Buffer+8, buffer);
+ msg.Buffer[tmp] = 0;
+
+ smprintf(s, "USSD reply: \"%s\"\n",buffer);
+
+ if (s->Phone.Data.EnableIncomingUSSD && s->User.IncomingUSSD!=NULL) {
+ EncodeUnicode(buffer2,buffer,strlen(buffer));
+ s->User.IncomingUSSD(s->CurrentConfig->Device, buffer2);
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error N6110_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ GSM_Error error;
+ unsigned char req1[] = {N6110_FRAME_HEADER, 0x42, 0x05, 0x01,
+ 0x07, 0xa2, 0x88, 0x81, 0x21, 0x15, 0x63, 0xa8,
+ 0x00, 0x00, 0x07, 0xa3, 0xb8, 0x81, 0x20, 0x15,
+ 0x63, 0x80};
+
+ if (!all) {
+ smprintf(s, "Answering call part 1\n");
+ error = GSM_WaitFor (s, req1, 24, 0x01, 5, ID_AnswerCall);
+ if (error != ERR_NONE) return error;
+ return DCT3DCT4_AnswerCall(s,ID);
+ }
+
+ return DCT3_AnswerAllCalls(s);
+}
+
+static GSM_Error N6110_DialVoice(GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber)
+{
+ unsigned int pos = 4;
+ unsigned char req[100] = {N6110_FRAME_HEADER,0x01,
+ 0x0c}; /* Length of number */
+
+ if (ShowNumber == GSM_CALL_DefaultNumberPresence) return DCT3_DialVoice(s,number,ShowNumber);
+
+ req[pos++] = strlen(number);
+ memcpy(req+pos,number,strlen(number));
+ pos += strlen(number);
+ req[pos++] = 0x05; /* call type: voice - 0x05, data - 0x01 */
+ req[pos++] = 0x01;
+ req[pos++] = 0x01;
+ req[pos++] = 0x05;
+ req[pos++] = 0x81;
+ switch (ShowNumber) {
+ case GSM_CALL_HideNumber:
+ req[pos++] = 0x02;
+ break;
+ case GSM_CALL_ShowNumber:
+ req[pos++] = 0x03;
+ break;
+ case GSM_CALL_DefaultNumberPresence:
+ req[pos++] = 0x01;
+ break;
+ }
+ req[pos++] = 0x00;
+ req[pos++] = 0x00;
+
+ smprintf(s, "Making voice call\n");
+ return GSM_WaitFor (s, req, pos, 0x01, 4, ID_DialVoice);
+}
+
+GSM_Error N6110_UnholdCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x24, 0x00, 0x02};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Unholding call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_UnholdCall);
+}
+
+GSM_Error N6110_HoldCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x22, 0x00, 0x00};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Unholding call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_HoldCall);
+}
+
+/* Joining selected call to current (and making conference) */
+GSM_Error N6110_ConferenceCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x28, 0x00, 0x01};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Conference call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_ConferenceCall);
+}
+
+/* Removing selected call from conference and making private call with it
+ * (conference call is on hold) */
+GSM_Error N6110_SplitCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x2A, 0x00, 0x01};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Split call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_SplitCall);
+}
+
+/* This probably need more investigation */
+GSM_Error N6110_SwitchCall(GSM_StateMachine *s, int ID, bool next)
+{
+// unsigned char req[] = {N6110_FRAME_HEADER, 0x20}; calls info
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x26, 0x00};
+
+ s->Phone.Data.CallID = ID;
+
+ if (next) {
+ smprintf(s, "Switch call\n");
+ return GSM_WaitFor (s, req, 4, 0x01, 4, ID_SwitchCall);
+ } else {
+ req[4] = (unsigned char)ID;
+
+ smprintf(s, "Switch call\n");
+ return GSM_WaitFor (s, req, 5, 0x01, 4, ID_SwitchCall);
+ }
+}
+
+/* This probably need more investigation */
+GSM_Error N6110_TransferCall(GSM_StateMachine *s, int ID, bool next)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x2C, 0x00};
+
+ s->Phone.Data.CallID = ID;
+
+ if (next) {
+ smprintf(s, "Transfer call\n");
+ return GSM_WaitFor (s, req, 4, 0x01, 4, ID_TransferCall);
+ } else {
+ req[4] = (unsigned char)ID;
+
+ smprintf(s, "Transfer call\n");
+ return GSM_WaitFor (s, req, 5, 0x01, 4, ID_TransferCall);
+ }
+}
+
+static GSM_Reply_Function N6110ReplyFunctions[] = {
+ {N6110_ReplyCallInfo, "\x01",0x03,0x02,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x03,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x04,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x05,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x07,ID_AnswerCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x07,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x09,ID_CancelCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x09,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x0A,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x23,ID_HoldCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x23,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x25,ID_UnholdCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x25,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x27,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x29,ID_ConferenceCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x29,ID_IncomingFrame },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x2B,ID_SplitCall },
+ {N6110_ReplyCallInfo, "\x01",0x03,0x2B,ID_IncomingFrame },
+ {N6110_ReplySendDTMF, "\x01",0x03,0x40,ID_SendDTMF },
+ {NoneReply, "\x01",0x03,0x40,ID_DialVoice },
+ {NoneReply, "\x01",0x03,0x40,ID_IncomingFrame },
+ {NoneReply, "\x01",0x03,0x43,ID_AnswerCall },
+ {N6110_ReplySendDTMF, "\x01",0x03,0x51,ID_SendDTMF },
+
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x02,ID_IncomingFrame },
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x03,ID_IncomingFrame },
+ {N6110_ReplyIncomingSMS, "\x02",0x03,0x10,ID_IncomingFrame },
+#ifdef GSM_ENABLE_CELLBROADCAST
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x21,ID_SetIncomingCB },
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x22,ID_SetIncomingCB },
+ {DCT3_ReplyIncomingCB, "\x02",0x03,0x23,ID_IncomingFrame },
+#endif
+ {DCT3_ReplySetSMSC, "\x02",0x03,0x31,ID_SetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x34,ID_GetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x35,ID_GetSMSC },
+
+ {N6110_ReplyGetMemory, "\x03",0x03,0x02,ID_GetMemory },
+ {N6110_ReplyGetMemory, "\x03",0x03,0x03,ID_GetMemory },
+ {N6110_ReplySetMemory, "\x03",0x03,0x05,ID_SetMemory },
+ {N6110_ReplySetMemory, "\x03",0x03,0x06,ID_SetMemory },
+ {N6110_ReplyGetMemoryStatus, "\x03",0x03,0x08,ID_GetMemoryStatus },
+ {N6110_ReplyGetMemoryStatus, "\x03",0x03,0x09,ID_GetMemoryStatus },
+ {N6110_ReplyGetCallerLogo, "\x03",0x03,0x11,ID_GetBitmap },
+ {N6110_ReplyGetCallerLogo, "\x03",0x03,0x12,ID_GetBitmap },
+ {N6110_ReplySetCallerLogo, "\x03",0x03,0x14,ID_SetBitmap },
+ {N6110_ReplySetCallerLogo, "\x03",0x03,0x15,ID_SetBitmap },
+ {N6110_ReplyGetSpeedDial, "\x03",0x03,0x17,ID_GetSpeedDial },
+ {N6110_ReplyGetSpeedDial, "\x03",0x03,0x18,ID_GetSpeedDial },
+ /* 0x1A, 0x1B - reply set speed dial */
+
+ {N6110_ReplyGetStatus, "\x04",0x03,0x02,ID_GetSignalQuality },
+ {N6110_ReplyGetStatus, "\x04",0x03,0x02,ID_GetBatteryCharge },
+
+ {N6110_ReplySetProfileFeature, "\x05",0x03,0x11,ID_SetProfile },
+ {N6110_ReplySetProfileFeature, "\x05",0x03,0x12,ID_SetProfile },
+ {N6110_ReplyGetProfileFeature, "\x05",0x03,0x14,ID_GetProfile },
+ {N6110_ReplyGetPhoneLanguage, "\x05",0x03,0x14,ID_GetLanguage },
+ {N6110_ReplyGetProfileFeature, "\x05",0x03,0x15,ID_GetProfile },
+ {N6110_ReplyGetPhoneLanguage, "\x05",0x03,0x15,ID_GetLanguage },
+ {N6110_ReplyGetStartup, "\x05",0x03,0x17,ID_GetBitmap },
+ {N6110_ReplySetStartup, "\x05",0x03,0x19,ID_SetBitmap },
+ {N6110_ReplyGetProfileFeature, "\x05",0x03,0x1b,ID_GetProfile },
+ {N61_91_ReplySetOpLogo, "\x05",0x03,0x31,ID_SetBitmap },
+ {N61_91_ReplySetOpLogo, "\x05",0x03,0x32,ID_SetBitmap },
+ {N6110_ReplyGetOpLogo, "\x05",0x03,0x34,ID_GetBitmap },
+ {N6110_ReplySetRingtone, "\x05",0x03,0x37,ID_SetRingtone },
+ {N6110_ReplySetRingtone, "\x05",0x03,0x38,ID_SetRingtone },
+
+ {DCT3DCT4_ReplyCallDivert, "\x06",0x03,0x02,ID_Divert },
+ {DCT3DCT4_ReplyCallDivert, "\x06",0x03,0x03,ID_Divert },
+ {N6110_ReplyUSSDInfo, "\x06",0x03,0x05,ID_IncomingFrame },
+ {NoneReply, "\x06",0x03,0x06,ID_IncomingFrame },//incoming call divert info
+
+ {N6110_ReplyGetSecurityStatus, "\x08",0x03,0x08,ID_GetSecurityStatus },
+ {N6110_ReplyEnterSecurityCode, "\x08",0x03,0x0b,ID_EnterSecurityCode },
+ {N6110_ReplyEnterSecurityCode, "\x08",0x03,0x0c,ID_EnterSecurityCode },
+
+ {DCT3_ReplySIMLogin, "\x09",0x03,0x80,ID_IncomingFrame },
+ {DCT3_ReplySIMLogout, "\x09",0x03,0x81,ID_IncomingFrame },
+
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_GetNetworkInfo },
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_IncomingFrame },
+
+ {N6110_ReplyGetDisplayStatus, "\x0D",0x03,0x52,ID_GetDisplayStatus },
+ {N6110_ReplyGetDisplayStatus, "\x0D",0x03,0x52,ID_IncomingFrame },
+
+ {DCT3_ReplySetDateTime, "\x11",0x03,0x61,ID_SetDateTime },
+ {DCT3_ReplyGetDateTime, "\x11",0x03,0x63,ID_GetDateTime },
+ {DCT3_ReplySetAlarm, "\x11",0x03,0x6C,ID_SetAlarm },
+ {DCT3_ReplyGetAlarm, "\x11",0x03,0x6E,ID_GetAlarm },
+
+ {N6110_ReplyAddCalendar, "\x13",0x03,0x65,ID_SetCalendarNote },
+ {N6110_ReplyAddCalendar, "\x13",0x03,0x65,ID_IncomingFrame },
+ {N6110_ReplyGetNextCalendar, "\x13",0x03,0x67,ID_GetCalendarNote },
+ {N6110_ReplyDeleteCalendar, "\x13",0x03,0x69,ID_DeleteCalendarNote },
+ {N6110_ReplyDeleteCalendar, "\x13",0x03,0x69,ID_IncomingFrame },
+
+ {N6110_ReplySaveSMSMessage, "\x14",0x03,0x05,ID_SaveSMSMessage },
+ {N6110_ReplySaveSMSMessage, "\x14",0x03,0x06,ID_SaveSMSMessage },
+ {N6110_ReplyGetSMSMessage, "\x14",0x03,0x08,ID_GetSMSMessage },
+ {N6110_ReplyGetSMSMessage, "\x14",0x03,0x09,ID_GetSMSMessage },
+ {DCT3_ReplyDeleteSMSMessage, "\x14",0x03,0x0B,ID_DeleteSMSMessage },
+ {DCT3_ReplyDeleteSMSMessage, "\x14",0x03,0x0C,ID_DeleteSMSMessage },
+ {N6110_ReplyGetSMSStatus, "\x14",0x03,0x37,ID_GetSMSStatus },
+ {N6110_ReplyGetSMSStatus, "\x14",0x03,0x38,ID_GetSMSStatus },
+
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x01,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x02,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x04,ID_DisableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x05,ID_DisableConnectFunc },
+ {DCT3_ReplyGetWAPBookmark, "\x3f",0x03,0x07,ID_GetWAPBookmark },
+ {DCT3_ReplyGetWAPBookmark, "\x3f",0x03,0x08,ID_GetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0A,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0B,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0D,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0E,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyGetActiveConnectSet,"\x3f",0x03,0x10,ID_GetConnectSet },
+ {DCT3DCT4_ReplySetActiveConnectSet,"\x3f",0x03,0x13,ID_SetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x16,ID_GetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x17,ID_GetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x19,ID_SetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x1A,ID_SetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x1C,ID_GetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x1D,ID_GetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x1F,ID_SetConnectSet },
+
+ {DCT3_ReplyEnableSecurity, "\x40",0x02,0x64,ID_EnableSecurity },
+ {N61_71_ReplyResetPhoneSettings, "\x40",0x02,0x65,ID_ResetPhoneSettings },
+ {DCT3_ReplyGetIMEI, "\x40",0x02,0x66,ID_GetIMEI },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_DialVoice },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_CancelCall },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_AnswerCall },
+ {DCT3_ReplyNetmonitor, "\x40",0x02,0x7E,ID_Netmonitor },
+ {DCT3_ReplyPlayTone, "\x40",0x02,0x8F,ID_PlayTone },
+ {N6110_ReplyGetRingtone, "\x40",0x02,0x9E,ID_GetRingtone },
+ {N6110_ReplySetBinRingtone, "\x40",0x02,0xA0,ID_SetRingtone },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetHardware },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetPPM },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCA,ID_GetProductCode },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetManufactureMonth},
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetOriginalIMEI },
+
+ {N6110_ReplyGetSetPicture, "\x47",0x03,0x02,ID_GetBitmap },
+ {N6110_ReplyGetSetPicture, "\x47",0x03,0x04,ID_SetBitmap },
+ {N6110_ReplyGetSetPicture, "\x47",0x03,0x05,ID_SetBitmap },
+ {N6110_ReplyGetSetPicture, "\x47",0x03,0x06,ID_GetBitmap },
+
+#ifndef ENABLE_LGPL
+ {N6110_ReplyGetMagicBytes, "\x64",0x00,0x00,ID_MakeAuthentication },
+#endif
+
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetFirmware },
+ {DCT3_ReplyPressKey, "\xD2",0x02,0x46,ID_PressKey },
+ {DCT3_ReplyPressKey, "\xD2",0x02,0x47,ID_PressKey },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N6110Phone = {
+ "2100|3210|3310|3330|3390|3410|3610|5110|5110i|5130|5190|5210|5510|6110|6130|6150|6190|8210|8250|8290|8850|8855|8890",
+ N6110ReplyFunctions,
+ N6110_Initialise,
+ PHONE_Terminate,
+ GSM_DispatchMessage,
+ N6110_ShowStartInfo,
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ DCT3_GetIMEI,
+ DCT3_GetOriginalIMEI,
+ DCT3_GetManufactureMonth,
+ DCT3_GetProductCode,
+ DCT3_GetHardware,
+ DCT3_GetPPM,
+ NOTSUPPORTED, /* GetSIMIMSI */
+ N6110_GetDateTime,
+ N6110_SetDateTime,
+ N6110_GetAlarm,
+ N6110_SetAlarm,
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ DCT3_PressKey,
+ DCT3_Reset,
+ N61_71_ResetPhoneSettings,
+ N6110_EnterSecurityCode,
+ N6110_GetSecurityStatus,
+ N6110_GetDisplayStatus,
+ NOTIMPLEMENTED, /* SetAutoNetworkLogin */
+ N6110_GetBatteryCharge,
+ N6110_GetSignalQuality,
+ DCT3_GetNetworkInfo,
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ N6110_GetMemoryStatus,
+ N6110_GetMemory,
+ NOTIMPLEMENTED, /* GetNextMemory */
+ N6110_SetMemory,
+ NOTIMPLEMENTED, /* AddMemory */
+ N6110_DeleteMemory,
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ N6110_GetSpeedDial,
+ NOTIMPLEMENTED, /* SetSpeedDial */
+ DCT3_GetSMSC,
+ DCT3_SetSMSC,
+ DCT3_GetSMSStatus,
+ N6110_GetSMSMessage,
+ N6110_GetNextSMSMessage,
+ N6110_SetSMS,
+ N6110_AddSMS,
+ N6110_DeleteSMSMessage,
+ DCT3_SendSMSMessage,
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOKIA_SetIncomingSMS,
+ DCT3_SetIncomingCB,
+ PHONE_GetSMSFolders,
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ N6110_DialVoice,
+ N6110_AnswerCall,
+ DCT3_CancelCall,
+ N6110_HoldCall,
+ N6110_UnholdCall,
+ N6110_ConferenceCall,
+ N6110_SplitCall,
+ N6110_TransferCall,
+ N6110_SwitchCall,
+ DCT3DCT4_GetCallDivert,
+ DCT3DCT4_SetCallDivert,
+ DCT3DCT4_CancelAllDiverts,
+ NOKIA_SetIncomingCall,
+ NOKIA_SetIncomingUSSD,
+ DCT3DCT4_SendDTMF,
+ N6110_GetRingtone,
+ N6110_SetRingtone,
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ DCT3_PlayTone,
+ DCT3_GetWAPBookmark,
+ DCT3_SetWAPBookmark,
+ DCT3_DeleteWAPBookmark,
+ DCT3_GetWAPSettings,
+ DCT3_SetWAPSettings,
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ N6110_GetBitmap,
+ N6110_SetBitmap,
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ NOTIMPLEMENTED, /* GetCalendarStatus */
+ NOTIMPLEMENTED, /* GetCalendar */
+ N6110_GetNextCalendarNote,
+ NOTIMPLEMENTED, /* SetCalendar */
+ N6110_AddCalendarNote,
+ N6110_DeleteCalendarNote,
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ N6110_GetProfile,
+ N6110_SetProfile,
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFile */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n6110.h b/gammu/emb/common/phone/nokia/dct3/n6110.h
new file mode 100644
index 0000000..d243766
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n6110.h
@@ -0,0 +1,45 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef n6110_h
+#define n6110_h
+
+#include "../../../config.h"
+#include "../../../service/sms/gsmsms.h"
+#include "dct3comm.h"
+
+typedef struct {
+#ifndef ENABLE_LGPL
+ unsigned char MagicBytes[4];
+#endif
+ int LastCalendarPos;
+ DCT3_WAPSettings_Locations WAPLocations;
+
+ GSM_SMSMemoryStatus LastSMSStatus;
+ int LastSMSRead;
+
+ int PhoneLanguage;
+} GSM_Phone_N6110Data;
+
+typedef enum {
+ N6110_Auto = 1,
+ N6110_Europe
+} N6110_Language;
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+#ifndef GSM_USED_FBUS2IRDA
+# define GSM_USED_FBUS2IRDA
+#endif
+#ifndef GSM_USED_IRDAPHONET
+# define GSM_USED_IRDAPHONET
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n7110.c b/gammu/emb/common/phone/nokia/dct3/n7110.c
new file mode 100644
index 0000000..5a02c9c
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n7110.c
@@ -0,0 +1,1724 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Markus Plail and Gnokii */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA7110
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../misc/coding/coding.h"
+#include "../../../gsmcomon.h"
+#include "../../../service/gsmlogo.h"
+#include "../../pfunc.h"
+#include "../nfunc.h"
+#include "../nfuncold.h"
+#include "n7110.h"
+#include "dct3func.h"
+
+static GSM_Error N7110_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ return DCT3_GetAlarm(s, alarm, 0x19);
+}
+
+static GSM_Error N7110_SetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ return DCT3_SetAlarm(s, alarm, 0x19);
+}
+
+static GSM_Error N7110_ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Phonebook entry received\n");
+ switch (msg.Buffer[6]) {
+ case 0x0f:
+ return N71_65_ReplyGetMemoryError(msg.Buffer[10], s);
+ default:
+ return N71_65_DecodePhonebook(s, Data->Memory,Data->Bitmap,Data->SpeedDial,msg.Buffer+18,msg.Length-18,false);
+ }
+ return ERR_UNKNOWN;
+}
+
+static GSM_Error N7110_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x07, 0x01, 0x01, 0x00, 0x01,
+ 0x02, /* memory type */
+ 0x05,
+ 0x00, 0x00, /* location */
+ 0x00, 0x00};
+
+ req[9] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[9]==0xff) return ERR_NOTSUPPORTED;
+
+ if (entry->Location==0x00) return ERR_INVALIDLOCATION;
+
+ req[10] = entry->Location / 256;
+ req[11] = entry->Location % 256;
+
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, 14, 0x03, 4, ID_GetMemory);
+}
+
+static GSM_Error N7110_ReplyGetMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Memory status received\n");
+ /* Quess ;-)) */
+ if (msg.Buffer[10]==0x10) {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[14]*256 + msg.Buffer[15];
+ } else {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[18];
+ }
+ smprintf(s, " Size : %i\n",Data->MemoryStatus->MemoryFree);
+ Data->MemoryStatus->MemoryUsed = msg.Buffer[16]*256 + msg.Buffer[17];
+ smprintf(s, " Used : %i\n",Data->MemoryStatus->MemoryUsed);
+ Data->MemoryStatus->MemoryFree -= Data->MemoryStatus->MemoryUsed;
+ smprintf(s, " Free : %i\n",Data->MemoryStatus->MemoryFree);
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x03, 0x02,
+ 0x05}; /* Memory type */
+
+ req[5] = NOKIA_GetMemoryType(s, Status->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[5]==0xff) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.MemoryStatus=Status;
+ smprintf(s, "Getting memory status\n");
+ return GSM_WaitFor (s, req, 6, 0x03, 4, ID_GetMemoryStatus);
+}
+
+static void N7110_GetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char *folderid, int *location)
+{
+ int ifolderid;
+
+ /* simulate flat SMS memory */
+ if (sms->Folder==0x00) {
+ ifolderid = sms->Location / PHONE_MAXSMSINFOLDER;
+ *folderid = (ifolderid + 1) * 0x08;
+ *location = sms->Location - ifolderid * PHONE_MAXSMSINFOLDER;
+ } else {
+ *folderid = sms->Folder * 0x08;
+ *location = sms->Location;
+ }
+ smprintf(s, "SMS folder %i & location %i -> 7110 folder %i & location %i\n",
+ sms->Folder,sms->Location,*folderid,*location);
+}
+
+static void N7110_SetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char folderid, int location)
+{
+ sms->Folder = 0;
+ sms->Location = (folderid / 0x08 - 1) * PHONE_MAXSMSINFOLDER + location;
+ smprintf(s, "7110 folder %i & location %i -> SMS folder %i & location %i\n",
+ folderid,location,sms->Folder,sms->Location);
+}
+
+static GSM_Error N7110_ReplyGetSMSFolders(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int j,current=5;
+ unsigned char buffer[200];
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x7B:
+ smprintf(s, "Names for SMS folders received\n");
+ Data->SMSFolders->Number=msg.Buffer[4];
+ for (j=0;j<msg.Buffer[4];j++) {
+ smprintf(s, "Folder index: %02x",msg.Buffer[current]);
+ current++;
+ smprintf(s, ", folder name: \"");
+ CopyUnicodeString(buffer,msg.Buffer+current);
+ if ((UnicodeLength(buffer))>GSM_MAX_SMS_FOLDER_NAME_LEN) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ CopyUnicodeString(Data->SMSFolders->Folder[j].Name,buffer);
+ smprintf(s, "%s\"\n",DecodeUnicodeString(buffer));
+ current=current+2+UnicodeLength(buffer)*2;
+ Data->SMSFolders->Folder[j].InboxFolder = false;
+ if (j==0) Data->SMSFolders->Folder[j].InboxFolder = true;
+ Data->SMSFolders->Folder[j].Memory = MEM_ME;
+ if (j==0 || j==1) Data->SMSFolders->Folder[j].InboxFolder = MEM_MT;
+ }
+ return ERR_NONE;
+ case 0x7C:
+ smprintf(s, "Security error ? No PIN ?\n");
+ return ERR_SECURITYERROR;
+ case 0xCA:
+ smprintf(s, "Wait a moment. Phone is during power on and busy now\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_GetSMSFolders(GSM_StateMachine *s, GSM_SMSFolders *folders)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x7A, 0x00, 0x00};
+
+ s->Phone.Data.SMSFolders=folders;
+ smprintf(s, "Getting SMS folders\n");
+ return GSM_WaitFor (s, req, 6, 0x14, 4, ID_GetSMSFolders);
+}
+
+static GSM_Error N7110_ReplyGetSMSFolderStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+
+ smprintf(s, "SMS folder status received\n");
+ Priv->LastSMSFolder.Number=msg.Buffer[4]*256+msg.Buffer[5];
+ smprintf(s, "Number of Entries: %i\n",Priv->LastSMSFolder.Number);
+ smprintf(s, "Locations: ");
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ Priv->LastSMSFolder.Location[i]=msg.Buffer[6+(i*2)]*256+msg.Buffer[(i*2)+7];
+ if (Priv->LastSMSFolder.Location[i] > PHONE_MAXSMSINFOLDER) {
+ smprintf(s, "Increase PHONE_MAXSMSINFOLDER\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ smprintf(s, "%i ",Priv->LastSMSFolder.Location[i]);
+ }
+ smprintf(s, "\n");
+ NOKIA_SortSMSFolderStatus(s, &Priv->LastSMSFolder);
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_PrivGetSMSFolderStatus(GSM_StateMachine *s, int folderid)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x6b,
+ 0x08, /* folderID */
+ 0x0F, 0x01};
+
+ req[4] = folderid;
+
+ smprintf(s, "Getting SMS folder status\n");
+ return GSM_WaitFor (s, req, 7, 0x14, 4, ID_GetSMSFolderStatus);
+}
+
+static GSM_Error N7110_GetSMSFolderStatus(GSM_StateMachine *s, int folderid)
+{
+ GSM_Error error;
+ int i;
+ GSM_NOKIASMSFolder folder;
+
+ error = N7110_PrivGetSMSFolderStatus(s,folderid);
+ /* 0x08 contais read Inbox, 0xf8 unread Inbox.
+ * we want all msg from Inbox, so read both 0x08 and 0xf8 */
+ if (folderid==0x08 && error==ERR_NONE) {
+ folder=s->Phone.Data.Priv.N7110.LastSMSFolder;
+ error = N7110_PrivGetSMSFolderStatus(s,0xf8);
+ if (error==ERR_NONE) {
+ for (i=0;i<folder.Number;i++) {
+ s->Phone.Data.Priv.N7110.LastSMSFolder.Location[s->Phone.Data.Priv.N7110.LastSMSFolder.Number++]=folder.Location[i];
+ }
+ }
+ }
+ return error;
+}
+
+static GSM_SMSMessageLayout N7110_SMSTemplate = {
+ 36 /* SMS Text */, 17 /* Phone number */,
+ 255 /* SMSC Number */, 15 /* TPDCS */,
+ 255 /* SendingDateTime */, 255 /* SMSCDateTime */,
+ 255 /* TPStatus */, 16 /* TPUDL */,
+ 255 /* TPVP */, 12 /* firstbyte */,
+ 13 /* TPMR */, 255 /* TPPID?? */};
+
+static GSM_Error N7110_ReplyGetSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ int Width, Height;
+ unsigned char output[500], output2[500];
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch(msg.Buffer[3]) {
+ case 0x08:
+ switch (msg.Buffer[8]) {
+ case 0x00:
+ case 0x01:
+ smprintf(s, "SMS message\n");
+ if (Data->RequestID == ID_GetSMSMessage) {
+ Data->GetSMSMessage->Number=1;
+ NOKIA_DecodeSMSState(s, msg.Buffer[4], &Data->GetSMSMessage->SMS[0]);
+ DCT3_DecodeSMSFrame(s, &Data->GetSMSMessage->SMS[0],msg.Buffer+9);
+ return ERR_NONE;
+ }
+ case 0x02:
+ smprintf(s, "SMS template\n");
+ if (Data->RequestID == ID_GetSMSMessage) {
+ Data->GetSMSMessage->Number=1;
+ NOKIA_DecodeSMSState(s, msg.Buffer[4], &Data->GetSMSMessage->SMS[0]);
+ Data->GetSMSMessage->SMS[0].PDU=SMS_Submit;
+ GSM_DecodeSMSFrame(&Data->GetSMSMessage->SMS[0],msg.Buffer+9,N7110_SMSTemplate);
+ return ERR_NONE;
+ }
+ case 0x07:
+ smprintf(s, "Picture Image\n");
+ switch (Data->RequestID) {
+ case ID_GetBitmap:
+ PHONE_GetBitmapWidthHeight(GSM_NokiaPictureImage, &Width, &Height);
+ Data->Bitmap->BitmapWidth = Width;
+ Data->Bitmap->BitmapHeight = Height;
+ PHONE_DecodeBitmap(GSM_NokiaPictureImage, msg.Buffer + 51, Data->Bitmap);
+ GSM_UnpackSemiOctetNumber(Data->Bitmap->Sender,msg.Buffer+22,true);
+#ifdef DEBUG
+ GSM_UnpackSemiOctetNumber(output,msg.Buffer+9,true);
+ smprintf(s, "SMSC : %s\n",DecodeUnicodeString(output));
+#endif
+ Data->Bitmap->Text[0] = 0;
+ Data->Bitmap->Text[1] = 0;
+ if (msg.Length!=304) {
+ GSM_UnpackEightBitsToSeven(0, msg.Length-304, msg.Length-304, msg.Buffer+52+PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0),output);
+ DecodeDefault(Data->Bitmap->Text, output, msg.Length - 304, true, NULL);
+ }
+ return ERR_NONE;
+ case ID_GetSMSMessage:
+ Data->GetSMSMessage->Number = 0;
+ i = 0;
+ output[i++] = 0x30; /* Smart Messaging 3.0 */
+ output[i++] = SM30_OTA;
+ output[i++] = 0x01; /* Length */
+ output[i++] = 0x00; /* Length */
+ output[i++] = 0x00;
+ PHONE_GetBitmapWidthHeight(GSM_NokiaPictureImage, &Width, &Height);
+ output[i++] = Width;
+ output[i++] = Height;
+ output[i++] = 0x01;
+ memcpy(output+i,msg.Buffer+51,PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0));
+ i = i + PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0);
+ if (msg.Length!=304) {
+ output[i++] = SM30_UNICODETEXT;
+ output[i++] = 0;
+ output[i++] = 0; /* Length - later changed */
+ GSM_UnpackEightBitsToSeven(0, msg.Length-304, msg.Length-304, msg.Buffer+52+PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0),output2);
+ DecodeDefault(output+i, output2, msg.Length - 304, true, NULL);
+ output[i - 1] = UnicodeLength(output+i) * 2;
+ i = i + output[i-1];
+ }
+ GSM_MakeMultiPartSMS(Data->GetSMSMessage,output,i,UDH_NokiaProfileLong,SMS_Coding_8bit,1,0);
+ for (i=0;i<3;i++) {
+ Data->GetSMSMessage->SMS[i].Number[0]=0;
+ Data->GetSMSMessage->SMS[i].Number[1]=0;
+ }
+ return ERR_NONE;
+ default:
+ smprintf(s, "Unknown SMS type: %i\n",msg.Buffer[8]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ default:
+ smprintf(s, "Unknown SMS type: %i\n",msg.Buffer[8]);
+ }
+ break;
+ case 0x09:
+ switch (msg.Buffer[4]) {
+ case 0x02:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ case 0x07:
+ smprintf(s, "Empty\n");
+ return ERR_EMPTY;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ case 0x6F:
+ smprintf(s, "SMS message info received\n");
+ if (msg.Length == 43) {
+ Data->GetSMSMessage->SMS[0].Name[0] = 0;
+ Data->GetSMSMessage->SMS[0].Name[1] = 0;
+ } else {
+ CopyUnicodeString(Data->GetSMSMessage->SMS[0].Name,msg.Buffer+43);
+ }
+ smprintf(s, "Name: \"%s\"\n",DecodeUnicodeString(Data->GetSMSMessage->SMS[0].Name));
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_PrivGetSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ GSM_Error error;
+ unsigned char folderid;
+ int location;
+ int i;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07,
+ 0x08, /* folder ID */
+ 0x00, 0x05, /* location */
+ 0x01, 0x65, 0x01};
+ unsigned char NameReq[] = {N6110_FRAME_HEADER, 0x6E,
+ 0x08, /* folder ID */
+ 0x00, 0x05}; /* location */
+
+ N7110_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+
+ req[4]=folderid;
+ req[5]=location / 256;
+ req[6]=location;
+
+ s->Phone.Data.GetSMSMessage=sms;
+ smprintf(s, "Getting sms\n");
+ error=GSM_WaitFor (s, req, 10, 0x14, 4, ID_GetSMSMessage);
+ if (error==ERR_NONE) {
+ NameReq[4] = folderid;
+ NameReq[5] = location / 256;
+ NameReq[6] = location;
+ smprintf(s, "Getting sms info\n");
+ error=GSM_WaitFor (s, NameReq, 7, 0x14, 4, ID_GetSMSMessage);
+ if (error != ERR_NONE) return error;
+ for (i=0;i<sms->Number;i++) {
+ N7110_SetSMSLocation(s, &sms->SMS[i], folderid, location);
+ sms->SMS[i].Folder = folderid/0x08;
+ sms->SMS[i].InboxFolder = true;
+ if (folderid/0x08 != 0x01) sms->SMS[i].InboxFolder = false;
+ CopyUnicodeString(sms->SMS[i].Name,sms->SMS[0].Name);
+ sms->SMS[i].Memory = MEM_ME;
+ if (folderid/0x08 == 0x01 || folderid/0x08 == 0x02) {
+ sms->SMS[i].Memory = MEM_MT;
+ if (folderid/0x08 == 0x01) { /* Inbox */
+ if (sms->SMS[i].State == SMS_Sent) sms->SMS[i].Memory = MEM_ME;
+ if (sms->SMS[i].State == SMS_UnSent) sms->SMS[i].Memory = MEM_ME;
+ if (sms->SMS[i].State == SMS_Read) sms->SMS[i].Memory = MEM_SM;
+ if (sms->SMS[i].State == SMS_UnRead) sms->SMS[i].Memory = MEM_SM;
+ }
+ if (folderid/0x08 == 0x02) { /* Outbox */
+ if (sms->SMS[i].State == SMS_Sent) sms->SMS[i].Memory = MEM_SM;
+ if (sms->SMS[i].State == SMS_UnSent) sms->SMS[i].Memory = MEM_SM;
+ if (sms->SMS[i].State == SMS_Read) sms->SMS[i].Memory = MEM_ME;
+ if (sms->SMS[i].State == SMS_UnRead) sms->SMS[i].Memory = MEM_ME;
+ }
+ }
+ }
+ }
+ return error;
+}
+
+static GSM_Error N7110_GetSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ GSM_Error error;
+ unsigned char folderid;
+ int location;
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+ int i;
+ bool found = false;
+
+ N7110_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+ error=N7110_GetSMSFolderStatus(s, folderid);
+ if (error!=ERR_NONE) return error;
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ if (Priv->LastSMSFolder.Location[i]==location) {
+ found = true;
+ break;
+ }
+ }
+ if (!found) return ERR_EMPTY;
+ return N7110_PrivGetSMSMessage(s,sms);
+}
+
+static GSM_Error N7110_GetNextSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start)
+{
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+ unsigned char folderid;
+ int location;
+ GSM_Error error;
+ int i;
+ bool findnextfolder = false;
+
+ if (start) {
+ folderid=0x00;
+ findnextfolder=true;
+ error=N7110_GetSMSFolders(s,&Priv->LastSMSFolders);
+ if (error!=ERR_NONE) return error;
+ } else {
+ N7110_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ if (Priv->LastSMSFolder.Location[i]==location) break;
+ }
+ /* Is this last location in this folder ? */
+ if (i==Priv->LastSMSFolder.Number-1) {
+ findnextfolder=true;
+ } else {
+ location=Priv->LastSMSFolder.Location[i+1];
+ }
+ }
+ if (findnextfolder) {
+ Priv->LastSMSFolder.Number=0;
+ while (Priv->LastSMSFolder.Number==0) {
+ folderid=folderid+0x08;
+ /* Too high folder number */
+ if ((folderid/0x08)>Priv->LastSMSFolders.Number) return ERR_EMPTY;
+ /* Get next folder status */
+ error=N7110_GetSMSFolderStatus(s, folderid);
+ if (error!=ERR_NONE) return error;
+ /* First location from this folder */
+ location=Priv->LastSMSFolder.Location[0];
+ }
+ }
+ N7110_SetSMSLocation(s, &sms->SMS[0], folderid, location);
+
+ return N7110_PrivGetSMSMessage(s, sms);
+}
+
+static int N7110_ReturnBinaryRingtoneLocation(char *model)
+{
+ if (strcmp(model,"NSE-5") == 0) return 0x72; /* first 0x72 - 7110 */
+ if (strcmp(model,"NPE-3") == 0) return 0x89; /* first 0x89 - 6210 */
+ if (strcmp(model,"NHM-3") == 0) return 0x89; /* quess for 6250 */
+ return 0;
+}
+
+static GSM_Error N7110_ReplyGetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int tmp,i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Ringtone received\n");
+ switch (msg.Buffer[3]) {
+ case 0x23:
+ tmp=0;i=4;
+ while (msg.Buffer[i]!=0 || msg.Buffer[i+1]!=0) {
+ tmp++;
+ i=i+2;
+ if (i>msg.Length) return ERR_EMPTY;
+ }
+ memcpy(Data->Ringtone->Name,msg.Buffer+6,tmp*2);
+ smprintf(s, "Name \"%s\"\n",DecodeUnicodeString(Data->Ringtone->Name));
+ /* Looking for end */
+ i=37;
+ while (true) {
+ if (msg.Buffer[i]==0x07 && msg.Buffer[i+1]==0x0b) {
+ i=i+2; break;
+ }
+ if (msg.Buffer[i]==0x0e && msg.Buffer[i+1]==0x0b) {
+ i=i+2; break;
+ }
+ i++;
+ if (i==msg.Length) return ERR_EMPTY;
+ }
+ /* Copying frame */
+ memcpy(Data->Ringtone->NokiaBinary.Frame,msg.Buffer+37,i-37);
+ Data->Ringtone->NokiaBinary.Length=i-37;
+ return ERR_NONE;
+ case 0x24:
+ smprintf(s, "Invalid location. Too high ?\n");
+ return ERR_INVALIDLOCATION;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_GetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x22, 0x00, 0x00};
+
+ if (PhoneRingtone) return ERR_NOTSUPPORTED;
+ if (Ringtone->Format == 0x00) Ringtone->Format = RING_NOKIABINARY;
+
+ switch (Ringtone->Format) {
+ case RING_NOTETONE:
+ /* In the future get binary and convert */
+ return ERR_NOTSUPPORTED;
+ case RING_NOKIABINARY:
+ req[5]=N7110_ReturnBinaryRingtoneLocation(s->Phone.Data.Model)+Ringtone->Location;
+ s->Phone.Data.Ringtone=Ringtone;
+ smprintf(s, "Getting binary ringtone\n");
+ return GSM_WaitFor (s, req, 6, 0x1f, 4, ID_GetRingtone);
+ case RING_MIDI:
+ return ERR_NOTSUPPORTED;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N7110_ReplyGetPictureImageInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+
+ smprintf(s, "Received info for Picture Images\n");
+ smprintf(s, "Number : %i\n",msg.Buffer[4]*256+msg.Buffer[5]);
+ smprintf(s, "Locations :");
+ Priv->LastPictureImageFolder.Number=msg.Buffer[4]*256+msg.Buffer[5];
+ for (i=0;i<Priv->LastPictureImageFolder.Number;i++) {
+ Priv->LastPictureImageFolder.Location[i]=msg.Buffer[6+i*2]*256+msg.Buffer[7+i*2];
+ smprintf(s, " %i",Priv->LastPictureImageFolder.Location[i]);
+ }
+ smprintf(s, "\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_GetPictureImageLocation(GSM_StateMachine *s, GSM_Bitmap *Bitmap, unsigned char *folder, int *location)
+{
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+ GSM_SMSFolders folders;
+ GSM_Error error;
+ int i, j = 0, count = 0;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x96,
+ 0x00, /* Folder ID */
+ 0x0f, 0x07};
+
+ error=N7110_GetSMSFolders (s, &folders);
+ if (error!=ERR_NONE) return error;
+
+ for (i=0;i<folders.Number;i++) {
+ req[4] = (i+1) * 0x08; /* SMS folder ID */
+ error = GSM_WaitFor (s, req, 7, 0x14, 4, ID_GetBitmap);
+ if (error!=ERR_NONE) return error;
+ for (j=0;j<Priv->LastPictureImageFolder.Number;j++) {
+ count++;
+ if (count==Bitmap->Location) break;
+ }
+ if (count==Bitmap->Location) break;
+ }
+ if (count!=Bitmap->Location) return ERR_INVALIDLOCATION;
+ *folder = (i+1) * 0x08; /* SMS Folder ID */
+ *location = Priv->LastPictureImageFolder.Location[j];
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_GetPictureImage(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char folder;
+ int location;
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07,
+ 0x00, /* Folder ID */
+ 0x00, 0x00, /* Location */
+ 0x00, 0x64};
+
+ error = N7110_GetPictureImageLocation(s, Bitmap, &folder, &location);
+ switch (error) {
+ case ERR_NONE:
+ req[4] = folder;
+ req[5] = location / 256;
+ req[6] = location % 256;
+ return GSM_WaitFor (s, req, 9, 0x14, 4, ID_GetBitmap);
+ default:
+ return error;
+ }
+}
+
+static GSM_Error N7110_GetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ GSM_MemoryEntry pbk;
+ GSM_Error error;
+ unsigned char OpReq[] = {N6110_FRAME_HEADER, 0x70};
+
+ s->Phone.Data.Bitmap=Bitmap;
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ smprintf(s, "Getting startup logo\n");
+ return N71_92_GetPhoneSetting(s, ID_GetBitmap, 0x15);
+ case GSM_WelcomeNote_Text:
+ smprintf(s, "Getting welcome note\n");
+ return N71_92_GetPhoneSetting(s, ID_GetBitmap, 0x02);
+ case GSM_DealerNote_Text:
+ smprintf(s, "Getting dealer note\n");
+ return N71_92_GetPhoneSetting(s, ID_GetBitmap, 0x17);
+ case GSM_CallerGroupLogo:
+ pbk.MemoryType = MEM7110_CG;
+ pbk.Location = Bitmap->Location;
+ smprintf(s, "Getting caller group logo\n");
+ error=N7110_GetMemory(s,&pbk);
+ if (error==ERR_NONE) NOKIA_GetDefaultCallerGroupName(s,Bitmap);
+ return error;
+ case GSM_OperatorLogo:
+ smprintf(s, "Getting operator logo\n");
+ /* This is like DCT3_GetNetworkInfo */
+ return GSM_WaitFor (s, OpReq, 4, 0x0a, 4, ID_GetBitmap);
+ case GSM_PictureImage:
+ /* 7110 doesn't support it */
+ if (strcmp(s->Phone.Data.Model,"NSE-5") == 0) return ERR_NOTSUPPORTED;
+ return N7110_GetPictureImage(s, Bitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N7110_SetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength)
+{
+ GSM_Ringtone dest;
+ GSM_Error error;
+ GSM_NetworkInfo NetInfo;
+ int size=200;
+ unsigned char req[1000] = {0x7C, 0x01, 0x00, 0x0D, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00}; /*Length*/
+ unsigned char req2[4000] = {N7110_FRAME_HEADER, 0x1F, 0x00,
+ 0x87, /* Location */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ switch (Ringtone->Format) {
+ case RING_NOTETONE:
+ if (Ringtone->Location==255) {
+ /* 7110 doesn't support it */
+ if (strcmp(s->Phone.Data.Model,"NSE-5") == 0) return ERR_NOTSUPPORTED;
+ *maxlength=GSM_EncodeNokiaRTTLRingtone(*Ringtone, req+11, &size);
+ req[10] = size;
+ error = s->Protocol.Functions->WriteMessage(s, req, size+11, 0x00);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+ /* We have to make something (not important, what) now */
+ /* no answer from phone*/
+ return DCT3_GetNetworkInfo(s,&NetInfo);
+ }
+ GSM_RingtoneConvert(&dest, Ringtone, RING_NOKIABINARY);
+ break;
+ case RING_NOKIABINARY:
+ memcpy(&dest,Ringtone,sizeof(GSM_Ringtone));
+ break;
+ default:
+ return ERR_NOTSUPPORTED;
+ }
+ req2[5]=N7110_ReturnBinaryRingtoneLocation(s->Phone.Data.Model)+Ringtone->Location;
+ CopyUnicodeString(req2+6,Ringtone->Name);
+ memcpy(req2+37,dest.NokiaBinary.Frame,dest.NokiaBinary.Length);
+ error = s->Protocol.Functions->WriteMessage(s, req2, 37+dest.NokiaBinary.Length, 0x1F);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+ /* We have to make something (not important, what) now */
+ /* no answer from phone*/
+ return DCT3_GetNetworkInfo(s,&NetInfo);
+}
+
+static GSM_Error N7110_ReplySaveSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x05:
+ smprintf(s, "SMS message saving status\n");
+ smprintf(s, "Saved in folder %i at location %i\n",msg.Buffer[4], msg.Buffer[5]*256+msg.Buffer[6]);
+ if (msg.Buffer[4] == 0xf8) {
+ N7110_SetSMSLocation(s, Data->SaveSMSMessage,0x08,msg.Buffer[5]*256+msg.Buffer[6]);
+ Data->SaveSMSMessage->Folder = 0x01;
+ } else {
+ N7110_SetSMSLocation(s, Data->SaveSMSMessage,msg.Buffer[4],msg.Buffer[5]*256+msg.Buffer[6]);
+ Data->SaveSMSMessage->Folder = msg.Buffer[4] / 0x08;
+ }
+ return ERR_NONE;
+ case 0x06:
+ smprintf(s, "SMS message saving status\n");
+ switch (msg.Buffer[4]) {
+ case 0x03:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ break;
+ case 0x84:
+ smprintf(s, "Name for SMS changed OK to \"%s\"\n",DecodeUnicodeString(msg.Buffer+7));
+ smprintf(s, "Saved in folder %i at location %i\n",msg.Buffer[4], msg.Buffer[5]*256+msg.Buffer[6]);
+ if (msg.Buffer[4] == 0xf8) {
+ N7110_SetSMSLocation(s, Data->SaveSMSMessage,0x08,msg.Buffer[5]*256+msg.Buffer[6]);
+ Data->SaveSMSMessage->Folder = 0x01;
+ } else {
+ N7110_SetSMSLocation(s, Data->SaveSMSMessage,msg.Buffer[4],msg.Buffer[5]*256+msg.Buffer[6]);
+ Data->SaveSMSMessage->Folder = msg.Buffer[4] / 0x08;
+ }
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_PrivSetSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int length, location;
+ unsigned char folderid, folder;
+ GSM_Error error;
+ unsigned char req[256] = {N6110_FRAME_HEADER, 0x04,
+ 0x03, /* sms status */
+ 0x10, /* folder */
+ 0x00,0x00, /* location */
+ 0x00};
+ unsigned char NameReq[200] = {N6110_FRAME_HEADER, 0x83};
+
+ switch (sms->State) {
+ case SMS_Read : req[4] = 0x01; break;
+ case SMS_UnRead : req[4] = 0x03; break;
+ case SMS_Sent : req[4] = 0x05; break;
+ case SMS_UnSent : req[4] = 0x07; break;
+ }
+
+ N7110_GetSMSLocation(s, sms, &folderid, &location);
+ req[5] = folderid;
+ req[6] = location / 256;
+ req[7] = location;
+
+ /* Outbox */
+ if (folderid == 0x10 && (sms->State == SMS_Sent || sms->State == SMS_UnSent)) {
+ /* We will use SIM Outbox */
+ sms->PDU = SMS_Submit;
+ }
+ /* Inbox */
+ if (folderid == 0x08 && sms->State == SMS_UnRead) {
+ /* We will use SIM Inbox */
+ req[5] = 0xf8;
+ }
+
+ switch (sms->PDU) {
+ case SMS_Deliver:
+ error = PHONE_EncodeSMSFrame(s,sms,req+9,PHONE_SMSDeliver,&length,true);
+ break;
+ case SMS_Submit:
+ smprintf(s, "Saving SMS template\n");
+ error = PHONE_EncodeSMSFrame(s,sms,req+9,N7110_SMSTemplate,&length,true);
+ req[8] = 0x02; /* SMS Template info */
+ break;
+ default:
+ return ERR_UNKNOWN;
+ }
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.SaveSMSMessage=sms;
+ smprintf(s, "Saving sms\n");
+ error=GSM_WaitFor (s, req, 9+length, 0x14, 4, ID_SaveSMSMessage);
+ if (error == ERR_NONE && UnicodeLength(sms->Name)!=0) {
+ folder = sms->Folder;
+ sms->Folder = 0;
+ N7110_GetSMSLocation(s, sms, &folderid, &location);
+ length = 4;
+ NameReq[length++] = folderid;
+ NameReq[length++] = location / 256;
+ NameReq[length++] = location;
+ CopyUnicodeString(NameReq+length, sms->Name);
+ length = length+UnicodeLength(sms->Name)*2;
+ NameReq[length++] = 0;
+ NameReq[length++] = 0;
+ error=GSM_WaitFor (s, NameReq, length, 0x14, 4, ID_SaveSMSMessage);
+ sms->Folder = folder;
+ }
+ return error;
+}
+
+static GSM_Error N7110_SetSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int location;
+ unsigned char folderid;
+
+ N7110_GetSMSLocation(s, sms, &folderid, &location);
+ if (location == 0) return ERR_INVALIDLOCATION;
+ return N7110_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N7110_AddSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int location;
+ unsigned char folderid;
+
+ N7110_GetSMSLocation(s, sms, &folderid, &location);
+ location = 0;
+ N7110_SetSMSLocation(s, sms, folderid, location);
+ return N7110_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N7110_ReplyClearOperatorLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Clearing operator logo.....\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_ReplySetOperatorLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Setting operator logo.....\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_SetCallerLogo(GSM_StateMachine *s, GSM_Bitmap *bitmap)
+{
+ int block=0, i, Width, Height;
+ unsigned int count = 18;
+ char string[500];
+ unsigned char req[500] = {N6110_FRAME_HEADER, 0x0b, 0x00,
+ 0x01, 0x01, 0x00, 0x00, 0x0c,
+ 0x00, 0x10, /* memory type */
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00};
+
+ req[13] = bitmap->Location;
+
+ /* Enabling/disabling logo */
+ if (bitmap->DefaultBitmap) {
+ string[0] = 0; //disabling
+ } else {
+ string[0] = bitmap->BitmapEnabled?1:0;
+ }
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_LOGOON, 2, block++, string, req + count);
+
+ /* Ringtone */
+ if (!bitmap->DefaultRingtone) {
+ string[0] = bitmap->RingtoneID;
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_RINGTONE_ID, 2, block++, string, req + count);
+ }
+
+ /* Number of group */
+ string[0] = bitmap->Location;
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_GROUP, 2, block++, string, req + count);
+
+ /* Name */
+ if (!bitmap->DefaultName) {
+ i = UnicodeLength(bitmap->Text) * 2;
+ string[0] = i + 2;
+ memcpy(string + 1, bitmap->Text, i);
+ string[i + 1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_NAME, i + 2, block++, string, req + count);
+ }
+
+ /* Logo */
+ if (bitmap->DefaultBitmap) {
+ bitmap->BitmapWidth = 72;
+ bitmap->BitmapHeight = 14;
+ GSM_ClearBitmap(bitmap);
+ }
+ PHONE_GetBitmapWidthHeight(GSM_NokiaCallerLogo, &Width, &Height);
+ string[0] = Width;
+ string[1] = Height;
+ string[2] = 0;
+ string[3] = 0;
+ string[4] = PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0);
+ PHONE_EncodeBitmap(GSM_NokiaCallerLogo, string + 5, bitmap);
+ count += N71_65_PackPBKBlock(s, N7110_PBK_GROUPLOGO, PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 5, block++, string, req + count);
+
+ req[17] = block;
+
+ return GSM_WaitFor (s, req, count, 0x03, 4, ID_SetBitmap);
+}
+
+static GSM_Error N7110_ReplySetPicture(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Picture Image written OK, folder %i, location %i\n",msg.Buffer[4],msg.Buffer[5]*256+msg.Buffer[6]);
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_SetPictureImage(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char folder;
+ GSM_Error error;
+ int location, i, count, Width, Height;
+ GSM_Phone_Bitmap_Types Type = GSM_NokiaPictureImage;
+ unsigned char req[500] = {N6110_FRAME_HEADER, 0x50, 0x07,
+ 0x00, /* location */
+ 0x00, 0x00, /* index */
+ 0x07};
+
+ error=N7110_GetPictureImageLocation(s, Bitmap, &folder, &location);
+ switch (error) {
+ case ERR_NONE:
+ req[5] = folder;
+ req[6] = location / 256;
+ req[7] = location % 256;
+ break;
+ case ERR_INVALIDLOCATION:
+ req[5] = 0x21; /* Save in Templates folder */
+ req[6] = 0;
+ req[7] = 0;
+ break;
+ default:
+ return error;
+ }
+
+ /* Cleaning */
+ for (i=0;i<36;i++) req[i+9]=0;
+
+ count=8;
+ if (UnicodeLength(Bitmap->Text)==0) {
+ count+=2 ;req[count]=0x0c;
+ count+=2 ;req[count]=0x0d;
+ count+=2 ;req[count]=0x0e;
+ count+=2 ;req[count]=0x0f;
+ count+=2 ;req[count]=0x10;
+ count+=2 ;req[count]=0x11;
+ count+=23;req[count]=0x02;
+ count++ ;
+ } else {
+ count+=2 ;req[count]=0x54;
+ count++ ;req[count]=0xd4;
+ count++ ;req[count]=0x0d;
+ count+=2 ;req[count]=0x0e;
+ count+=2 ;req[count]=0x0f;
+ count+=2 ;req[count]=0x10;
+ count+=2 ;req[count]=0x11;
+ count+=21;req[count]=0x01;
+ count+=3 ;
+ }
+ req[count] = 0x01;
+ count+=2;
+ req[count++] = 0x01;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ req[count++] = Width;
+ req[count++] = Height;
+ req[count++] = PHONE_GetBitmapSize(Type,0,0) / 256;
+ req[count++] = PHONE_GetBitmapSize(Type,0,0) % 256;
+ PHONE_EncodeBitmap(Type, req + count, Bitmap);
+ count += PHONE_GetBitmapSize(Type,0,0);
+ if (UnicodeLength(Bitmap->Text)!=0) {
+ req[count] = UnicodeLength(Bitmap->Text);
+ GSM_PackSevenBitsToEight(0, Bitmap->Text, req+count+1,strlen(Bitmap->Text));
+ count = count + req[count];
+ } else {
+ req[count++]=0x00;
+ }
+ req[count++]=0x00;
+ smprintf(s, "Setting Picture Image\n");
+ return GSM_WaitFor (s, req, count, 0x14, 4, ID_SetBitmap);
+}
+
+static GSM_Error N7110_SetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ GSM_Error error;
+ GSM_Phone_Bitmap_Types Type;
+ int Width, Height, i;
+ unsigned char reqStartup[1000] = {N7110_FRAME_HEADER, 0xec,
+ 0x15, /* Startup Logo setting */
+ 0x00, 0x00, 0x00, 0x04, 0xc0, 0x02, 0x00,
+ 0x00, /* Bitmap height */
+ 0xc0, 0x03, 0x00,
+ 0x00, /* Bitmap width */
+ 0xc0, 0x04, 0x03, 0x00};
+ unsigned char reqOp[1000] = {N7110_FRAME_HEADER, 0xa3, 0x01,
+ 0x00, /* logo disabled */
+ 0x00, 0xf0, 0x00, /* network code (000 00) */
+ 0x00 ,0x04,
+ 0x08, /* length of rest */
+ 0x00, 0x00, /* Bitmap width / height */
+ 0x00,
+ 0x00, /* Bitmap size */
+ 0x00, 0x00};
+ unsigned char reqClrOp[] = {0x00, 0x01, 0x00, 0xaf, 0x00};
+ unsigned char reqStartupText[500] = {N7110_FRAME_HEADER, 0xec,
+ 0x02}; /* Startup Text setting */
+
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ if (Bitmap->Location!=1) return ERR_NOTSUPPORTED;
+ Type=GSM_Nokia6210StartupLogo;
+ if (strcmp(s->Phone.Data.Model,"NSE-5") == 0) Type=GSM_Nokia7110StartupLogo;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ reqStartup[12] = Height;
+ reqStartup[16] = Width;
+ PHONE_EncodeBitmap(Type, reqStartup + 21, Bitmap);
+ smprintf(s, "Setting startup logo\n");
+ return GSM_WaitFor (s, reqStartup, 21+PHONE_GetBitmapSize(Type,0,0), 0x7A, 4, ID_SetBitmap);
+ case GSM_WelcomeNote_Text:
+ CopyUnicodeString(reqStartupText + 5, Bitmap->Text);
+ i = 6 + UnicodeLength(Bitmap->Text) * 2;
+ reqStartupText[i++] = 0;
+ reqStartupText[i++] = 0;
+ return GSM_WaitFor (s, reqStartupText, i, 0x7A, 4, ID_SetBitmap);
+ case GSM_DealerNote_Text:
+ reqStartupText[4] = 0x17;
+ CopyUnicodeString(reqStartupText + 5, Bitmap->Text);
+ i = 6 + UnicodeLength(Bitmap->Text) * 2;
+ reqStartupText[i++] = 0;
+ reqStartupText[i++] = 0;
+ return GSM_WaitFor (s, reqStartupText, i, 0x7A, 4, ID_SetBitmap);
+ case GSM_OperatorLogo:
+ /* We want to set operator logo, not clear */
+ if (strcmp(Bitmap->NetworkCode,"000 00")) {
+ reqOp[5] = 0x01; /* Logo enabled */
+ NOKIA_EncodeNetworkCode(reqOp+6, Bitmap->NetworkCode);
+ Type = GSM_Nokia7110OperatorLogo;
+ reqOp[11] = 8 + PHONE_GetBitmapSize(Type,0,0);
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ reqOp[12]=Width;
+ reqOp[13]=Height;
+ reqOp[15]=PHONE_GetBitmapSize(Type,0,0);
+ PHONE_EncodeBitmap(Type, reqOp + 18, Bitmap);
+ smprintf(s, "Setting operator logo\n");
+ return GSM_WaitFor (s, reqOp, 18+PHONE_GetBitmapSize(Type,0,0), 0x0A, 4, ID_SetBitmap);
+ } else {
+ smprintf(s, "Clearing operator logo\n");
+ for (i=0;i<5;i++) {
+ reqClrOp[4]=i;
+ error=GSM_WaitFor (s, reqClrOp, 5, 0x0A, 4, ID_SetBitmap);
+ if (error!=ERR_NONE) return error;
+ }
+ return GSM_WaitFor (s, reqOp, 18, 0x0A, 4, ID_SetBitmap);
+ }
+ case GSM_CallerGroupLogo:
+ return N7110_SetCallerLogo(s,Bitmap);
+ case GSM_PictureImage:
+ return N7110_SetPictureImage(s,Bitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N7110_ReplyDeleteMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phonebook entry deleted\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_DeleteMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x0f, 0x00, 0x01,
+ 0x04, 0x00, 0x00, 0x0c, 0x01, 0xff,
+ 0x00, 0x01, /* location */
+ 0x05, /* memory type */
+ 0x00, 0x00, 0x00};
+
+ req[12] = entry->Location / 256;
+ req[13] = entry->Location % 256;
+
+ req[14] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[14]==0xff) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Deleting phonebook entry\n");
+ return GSM_WaitFor (s, req, 18, 0x03, 4, ID_SetMemory);
+}
+
+static GSM_Error N7110_SetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ int count = 18, blocks;
+ unsigned char req[500] = {N7110_FRAME_HEADER, 0x0b, 0x00,
+ 0x01, 0x01, 0x00, 0x00, 0x0c,
+ 0x00, 0x00, /* memory type */
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00};
+
+ if (entry->Location == 0) return ERR_NOTSUPPORTED;
+
+ req[11] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[11]==0xff) return ERR_NOTSUPPORTED;
+
+ req[12] = entry->Location >> 8;
+ req[13] = entry->Location & 0xff;
+
+ count = count + N71_65_EncodePhonebookFrame(s, req+18, *entry, &blocks, false, IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_VOICETAGS));
+ req[17] = blocks;
+
+ smprintf(s, "Writing phonebook entry\n");
+ return GSM_WaitFor (s, req, count, 0x03, 4, ID_SetMemory);
+}
+
+static GSM_Error N7110_DeleteSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ unsigned char folderid;
+ int location;
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x0a,
+ 0x00, /* folder */
+ 0x00, 0x00, /* location */
+ 0x01};
+
+ N7110_GetSMSLocation(s, sms, &folderid, &location);
+ req[4] = folderid;
+ req[5] = location / 256;
+ req[6] = location;
+
+ smprintf(s, "Deleting sms\n");
+ return GSM_WaitFor (s, req, 8, 0x14, 4, ID_DeleteSMSMessage);
+}
+
+static GSM_Error N7110_ReplyGetSMSStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "SMS status received\n");
+ switch (msg.Buffer[3]) {
+ case 0x37:
+ smprintf(s, "SIM size : %i\n",msg.Buffer[8]*256+msg.Buffer[9]);
+ smprintf(s, "Used in phone memory : %i\n",msg.Buffer[10]*256+msg.Buffer[11]);
+ smprintf(s, "Unread in phone memory : %i\n",msg.Buffer[12]*256+msg.Buffer[13]);
+ smprintf(s, "Used in SIM : %i\n",msg.Buffer[14]*256+msg.Buffer[15]);
+ smprintf(s, "Unread in SIM : %i\n",msg.Buffer[16]*256+msg.Buffer[17]);
+ Data->SMSStatus->SIMSize = msg.Buffer[8]*256+msg.Buffer[9];
+ Data->SMSStatus->PhoneUsed = msg.Buffer[10]*256+msg.Buffer[11];
+ Data->SMSStatus->PhoneUnRead = msg.Buffer[12]*256+msg.Buffer[13];
+ Data->SMSStatus->PhoneSize = 150;
+ Data->SMSStatus->SIMUsed = msg.Buffer[14]*256+msg.Buffer[15];
+ Data->SMSStatus->SIMUnRead = msg.Buffer[16]*256+msg.Buffer[17];
+ return ERR_NONE;
+ case 0x38:
+ smprintf(s, "Error. No PIN ?\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_GetSMSStatus(GSM_StateMachine *s, GSM_SMSMemoryStatus *status)
+{
+ GSM_Error error;
+ GSM_Phone_N7110Data *Priv = &s->Phone.Data.Priv.N7110;
+
+ error = DCT3_GetSMSStatus(s,status);
+ if (error != ERR_NONE) return error;
+
+ /* 6210 family doesn't show in frame with SMS status info
+ * about Templates. We get separately info about this SMS folder.
+ */
+ error = N7110_GetSMSFolderStatus(s, 0x20);
+ if (error != ERR_NONE) return error;
+ status->TemplatesUsed = Priv->LastSMSFolder.Number;
+
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_ReplyGetProfileFeature(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s, "Profile feature %02x with value %02x\n",msg.Buffer[6],msg.Buffer[10]);
+ switch (msg.Buffer[6]) {
+ case 0x03:
+ smprintf(s, "Ringtone ID\n");
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_RingtoneID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = msg.Buffer[10];
+ Data->Profile->FeaturesNumber++;
+ break;
+ case 0x08: /* Caller groups */
+ NOKIA_FindFeatureValue(s, Profile71_65,msg.Buffer[6],msg.Buffer[10],Data,true);
+ break;
+ case 0x09: /* Autoanswer */
+ if (Data->Profile->CarKitProfile || Data->Profile->HeadSetProfile) {
+ NOKIA_FindFeatureValue(s, Profile71_65,msg.Buffer[6],msg.Buffer[10],Data,false);
+ }
+ break;
+ case 0xff :
+ CopyUnicodeString(Data->Profile->Name, msg.Buffer+10);
+ smprintf(s, "profile Name: \"%s\"\n", DecodeUnicodeString(Data->Profile->Name));
+ Data->Profile->DefaultName = false;
+ break;
+ default:
+ NOKIA_FindFeatureValue(s, Profile71_65,msg.Buffer[6],msg.Buffer[10],Data,false);
+ }
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N7110_GetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ GSM_Error error;
+ int i;
+ unsigned char Features[12] = {0x00,0x02,0x03,0x04,0x05,0x06,
+ 0x07,0x08,0x09,0xff,
+ 0x0a,0x22};
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x01, 0x01, 0x01, 0x01,
+ 0x00, /* Profile Location */
+ 0xff}; /* Feature number */
+
+ if (Profile->Location > 7) return ERR_INVALIDLOCATION;
+
+ Profile->CarKitProfile = false;
+ Profile->HeadSetProfile = false;
+ if (Profile->Location == 6) Profile->CarKitProfile = true;
+ if (Profile->Location == 7) Profile->HeadSetProfile = true;
+
+ Profile->FeaturesNumber = 0;
+
+ s->Phone.Data.Profile=Profile;
+ for (i = 0; i < 10; i++) {
+ req[7] = Profile->Location;
+ req[8] = Features[i];
+ smprintf(s, "Getting profile feature\n");
+ error = GSM_WaitFor (s, req, 9, 0x39, 4, ID_GetProfile);
+ if (error!=ERR_NONE) return error;
+ }
+ NOKIA_GetDefaultProfileName(s, Profile);
+ Profile->Active = false;
+ return error;
+}
+
+static GSM_Error N7110_ReplySetProfileFeature(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Profile feature set\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_SetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ int i;
+ bool found;
+ GSM_Error error;
+ unsigned char ID,Value;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x03, 0x01, 0x01, 0x03,
+ 0x02, /* feature number */
+ 0x01, /* Profile Location */
+ 0x01,
+ 0xff}; /* Value */
+
+ for (i=0;i<Profile->FeaturesNumber;i++) {
+ found = false;
+ switch (Profile->FeatureID[i]) {
+ case Profile_RingtoneID:
+ ID = 0x03;
+ Value = Profile->FeatureValue[i];
+ found = true;
+ break;
+ default:
+ found=NOKIA_FindPhoneFeatureValue(
+ s,
+ Profile71_65,
+ Profile->FeatureID[i],Profile->FeatureValue[i],
+ &ID,&Value);
+ }
+ if (found) {
+ req[7] = ID;
+ req[8] = Profile->Location;
+ req[10] = Value;
+ smprintf(s, "Setting profile feature\n");
+ error = GSM_WaitFor (s, req, 11, 0x39, 4, ID_SetProfile);
+ if (error!=ERR_NONE) return error;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_GetSpeedDial(GSM_StateMachine *s, GSM_SpeedDial *SpeedDial)
+{
+ GSM_MemoryEntry pbk;
+ GSM_Error error;
+
+ pbk.MemoryType = MEM7110_SP;
+ pbk.Location = SpeedDial->Location;
+ SpeedDial->MemoryLocation = 0;
+ s->Phone.Data.SpeedDial = SpeedDial;
+
+ smprintf(s, "Getting speed dial\n");
+ error=N7110_GetMemory(s,&pbk);
+ switch (error) {
+ case ERR_NOTSUPPORTED:
+ smprintf(s, "No speed dials set in phone\n");
+ return ERR_EMPTY;
+ case ERR_NONE:
+ if (SpeedDial->MemoryLocation == 0) {
+ smprintf(s, "Speed dial not assigned or error in firmware\n");
+ return ERR_EMPTY;
+ }
+ return ERR_NONE;
+ default:
+ return error;
+ }
+}
+
+static GSM_Error N7110_ReplyIncomingSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SMSMessage sms;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+#ifdef DEBUG
+ smprintf(s, "SMS message received\n");
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+8);
+#endif
+ if (Data->EnableIncomingSMS && s->User.IncomingSMS!=NULL) {
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+8);
+
+ s->User.IncomingSMS(s->CurrentConfig->Device,sms);
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_Initialise (GSM_StateMachine *s)
+{
+#ifdef DEBUG
+ DCT3_SetIncomingCB(s,true);
+#endif
+#ifdef GSM_ENABLE_N71_92INCOMINGINFO
+ /* Enables various things like incoming SMS, call info, etc. */
+ return N71_65_EnableFunctions (s, "\x01\x02\x06\x0A\x14\x17", 6);
+#endif
+ return ERR_NONE;
+}
+
+static GSM_Error N7110_ReplyGetCalendarInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ /* Old method 1 for accessing calendar */
+ return N71_65_ReplyGetCalendarInfo1(msg, s, &s->Phone.Data.Priv.N7110.LastCalendar);
+}
+
+#ifdef DEBUG
+static GSM_Error N7110_ReplyGetCalendarNotePos(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ /* Old method 1 for accessing calendar */
+ return N71_65_ReplyGetCalendarNotePos1(msg, s, &s->Phone.Data.Priv.N7110.FirstCalendarPos);
+}
+#endif
+
+static GSM_Error N7110_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ return N71_65_GetNextCalendar1(s,Note,start,&s->Phone.Data.Priv.N7110.LastCalendar,&s->Phone.Data.Priv.N7110.LastCalendarYear,&s->Phone.Data.Priv.N7110.LastCalendarPos);
+// return N71_65_GetNextCalendar2(s,Note,start,&s->Phone.Data.Priv.N7110.LastCalendarYear,&s->Phone.Data.Priv.N7110.LastCalendarPos);
+}
+
+static GSM_Error N7110_GetCalendarStatus(GSM_StateMachine *s, GSM_CalendarStatus *Status)
+{
+ GSM_Error error;
+
+ /* Method 1 */
+ error=N71_65_GetCalendarInfo1(s, &s->Phone.Data.Priv.N7110.LastCalendar);
+ if (error!=ERR_NONE) return error;
+ Status->Used = s->Phone.Data.Priv.N6510.LastCalendar.Number;
+ return ERR_NONE;
+
+ /* Method 2 */
+// return GE_NOTSUPPORTED;
+}
+
+static GSM_Error N7110_AddCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+// return N71_65_AddCalendar1(s, Note, NULL);
+ return N71_65_AddCalendar2(s,Note);
+}
+
+static GSM_Error N7110_ReplyGetNetworkInfoError(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Probably means no PIN\n");
+ return ERR_SECURITYERROR;
+}
+
+static GSM_Error N7110_SetIncomingCall(GSM_StateMachine *s, bool enable)
+{
+#ifndef GSM_ENABLE_N71_92INCOMINGINFO
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+ return NOKIA_SetIncomingCall(s,enable);
+}
+
+static GSM_Error N7110_SetIncomingUSSD(GSM_StateMachine *s, bool enable)
+{
+#ifndef GSM_ENABLE_N71_92INCOMINGINFO
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+ return NOKIA_SetIncomingUSSD(s,enable);
+}
+
+static GSM_Error N7110_SetIncomingSMS(GSM_StateMachine *s, bool enable)
+{
+#ifndef GSM_ENABLE_N71_92INCOMINGINFO
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+ return NOKIA_SetIncomingSMS(s,enable);
+}
+
+GSM_Error N7110_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (!all) return DCT3DCT4_AnswerCall(s,ID);
+ return DCT3_AnswerAllCalls(s);
+}
+
+GSM_Error N7110_SetCallDivert(GSM_StateMachine *s, GSM_MultiCallDivert *divert)
+{
+ GSM_Error error;
+ int i;
+
+ /* No answer from phone side */
+ i = s->ReplyNum;
+ s->ReplyNum = 1;
+ error = DCT3DCT4_SetCallDivert(s,divert);
+ s->ReplyNum = i;
+ return error;
+}
+
+GSM_Error N7110_CancelAllDiverts(GSM_StateMachine *s)
+{
+ GSM_Error error;
+ int i;
+
+ /* No answer from phone side */
+ i = s->ReplyNum;
+ s->ReplyNum = 1;
+ error = DCT3DCT4_CancelAllDiverts(s);
+ s->ReplyNum = i;
+ return error;
+}
+
+static GSM_Reply_Function N7110ReplyFunctions[] = {
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x02,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x03,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x04,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x05,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x07,ID_AnswerCall },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x07,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x09,ID_CancelCall },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x09,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0A,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0B,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0C,ID_IncomingFrame },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x51,ID_SendDTMF },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x53,ID_IncomingFrame },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x59,ID_SendDTMF },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x5E,ID_SendDTMF },
+
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x02,ID_IncomingFrame },
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x03,ID_IncomingFrame },
+ {N7110_ReplyIncomingSMS, "\x02",0x03,0x10,ID_IncomingFrame },
+#ifdef GSM_ENABLE_CELLBROADCAST
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x21,ID_SetIncomingCB },
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x22,ID_SetIncomingCB },
+ {DCT3_ReplyIncomingCB, "\x02",0x03,0x23,ID_IncomingFrame },
+#endif
+ {DCT3_ReplySetSMSC, "\x02",0x03,0x31,ID_SetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x34,ID_GetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x35,ID_GetSMSC },
+#ifdef GSM_ENABLE_CELLBROADCAST
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0xCA,ID_SetIncomingCB },
+#endif
+
+ {N7110_ReplyGetMemoryStatus, "\x03",0x03,0x04,ID_GetMemoryStatus },
+ {N7110_ReplyGetMemory, "\x03",0x03,0x08,ID_GetMemory },
+ {N7110_ReplyDeleteMemory, "\x03",0x03,0x10,ID_SetMemory },
+ {N71_65_ReplyWritePhonebook, "\x03",0x03,0x0C,ID_SetBitmap },
+ {N71_65_ReplyWritePhonebook, "\x03",0x03,0x0C,ID_SetMemory },
+
+ {N71_65_ReplyUSSDInfo, "\x06",0x03,0x03,ID_IncomingFrame },
+ {NoneReply, "\x06",0x03,0x06,ID_IncomingFrame },
+
+ {DCT3_ReplySIMLogin, "\x09",0x03,0x80,ID_IncomingFrame },
+ {DCT3_ReplySIMLogout, "\x09",0x03,0x81,ID_IncomingFrame },
+
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_GetNetworkInfo },
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_GetBitmap },
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_IncomingFrame },
+ {N7110_ReplyGetNetworkInfoError, "\x0A",0x03,0x72,ID_GetNetworkInfo },
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x73,ID_IncomingFrame },
+ {N71_92_ReplyGetSignalQuality, "\x0A",0x03,0x82,ID_GetSignalQuality },
+ {N7110_ReplySetOperatorLogo, "\x0A",0x03,0xA4,ID_SetBitmap },
+ {N7110_ReplyClearOperatorLogo, "\x0A",0x03,0xB0,ID_SetBitmap },
+ {NoneReply, "\x0A",0x03,0xB5,ID_IncomingFrame },
+
+#ifdef DEBUG
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x02,ID_SetCalendarNote },/*method 1*/
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x04,ID_SetCalendarNote },/*method 1*/
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x06,ID_SetCalendarNote },/*method 1*/
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x08,ID_SetCalendarNote },/*method 1*/
+#endif
+ {N71_65_ReplyDelCalendar, "\x13",0x03,0x0C,ID_DeleteCalendarNote },
+ {N71_65_ReplyGetNextCalendar1, "\x13",0x03,0x1A,ID_GetCalendarNote },/*method 1*/
+#ifdef DEBUG
+ {N7110_ReplyGetCalendarNotePos, "\x13",0x03,0x32,ID_GetCalendarNotePos },/*method 1*/
+#endif
+ {N7110_ReplyGetCalendarInfo, "\x13",0x03,0x3B,ID_GetCalendarNotesInfo},/*method 1*/
+#ifdef DEBUG
+ {N71_65_ReplyGetNextCalendar2, "\x13",0x03,0x3F,ID_GetCalendarNote },/*method 2*/
+#endif
+ {N71_65_ReplyAddCalendar2, "\x13",0x03,0x41,ID_SetCalendarNote },/*method 2*/
+
+ {N7110_ReplySaveSMSMessage, "\x14",0x03,0x05,ID_SaveSMSMessage },
+ {N7110_ReplySaveSMSMessage, "\x14",0x03,0x06,ID_SaveSMSMessage },
+ {N7110_ReplyGetSMSMessage, "\x14",0x03,0x08,ID_GetSMSMessage },
+ {N7110_ReplyGetSMSMessage, "\x14",0x03,0x08,ID_GetBitmap },
+ {N7110_ReplyGetSMSMessage, "\x14",0x03,0x09,ID_GetSMSMessage },
+ {DCT3_ReplyDeleteSMSMessage, "\x14",0x03,0x0B,ID_DeleteSMSMessage },
+ {DCT3_ReplyDeleteSMSMessage, "\x14",0x03,0x0C,ID_DeleteSMSMessage },
+ {N7110_ReplyGetSMSStatus, "\x14",0x03,0x37,ID_GetSMSStatus },
+ {N7110_ReplyGetSMSStatus, "\x14",0x03,0x38,ID_GetSMSStatus },
+ {N7110_ReplySetPicture, "\x14",0x03,0x51,ID_SetBitmap },
+ {N7110_ReplyGetSMSFolderStatus, "\x14",0x03,0x6C,ID_GetSMSFolderStatus },
+ {N7110_ReplyGetSMSMessage, "\x14",0x03,0x6F,ID_GetSMSMessage },
+ {N7110_ReplyGetSMSFolders, "\x14",0x03,0x7B,ID_GetSMSFolders },
+ {N7110_ReplyGetSMSFolders, "\x14",0x03,0x7C,ID_GetSMSFolders },
+ {N7110_ReplySaveSMSMessage, "\x14",0x03,0x84,ID_SaveSMSMessage },
+ {N7110_ReplyGetPictureImageInfo, "\x14",0x03,0x97,ID_GetBitmap },
+ {N7110_ReplyGetSMSFolders, "\x14",0x03,0xCA,ID_GetSMSFolders },
+
+ {N71_92_ReplyGetBatteryCharge, "\x17",0x03,0x03,ID_GetBatteryCharge },
+
+ {DCT3_ReplySetDateTime, "\x19",0x03,0x61,ID_SetDateTime },
+ {DCT3_ReplyGetDateTime, "\x19",0x03,0x63,ID_GetDateTime },
+ {DCT3_ReplySetAlarm, "\x19",0x03,0x6C,ID_SetAlarm },
+ {DCT3_ReplyGetAlarm, "\x19",0x03,0x6E,ID_GetAlarm },
+
+ {N7110_ReplyGetRingtone, "\x1f",0x03,0x23,ID_GetRingtone },
+ {N7110_ReplyGetRingtone, "\x1f",0x03,0x24,ID_GetRingtone },
+
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x01,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x02,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x04,ID_DisableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x05,ID_DisableConnectFunc },
+ {DCT3_ReplyGetWAPBookmark, "\x3f",0x03,0x07,ID_GetWAPBookmark },
+ {DCT3_ReplyGetWAPBookmark, "\x3f",0x03,0x08,ID_GetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0A,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0B,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0D,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0E,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyGetActiveConnectSet,"\x3f",0x03,0x10,ID_GetConnectSet },
+ {DCT3DCT4_ReplySetActiveConnectSet,"\x3f",0x03,0x13,ID_SetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x16,ID_GetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x17,ID_GetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x19,ID_SetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x1A,ID_SetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x1C,ID_GetConnectSet },
+ {DCT3_ReplyGetWAPSettings, "\x3f",0x03,0x1D,ID_GetConnectSet },
+ {DCT3_ReplySetWAPSettings, "\x3f",0x03,0x1F,ID_SetConnectSet },
+
+ {N7110_ReplyGetProfileFeature, "\x39",0x03,0x02,ID_GetProfile },
+ {N7110_ReplySetProfileFeature, "\x39",0x03,0x04,ID_SetProfile },
+
+ {DCT3_ReplyEnableSecurity, "\x40",0x02,0x64,ID_EnableSecurity },
+ {N61_71_ReplyResetPhoneSettings, "\x40",0x02,0x65,ID_ResetPhoneSettings },
+ {DCT3_ReplyGetIMEI, "\x40",0x02,0x66,ID_GetIMEI },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_DialVoice },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_CancelCall },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_AnswerCall },
+ {DCT3_ReplyNetmonitor, "\x40",0x02,0x7E,ID_Netmonitor },
+ {DCT3_ReplyPlayTone, "\x40",0x02,0x8F,ID_PlayTone },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetHardware },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetPPM },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCA,ID_GetProductCode },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetManufactureMonth },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetOriginalIMEI },
+ {NoneReply, "\x40",0x02,0xFF,ID_IncomingFrame },
+
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x02,ID_GetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x02,ID_SetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x15,ID_GetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x15,ID_SetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x17,ID_GetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x17,ID_SetBitmap },
+
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetFirmware },
+ {DCT3_ReplyPressKey, "\xD2",0x02,0x46,ID_PressKey },
+ {DCT3_ReplyPressKey, "\xD2",0x02,0x47,ID_PressKey },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N7110Phone = {
+ "6210|6250|7110|7190",
+ N7110ReplyFunctions,
+ N7110_Initialise,
+ PHONE_Terminate,
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ DCT3_GetIMEI,
+ DCT3_GetOriginalIMEI,
+ DCT3_GetManufactureMonth,
+ DCT3_GetProductCode,
+ DCT3_GetHardware,
+ DCT3_GetPPM,
+ NOTSUPPORTED, /* GetSIMIMSI */
+ N71_92_GetDateTime,
+ N71_92_SetDateTime,
+ N7110_GetAlarm,
+ N7110_SetAlarm,
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ DCT3_PressKey,
+ DCT3_Reset,
+ N61_71_ResetPhoneSettings,
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTIMPLEMENTED, /* SetAutoNetworkLogin */
+ N71_92_GetBatteryCharge,
+ N71_92_GetSignalQuality,
+ DCT3_GetNetworkInfo,
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ N7110_GetMemoryStatus,
+ N7110_GetMemory,
+ NOTIMPLEMENTED, /* GetNextMemory */
+ N7110_SetMemory,
+ NOTIMPLEMENTED, /* AddMemory */
+ N7110_DeleteMemory,
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ N7110_GetSpeedDial,
+ NOTIMPLEMENTED, /* SetSpeedDial */
+ DCT3_GetSMSC,
+ DCT3_SetSMSC,
+ N7110_GetSMSStatus,
+ N7110_GetSMSMessage,
+ N7110_GetNextSMSMessage,
+ N7110_SetSMS,
+ N7110_AddSMS,
+ N7110_DeleteSMS,
+ DCT3_SendSMSMessage,
+ NOTSUPPORTED, /* SendSavedSMS */
+ N7110_SetIncomingSMS,
+ DCT3_SetIncomingCB,
+ N7110_GetSMSFolders,
+ NOTIMPLEMENTED, /* AddSMSFolder */
+ NOTIMPLEMENTED, /* DeleteSMSFolder */
+ DCT3_DialVoice,
+ N7110_AnswerCall,
+ DCT3_CancelCall,
+ NOTIMPLEMENTED, /* HoldCall */
+ NOTIMPLEMENTED, /* UnholdCall */
+ NOTIMPLEMENTED, /* ConferenceCall */
+ NOTIMPLEMENTED, /* SplitCall */
+ NOTIMPLEMENTED, /* TransferCall */
+ NOTIMPLEMENTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ N7110_SetCallDivert,
+ N7110_CancelAllDiverts,
+ N7110_SetIncomingCall,
+ N7110_SetIncomingUSSD,
+ DCT3DCT4_SendDTMF,
+ N7110_GetRingtone,
+ N7110_SetRingtone,
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ DCT3_PlayTone,
+ DCT3_GetWAPBookmark,
+ DCT3_SetWAPBookmark,
+ DCT3_DeleteWAPBookmark,
+ DCT3_GetWAPSettings,
+ DCT3_SetWAPSettings,
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ N7110_GetBitmap,
+ N7110_SetBitmap,
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ N7110_GetCalendarStatus,
+ NOTIMPLEMENTED, /* GetCalendar */
+ N7110_GetNextCalendar,
+ NOTIMPLEMENTED, /* SetCalendar */
+ N7110_AddCalendar,
+ N71_65_DelCalendar,
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ N7110_GetProfile,
+ N7110_SetProfile,
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFile */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n7110.h b/gammu/emb/common/phone/nokia/dct3/n7110.h
new file mode 100644
index 0000000..a7934c2
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n7110.h
@@ -0,0 +1,45 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef n7110_h
+#define n7110_h
+
+#include "../ncommon.h"
+#include "dct3comm.h"
+
+typedef struct {
+ int LastCalendarYear;
+ int LastCalendarPos;
+ GSM_NOKIACalToDoLocations LastCalendar;
+ int FirstCalendarPos;
+
+ GSM_NOKIASMSFolder LastSMSFolder;
+ GSM_SMSFolders LastSMSFolders;
+ GSM_NOKIASMSFolder LastPictureImageFolder;
+
+ DCT3_WAPSettings_Locations WAPLocations;
+} GSM_Phone_N7110Data;
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+#ifndef GSM_USED_FBUS2DLR3
+# define GSM_USED_FBUS2DLR3
+#endif
+#ifndef GSM_USED_FBUS2BLUE
+# define GSM_USED_FBUS2BLUE
+#endif
+#ifndef GSM_USED_IRDAPHONET
+# define GSM_USED_IRDAPHONET
+#endif
+#ifndef GSM_USED_BLUEFBUS2
+# define GSM_USED_BLUEFBUS2
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n9210.c b/gammu/emb/common/phone/nokia/dct3/n9210.c
new file mode 100644
index 0000000..e82d530
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n9210.c
@@ -0,0 +1,396 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA9210
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../misc/coding/coding.h"
+#include "../../../gsmcomon.h"
+#include "../../../service/gsmlogo.h"
+#include "../../pfunc.h"
+#include "../nfunc.h"
+#include "n9210.h"
+#include "dct3func.h"
+
+static GSM_Error N9210_GetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char OpReq[] = {N6110_FRAME_HEADER, 0x70};
+
+ s->Phone.Data.Bitmap=Bitmap;
+ switch (Bitmap->Type) {
+ case GSM_OperatorLogo:
+ smprintf(s, "Getting operator logo\n");
+ /* This is like DCT3_GetNetworkInfo */
+ return GSM_WaitFor (s, OpReq, 4, 0x0a, 4, ID_GetBitmap);
+ case GSM_StartupLogo:
+ smprintf(s, "Getting startup logo\n");
+ return N71_92_GetPhoneSetting(s, ID_GetBitmap, 0x15);
+ case GSM_WelcomeNote_Text:
+ smprintf(s, "Getting welcome note\n");
+ return N71_92_GetPhoneSetting(s, ID_GetBitmap, 0x02);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N9210_ReplySetOpLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Operator logo clear/set\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N9210_SetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ GSM_Error error;
+ GSM_Phone_Bitmap_Types Type;
+ int Width, Height, i,count=3;
+ unsigned char req[600] = { N7110_FRAME_HEADER };
+ unsigned char reqStartup[1000] = {
+ N6110_FRAME_HEADER, 0xec,
+ 0x15, /* Startup Logo setting */
+ 0x04, 0x00, 0x00, 0x00, 0x30, 0x00,
+ 0x02, 0xc0, 0x54, 0x00, 0x03, 0xc0,
+ 0xf8, 0xf8, 0x01, 0x04};
+ unsigned char reqStartupText[500] = {
+ N7110_FRAME_HEADER, 0xec,
+ 0x02}; /* Startup Text setting */
+ unsigned char reqClrOp[] = {
+ N7110_FRAME_HEADER, 0xAF,
+ 0x02}; /* Number of logo = 0 - 0x04 */
+
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ if (Bitmap->Location!=1) return ERR_NOTSUPPORTED;
+ Type=GSM_NokiaStartupLogo;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ PHONE_EncodeBitmap(Type, reqStartup + 21, Bitmap);
+ smprintf(s, "Setting startup logo\n");
+ return GSM_WaitFor (s, reqStartup, 21+PHONE_GetBitmapSize(Type,0,0), 0x7A, 4, ID_SetBitmap);
+ case GSM_WelcomeNote_Text:
+ /* Nokia bug: Unicode text is moved one char to left */
+ CopyUnicodeString(reqStartupText + 4, Bitmap->Text);
+ reqStartupText[4] = 0x02;
+ i = 5 + UnicodeLength(Bitmap->Text) * 2;
+ reqStartupText[i++] = 0;
+ reqStartupText[i++] = 0;
+ return GSM_WaitFor (s, reqStartupText, i, 0x7A, 4, ID_SetBitmap);
+ case GSM_OperatorLogo:
+ /* First part for clearing logo */
+ if (!strcmp(Bitmap->NetworkCode,"000 00")) {
+ for (i=0;i<5;i++) {
+ reqClrOp[4] = i;
+ error=GSM_WaitFor (s, reqClrOp, 5, 0x0A, 4, ID_SetBitmap);
+ if (error != ERR_NONE) return error;
+ }
+ }
+ Type=GSM_NokiaOperatorLogo;
+ req[count++] = 0xA3;
+ req[count++] = 0x01;
+ req[count++] = 0x00; /* Logo removed */
+ NOKIA_EncodeNetworkCode(req+count, "000 00");
+ count = count + 3;
+ req[count++] = 0x00;
+ req[count++] = 0x04;
+ req[count++] = 0x08; /* Length of rest + 2 */
+ memcpy(req+count, "\x00\x00\x00\x00\x00\x00", 6);
+ count += 6;
+ error=GSM_WaitFor (s, req, count, 0x0A, 4, ID_SetBitmap);
+ if (error != ERR_NONE) return error;
+ /* We wanted only clear - now exit */
+ if (!strcmp(Bitmap->NetworkCode,"000 00")) return error;
+
+ /* Now setting logo */
+ count=3;
+ req[count++] = 0xA3;
+ req[count++] = 0x01;
+ req[count++] = 0x01; /* Logo set */
+ NOKIA_EncodeNetworkCode(req+count, Bitmap->NetworkCode);
+ count = count + 3;
+ req[count++] = 0x00;
+ req[count++] = 0x04;
+ req[count++] = PHONE_GetBitmapSize(Type,0,0)+8;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ req[count++] = Width;
+ req[count++] = Height;
+ req[count++] = PHONE_GetBitmapSize(Type,0,0);
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ PHONE_EncodeBitmap(Type, req+count, Bitmap);
+ return GSM_WaitFor (s, req, count+PHONE_GetBitmapSize(Type,0,0), 0x0A, 4, ID_SetBitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N9210_ReplyIncomingSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SMSMessage sms;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+#ifdef DEBUG
+ smprintf(s, "SMS message received\n");
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+5);
+#endif
+ if (Data->EnableIncomingSMS && s->User.IncomingSMS!=NULL) {
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+ DCT3_DecodeSMSFrame(s, &sms,msg.Buffer+5);
+
+ s->User.IncomingSMS(s->CurrentConfig->Device,sms);
+ }
+ return ERR_NONE;
+}
+
+#ifdef GSM_ENABLE_N71_92INCOMINGINFO
+static GSM_Error N9210_ReplySetIncomingSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x0e:
+ s->Phone.Data.EnableIncomingSMS = true;
+ smprintf(s, "Incoming SMS enabled\n");
+ return ERR_NONE;
+ case 0x0f:
+ smprintf(s, "Error enabling incoming SMS\n");
+ switch (msg.Buffer[4]) {
+ case 0x0c:
+ smprintf(s, "No PIN ?\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+#endif
+
+static GSM_Error N9210_SetIncomingSMS(GSM_StateMachine *s, bool enable)
+{
+#ifdef GSM_ENABLE_N71_92INCOMINGINFO
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0d, 0x00, 0x00, 0x02};
+
+ if (enable!=s->Phone.Data.EnableIncomingSMS) {
+ if (enable) {
+ smprintf(s, "Enabling incoming SMS\n");
+ return GSM_WaitFor (s, req, 7, 0x02, 4, ID_SetIncomingSMS);
+ } else {
+ s->Phone.Data.EnableIncomingSMS = false;
+ smprintf(s, "Disabling incoming SMS\n");
+ }
+ }
+ return ERR_NONE;
+#else
+ return ERR_SOURCENOTAVAILABLE;
+#endif
+}
+
+static GSM_Error N9210_Initialise (GSM_StateMachine *s)
+{
+#ifdef DEBUG
+ DCT3_SetIncomingCB(s,true);
+
+#ifdef GSM_ENABLE_N71_92INCOMINGINFO
+ N9210_SetIncomingSMS(s,true);
+#endif
+
+#endif
+ return ERR_NONE;
+}
+
+GSM_Error N9210_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (!all) return DCT3DCT4_AnswerCall(s,ID);
+ return DCT3_AnswerAllCalls(s);
+}
+
+static GSM_Reply_Function N9210ReplyFunctions[] = {
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x02,ID_IncomingFrame },
+ {DCT3_ReplySendSMSMessage, "\x02",0x03,0x03,ID_IncomingFrame },
+#ifdef GSM_ENABLE_N71_92INCOMINGINFO
+ {N9210_ReplySetIncomingSMS, "\x02",0x03,0x0E,ID_SetIncomingSMS },
+ {N9210_ReplySetIncomingSMS, "\x02",0x03,0x0F,ID_SetIncomingSMS },
+#endif
+ {N9210_ReplyIncomingSMS, "\x02",0x03,0x11,ID_IncomingFrame },
+#ifdef GSM_ENABLE_CELLBROADCAST
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x21,ID_SetIncomingCB },
+ {DCT3_ReplySetIncomingCB, "\x02",0x03,0x22,ID_SetIncomingCB },
+ {DCT3_ReplyIncomingCB, "\x02",0x03,0x23,ID_IncomingFrame },
+#endif
+ {DCT3_ReplySetSMSC, "\x02",0x03,0x31,ID_SetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x34,ID_GetSMSC },
+ {DCT3_ReplyGetSMSC, "\x02",0x03,0x35,ID_GetSMSC },
+
+ {N61_91_ReplySetOpLogo, "\x05",0x03,0x31,ID_SetBitmap },
+ {N61_91_ReplySetOpLogo, "\x05",0x03,0x32,ID_SetBitmap },
+
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_GetNetworkInfo },
+ {DCT3_ReplyGetNetworkInfo, "\x0A",0x03,0x71,ID_IncomingFrame },
+ {N71_92_ReplyGetSignalQuality, "\x0A",0x03,0x82,ID_GetSignalQuality },
+ {N9210_ReplySetOpLogo, "\x0A",0x03,0xA4,ID_SetBitmap },
+ {N9210_ReplySetOpLogo, "\x0A",0x03,0xB0,ID_SetBitmap },
+
+ {N71_92_ReplyGetBatteryCharge, "\x17",0x03,0x03,ID_GetBatteryCharge },
+
+ {DCT3_ReplySetDateTime, "\x19",0x03,0x61,ID_SetDateTime },
+ {DCT3_ReplyGetDateTime, "\x19",0x03,0x63,ID_GetDateTime },
+
+ {DCT3_ReplyEnableSecurity, "\x40",0x02,0x64,ID_EnableSecurity },
+ {DCT3_ReplyGetIMEI, "\x40",0x02,0x66,ID_GetIMEI },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_DialVoice },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_CancelCall },
+ {DCT3_ReplyDialCommand, "\x40",0x02,0x7C,ID_AnswerCall },
+ {DCT3_ReplyNetmonitor, "\x40",0x02,0x7E,ID_Netmonitor },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetHardware },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xC8,ID_GetPPM },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCA,ID_GetProductCode },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetManufactureMonth },
+ {NOKIA_ReplyGetPhoneString, "\x40",0x02,0xCC,ID_GetOriginalIMEI },
+
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x02,ID_GetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x02,ID_SetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x15,ID_GetBitmap },
+ {N71_92_ReplyPhoneSetting, "\x7a",0x04,0x15,ID_SetBitmap },
+
+ {DCT3DCT4_ReplyGetModelFirmware,"\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware,"\xD2",0x02,0x00,ID_GetFirmware },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N9210Phone = {
+ "9210|9210i",
+ N9210ReplyFunctions,
+ N9210_Initialise,
+ PHONE_Terminate,
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ DCT3_GetIMEI,
+ DCT3_GetOriginalIMEI,
+ DCT3_GetManufactureMonth,
+ DCT3_GetProductCode,
+ DCT3_GetHardware,
+ DCT3_GetPPM,
+ NOTSUPPORTED, /* GetSIMIMSI */
+ N71_92_GetDateTime,
+ N71_92_SetDateTime,
+ NOTIMPLEMENTED, /* GetAlarm */
+ NOTIMPLEMENTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTIMPLEMENTED, /* PressKey */
+ NOTIMPLEMENTED, /* Reset */
+ NOTIMPLEMENTED, /* ResetPhoneSettings */
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTIMPLEMENTED, /* SetAutoNetworkLogin */
+ N71_92_GetBatteryCharge,
+ N71_92_GetSignalQuality,
+ DCT3_GetNetworkInfo,
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ NOTIMPLEMENTED, /* GetMemoryStatus */
+ NOTIMPLEMENTED, /* GetMemory */
+ NOTIMPLEMENTED, /* GetNextMemory */
+ NOTIMPLEMENTED, /* SetMemory */
+ NOTIMPLEMENTED, /* AddMemory */
+ NOTIMPLEMENTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTIMPLEMENTED, /* GetSpeedDial */
+ NOTIMPLEMENTED, /* SetSpeedDial */
+ DCT3_GetSMSC,
+ DCT3_SetSMSC, /* FIXME: test it */
+ NOTIMPLEMENTED, /* GetSMSStatus */
+ NOTIMPLEMENTED, /* GetSMS */
+ NOTIMPLEMENTED, /* GetNextSMS */
+ NOTIMPLEMENTED, /* SetSMS */
+ NOTIMPLEMENTED, /* AddSMS */
+ NOTIMPLEMENTED, /* DeleteSMS */
+ DCT3_SendSMSMessage,
+ NOTSUPPORTED, /* SendSavedSMS */
+ N9210_SetIncomingSMS,
+ DCT3_SetIncomingCB,
+ NOTIMPLEMENTED, /* GetSMSFolders */
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ DCT3_DialVoice,
+ N9210_AnswerCall,
+ DCT3_CancelCall,
+ NOTSUPPORTED, /* HoldCall */
+ NOTSUPPORTED, /* UnholdCall */
+ NOTSUPPORTED, /* ConferenceCall */
+ NOTSUPPORTED, /* SplitCall */
+ NOTSUPPORTED, /* TransferCall */
+ NOTSUPPORTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NOTSUPPORTED, /* SetIncomingCall */
+ NOTIMPLEMENTED, /* SetIncomingUSSD */
+ NOTSUPPORTED, /* SendDTMF */
+ NOTIMPLEMENTED, /* GetRingtone */
+ NOTIMPLEMENTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTIMPLEMENTED, /* GetWAPBookmark */
+ NOTIMPLEMENTED, /* SetWAPBookmark */
+ NOTIMPLEMENTED, /* DeleteWAPBookmark */
+ NOTIMPLEMENTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ N9210_GetBitmap,
+ N9210_SetBitmap,
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ NOTSUPPORTED, /* GetCalendarStatus */
+ NOTSUPPORTED, /* GetCalendar */
+ NOTSUPPORTED, /* GetNextCalendar */
+ NOTSUPPORTED, /* SetCalendar */
+ NOTSUPPORTED, /* AddCalendar */
+ NOTSUPPORTED, /* DeleteCalendar */
+ NOTSUPPORTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTIMPLEMENTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFile */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct3/n9210.h b/gammu/emb/common/phone/nokia/dct3/n9210.h
new file mode 100644
index 0000000..8998532
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct3/n9210.h
@@ -0,0 +1,17 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef n9210_h
+#define n9210_h
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/dct4func.c b/gammu/emb/common/phone/nokia/dct4/dct4func.c
new file mode 100644
index 0000000..9fa931f
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/dct4func.c
@@ -0,0 +1,115 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#include "../../../gsmstate.h"
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../gsmcomon.h"
+#include "../../../misc/coding/coding.h"
+#include "../../../service/gsmlogo.h"
+#include "../nfunc.h"
+#include "../nfuncold.h"
+#include "../../pfunc.h"
+#include "dct4func.h"
+
+#ifdef GSM_ENABLE_NOKIA_DCT4
+
+GSM_Error DCT4_ReplyGetPhoneMode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ s->Phone.Data.PhoneString[0] = msg.Buffer[4];
+ return ERR_NONE;
+}
+
+GSM_Error DCT4_GetPhoneMode(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x02, 0x00, 0x00};
+
+ smprintf(s,"Getting phone mode\n");
+
+ return GSM_WaitFor (s, req, 6, 0x15, 4, ID_Reset);
+}
+
+GSM_Error DCT4_ReplySetPhoneMode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+GSM_Error DCT4_SetPhoneMode(GSM_StateMachine *s, DCT4_PHONE_MODE mode)
+{
+ unsigned char PhoneMode[10];
+ int i;
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x01,
+ 0x04, /* phone mode */
+ 0x00};
+
+ if (s->ConnectionType != GCT_FBUS2) return ERR_OTHERCONNECTIONREQUIRED;
+
+ s->Phone.Data.PhoneString = PhoneMode;
+ req[4] = mode;
+
+ while (1) {
+ smprintf(s,"Going to phone mode %i\n",mode);
+ error = GSM_WaitFor (s, req, 6, 0x15, 4, ID_Reset);
+ if (error != ERR_NONE) return error;
+ for (i=0;i<20;i++) {
+ error=DCT4_GetPhoneMode(s);
+ if (error != ERR_NONE) return error;
+ if (PhoneMode[0] == mode) return ERR_NONE;
+ my_sleep(500);
+ }
+ }
+ return ERR_NONE;
+}
+
+GSM_Error DCT4_ReplyGetIMEI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ memcpy(s->Phone.Data.IMEI,msg.Buffer + 10, 16);
+ smprintf(s, "Received IMEI %s\n",s->Phone.Data.IMEI);
+ return ERR_NONE;
+}
+
+GSM_Error DCT4_GetIMEI (GSM_StateMachine *s)
+{
+ unsigned char req[5] = {N6110_FRAME_HEADER, 0x00, 0x41};
+
+ smprintf(s, "Getting IMEI\n");
+ return GSM_WaitFor (s, req, 5, 0x1B, 2, ID_GetIMEI);
+}
+
+GSM_Error DCT4_GetHardware(GSM_StateMachine *s, char *value)
+{
+ return NOKIA_GetPhoneString(s,"\x00\x03\x02\x07\x00\x02",6,0x1b,value,ID_GetHardware,10);
+}
+
+GSM_Error DCT4_GetProductCode(GSM_StateMachine *s, char *value)
+{
+ return NOKIA_GetPhoneString(s,"\x00\x03\x04\x0b\x00\x02",6,0x1b,value,ID_GetProductCode,10);
+}
+
+GSM_Error DCT4_Reset(GSM_StateMachine *s, bool hard)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x05,
+ 0x80, /* 0x80 - reset, 0x00 - off */
+ 0x00};
+// unsigned char TimeReq[] = {N6110_FRAME_HEADER, 0x0E, 0x00, 0x00};
+
+ if (hard) return ERR_NOTSUPPORTED;
+
+// error = DCT4_SetPhoneMode(s, DCT4_MODE_TEST);
+// if (error != ERR_NONE) return error;
+// error = DCT4_SetPhoneMode(s, DCT4_MODE_NORMAL);
+// if (error != ERR_NONE) return error;
+
+ s->Phone.Data.EnableIncomingSMS = false;
+ s->Phone.Data.EnableIncomingCB = false;
+
+ return GSM_WaitFor (s, req, 6, 0x15, 2, ID_Reset);
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/dct4func.h b/gammu/emb/common/phone/nokia/dct4/dct4func.h
new file mode 100644
index 0000000..ad802e8
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/dct4func.h
@@ -0,0 +1,30 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef dct4func_h
+#define dct4func_h
+
+#include "../ncommon.h"
+#include "../../../service/sms/gsmsms.h"
+
+typedef enum {
+ DCT4_MODE_NORMAL = 0x01,
+ DCT4_MODE_TEST = 0x04,
+ DCT4_MODE_LOCAL = 0x05
+} DCT4_PHONE_MODE;
+
+GSM_Error DCT4_ReplyGetPhoneMode (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT4_ReplySetPhoneMode (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT4_ReplyGetIMEI (GSM_Protocol_Message msg, GSM_StateMachine *s);
+
+GSM_Error DCT4_GetPhoneMode (GSM_StateMachine *s);
+GSM_Error DCT4_SetPhoneMode (GSM_StateMachine *s, DCT4_PHONE_MODE mode);
+GSM_Error DCT4_GetIMEI (GSM_StateMachine *s);
+GSM_Error DCT4_GetHardware (GSM_StateMachine *s, char *value);
+GSM_Error DCT4_GetProductCode (GSM_StateMachine *s, char *value);
+GSM_Error DCT4_Reset (GSM_StateMachine *s, bool hard);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n3320.c b/gammu/emb/common/phone/nokia/dct4/n3320.c
new file mode 100644
index 0000000..51e6f18
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n3320.c
@@ -0,0 +1,271 @@
+/* (c) 2004 by Marcin Wiacek */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA3320
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../gsmcomon.h"
+#include "../../../misc/coding/coding.h"
+#include "../../../service/gsmlogo.h"
+#include "../nfunc.h"
+#include "../nfuncold.h"
+#include "../../pfunc.h"
+#include "n3320.h"
+
+static GSM_Error N3320_ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phonebook entry received\n");
+ switch (msg.Buffer[6]) {
+ case 0x0f:
+ return N71_65_ReplyGetMemoryError(msg.Buffer[10], s);
+ default:
+ return N71_65_DecodePhonebook(s, s->Phone.Data.Memory, s->Phone.Data.Bitmap, s->Phone.Data.SpeedDial, msg.Buffer+22, msg.Length-22,true);
+ }
+ return ERR_UNKNOWN;
+}
+
+static GSM_Error N3320_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07, 0x01, 0x01, 0x00, 0x01,
+ 0xfe, 0x10, /* memory type */
+ 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x01, /* location */
+ 0x00, 0x00, 0x01};
+
+ req[9] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (entry->MemoryType == MEM_SM) return ERR_NOTSUPPORTED;
+ if (req[9]==0xff) return ERR_NOTSUPPORTED;
+
+ if (entry->Location==0x00) return ERR_INVALIDLOCATION;
+
+ req[14] = entry->Location / 256;
+ req[15] = entry->Location % 256;
+
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, 19, 0x03, 4, ID_GetMemory);
+}
+
+static GSM_Error N3320_ReplyGetMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Memory status received\n");
+ /* Quess ;-)) */
+ if (msg.Buffer[14]==0x10) {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[18]*256 + msg.Buffer[19];
+ } else {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[17];
+ }
+ smprintf(s, "Size : %i\n",Data->MemoryStatus->MemoryFree);
+ Data->MemoryStatus->MemoryUsed = msg.Buffer[20]*256 + msg.Buffer[21];
+ smprintf(s, "Used : %i\n",Data->MemoryStatus->MemoryUsed);
+ Data->MemoryStatus->MemoryFree -= Data->MemoryStatus->MemoryUsed;
+ smprintf(s, "Free : %i\n",Data->MemoryStatus->MemoryFree);
+ return ERR_NONE;
+}
+
+static GSM_Error N3320_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x03, 0x02,
+ 0x00, /* memory type */
+ 0x55, 0x55, 0x55, 0x00};
+
+ req[5] = NOKIA_GetMemoryType(s, Status->MemoryType,N71_65_MEMORY_TYPES);
+ if (Status->MemoryType == MEM_SM) return ERR_NOTSUPPORTED;
+ if (req[5]==0xff) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.MemoryStatus=Status;
+ smprintf(s, "Getting memory status\n");
+ return GSM_WaitFor (s, req, 10, 0x03, 4, ID_GetMemoryStatus);
+}
+
+static GSM_Error N3320_ReplyGetDateTime(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Date & time received\n");
+ if (msg.Buffer[4]==0x01) {
+ NOKIA_DecodeDateTime(s, msg.Buffer+10, s->Phone.Data.DateTime);
+ return ERR_NONE;
+ }
+ smprintf(s, "Not set in phone\n");
+ return ERR_EMPTY;
+}
+
+static GSM_Error N3320_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0A, 0x00, 0x00};
+
+ s->Phone.Data.DateTime=date_time;
+ smprintf(s, "Getting date & time\n");
+ return GSM_WaitFor (s, req, 6, 0x19, 4, ID_GetDateTime);
+}
+
+static GSM_Error N3320_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+ return N71_65_GetNextCalendar1(s,Note,start,&s->Phone.Data.Priv.N3320.LastCalendar,&s->Phone.Data.Priv.N3320.LastCalendarYear,&s->Phone.Data.Priv.N3320.LastCalendarPos);
+}
+
+static GSM_Error N3320_GetCalendarStatus(GSM_StateMachine *s, GSM_CalendarStatus *Status)
+{
+ GSM_Error error;
+
+ /* Method 1 */
+ error=N71_65_GetCalendarInfo1(s, &s->Phone.Data.Priv.N3320.LastCalendar);
+ if (error!=ERR_NONE) return error;
+ Status->Used = s->Phone.Data.Priv.N3320.LastCalendar.Number;
+ return ERR_NONE;
+}
+
+static GSM_Error N3320_ReplyGetCalendarInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return N71_65_ReplyGetCalendarInfo1(msg, s, &s->Phone.Data.Priv.N3320.LastCalendar);
+}
+
+static GSM_Reply_Function N3320ReplyFunctions[] = {
+ {N3320_ReplyGetMemoryStatus, "\x03",0x03,0x04,ID_GetMemoryStatus },
+ {N3320_ReplyGetMemory, "\x03",0x03,0x08,ID_GetMemory },
+
+ {N71_65_ReplyGetNextCalendar1, "\x13",0x03,0x1A,ID_GetCalendarNote },/*method 1*/
+ {N3320_ReplyGetCalendarInfo, "\x13",0x03,0x3B,ID_GetCalendarNotesInfo},/*method 1*/
+
+ {N3320_ReplyGetDateTime, "\x19",0x03,0x0B,ID_GetDateTime },
+
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetFirmware },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N3320Phone = {
+ "3320",
+ N3320ReplyFunctions,
+ NONEFUNCTION, /* Initialise */
+ NONEFUNCTION, /* Terminate */
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ NOTSUPPORTED, /* GetIMEI */
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ NOTSUPPORTED, /* GetProductCode */
+ NOTSUPPORTED, /* GetHardware */
+ NOTSUPPORTED, /* GetPPM */
+ NOTSUPPORTED, /* GetSIMIMSI */
+ N3320_GetDateTime,
+ NOTSUPPORTED, /* SetDateTime */
+ NOTSUPPORTED, /* GetAlarm */
+ NOTSUPPORTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTSUPPORTED, /* PressKey */
+ NOTSUPPORTED, /* Reset */
+ NOTSUPPORTED, /* ResetPhoneSettings */
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTSUPPORTED, /* SetAutoNetworkLogin */
+ NOTSUPPORTED, /* GetBatteryCharge */
+ NOTSUPPORTED, /* GetSignalQuality */
+ NOTSUPPORTED, /* GetNetworkInfo */
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ N3320_GetMemoryStatus,
+ N3320_GetMemory,
+ NOTSUPPORTED, /* GetNextMemory */
+ NOTSUPPORTED, /* SetMemory */
+ NOTSUPPORTED, /* AddMemory */
+ NOTSUPPORTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ NOTSUPPORTED, /* GetSMSC */
+ NOTSUPPORTED, /* SetSMSC */
+ NOTSUPPORTED, /* GetSMSStatus */
+ NOTSUPPORTED, /* GetSMS */
+ NOTSUPPORTED, /* GetNextSMS */
+ NOTSUPPORTED, /* SetSMS */
+ NOTSUPPORTED, /* AddSMS */
+ NOTSUPPORTED, /* DeleteSMS */
+ NOTSUPPORTED, /* SendSMS */
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOTSUPPORTED, /* SetIncomingSMS */
+ NOTSUPPORTED, /* SetIncomingCB */
+ NOTSUPPORTED, /* GetSMSFolders */
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ NOTIMPLEMENTED, /* DialVoice */
+ NOTIMPLEMENTED, /* AnswerCall */
+ NOTIMPLEMENTED, /* CancelCall */
+ NOTIMPLEMENTED, /* HoldCall */
+ NOTIMPLEMENTED, /* UnholdCall */
+ NOTIMPLEMENTED, /* ConferenceCall */
+ NOTIMPLEMENTED, /* SplitCall */
+ NOTIMPLEMENTED, /* TransferCall */
+ NOTIMPLEMENTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NOTIMPLEMENTED, /* SetIncomingCall */
+ NOTIMPLEMENTED, /* SetIncomingUSSD */
+ NOTSUPPORTED, /* SendDTMF */
+ NOTSUPPORTED, /* GetRingtone */
+ NOTSUPPORTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTIMPLEMENTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ NOTSUPPORTED, /* GetBitmap */
+ NOTSUPPORTED, /* SetBitmap */
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ N3320_GetCalendarStatus,
+ NOTIMPLEMENTED, /* GetCalendar */
+ N3320_GetNextCalendar,
+ NOTIMPLEMENTED, /* SetCalendar */
+ NOTSUPPORTED, /* AddCalendar */
+ NOTSUPPORTED, /* DeleteCalendar */
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTIMPLEMENTED, /* AddFilePart */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTIMPLEMENTED, /* DeleteFile */
+ NOTIMPLEMENTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n3320.h b/gammu/emb/common/phone/nokia/dct4/n3320.h
new file mode 100644
index 0000000..13825a9
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n3320.h
@@ -0,0 +1,30 @@
+/* (c) 2004 by Marcin Wiacek */
+
+#ifndef n3320_h
+#define n3320_h
+
+#include "../ncommon.h"
+#include "../../../service/sms/gsmsms.h"
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+
+typedef struct {
+ int LastCalendarYear;
+ int LastCalendarPos;
+ GSM_NOKIACalToDoLocations LastCalendar;
+ int FirstCalendarPos;
+ unsigned char CalendarIcons[10];
+ GSM_CalendarNoteType CalendarIconsTypes[10];
+ int CalendarIconsNum;
+} GSM_Phone_N3320Data;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n3650.c b/gammu/emb/common/phone/nokia/dct4/n3650.c
new file mode 100644
index 0000000..2da55bf
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n3650.c
@@ -0,0 +1,392 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA3650
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../gsmcomon.h"
+#include "../../../misc/coding/coding.h"
+#include "../../../service/gsmlogo.h"
+#include "../nfunc.h"
+#include "../nfuncold.h"
+#include "../../pfunc.h"
+#include "dct4func.h"
+#include "n3650.h"
+
+static GSM_Error N3650_ReplyGetFilePart(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int old;
+
+ smprintf(s,"File part received\n");
+ old = s->Phone.Data.File->Used;
+
+ if (msg.Length < 10) {
+ if (old == 0) return ERR_UNKNOWN;
+ return ERR_EMPTY;
+ }
+
+ s->Phone.Data.File->Used += msg.Buffer[10]*256*256*256+
+ msg.Buffer[11]*256*256+
+ msg.Buffer[12]*256+
+ msg.Buffer[13];
+ smprintf(s,"Length: %i\n",
+ msg.Buffer[10]*256*256*256+
+ msg.Buffer[11]*256*256+
+ msg.Buffer[12]*256+
+ msg.Buffer[13]);
+ s->Phone.Data.File->Buffer = (unsigned char *)realloc(s->Phone.Data.File->Buffer,s->Phone.Data.File->Used);
+ memcpy(s->Phone.Data.File->Buffer+old,msg.Buffer+18,s->Phone.Data.File->Used-old);
+ if (s->Phone.Data.File->Used-old < 0x03 * 256 + 0xD4) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+static GSM_Error N3650_GetFilePart(GSM_StateMachine *s, GSM_File *File)
+{
+ unsigned int len=10,i;
+ GSM_Error error;
+ unsigned char StartReq[500] = {
+ N7110_FRAME_HEADER, 0x0D, 0x10, 0x01, 0x07,
+ 0x24, /* len1 */
+ 0x12, /* len2 */
+ 0x0E, /* len3 */
+ 0x00}; /* File name */
+ unsigned char ContinueReq[] = {
+ N7110_FRAME_HEADER, 0x0D, 0x20, 0x01, 0xF0,
+ 0x08, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ if (File->Used == 0) {
+ sprintf(StartReq+10,"%s",File->ID_FullName);
+ len+=strlen(File->ID_FullName)-1;
+ StartReq[7] = strlen(File->ID_FullName) + 3;
+
+ StartReq[8] = strlen(File->ID_FullName);
+ StartReq[9] = 0;
+ while (File->ID_FullName[StartReq[8]] != '\\') {
+ StartReq[8]--;
+ StartReq[9]++;
+ }
+ for (i=StartReq[8];i<strlen(File->ID_FullName);i++) {
+ StartReq[i+10] = StartReq[i+1+10];
+ }
+ StartReq[9]--;
+
+ EncodeUnicode(File->Name,File->ID_FullName+StartReq[8]+1,StartReq[9]);
+ File->Folder = false;
+
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_TEST);
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.File = File;
+ return GSM_WaitFor (s, StartReq, len, 0x58, 4, ID_GetFile);
+ }
+
+ s->Phone.Data.File = File;
+ error = GSM_WaitFor (s, ContinueReq, 14, 0x58, 4, ID_GetFile);
+
+// if (error == GE_EMPTY) {
+// error = DCT4_SetPhoneMode(s, DCT4_MODE_NORMAL);
+// if (error != ERR_NONE) return error;
+// return GE_EMPTY;
+// }
+
+ return error;
+}
+
+static GSM_Error N3650_ReplyGetFolderInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_File *File = s->Phone.Data.FileInfo;
+ GSM_Phone_N3650Data *Priv = &s->Phone.Data.Priv.N3650;
+ int i,pos = 6;
+
+ i = Priv->FilesLocationsUsed-1;
+ while (1) {
+ if (i==Priv->FilesLocationsCurrent-1) break;
+ dbgprintf("Copying %i to %i, max %i, current %i\n",
+ i,i+msg.Buffer[5],
+ Priv->FilesLocationsUsed,Priv->FilesLocationsCurrent);
+ memcpy(Priv->Files[i+msg.Buffer[5]],Priv->Files[i],sizeof(GSM_File));
+ i--;
+ }
+ Priv->FileEntries = msg.Buffer[5];
+ Priv->FilesLocationsUsed += msg.Buffer[5];
+ for (i=0;i<msg.Buffer[5];i++) {
+ Priv->Files[Priv->FilesLocationsCurrent+i]->Folder = true;
+ if (msg.Buffer[pos+2] == 0x01) {
+ Priv->Files[Priv->FilesLocationsCurrent+i]->Folder = false;
+ smprintf(s,"File ");
+ }
+ EncodeUnicode(Priv->Files[Priv->FilesLocationsCurrent+i]->Name,msg.Buffer+pos+9,msg.Buffer[pos+8]);
+ smprintf(s,"%s\n",DecodeUnicodeString(Priv->Files[Priv->FilesLocationsCurrent+i]->Name));
+ Priv->Files[Priv->FilesLocationsCurrent+i]->Level = File->Level+1;
+ sprintf(Priv->Files[Priv->FilesLocationsCurrent+i]->ID_FullName,"%s\\%s",File->ID_FullName,msg.Buffer+pos+9);
+ pos+=msg.Buffer[pos+1];
+ }
+ dbgprintf("\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N3650_GetFolderInfo(GSM_StateMachine *s, GSM_File *File)
+{
+ int len=10;
+ unsigned char req[500] = {
+ N7110_FRAME_HEADER, 0x0B, 0x00, 0x01, 0x07,
+ 0x18, /* folder name length + 6 */
+ 0x12, /* folder name length */
+ 0x00,
+ 0x00}; /* folder name */
+
+ sprintf(req+10,File->ID_FullName);
+ len +=strlen(File->ID_FullName);
+ req[7] = strlen(File->ID_FullName) + 6;
+ req[8] = strlen(File->ID_FullName);
+ req[len++] = 0x00;
+ req[len++] = 0x00;
+
+ s->Phone.Data.FileInfo = File;
+ return GSM_WaitFor (s, req, len, 0x58, 4, ID_GetFile);
+}
+
+static GSM_Error N3650_GetNextFileFolder(GSM_StateMachine *s, GSM_File *File, bool start)
+{
+ GSM_Error error;
+ GSM_Phone_N3650Data *Priv = &s->Phone.Data.Priv.N3650;
+
+ if (start) {
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_LOCAL);
+ if (error != ERR_NONE) return error;
+
+ Priv->Files[0]->Folder = true;
+ Priv->Files[0]->Level = 1;
+ Priv->Files[0]->Name[0] = 0;
+ Priv->Files[0]->Name[1] = 0;
+ Priv->Files[0]->ID_FullName[0] = 'Z';
+ Priv->Files[0]->ID_FullName[1] = ':';
+ Priv->Files[0]->ID_FullName[2] = 0;
+
+ Priv->Files[1]->Folder = true;
+ Priv->Files[1]->Level = 1;
+ Priv->Files[1]->Name[0] = 0;
+ Priv->Files[1]->Name[1] = 0;
+ Priv->Files[1]->ID_FullName[0] = 'E';
+ Priv->Files[1]->ID_FullName[1] = ':';
+ Priv->Files[1]->ID_FullName[2] = 0;
+
+ Priv->Files[2]->Folder = true;
+ Priv->Files[2]->Level = 1;
+ Priv->Files[2]->Name[0] = 0;
+ Priv->Files[2]->Name[1] = 0;
+ Priv->Files[2]->ID_FullName[0] = 'C';
+ Priv->Files[2]->ID_FullName[1] = ':';
+ Priv->Files[2]->ID_FullName[2] = 0;
+
+ Priv->FilesLocationsUsed = 3;
+ Priv->FilesLocationsCurrent = 0;
+ Priv->FileLev = 1;
+ }
+
+ if (Priv->FilesLocationsCurrent == Priv->FilesLocationsUsed) {
+// error = DCT4_SetPhoneMode(s, DCT4_MODE_NORMAL);
+// if (error != ERR_NONE) return error;
+
+ return ERR_EMPTY;
+ }
+
+ strcpy(File->ID_FullName,Priv->Files[Priv->FilesLocationsCurrent]->ID_FullName);
+ File->Level = Priv->Files[Priv->FilesLocationsCurrent]->Level;
+ File->Folder = Priv->Files[Priv->FilesLocationsCurrent]->Folder;
+ CopyUnicodeString(File->Name,Priv->Files[Priv->FilesLocationsCurrent]->Name);
+ Priv->FilesLocationsCurrent++;
+
+ if (!File->Folder) return ERR_NONE;
+
+ if (Priv->FilesLocationsCurrent > 1) {
+ if (File->ID_FullName[0]!=Priv->Files[Priv->FilesLocationsCurrent-2]->ID_FullName[0]) {
+ if (File->ID_FullName[0] == 'E') {
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_TEST);
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_TEST);
+ }
+ if (File->ID_FullName[0] == 'C') {
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_LOCAL);
+ error = DCT4_SetPhoneMode(s, DCT4_MODE_LOCAL);
+ }
+// if (error != ERR_NONE) return error;
+ }
+ }
+
+ File->ReadOnly = false;
+ File->System = false;
+ File->Protected = false;
+ File->Hidden = false;
+
+ return N3650_GetFolderInfo(s, File);
+}
+
+static GSM_Error N3650_Initialise (GSM_StateMachine *s)
+{
+ GSM_Phone_N3650Data *Priv = &s->Phone.Data.Priv.N3650;
+ int i;
+
+ for (i=0;i<10000;i++) {
+ Priv->Files[i] = malloc(sizeof(GSM_File));
+ if (Priv->Files[i] == NULL) return ERR_MOREMEMORY;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N3650_Terminate(GSM_StateMachine *s)
+{
+ GSM_Phone_N3650Data *Priv = &s->Phone.Data.Priv.N3650;
+ int i;
+
+ for (i=0;i<10000;i++) free(Priv->Files[i]);
+ return ERR_NONE;
+}
+
+static GSM_Reply_Function N3650ReplyFunctions[] = {
+ {DCT4_ReplySetPhoneMode, "\x15",0x03,0x64,ID_Reset },
+ {DCT4_ReplyGetPhoneMode, "\x15",0x03,0x65,ID_Reset },
+ {NoneReply, "\x15",0x03,0x68,ID_Reset },
+
+ {DCT4_ReplyGetIMEI, "\x1B",0x03,0x01,ID_GetIMEI },
+ {NOKIA_ReplyGetPhoneString, "\x1B",0x03,0x08,ID_GetHardware },
+ {NOKIA_ReplyGetPhoneString, "\x1B",0x03,0x0C,ID_GetProductCode },
+
+ {N3650_ReplyGetFolderInfo, "\x58",0x03,0x0C,ID_GetFile },
+ {N3650_ReplyGetFilePart, "\x58",0x03,0x0E,ID_GetFile },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N3650Phone = {
+ "3650|NGAGE",
+ N3650ReplyFunctions,
+ N3650_Initialise,
+ N3650_Terminate,
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ DCT4_GetIMEI,
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ DCT4_GetProductCode,
+ DCT4_GetHardware,
+ NOTSUPPORTED, /* GetPPM */
+ NOTSUPPORTED, /* GetSIMIMSI */
+ NOTSUPPORTED, /* GetDateTime */
+ NOTSUPPORTED, /* SetDateTime */
+ NOTSUPPORTED, /* GetAlarm */
+ NOTSUPPORTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTSUPPORTED, /* PressKey */
+ DCT4_Reset,
+ NOTSUPPORTED, /* ResetPhoneSettings */
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTSUPPORTED, /* SetAutoNetworkLogin */
+ NOTSUPPORTED, /* GetBatteryCharge */
+ NOTSUPPORTED, /* GetSignalQuality */
+ NOTSUPPORTED, /* GetNetworkInfo */
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ NOTSUPPORTED, /* GetMemoryStatus */
+ NOTSUPPORTED, /* GetMemory */
+ NOTSUPPORTED, /* GetNextMemory */
+ NOTSUPPORTED, /* SetMemory */
+ NOTSUPPORTED, /* AddMemory */
+ NOTSUPPORTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ NOTSUPPORTED, /* GetSMSC */
+ NOTSUPPORTED, /* SetSMSC */
+ NOTSUPPORTED, /* GetSMSStatus */
+ NOTSUPPORTED, /* GetSMS */
+ NOTSUPPORTED, /* GetNextSMS */
+ NOTSUPPORTED, /* SetSMS */
+ NOTSUPPORTED, /* AddSMS */
+ NOTSUPPORTED, /* DeleteSMS */
+ NOTSUPPORTED, /* SendSMS */
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOTSUPPORTED, /* SetIncomingSMS */
+ NOTSUPPORTED, /* SetIncomingCB */
+ NOTSUPPORTED, /* GetSMSFolders */
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ NOTIMPLEMENTED, /* DialVoice */
+ NOTIMPLEMENTED, /* AnswerCall */
+ NOTIMPLEMENTED, /* CancelCall */
+ NOTIMPLEMENTED, /* HoldCall */
+ NOTIMPLEMENTED, /* UnholdCall */
+ NOTIMPLEMENTED, /* ConferenceCall */
+ NOTIMPLEMENTED, /* SplitCall */
+ NOTIMPLEMENTED, /* TransferCall */
+ NOTIMPLEMENTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NOTIMPLEMENTED, /* SetIncomingCall */
+ NOTIMPLEMENTED, /* SetIncomingUSSD */
+ NOTSUPPORTED, /* SendDTMF */
+ NOTSUPPORTED, /* GetRingtone */
+ NOTSUPPORTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTIMPLEMENTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ NOTSUPPORTED, /* GetBitmap */
+ NOTSUPPORTED, /* SetBitmap */
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ NOTIMPLEMENTED, /* GetCalendarStatus */
+ NOTIMPLEMENTED, /* GetCalendar */
+ NOTSUPPORTED, /* GetNextCalendar */
+ NOTIMPLEMENTED, /* SetCalendar */
+ NOTSUPPORTED, /* AddCalendar */
+ NOTSUPPORTED, /* DeleteCalendar */
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ N3650_GetNextFileFolder,
+ N3650_GetFilePart,
+ NOTIMPLEMENTED, /* AddFilePart */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTIMPLEMENTED, /* DeleteFile */
+ NOTIMPLEMENTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n3650.h b/gammu/emb/common/phone/nokia/dct4/n3650.h
new file mode 100644
index 0000000..7769ba5
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n3650.h
@@ -0,0 +1,30 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef n3650_h
+#define n3650_h
+
+#include "../../../gsmcomon.h"
+#include "../../../gsmstate.h"
+#include "../../../service/gsmmisc.h"
+#include "../../../service/sms/gsmsms.h"
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+
+typedef struct {
+ int FileLev;
+ int FilesLocationsUsed;
+ int FilesLocationsCurrent;
+ GSM_File *Files[10000];
+ int FileEntries;
+} GSM_Phone_N3650Data;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n6510.c b/gammu/emb/common/phone/nokia/dct4/n6510.c
new file mode 100644
index 0000000..67fe492
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n6510.c
@@ -0,0 +1,5782 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* based on some work from Markus Plail, Pawel Kot and Gnokii */
+/* function for making CRC for filesystem (c) 2003 by Michael Schroeder */
+
+#include "../../../gsmstate.h"
+
+#ifdef GSM_ENABLE_NOKIA6510
+
+#include <string.h>
+#include <time.h>
+
+#include "../../../misc/coding/coding.h"
+#include "../../../gsmcomon.h"
+#include "../../../service/gsmlogo.h"
+#include "../nfunc.h"
+#include "../nfuncold.h"
+#include "../../pfunc.h"
+#include "dct4func.h"
+#include "n6510.h"
+
+static GSM_Error N6510_Initialise (GSM_StateMachine *s)
+{
+ s->Phone.Data.Priv.N6510.CalendarIconsNum = 0;
+
+ /* Enables various things like incoming SMS, call info, etc. */
+ return N71_65_EnableFunctions (s, "\x01\x02\x06\x0A\x14\x17\x39", 7);
+}
+
+static GSM_Error N6510_ReplyGetMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phonebook entry received\n");
+ switch (msg.Buffer[6]) {
+ case 0x0f:
+ return N71_65_ReplyGetMemoryError(msg.Buffer[10], s);
+ default:
+ return N71_65_DecodePhonebook(s, s->Phone.Data.Memory, s->Phone.Data.Bitmap, s->Phone.Data.SpeedDial, msg.Buffer+22, msg.Length-22,false);
+ }
+ return ERR_UNKNOWN;
+}
+
+static GSM_Error N6510_GetMemory (GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x07, 0x01, 0x01, 0x00, 0x01,
+ 0xfe, 0x10, /* memory type */
+ 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x01, /* location */
+ 0x00, 0x00, 0x01};
+
+ req[9] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[9]==0xff) return ERR_NOTSUPPORTED;
+
+ if (entry->Location==0x00) return ERR_INVALIDLOCATION;
+
+ req[14] = entry->Location / 256;
+ req[15] = entry->Location % 256;
+
+ s->Phone.Data.Memory=entry;
+ smprintf(s, "Getting phonebook entry\n");
+ return GSM_WaitFor (s, req, 19, 0x03, 4, ID_GetMemory);
+}
+
+static GSM_Error N6510_ReplyGetMemoryStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Memory status received\n");
+ /* Quess ;-)) */
+ if (msg.Buffer[14]==0x10) {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[18]*256 + msg.Buffer[19];
+ } else {
+ Data->MemoryStatus->MemoryFree = msg.Buffer[17];
+ }
+ smprintf(s, "Size : %i\n",Data->MemoryStatus->MemoryFree);
+ Data->MemoryStatus->MemoryUsed = msg.Buffer[20]*256 + msg.Buffer[21];
+ smprintf(s, "Used : %i\n",Data->MemoryStatus->MemoryUsed);
+ Data->MemoryStatus->MemoryFree -= Data->MemoryStatus->MemoryUsed;
+ smprintf(s, "Free : %i\n",Data->MemoryStatus->MemoryFree);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetMemoryStatus(GSM_StateMachine *s, GSM_MemoryStatus *Status)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x03, 0x02,
+ 0x00, /* memory type */
+ 0x55, 0x55, 0x55, 0x00};
+
+ req[5] = NOKIA_GetMemoryType(s, Status->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[5]==0xff) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.MemoryStatus=Status;
+ smprintf(s, "Getting memory status\n");
+ return GSM_WaitFor (s, req, 10, 0x03, 4, ID_GetMemoryStatus);
+}
+
+static GSM_Error N6510_ReplyGetSMSC(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i, current, j;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "SMSC received\n");
+ break;
+ case 0x02:
+ smprintf(s, "SMSC empty\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "Unknown SMSC state: %02x\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ memset(Data->SMSC,0,sizeof(GSM_SMSC));
+ Data->SMSC->Location = msg.Buffer[8];
+ Data->SMSC->Format = SMS_FORMAT_Text;
+ switch (msg.Buffer[10]) {
+ case 0x00: Data->SMSC->Format = SMS_FORMAT_Text; break;
+ case 0x22: Data->SMSC->Format = SMS_FORMAT_Fax; break;
+ case 0x26: Data->SMSC->Format = SMS_FORMAT_Pager; break;
+ case 0x32: Data->SMSC->Format = SMS_FORMAT_Email; break;
+ }
+ Data->SMSC->Validity.Format = SMS_Validity_RelativeFormat;
+ Data->SMSC->Validity.Relative = msg.Buffer[12];
+ current = 14;
+ for (i=0;i<msg.Buffer[13];i++) {
+ switch (msg.Buffer[current]) {
+ case 0x81:
+ j=current+4;
+ while (msg.Buffer[j]!=0) {j++;}
+ j=j-33;
+ if (j>GSM_MAX_SMSC_NAME_LENGTH) {
+ smprintf(s, "Too long name\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ CopyUnicodeString(Data->SMSC->Name,msg.Buffer+current+4);
+ smprintf(s, " Name \"%s\"\n", DecodeUnicodeString(Data->SMSC->Name));
+ break;
+ case 0x82:
+ switch (msg.Buffer[current+2]) {
+ case 0x01:
+ GSM_UnpackSemiOctetNumber(Data->SMSC->DefaultNumber,msg.Buffer+current+4,true);
+ smprintf(s, " Default number \"%s\"\n", DecodeUnicodeString(Data->SMSC->DefaultNumber));
+ break;
+ case 0x02:
+ GSM_UnpackSemiOctetNumber(Data->SMSC->Number,msg.Buffer+current+4,false);
+ smprintf(s, " Number \"%s\"\n", DecodeUnicodeString(Data->SMSC->Number));
+ break;
+ default:
+ smprintf(s, "Unknown SMSC number: %02x\n",msg.Buffer[current+2]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ break;
+ default:
+ smprintf(s, "Unknown SMSC block: %02x\n",msg.Buffer[current]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ current = current + msg.Buffer[current+1];
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x14,
+ 0x01, /* location */
+ 0x00};
+
+ if (smsc->Location==0x00) return ERR_INVALIDLOCATION;
+
+ req[4]=smsc->Location;
+
+ s->Phone.Data.SMSC=smsc;
+ smprintf(s, "Getting SMSC\n");
+ return GSM_WaitFor (s, req, 6, 0x02, 4, ID_GetSMSC);
+}
+
+static GSM_Error N6510_ReplySetSMSC(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "SMSC set OK\n");
+ return ERR_NONE;
+ case 0x02:
+ smprintf(s, "Invalid SMSC location\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "Unknown SMSC state: %02x\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_SetSMSC(GSM_StateMachine *s, GSM_SMSC *smsc)
+{
+ int count = 13,i;
+ unsigned char req[256] = {N6110_FRAME_HEADER,
+ 0x12, 0x55, 0x01, 0x0B, 0x34,
+ 0x05, /* Location */
+ 0x00,
+ 0x00, /* Format */
+ 0x00,
+ 0xFF}; /* Validity */
+
+ req[8] = smsc->Location;
+ switch (smsc->Format) {
+ case SMS_FORMAT_Text: req[10] = 0x00; break;
+ case SMS_FORMAT_Fax: req[10] = 0x22; break;
+ case SMS_FORMAT_Pager: req[10] = 0x26; break;
+ case SMS_FORMAT_Email: req[10] = 0x32; break;
+ }
+ req[12] = smsc->Validity.Relative;
+
+ /* We have now blocks. Number of blocks = 3 */
+ req[count++] = 0x03;
+
+ /* -------------- SMSC number ----------------- */
+ /* Block type: number */
+ req[count++] = 0x82;
+ /* Offset to next block */
+ req[count++] = 0x1A;
+ /* Type of number: SMSC number */
+ req[count++] = 0x02;
+ req[count] = GSM_PackSemiOctetNumber(smsc->Number, req+count+2, false) + 1;
+ if (req[count]>18) {
+ smprintf(s, "Too long SMSC number in frame\n");
+ return ERR_UNKNOWN;
+ }
+ req[count+1] = req[count] - 1;
+ count += 23;
+
+ /* --------------- Default number ------------- */
+ /* Block type: number */
+ req[count++] = 0x82;
+ /* Offset to next block */
+ req[count++] = 0x14;
+ /* Type of number: default number */
+ req[count++] = 0x01;
+ req[count] = GSM_PackSemiOctetNumber(smsc->DefaultNumber, req+count+2, true) + 1;
+ if (req[count]*2>12) {
+ smprintf(s, "Too long SMSC number in frame\n");
+ return ERR_UNKNOWN;
+ }
+ req[count+1] = req[count] - 1;
+ count += 17;
+
+ /* -------------- SMSC name ------------------- */
+ req[count++] = 0x81;
+ req[count++] = UnicodeLength(smsc->Name)*2 + 2 + 4;
+ req[count++] = UnicodeLength(smsc->Name)*2 + 2;
+ req[count++] = 0x00;
+ /* Can't make CopyUnicodeString(req+count,sms->Name) !!!!
+ * with MSVC6 count is changed then
+ */
+ i = count;
+ CopyUnicodeString(req+i,smsc->Name);
+ count += UnicodeLength(smsc->Name)*2 + 2;
+
+ smprintf(s, "Setting SMSC\n");
+ return GSM_WaitFor (s, req, count, 0x02, 4, ID_SetSMSC);
+}
+
+static GSM_Error N6510_ReplyGetNetworkInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int current = msg.Buffer[7]+7, tmp;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+#ifdef DEBUG
+ char name[100];
+ GSM_NetworkInfo NetInfo;
+
+ smprintf(s, "Network status: ");
+ switch (msg.Buffer[8]) {
+ case 0x00 : smprintf(s, "home network\n"); break;
+ case 0x01 : smprintf(s, "roaming network\n"); break;
+ case 0x04 : smprintf(s, "not logged"); break;
+ case 0x06 : smprintf(s, "SIM card rejected\n"); break;
+ case 0x09 : smprintf(s, "not logged"); break;
+ default : smprintf(s, "unknown %i!\n",msg.Buffer[8]); break;
+ }
+ if (msg.Buffer[8]==0x00 || msg.Buffer[8] == 0x01) {
+ NOKIA_DecodeNetworkCode(msg.Buffer + (current + 7),NetInfo.NetworkCode);
+ smprintf(s, "Network code : %s\n", NetInfo.NetworkCode);
+ smprintf(s, "Network name for Gammu : %s ",
+ DecodeUnicodeString(GSM_GetNetworkName(NetInfo.NetworkCode)));
+ smprintf(s, "(%s)\n",DecodeUnicodeString(GSM_GetCountryName(NetInfo.NetworkCode)));
+
+ sprintf(NetInfo.LAC, "%02x%02x", msg.Buffer[current+1], msg.Buffer[current+2]);
+ smprintf(s, "LAC : %s\n", NetInfo.LAC);
+
+ sprintf(NetInfo.CID, "%02x%02x", msg.Buffer[current+5], msg.Buffer[current+6]);
+ smprintf(s, "CID : %s\n", NetInfo.CID);
+
+ tmp = 10;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,name,true);
+ smprintf(s, "Network name for phone : %s\n",DecodeUnicodeString(name));
+ }
+#endif
+ if (Data->RequestID==ID_GetNetworkInfo) {
+ Data->NetworkInfo->NetworkName[0] = 0x00;
+ Data->NetworkInfo->NetworkName[1] = 0x00;
+ Data->NetworkInfo->State = 0;
+ switch (msg.Buffer[8]) {
+ case 0x00: Data->NetworkInfo->State = GSM_HomeNetwork; break;
+ case 0x01: Data->NetworkInfo->State = GSM_RoamingNetwork; break;
+ case 0x04:
+ case 0x06:
+ case 0x09: Data->NetworkInfo->State = GSM_NoNetwork; break;
+ }
+ if (Data->NetworkInfo->State == GSM_HomeNetwork || Data->NetworkInfo->State == GSM_RoamingNetwork) {
+ tmp = 10;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,Data->NetworkInfo->NetworkName,true);
+ sprintf(Data->NetworkInfo->LAC, "%02x%02x", msg.Buffer[current+1], msg.Buffer[current+2]);
+ sprintf(Data->NetworkInfo->CID, "%02x%02x", msg.Buffer[current+5], msg.Buffer[current+6]);
+ NOKIA_DecodeNetworkCode(msg.Buffer + (current+7),Data->NetworkInfo->NetworkCode);
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetNetworkInfo(GSM_StateMachine *s, GSM_NetworkInfo *netinfo)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x00, 0x00};
+
+ s->Phone.Data.NetworkInfo=netinfo;
+ smprintf(s, "Getting network info\n");
+ return GSM_WaitFor (s, req, 5, 0x0a, 4, ID_GetNetworkInfo);
+}
+
+static GSM_Error N6510_EncodeSMSFrame(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char *req, GSM_SMSMessageLayout *Layout, int *length)
+{
+ int start, count = 0, pos1, pos2, pos3, pos4, pos5;
+ GSM_Error error;
+
+ memset(Layout,255,sizeof(GSM_SMSMessageLayout));
+
+ start = *length;
+
+ req[count++] = 0x01;
+ if (sms->PDU != SMS_Deliver) {
+ req[count++] = 0x02;
+ } else {
+ req[count++] = 0x00;
+ }
+
+ pos1 = count; count++;
+ /* firstbyte set in SMS Layout */
+ Layout->firstbyte = count; count++;
+ if (sms->PDU != SMS_Deliver) {
+ Layout->TPMR = count; count++;
+
+ Layout->TPPID = count; count++;
+
+ /* TP.DCS set in SMS layout */
+ Layout->TPDCS = count; count++;
+ req[count++] = 0x00;
+ } else {
+ Layout->TPPID = count; count++;
+ /* TP.DCS set in SMS layout */
+ Layout->TPDCS = count; count++;
+ Layout->DateTime = count; count += 7;
+ req[count++] = 0x55;
+ req[count++] = 0x55;
+ req[count++] = 0x55;
+ }
+
+ /* We have now blocks. Number of blocks = 3 or 4 */
+ if (sms->PDU != SMS_Deliver) {
+ req[count++] = 0x04;
+ } else {
+ req[count++] = 0x03;
+ }
+
+ /* -------------- Phone number ------------- */
+ /* Block type: number */
+ req[count++] = 0x82;
+ /* Offset to next block */
+ req[count++] = 0x10;
+ /* Type of number: default number */
+ req[count++] = 0x01;
+ pos4 = count; count++;
+ /* now coded Number in SMS Layout */
+ Layout->Number = count; count+= 12;
+
+ /* -------------- SMSC number -------------- */
+ /* Block type: number */
+ req[count++] = 0x82;
+ /* Offset to next block */
+ req[count++] = 0x10;
+ /* Type of number: SMSC number */
+ req[count++] = 0x02;
+ pos5 = count; count++;
+ /* now coded SMSC number in SMS Layout */
+ Layout->SMSCNumber = count; count += 12;
+
+ /* -------------- SMS validity ------------- */
+ if (sms->PDU != SMS_Deliver) {
+ /* Block type: validity */
+ req[count++] = 0x08;
+ req[count++] = 0x04;
+ /* data length */
+ req[count++] = 0x01;
+ Layout->TPVP = count; count++;
+ }
+
+ /* --------------- SMS text ---------------- */
+ /* Block type: SMS text */
+ req[count++] = 0x80;
+ /* this the same as req[11] but starting from req[42] */
+ pos2 = count; count++;
+ pos3 = count; count++;
+ /* FIXME*/
+ Layout->TPUDL = count; count++;
+ /* SMS text and UDH coded in SMS Layout */
+ Layout->Text = count;
+
+ error = PHONE_EncodeSMSFrame(s,sms,req,*Layout,length,false);
+ if (error != ERR_NONE) return error;
+
+ req[pos1] = *length - 1;
+ req[pos2] = *length - Layout->Text + 6;
+ req[pos3] = *length - Layout->Text;
+
+ /* Convert number of semioctets to number of chars */
+ req[pos4] = req[Layout->Number] + 4;
+ if (req[pos4] % 2) req[pos4]++;
+ req[pos4] /= 2;
+
+ req[pos5] = req[Layout->SMSCNumber] + 1;
+
+ if (req[pos4]>12 || req[pos5]>12) {
+ smprintf(s, "Too long phone number in frame\n");
+ return ERR_UNKNOWN;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplyGetSMSFolders(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int j, num = 0, pos;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x13:
+ smprintf(s, "SMS folders names received\n");
+ Data->SMSFolders->Number = msg.Buffer[5]+2;
+ pos = 6;
+ for (j=0;j<msg.Buffer[5];j++) {
+ while (true) {
+ if (msg.Buffer[pos] == msg.Buffer[6] &&
+ msg.Buffer[pos+1] == msg.Buffer[7]) break;
+ if (pos+4 > msg.Length) return ERR_UNKNOWNRESPONSE;
+ pos++;
+ }
+ pos+=4;
+ smprintf(s, "Folder index: %02x",msg.Buffer[pos - 2]);
+ if (msg.Buffer[pos - 1]>GSM_MAX_SMS_FOLDER_NAME_LEN) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ CopyUnicodeString(Data->SMSFolders->Folder[num].Name,msg.Buffer + pos);
+ smprintf(s, ", folder name: \"%s\"\n",DecodeUnicodeString(Data->SMSFolders->Folder[num].Name));
+ Data->SMSFolders->Folder[num].InboxFolder = false;
+ Data->SMSFolders->Folder[num].Memory = MEM_ME;
+ if (num == 0x01) { /* OUTBOX SIM */
+ Data->SMSFolders->Folder[0].Memory = MEM_SM;
+ Data->SMSFolders->Folder[0].InboxFolder = true;
+
+ Data->SMSFolders->Folder[1].Memory = MEM_SM;
+
+ CopyUnicodeString(Data->SMSFolders->Folder[2].Name,Data->SMSFolders->Folder[0].Name);
+ Data->SMSFolders->Folder[2].Memory = MEM_ME;
+ Data->SMSFolders->Folder[2].InboxFolder = true;
+
+ CopyUnicodeString(Data->SMSFolders->Folder[3].Name,Data->SMSFolders->Folder[1].Name);
+ Data->SMSFolders->Folder[3].Memory = MEM_ME;
+ Data->SMSFolders->Folder[3].InboxFolder = false;
+
+ num+=2;
+ }
+ num++;
+ }
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetSMSFolders(GSM_StateMachine *s, GSM_SMSFolders *folders)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x12, 0x00, 0x00};
+
+ s->Phone.Data.SMSFolders=folders;
+ smprintf(s, "Getting SMS folders\n");
+ return GSM_WaitFor (s, req, 6, 0x14, 4, ID_GetSMSFolders);
+}
+
+static GSM_Error N6510_ReplyGetSMSFolderStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+
+ smprintf(s, "SMS folder status received\n");
+ Priv->LastSMSFolder.Number=msg.Buffer[6]*256+msg.Buffer[7];
+ smprintf(s, "Number of Entries: %i\n",Priv->LastSMSFolder.Number);
+ smprintf(s, "Locations: ");
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ Priv->LastSMSFolder.Location[i]=msg.Buffer[8+(i*2)]*256+msg.Buffer[(i*2)+9];
+ smprintf(s, "%i ",Priv->LastSMSFolder.Location[i]);
+ }
+ smprintf(s, "\n");
+ NOKIA_SortSMSFolderStatus(s, &Priv->LastSMSFolder);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetSMSFolderStatus(GSM_StateMachine *s, int folderid)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x0C,
+ 0x01, /* 0x01=SIM, 0x02=ME */
+ 0x00, /* Folder ID */
+ 0x0f, 0x55, 0x55, 0x55};
+
+ switch (folderid) {
+ case 0x01: req[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: req[5] = 0x03; break; /* OUTBOX SIM */
+ default : req[5] = folderid - 1; req[4] = 0x02; break; /* ME folders */
+ }
+
+ smprintf(s, "Getting SMS folder status\n");
+ return GSM_WaitFor (s, req, 10, 0x14, 4, ID_GetSMSFolderStatus);
+}
+
+static void N6510_GetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char *folderid, int *location)
+{
+ int ifolderid;
+
+ /* simulate flat SMS memory */
+ if (sms->Folder==0x00) {
+ ifolderid = sms->Location / PHONE_MAXSMSINFOLDER;
+ *folderid = ifolderid + 0x01;
+ *location = sms->Location - ifolderid * PHONE_MAXSMSINFOLDER;
+ } else {
+ *folderid = sms->Folder;
+ *location = sms->Location;
+ }
+ smprintf(s, "SMS folder %i & location %i -> 6510 folder %i & location %i\n",
+ sms->Folder,sms->Location,*folderid,*location);
+}
+
+static void N6510_SetSMSLocation(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char folderid, int location)
+{
+ sms->Folder = 0;
+ sms->Location = (folderid - 0x01) * PHONE_MAXSMSINFOLDER + location;
+ smprintf(s, "6510 folder %i & location %i -> SMS folder %i & location %i\n",
+ folderid,location,sms->Folder,sms->Location);
+}
+
+static GSM_Error N6510_DecodeSMSFrame(GSM_StateMachine *s, GSM_SMSMessage *sms, unsigned char *buffer)
+{
+ int i, current, blocks=0, SMSTemplateDateTime = 0;
+ GSM_SMSMessageLayout Layout;
+ GSM_Error error;
+
+ memset(&Layout,255,sizeof(GSM_SMSMessageLayout));
+ Layout.firstbyte = 2;
+ switch (buffer[0]) {
+ case 0x00:
+ smprintf(s, "SMS deliver\n");
+ sms->PDU = SMS_Deliver;
+ Layout.TPPID = 3;
+ Layout.TPDCS = 4;
+ Layout.DateTime = 5;
+ blocks = 15;
+ break;
+ case 0x01:
+ smprintf(s, "Delivery report\n");
+ sms->PDU = SMS_Status_Report;
+ Layout.TPMR = 3;
+ Layout.TPStatus = 4;
+ Layout.DateTime = 5;
+ Layout.SMSCTime = 12;
+ blocks = 19;
+ break;
+ case 0x02:
+ smprintf(s, "SMS template\n");
+ sms->PDU = SMS_Submit;
+ Layout.TPMR = 3;
+ Layout.TPPID = 4;
+ Layout.TPDCS = 5;
+ blocks = 7;
+ break;
+ }
+ current = blocks + 1;
+ for (i=0;i<buffer[blocks];i++) {
+ switch (buffer[current]) {
+ case 0x80:
+ smprintf(s, "SMS text\n");
+ if (buffer[current + 2] > buffer[current + 3]) {
+ Layout.TPUDL = current + 2;
+ } else {
+ Layout.TPUDL = current + 3;
+ }
+ Layout.Text = current + 4;
+ break;
+ case 0x82:
+ switch (buffer[current+2]) {
+ case 0x01:
+ smprintf(s, "Phone number\n");
+ Layout.Number = current + 4;
+ break;
+ case 0x02:
+ smprintf(s, "SMSC number\n");
+ Layout.SMSCNumber = current + 4;
+ break;
+ default:
+ smprintf(s, "Unknown number\n");
+ break;
+ }
+ break;
+ case 0x84:
+ smprintf(s, "Date and time of saving for SMS template\n");
+ SMSTemplateDateTime = current + 2;
+ break;
+ default:
+ smprintf(s, "Unknown block %02x\n",buffer[current]);
+ }
+ current = current + buffer[current + 1];
+ }
+ error=GSM_DecodeSMSFrame(sms,buffer,Layout);
+ if (SMSTemplateDateTime != 0) {
+ sms->PDU = SMS_Deliver;
+ NOKIA_DecodeDateTime(s, buffer+SMSTemplateDateTime, &sms->DateTime);
+ sms->DateTime.Timezone = 0;
+ }
+ return error;
+}
+
+static GSM_Error N6510_ReplyGetSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ int Width, Height;
+ unsigned char output[500]; //output2[500];
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch(msg.Buffer[3]) {
+ case 0x03:
+ smprintf(s, "SMS Message received\n");
+ Data->GetSMSMessage->Number=1;
+ NOKIA_DecodeSMSState(s, msg.Buffer[5], &Data->GetSMSMessage->SMS[0]);
+ switch (msg.Buffer[14]) {
+ case 0x00:
+ case 0x01:
+ case 0x02:
+ return N6510_DecodeSMSFrame(s, &Data->GetSMSMessage->SMS[0],msg.Buffer+14);
+ case 0xA0:
+ smprintf(s, "Picture Image\n");
+ Data->GetSMSMessage->Number = 0;
+ i = 0;
+ output[i++] = 0x30; /* Smart Messaging 3.0 */
+ output[i++] = SM30_OTA;
+ output[i++] = 0x01; /* Length */
+ output[i++] = 0x00; /* Length */
+ output[i++] = 0x00;
+ PHONE_GetBitmapWidthHeight(GSM_NokiaPictureImage, &Width, &Height);
+ output[i++] = Width;
+ output[i++] = Height;
+ output[i++] = 0x01;
+ memcpy(output+i,msg.Buffer+30,PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0));
+ i = i + PHONE_GetBitmapSize(GSM_NokiaPictureImage,0,0);
+// if (msg.Length!=282) {
+// output[i++] = SM30_UNICODETEXT;
+// output[i++] = 0;
+// output[i++] = 0; /* Length - later changed */
+// GSM_UnpackEightBitsToSeven(0, msg.Length-282, msg.Length-304, msg.Buffer+282,output2);
+// DecodeDefault(output+i, output2, msg.Length - 282, true);
+// output[i - 1] = UnicodeLength(output+i) * 2;
+// i = i + output[i-1];
+// }
+ GSM_MakeMultiPartSMS(Data->GetSMSMessage,output,i,UDH_NokiaProfileLong,SMS_Coding_8bit,1,0);
+ for (i=0;i<3;i++) {
+ Data->GetSMSMessage->SMS[i].Number[0]=0;
+ Data->GetSMSMessage->SMS[i].Number[1]=0;
+ }
+ if (Data->Bitmap != NULL) {
+ Data->Bitmap->Location = 0;
+ PHONE_GetBitmapWidthHeight(GSM_NokiaPictureImage, &Width, &Height);
+ Data->Bitmap->BitmapWidth = Width;
+ Data->Bitmap->BitmapHeight = Height;
+ PHONE_DecodeBitmap(GSM_NokiaPictureImage, msg.Buffer + 30, Data->Bitmap);
+ Data->Bitmap->Sender[0] = 0x00;
+ Data->Bitmap->Sender[1] = 0x00;
+ Data->Bitmap->Text[0] = 0;
+ Data->Bitmap->Text[1] = 0;
+ }
+ return ERR_NONE;
+ default:
+ smprintf(s, "Unknown SMS type: %i\n",msg.Buffer[8]);
+ }
+ break;
+ case 0x0f:
+ smprintf(s, "SMS message info received\n");
+ CopyUnicodeString(Data->GetSMSMessage->SMS[0].Name,msg.Buffer+52);
+ smprintf(s, "Name: \"%s\"\n",DecodeUnicodeString(Data->GetSMSMessage->SMS[0].Name));
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_PrivGetSMSMessageBitmap(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, GSM_Bitmap *bitmap)
+{
+ GSM_Error error;
+ unsigned char folderid,namebuffer[200];
+ int location;
+ int i;
+ unsigned char req[] = {
+ N6110_FRAME_HEADER,
+ 0x02, /* msg type: 0x02 for getting sms, 0x0e for sms status */
+ 0x01, /* 0x01=SIM, 0x02=ME */
+ 0x00, /* FolderID */
+ 0x00, 0x02, /* Location */
+ 0x01, 0x00};
+
+ N6510_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+
+ switch (folderid) {
+ case 0x01: req[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: req[5] = 0x03; break; /* OUTBOX SIM */
+ default : req[5] = folderid - 1; req[4] = 0x02; break; /* ME folders */
+ }
+ req[6]=location / 256;
+ req[7]=location;
+
+ s->Phone.Data.GetSMSMessage = sms;
+ s->Phone.Data.Bitmap = bitmap;
+ smprintf(s, "Getting sms message info\n");
+ req[3] = 0x0e; req[8] = 0x55; req[9] = 0x55;
+ error=GSM_WaitFor (s, req, 10, 0x14, 4, ID_GetSMSMessage);
+ if (error!=ERR_NONE) return error;
+ CopyUnicodeString(namebuffer,sms->SMS[0].Name);
+
+ smprintf(s, "Getting sms\n");
+ req[3] = 0x02; req[8] = 0x01; req[9] = 0x00;
+ error=GSM_WaitFor (s, req, 10, 0x14, 4, ID_GetSMSMessage);
+ if (error==ERR_NONE) {
+ for (i=0;i<sms->Number;i++) {
+ N6510_SetSMSLocation(s, &sms->SMS[i], folderid, location);
+ sms->SMS[i].Folder = folderid;
+ sms->SMS[i].InboxFolder = true;
+ if (folderid != 0x01 && folderid != 0x03) sms->SMS[i].InboxFolder = false;
+ sms->SMS[i].Memory = MEM_ME;
+ if (folderid == 0x01 || folderid == 0x02) sms->SMS[i].Memory = MEM_SM;
+ CopyUnicodeString(sms->SMS[i].Name,namebuffer);
+ }
+ }
+ return error;
+}
+
+static GSM_Error N6510_GetSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms)
+{
+ GSM_Error error;
+ unsigned char folderid;
+ int location;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ int i;
+ bool found = false;
+
+ N6510_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+ error=N6510_GetSMSFolderStatus(s, folderid);
+ if (error!=ERR_NONE) return error;
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ if (Priv->LastSMSFolder.Location[i]==location) {
+ found = true;
+ break;
+ }
+ }
+ if (!found) return ERR_EMPTY;
+ return N6510_PrivGetSMSMessageBitmap(s,sms,NULL);
+}
+
+static GSM_Error N6510_GetNextSMSMessageBitmap(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start, GSM_Bitmap *bitmap)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ unsigned char folderid;
+ int location;
+ GSM_Error error;
+ int i;
+ bool findnextfolder = false;
+
+ if (start) {
+ folderid = 0x00;
+ findnextfolder = true;
+ error=N6510_GetSMSFolders(s,&Priv->LastSMSFolders);
+ if (error!=ERR_NONE) return error;
+ } else {
+ N6510_GetSMSLocation(s, &sms->SMS[0], &folderid, &location);
+ for (i=0;i<Priv->LastSMSFolder.Number;i++) {
+ if (Priv->LastSMSFolder.Location[i]==location) break;
+ }
+ /* Is this last location in this folder ? */
+ if (i==Priv->LastSMSFolder.Number-1) {
+ findnextfolder=true;
+ } else {
+ location=Priv->LastSMSFolder.Location[i+1];
+ }
+ }
+ if (findnextfolder) {
+ Priv->LastSMSFolder.Number=0;
+ while (Priv->LastSMSFolder.Number==0) {
+ folderid++;
+ /* Too high folder number */
+ if ((folderid-1)>Priv->LastSMSFolders.Number) return ERR_EMPTY;
+ /* Get next folder status */
+ error=N6510_GetSMSFolderStatus(s, folderid);
+ if (error!=ERR_NONE) return error;
+ /* First location from this folder */
+ location=Priv->LastSMSFolder.Location[0];
+ }
+ }
+ N6510_SetSMSLocation(s, &sms->SMS[0], folderid, location);
+
+ return N6510_PrivGetSMSMessageBitmap(s, sms, bitmap);
+}
+
+static GSM_Error N6510_GetNextSMSMessage(GSM_StateMachine *s, GSM_MultiSMSMessage *sms, bool start)
+{
+ return N6510_GetNextSMSMessageBitmap(s, sms, start, NULL);
+}
+
+static GSM_Error N6510_ReplyStartupNoteLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ if (Data->RequestID == ID_GetBitmap) {
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Welcome note text received\n");
+ CopyUnicodeString(Data->Bitmap->Text,msg.Buffer+6);
+ smprintf(s, "Text is \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ return ERR_NONE;
+ case 0x10:
+ smprintf(s, "Dealer note text received\n");
+ CopyUnicodeString(Data->Bitmap->Text,msg.Buffer+6);
+ smprintf(s, "Text is \"%s\"\n",DecodeUnicodeString(Data->Bitmap->Text));
+ return ERR_NONE;
+ case 0x0f:
+ smprintf(s, "Startup logo received\n");
+ PHONE_DecodeBitmap(GSM_Nokia7110StartupLogo, msg.Buffer + 22, Data->Bitmap);
+ return ERR_NONE;
+ }
+ }
+ if (Data->RequestID == ID_SetBitmap) {
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ case 0x10:
+ case 0x0f:
+ case 0x25:
+ return ERR_NONE;
+ }
+ }
+ return ERR_UNKNOWN;
+}
+
+static GSM_Error N6510_GetPictureImage(GSM_StateMachine *s, GSM_Bitmap *Bitmap, int *location)
+{
+ GSM_MultiSMSMessage sms;
+ int Number;
+ GSM_Bitmap bitmap;
+ GSM_Error error;
+
+ sms.SMS[0].Folder = 0;
+ Number = 0;
+ bitmap.Location = 255;
+ error=N6510_GetNextSMSMessageBitmap(s, &sms, true, &bitmap);
+ while (error == ERR_NONE) {
+ if (bitmap.Location != 255) {
+ Number++;
+ if (Number == Bitmap->Location) {
+ bitmap.Location = Bitmap->Location;
+ memcpy(Bitmap,&bitmap,sizeof(GSM_Bitmap));
+ *location = sms.SMS[0].Location;
+ return ERR_NONE;
+ }
+ }
+ bitmap.Location = 255;
+ sms.SMS[0].Folder = 0;
+ error=N6510_GetNextSMSMessageBitmap(s, &sms, false, &bitmap);
+ }
+ return ERR_INVALIDLOCATION;
+}
+
+static GSM_Error N6510_GetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ unsigned char reqOp [] = {N6110_FRAME_HEADER, 0x23, 0x00, 0x00, 0x55, 0x55, 0x55};
+ unsigned char reqStartup[] = {N6110_FRAME_HEADER, 0x02, 0x0f};
+ unsigned char reqNote [] = {N6110_FRAME_HEADER, 0x02, 0x01, 0x00};
+ GSM_MemoryEntry pbk;
+ GSM_Error error;
+ int Location;
+
+ s->Phone.Data.Bitmap=Bitmap;
+ switch (Bitmap->Type) {
+ case GSM_StartupLogo:
+ Bitmap->BitmapWidth = 96;
+ Bitmap->BitmapHeight = 65;
+ GSM_ClearBitmap(Bitmap);
+ smprintf(s, "Getting startup logo\n");
+ return GSM_WaitFor (s, reqStartup, 5, 0x7A, 4, ID_GetBitmap);
+ case GSM_DealerNote_Text:
+ reqNote[4] = 0x10;
+ smprintf(s, "Getting dealer note\n");
+ return GSM_WaitFor (s, reqNote, 6, 0x7A, 4, ID_GetBitmap);
+ case GSM_WelcomeNote_Text:
+ smprintf(s, "Getting welcome note\n");
+ return GSM_WaitFor (s, reqNote, 6, 0x7A, 4, ID_GetBitmap);
+ case GSM_CallerGroupLogo:
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBK35)) return ERR_NOTSUPPORTED;
+ Bitmap->BitmapWidth = 72;
+ Bitmap->BitmapHeight = 14;
+ GSM_ClearBitmap(Bitmap);
+ pbk.MemoryType = MEM7110_CG;
+ pbk.Location = Bitmap->Location;
+ smprintf(s, "Getting caller group logo\n");
+ error=N6510_GetMemory(s,&pbk);
+ if (error==ERR_NONE) NOKIA_GetDefaultCallerGroupName(s, Bitmap);
+ return error;
+ case GSM_OperatorLogo:
+ smprintf(s, "Getting operator logo\n");
+ return GSM_WaitFor (s, reqOp, 9, 0x0A, 4, ID_GetBitmap);
+ case GSM_PictureImage:
+ return N6510_GetPictureImage(s, Bitmap, &Location);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6510_ReplyGetIncSignalQuality(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Network level changed to: %i\n",msg.Buffer[4]);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplyGetSignalQuality(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Network level received: %i\n",msg.Buffer[8]);
+ Data->SignalQuality->SignalStrength = -1;
+ Data->SignalQuality->SignalPercent = ((int)msg.Buffer[8]);
+ Data->SignalQuality->BitErrorRate = -1;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetSignalQuality(GSM_StateMachine *s, GSM_SignalQuality *sig)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0B, 0x00, 0x02, 0x00, 0x00, 0x00};
+
+ s->Phone.Data.SignalQuality = sig;
+ smprintf(s, "Getting network level\n");
+ return GSM_WaitFor (s, req, 9, 0x0a, 4, ID_GetSignalQuality);
+}
+
+static GSM_Error N6510_ReplyGetBatteryCharge(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Battery level received: %i\n",msg.Buffer[9]*100/7);
+ Data->BatteryCharge->BatteryPercent = ((int)(msg.Buffer[9]*100/7));
+ Data->BatteryCharge->ChargeState = 0;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetBatteryCharge(GSM_StateMachine *s, GSM_BatteryCharge *bat)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0A, 0x02, 0x00};
+
+ s->Phone.Data.BatteryCharge = bat;
+ smprintf(s, "Getting battery level\n");
+ return GSM_WaitFor (s, req, 6, 0x17, 4, ID_GetBatteryCharge);
+}
+
+static GSM_Error N6510_ReplyGetWAPBookmark(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return DCT3DCT4_ReplyGetWAPBookmark (msg, s, true);
+}
+
+static GSM_Error N6510_ReplyGetOperatorLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Operator logo received\n");
+ NOKIA_DecodeNetworkCode(msg.Buffer+12,Data->Bitmap->NetworkCode);
+ smprintf(s, "Network code %s\n",Data->Bitmap->NetworkCode);
+ Data->Bitmap->BitmapWidth = msg.Buffer[20];
+ Data->Bitmap->BitmapHeight = msg.Buffer[21];
+ if (msg.Length == 18) return ERR_EMPTY;
+ PHONE_DecodeBitmap(GSM_Nokia6510OperatorLogo,msg.Buffer+26,Data->Bitmap);
+ return ERR_NONE;
+}
+
+GSM_Error N6510_ReplyDeleteMemory(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phonebook entry deleted\n");
+ return ERR_NONE;
+}
+
+GSM_Error N6510_DeleteMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ unsigned char req[] = {N7110_FRAME_HEADER, 0x0f, 0x55, 0x01,
+ 0x04, 0x55, 0x00, 0x10, 0xFF, 0x02,
+ 0x00, 0x01, /* location */
+ 0x00, 0x00, 0x00, 0x00,
+ 0x05, /* memory type */
+ 0x55, 0x55, 0x55};
+
+ req[12] = entry->Location / 256;
+ req[13] = entry->Location % 256;
+
+ req[18] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[18]==0xff) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Deleting phonebook entry\n");
+ return GSM_WaitFor (s, req, 22, 0x03, 4, ID_SetMemory);
+}
+
+static GSM_Error N6510_SetMemory(GSM_StateMachine *s, GSM_MemoryEntry *entry)
+{
+ int count = 22, blocks;
+ unsigned char req[500] = {
+ N7110_FRAME_HEADER, 0x0b, 0x00, 0x01, 0x01, 0x00, 0x00, 0x10,
+ 0x02, 0x00, /* memory type */
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ if (entry->Location == 0) return ERR_NOTSUPPORTED;
+
+ req[11] = NOKIA_GetMemoryType(s, entry->MemoryType,N71_65_MEMORY_TYPES);
+ if (req[11]==0xff) return ERR_NOTSUPPORTED;
+
+ req[12] = entry->Location / 256;
+ req[13] = entry->Location % 256;
+
+ count = count + N71_65_EncodePhonebookFrame(s, req+22, *entry, &blocks, true, IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_VOICETAGS));
+ req[21] = blocks;
+
+ smprintf(s, "Writing phonebook entry\n");
+ return GSM_WaitFor (s, req, count, 0x03, 4, ID_SetMemory);
+}
+
+static GSM_Error N6510_ReplySetOperatorLogo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Operator logo set OK\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetCallerLogo(GSM_StateMachine *s, GSM_Bitmap *bitmap)
+{
+ char string[500];
+ int block=0, i, Width, Height;
+ unsigned int count = 22;
+ unsigned char req[500] = {
+ N6110_FRAME_HEADER, 0x0b, 0x00, 0x01, 0x01, 0x00, 0x00, 0x10,
+ 0xfe, 0x10, /* memory type */
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ req[13] = bitmap->Location;
+
+ /* Enabling/disabling logo */
+ string[0] = bitmap->BitmapEnabled?1:0;
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_LOGOON, 2, block++, string, req + count);
+
+ /* Ringtone */
+ if (!bitmap->DefaultRingtone) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBKTONEGAL)) {
+ } else {
+ string[0] = 0x00;
+ string[1] = 0x00;
+ string[2] = bitmap->RingtoneID;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_RINGTONE_ID, 3, block++, string, req + count);
+ count --;
+ req[count-5] = 8;
+ }
+ }
+
+ /* Number of group */
+ string[0] = bitmap->Location;
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_GROUP, 2, block++, string, req + count);
+
+ /* Name */
+ if (!bitmap->DefaultName) {
+ i = UnicodeLength(bitmap->Text) * 2;
+ string[0] = i + 2;
+ memcpy(string + 1, bitmap->Text, i);
+ string[i + 1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_NAME, i + 2, block++, string, req + count);
+ }
+
+ /* Logo */
+ if (!bitmap->DefaultBitmap) {
+ PHONE_GetBitmapWidthHeight(GSM_NokiaCallerLogo, &Width, &Height);
+ string[0] = Width;
+ string[1] = Height;
+ string[2] = 0;
+ string[3] = 0;
+ string[4] = PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0);
+ PHONE_EncodeBitmap(GSM_NokiaCallerLogo, string + 5, bitmap);
+ count += N71_65_PackPBKBlock(s, N7110_PBK_GROUPLOGO, PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 5, block++, string, req + count);
+ }
+
+ req[21] = block;
+
+ return GSM_WaitFor (s, req, count, 0x03, 4, ID_SetBitmap);
+}
+
+static GSM_Error N6510_ReplySetPicture(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+// smprintf(s, "Picture Image written OK, folder %i, location %i\n",msg.Buffer[4],msg.Buffer[5]*256+msg.Buffer[6]);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetBitmap(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ GSM_SMSMessage sms;
+ GSM_Phone_Bitmap_Types Type;
+ int Width, Height, i, count;
+#ifdef DEVELOP
+ unsigned char folderid;
+ int location;
+#endif
+ GSM_NetworkInfo NetInfo;
+ GSM_Error error;
+ unsigned char reqStartup[1000] = {
+ N7110_FRAME_HEADER, 0x04, 0x0F,
+ 0x00, 0x00, 0x00,
+ 0x04, 0xC0, 0x02, 0x00,
+ 0x41, 0xC0, 0x03, 0x00,
+ 0x60, 0xC0, 0x04};
+ unsigned char reqColourWallPaper[200] = {
+ N6110_FRAME_HEADER, 0x07, 0x00, 0x00, 0x00, 0xD5,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x01, 0x00,
+ 0x18}; /* Bitmap ID */
+ unsigned char reqColourStartup[200] = {
+ N6110_FRAME_HEADER, 0x04, 0x25, 0x00, 0x01, 0x00, 0x18};
+ unsigned char reqOp[1000] = {
+ N7110_FRAME_HEADER, 0x25, 0x01,
+ 0x55, 0x00, 0x00, 0x55,
+ 0x01, /* 0x01 - not set, 0x02 - set */
+ 0x0C, 0x08,
+ 0x62, 0xF0, 0x10, /* Network code */
+ 0x03, 0x55, 0x55};
+ unsigned char reqColourOp[200] = {
+ N6110_FRAME_HEADER,
+ 0x07, 0x00, 0x00, 0x00, 0xE7, 0x00, 0x00, 0x00, 0xF9, 0x00,
+ 0x08, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x01, 0x00,
+ 0x18, /* File ID */
+ 0x00,
+ 0x00, 0x00, 0x00}; /* Network code */
+ unsigned char reqNote[200] = {N6110_FRAME_HEADER, 0x04, 0x01};
+ unsigned char reqPicture[2000] = {
+ N6110_FRAME_HEADER, 0x00,
+ 0x02, 0x05, /* SMS folder */
+ 0x00, 0x00, /* location */
+ 0x01, 0x01, 0xa0, 0x02, 0x01, 0x40, 0x00, 0x34,
+ 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x55, 0x55, 0x55, 0x03, 0x82, 0x10,
+ 0x01, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x82, 0x10,
+ 0x02, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x04,
+ 0x00, 0x00, 0xa1, 0x55, 0x01, 0x08, 0x00, 0x00,
+ 0x00, 0x01, 0x48, 0x1c, 0x00, 0xfc, 0x00};
+
+ switch (Bitmap->Type) {
+ case GSM_ColourWallPaper_ID:
+ reqColourWallPaper[21] = Bitmap->ID;
+ smprintf(s, "Setting colour wall paper\n");
+ return GSM_WaitFor (s, reqColourWallPaper, 22, 0x43, 4, ID_SetBitmap);
+ case GSM_StartupLogo:
+ Type = GSM_Nokia7110StartupLogo;
+ switch (Bitmap->Location) {
+ case 1: PHONE_EncodeBitmap(Type, reqStartup + 22, Bitmap);
+ break;
+ case 2: memset(reqStartup+5,0x00,15);
+ PHONE_ClearBitmap(Type, reqStartup + 22,0,0);
+ break;
+ default: return ERR_NOTSUPPORTED;
+ }
+ smprintf(s, "Setting startup logo\n");
+ return GSM_WaitFor (s, reqStartup, 22+PHONE_GetBitmapSize(Type,0,0), 0x7A, 4, ID_SetBitmap);
+ case GSM_DealerNote_Text:
+ reqNote[4] = 0x10;
+ CopyUnicodeString(reqNote + 5, Bitmap->Text);
+ i = 6 + UnicodeLength(Bitmap->Text) * 2;
+ reqNote[i++] = 0;
+ reqNote[i] = 0;
+ return GSM_WaitFor (s, reqNote, i, 0x7A, 4, ID_SetBitmap);
+ case GSM_WelcomeNote_Text:
+ CopyUnicodeString(reqNote + 5, Bitmap->Text);
+ i = 6 + UnicodeLength(Bitmap->Text) * 2;
+ reqNote[i++] = 0;
+ reqNote[i] = 0;
+ return GSM_WaitFor (s, reqNote, i, 0x7A, 4, ID_SetBitmap);
+ case GSM_OperatorLogo:
+ /* We want to set operator logo, not clear */
+ if (strcmp(Bitmap->NetworkCode,"000 00")) {
+ memset(reqOp + 19, 0, 281);
+ NOKIA_EncodeNetworkCode(reqOp+12, Bitmap->NetworkCode);
+ Type = GSM_Nokia6510OperatorLogo;
+ reqOp[9] = 0x02; /* Logo enabled */
+ reqOp[18] = 0x1a; /* FIXME */
+ reqOp[19] = PHONE_GetBitmapSize(Type,0,0) + 8 + 29 + 2;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ reqOp[20] = Width;
+ reqOp[21] = Height;
+ reqOp[22] = 0x00;
+ reqOp[23] = PHONE_GetBitmapSize(Type,0,0) + 29;
+ reqOp[24] = 0x00;
+ reqOp[25] = PHONE_GetBitmapSize(Type,0,0) + 29;
+ PHONE_EncodeBitmap(Type, reqOp + 26, Bitmap);
+ smprintf(s, "Setting operator logo\n");
+ return GSM_WaitFor (s, reqOp, reqOp[19]+reqOp[11]+10, 0x0A, 4, ID_SetBitmap);
+ } else {
+ error=N6510_GetNetworkInfo(s,&NetInfo);
+ if (error != ERR_NONE) return error;
+ NOKIA_EncodeNetworkCode(reqOp+12, NetInfo.NetworkCode);
+ smprintf(s, "Clearing operator logo\n");
+ return GSM_WaitFor (s, reqOp, 18, 0x0A, 4, ID_SetBitmap);
+ }
+ case GSM_ColourOperatorLogo_ID:
+ /* We want to set operator logo, not clear */
+ if (strcmp(Bitmap->NetworkCode,"000 00")) {
+ EncodeBCD(reqColourOp+23, Bitmap->NetworkCode, 6, false);
+ reqColourOp[21] = Bitmap->ID;
+ }
+ smprintf(s, "Setting colour operator logo\n");
+ return GSM_WaitFor (s, reqColourOp, 26, 0x43, 4, ID_SetBitmap);
+ case GSM_ColourStartupLogo_ID:
+ switch (Bitmap->Location) {
+ case 0: reqColourStartup[6] = 0x00;
+ reqColourStartup[8] = 0x00;
+ smprintf(s, "Setting colour startup logo\n");
+ return GSM_WaitFor (s, reqColourStartup, 9, 0x7A, 4, ID_SetBitmap);
+ case 1: reqColourStartup[8] = Bitmap->ID;
+ smprintf(s, "Setting colour startup logo\n");
+ return GSM_WaitFor (s, reqColourStartup, 9, 0x7A, 4, ID_SetBitmap);
+ default:return ERR_NOTSUPPORTED;
+ }
+ case GSM_CallerGroupLogo:
+ return N6510_SetCallerLogo(s,Bitmap);
+ case GSM_PictureImage:
+ error = N6510_GetPictureImage(s, Bitmap, &sms.Location);
+ if (error == ERR_NONE) {
+#ifdef DEVELOP
+ sms.Folder = 0;
+ N6510_GetSMSLocation(s, &sms, &folderid, &location);
+ switch (folderid) {
+ case 0x01: reqPicture[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: reqPicture[5] = 0x03; break; /* OUTBOX SIM */
+ default : reqPicture[5] = folderid - 1; reqPicture[4] = 0x02; break; /* ME folders */
+ }
+ reqPicture[6]=location / 256;
+ reqPicture[7]=location;
+#else
+ return ERR_NOTSUPPORTED;
+#endif
+ }
+ Type = GSM_NokiaPictureImage;
+ count = 78;
+ PHONE_EncodeBitmap(Type, reqPicture + count, Bitmap);
+ count += PHONE_GetBitmapSize(Type,0,0);
+ smprintf(s, "Setting Picture Image\n");
+ return GSM_WaitFor (s, reqPicture, count, 0x14, 4, ID_SetBitmap);
+ default:
+ break;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6510_ReplyGetRingtoneID(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+
+ smprintf(s, "Ringtone ID received\n");
+ Priv->RingtoneID = msg.Buffer[15];
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplySetBinRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Binary ringtone set\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, int *maxlength)
+{
+ GSM_Error error;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ GSM_NetworkInfo NetInfo;
+ int size=200, current;
+ unsigned char GetIDReq[] = {
+ N7110_FRAME_HEADER, 0x01, 0x00, 0x00,
+ 0x00, 0xFF, 0x06, 0xE1, 0x00,
+ 0xFF, 0x06, 0xE1, 0x01, 0x42};
+ unsigned char SetPreviewReq[1000] = {
+ 0xAE, /* Ringtone ID */
+ 0x01, 0x00, 0x0D, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00}; /*Length*/
+ unsigned char AddBinaryReq[33000] = {
+ N7110_FRAME_HEADER, 0x0E, 0x7F, 0xFF, 0xFE};
+
+ if (Ringtone->Format == RING_NOTETONE && Ringtone->Location==255)
+ {
+ smprintf(s, "Getting ringtone ID\n");
+ error=GSM_WaitFor (s, GetIDReq, 14, 0xDB, 4, ID_SetRingtone);
+ if (error != ERR_NONE) return error;
+ *maxlength=GSM_EncodeNokiaRTTLRingtone(*Ringtone, SetPreviewReq+11, &size);
+ SetPreviewReq[0] = Priv->RingtoneID;
+ SetPreviewReq[10] = size;
+ smprintf(s, "Setting ringtone\n");
+ error = s->Protocol.Functions->WriteMessage(s, SetPreviewReq, size+11, 0x00);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+ /* We have to make something (not important, what) now */
+ /* no answer from phone*/
+ return s->Phone.Functions->GetNetworkInfo(s,&NetInfo);
+ }
+ if (Ringtone->Format == RING_NOKIABINARY) {
+ AddBinaryReq[7] = UnicodeLength(Ringtone->Name);
+ CopyUnicodeString(AddBinaryReq+8,Ringtone->Name);
+ current = 8 + UnicodeLength(Ringtone->Name)*2;
+ AddBinaryReq[current++] = Ringtone->NokiaBinary.Length/256 + 1;
+ AddBinaryReq[current++] = Ringtone->NokiaBinary.Length%256 + 1;
+ AddBinaryReq[current++] = 0x00;
+ memcpy(AddBinaryReq+current,Ringtone->NokiaBinary.Frame,Ringtone->NokiaBinary.Length);
+ current += Ringtone->NokiaBinary.Length;
+ smprintf(s, "Adding binary ringtone\n");
+ return GSM_WaitFor (s, AddBinaryReq, current, 0x1F, 4, ID_SetRingtone);
+ }
+ if (Ringtone->Format == RING_MIDI) {
+ AddBinaryReq[7] = UnicodeLength(Ringtone->Name);
+ CopyUnicodeString(AddBinaryReq+8,Ringtone->Name);
+ current = 8 + UnicodeLength(Ringtone->Name)*2;
+ AddBinaryReq[current++] = Ringtone->NokiaBinary.Length/256;
+ AddBinaryReq[current++] = Ringtone->NokiaBinary.Length%256;
+ memcpy(AddBinaryReq+current,Ringtone->NokiaBinary.Frame,Ringtone->NokiaBinary.Length);
+ current += Ringtone->NokiaBinary.Length;
+ AddBinaryReq[current++] = 0x00;
+ AddBinaryReq[current++] = 0x00;
+ smprintf(s, "Adding binary or MIDI ringtone\n");
+ return GSM_WaitFor (s, AddBinaryReq, current, 0x1F, 4, ID_SetRingtone);
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6510_ReplyDeleteRingtones(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Ringtones deleted\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_DeleteUserRingtones(GSM_StateMachine *s)
+{
+ unsigned char DelAllRingtoneReq[] = {N7110_FRAME_HEADER, 0x10, 0x7F, 0xFE};
+
+ smprintf(s, "Deleting all user ringtones\n");
+ return GSM_WaitFor (s, DelAllRingtoneReq, 6, 0x1F, 4, ID_SetRingtone);
+}
+
+static GSM_Error N6510_PressKey(GSM_StateMachine *s, GSM_KeyCode Key, bool Press)
+{
+#ifdef DEVELOP
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x11, 0x00, 0x01, 0x00, 0x00,
+ 0x00, /* Event */
+ 0x01}; /* Number of presses */
+
+// req[7] = Key;
+ if (Press) {
+ req[8] = NOKIA_PRESSPHONEKEY;
+ s->Phone.Data.PressKey = true;
+ smprintf(s, "Pressing key\n");
+ } else {
+ req[8] = NOKIA_RELEASEPHONEKEY;
+ s->Phone.Data.PressKey = false;
+ smprintf(s, "Releasing key\n");
+ }
+ return GSM_WaitFor (s, req, 10, 0x0c, 4, ID_PressKey);
+#else
+ return ERR_NOTSUPPORTED;
+#endif
+}
+
+static GSM_Error N6510_EnableConnectionFunctions(GSM_StateMachine *s, N6510_Connection_Settings Type)
+{
+ GSM_Error error;
+ unsigned char req2[] = {N6110_FRAME_HEADER, 0x00, 0x01};
+ unsigned char req3[] = {N6110_FRAME_HEADER, 0x00, 0x03};
+ unsigned char req4[] = {N6110_FRAME_HEADER, 0x00, 0x04};
+
+ if (Type == N6510_MMS_SETTINGS && IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOMMS)) return ERR_NOTSUPPORTED;
+
+ error=DCT3DCT4_DisableConnectionFunctions(s);
+ if (error!=ERR_NONE) return error;
+
+ switch (Type) {
+ case N6510_WAP_SETTINGS:
+ return DCT3DCT4_EnableWAPFunctions(s);
+ case N6510_MMS_SETTINGS:
+ dbgprintf("Enabling MMS\n");
+ return GSM_WaitFor (s, req2, 5, 0x3f, 4, ID_EnableConnectFunc);
+ case N6510_SYNCML_SETTINGS:
+ dbgprintf("Enabling SyncML\n");
+ return GSM_WaitFor (s, req3, 5, 0x3f, 5, ID_EnableConnectFunc);
+ case N6510_CHAT_SETTINGS:
+ dbgprintf("Enabling Chat\n");
+ return GSM_WaitFor (s, req4, 5, 0x3f, 5, ID_EnableConnectFunc);
+ default:
+ return ERR_UNKNOWN;
+ }
+}
+
+static GSM_Error N6510_ReplyGetConnectionSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ int tmp,num=0,i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ unsigned char buff[2000];
+
+ switch(msg.Buffer[3]) {
+ case 0x16:
+ smprintf(s, "Connection settings received OK\n");
+
+ Data->WAPSettings->Number = Priv->BearerNumber;
+
+ Data->WAPSettings->Proxy[0] = 0x00;
+ Data->WAPSettings->Proxy[1] = 0x00;
+ Data->WAPSettings->ProxyPort = 8080;
+
+ Data->WAPSettings->Proxy2[0] = 0x00;
+ Data->WAPSettings->Proxy2[1] = 0x00;
+ Data->WAPSettings->Proxy2Port = 8080;
+
+ tmp = 4;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].Title,true);
+ CopyUnicodeString(Data->WAPSettings->Settings[1].Title,Data->WAPSettings->Settings[0].Title);
+ smprintf(s, "Title: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].Title));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].HomePage,true);
+ CopyUnicodeString(Data->WAPSettings->Settings[1].HomePage,Data->WAPSettings->Settings[0].HomePage);
+ smprintf(s, "Homepage: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].HomePage));
+
+#ifdef DEBUG
+ smprintf(s, "Connection type: ");
+ switch (msg.Buffer[tmp]) {
+ case 0x00: smprintf(s, "temporary\n"); break;
+ case 0x01: smprintf(s, "continuous\n"); break;
+ default: smprintf(s, "unknown\n");
+ }
+ smprintf(s, "Connection security: ");
+ switch (msg.Buffer[tmp+1]) {
+ case 0x00: smprintf(s, "off\n"); break;
+ case 0x01: smprintf(s, "on\n"); break;
+ default: smprintf(s, "unknown\n");
+ }
+ smprintf(s, "Bearer: ");
+ switch (msg.Buffer[tmp+2]) {
+ case 0x01: smprintf(s, "GSM data\n"); break;
+ case 0x03: smprintf(s, "GPRS\n"); break;
+ default: smprintf(s, "unknown\n");
+ }
+ if (msg.Buffer[tmp+3] == 0x01) smprintf(s, "locked\n");
+#endif
+ Data->WAPSettings->Settings[0].IsContinuous = false;
+ if (msg.Buffer[tmp] == 0x01) Data->WAPSettings->Settings[0].IsContinuous = true;
+ Data->WAPSettings->Settings[1].IsContinuous = Data->WAPSettings->Settings[0].IsContinuous;
+
+ Data->WAPSettings->Settings[0].IsSecurity = false;
+ if (msg.Buffer[tmp+1] == 0x01) Data->WAPSettings->Settings[0].IsSecurity = true;
+ Data->WAPSettings->Settings[1].IsSecurity = Data->WAPSettings->Settings[0].IsSecurity;
+
+ Data->WAPSettings->ActiveBearer = WAPSETTINGS_BEARER_DATA;
+ if (msg.Buffer[tmp+2] == 0x03) Data->WAPSettings->ActiveBearer = WAPSETTINGS_BEARER_GPRS;
+
+ Data->WAPSettings->ReadOnly = false;
+ if (msg.Buffer[tmp+3] == 0x01) Data->WAPSettings->ReadOnly = true;
+
+ tmp+=3;
+
+ if (Priv->BearerNumber == 2) {
+ /* Here starts settings for data bearer */
+ Data->WAPSettings->Settings[0].Bearer = WAPSETTINGS_BEARER_DATA;
+ while ((msg.Buffer[tmp] != 0x01) || (msg.Buffer[tmp + 1] != 0x00)) tmp++;
+ tmp += 4;
+
+#ifdef DEBUG
+ smprintf(s, "Authentication type: ");
+ switch (msg.Buffer[tmp]) {
+ case 0x00: smprintf(s, "normal\n"); break;
+ case 0x01: smprintf(s, "secure\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Data call type: ");
+ switch (msg.Buffer[tmp+1]) {
+ case 0x00: smprintf(s, "analogue\n"); break;
+ case 0x01: smprintf(s, "ISDN\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Data call speed: ");
+ switch (msg.Buffer[tmp+2]) {
+ case 0x00: smprintf(s, "automatic\n"); break;
+ case 0x01: smprintf(s, "9600\n"); break;
+ case 0x02: smprintf(s, "14400\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Login Type: ");
+ switch (msg.Buffer[tmp+4]) {
+ case 0x00: smprintf(s, "manual\n"); break;
+ case 0x01: smprintf(s, "automatic\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+#endif
+ Data->WAPSettings->Settings[0].IsNormalAuthentication=true;
+ if (msg.Buffer[tmp]==0x01) Data->WAPSettings->Settings[0].IsNormalAuthentication=false;
+
+ Data->WAPSettings->Settings[0].IsISDNCall=false;
+ if (msg.Buffer[tmp+1]==0x01) Data->WAPSettings->Settings[0].IsISDNCall=true;
+
+ switch (msg.Buffer[tmp+2]) {
+ case 0x00: Data->WAPSettings->Settings[0].Speed=WAPSETTINGS_SPEED_AUTO; break;
+ case 0x01: Data->WAPSettings->Settings[0].Speed=WAPSETTINGS_SPEED_9600; break;
+ case 0x02: Data->WAPSettings->Settings[0].Speed=WAPSETTINGS_SPEED_14400; break;
+ }
+
+ Data->WAPSettings->Settings[0].ManualLogin=false;
+ if (msg.Buffer[tmp+4]==0x00) Data->WAPSettings->Settings[0].ManualLogin = true;
+
+ tmp+=5;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].IPAddress,false);
+ smprintf(s, "IP address: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].IPAddress));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].DialUp,true);
+ smprintf(s, "Dial-up number: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].DialUp));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].User,true);
+ smprintf(s, "User name: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].User));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[0].Password,true);
+ smprintf(s, "Password: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[0].Password));
+
+ num = 1;
+ } else {
+ num = 0;
+ }
+
+ /* Here starts settings for gprs bearer */
+ Data->WAPSettings->Settings[num].Bearer = WAPSETTINGS_BEARER_GPRS;
+ while (msg.Buffer[tmp] != 0x03) tmp++;
+ tmp += 4;
+
+#ifdef DEBUG
+ smprintf(s, "Authentication type: ");
+ switch (msg.Buffer[tmp]) {
+ case 0x00: smprintf(s, "normal\n"); break;
+ case 0x01: smprintf(s, "secure\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "GPRS connection: ");
+ switch (msg.Buffer[tmp+1]) {
+ case 0x00: smprintf(s, "ALWAYS online\n"); break;
+ case 0x01: smprintf(s, "when needed\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+ smprintf(s, "Login Type: ");
+ switch (msg.Buffer[tmp+2]) {
+ case 0x00: smprintf(s, "manual\n"); break;
+ case 0x01: smprintf(s, "automatic\n"); break;
+ default: smprintf(s, "unknown\n"); break;
+ }
+#endif
+ Data->WAPSettings->Settings[num].IsNormalAuthentication=true;
+ if (msg.Buffer[tmp]==0x01) Data->WAPSettings->Settings[num].IsNormalAuthentication=false;
+
+ Data->WAPSettings->Settings[num].IsContinuous = true;
+ if (msg.Buffer[tmp+1] == 0x01) Data->WAPSettings->Settings[num].IsContinuous = false;
+
+ Data->WAPSettings->Settings[num].ManualLogin=false;
+ if (msg.Buffer[tmp+2]==0x00) Data->WAPSettings->Settings[num].ManualLogin = true;
+
+ tmp+=3;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[num].DialUp,false);
+ smprintf(s, "Access point: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[num].DialUp));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[num].IPAddress,true);
+ smprintf(s, "IP address: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[num].IPAddress));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[num].User,true);
+ smprintf(s, "User name: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[num].User));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPSettings->Settings[num].Password,true);
+ smprintf(s, "Password: \"%s\"\n",DecodeUnicodeString(Data->WAPSettings->Settings[num].Password));
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_WAPMMSPROXY)) {
+ if (msg.Buffer[tmp] == 0x00 && msg.Buffer[tmp+1] == 0x00) tmp = tmp+2;
+
+ memcpy(buff,msg.Buffer+tmp+10,msg.Buffer[tmp+4]);
+ buff[msg.Buffer[tmp+4]] = 0x00;
+ smprintf(s, "Proxy 1: \"%s\", port %i\n",buff,msg.Buffer[tmp+6]*256+msg.Buffer[tmp+7]);
+ EncodeUnicode(Data->WAPSettings->Proxy,buff,strlen(buff));
+ Data->WAPSettings->ProxyPort = msg.Buffer[tmp+6]*256+msg.Buffer[tmp+7];
+
+ memcpy(buff,msg.Buffer+tmp+10+msg.Buffer[tmp+4],msg.Buffer[tmp+5]);
+ buff[msg.Buffer[tmp+5]] = 0x00;
+ smprintf(s, "Proxy 2: \"%s\", port %i\n",buff,msg.Buffer[tmp+8]*256+msg.Buffer[tmp+9]);
+ EncodeUnicode(Data->WAPSettings->Proxy2,buff,strlen(buff));
+ Data->WAPSettings->Proxy2Port = msg.Buffer[tmp+8]*256+msg.Buffer[tmp+9];
+
+ tmp = tmp + msg.Buffer[tmp+3] + 19;
+
+ for (i=0;i<4;i++) {
+#ifdef DEBUG
+ smprintf(s, "Proxy data %i\n",i+1);
+ if (msg.Buffer[tmp+2]!=0) memcpy(buff,msg.Buffer+tmp+9,msg.Buffer[tmp+2]*2);
+ buff[msg.Buffer[tmp+2]*2] =0;
+ buff[msg.Buffer[tmp+2]*2+1]=0;
+ smprintf(s, "IP: \"%s\"",DecodeUnicodeString(buff));
+ smprintf(s, ", port %i\n",msg.Buffer[tmp+3]*256+msg.Buffer[tmp+4]);
+#endif
+ tmp = tmp + msg.Buffer[tmp];
+ }
+
+#ifdef DEBUG
+ smprintf(s, "%02x %02x\n",msg.Buffer[tmp],msg.Buffer[tmp+1]);
+ smprintf(s, "Port %i\n",msg.Buffer[tmp+3]*256+msg.Buffer[tmp+4]);
+ tmp = tmp + msg.Buffer[tmp];
+#endif
+ }
+
+ return ERR_NONE;
+ case 0x17:
+ smprintf(s, "Connection settings receiving error\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside phone settings menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x02:
+ smprintf(s, "Invalid or empty\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetConnectionSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings, N6510_Connection_Settings Type)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x15,
+ 0x00}; /* Location */
+
+ error = N6510_EnableConnectionFunctions(s, Type);
+ if (error!=ERR_NONE) return error;
+
+ req[4] = settings->Location-1;
+ s->Phone.Data.WAPSettings = settings;
+
+ switch (Type) {
+ case N6510_MMS_SETTINGS:
+ smprintf(s, "Getting MMS settings\n");
+ Priv->BearerNumber = 1;
+ break;
+ case N6510_WAP_SETTINGS:
+ smprintf(s, "Getting WAP settings\n");
+ Priv->BearerNumber = 2;
+ break;
+ case N6510_SYNCML_SETTINGS:
+ smprintf(s, "Getting SyncML settings\n");
+ Priv->BearerNumber = 2;
+ break;
+ case N6510_CHAT_SETTINGS:
+ smprintf(s, "Getting Chat settings\n");
+ Priv->BearerNumber = 1;
+ break;
+ }
+
+ error=GSM_WaitFor (s, req, 5, 0x3f, 4, ID_GetConnectSet);
+ if (error != ERR_NONE) {
+ if (error == ERR_INVALIDLOCATION || error == ERR_INSIDEPHONEMENU) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ error=DCT3DCT4_GetActiveConnectSet(s);
+ if (error != ERR_NONE) return error;
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+static GSM_Error N6510_GetWAPSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+ return N6510_GetConnectionSettings(s, settings, N6510_WAP_SETTINGS);
+}
+
+static GSM_Error N6510_GetMMSSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+ return N6510_GetConnectionSettings(s, settings, N6510_MMS_SETTINGS);
+}
+
+static GSM_Error N6510_ReplyGetSyncMLSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SyncMLSettings *Sett = s->Phone.Data.SyncMLSettings;
+
+ smprintf(s, "SyncML settings received OK\n");
+ CopyUnicodeString(Sett->User,msg.Buffer+18);
+ CopyUnicodeString(Sett->Password,msg.Buffer+86);
+ CopyUnicodeString(Sett->PhonebookDataBase,msg.Buffer+130);
+ CopyUnicodeString(Sett->CalendarDataBase,msg.Buffer+234);
+ CopyUnicodeString(Sett->Server,msg.Buffer+338);
+
+ Sett->SyncPhonebook = false;
+ Sett->SyncCalendar = false;
+ if ((msg.Buffer[598] & 0x02)==0x02) Sett->SyncCalendar = true;
+ if ((msg.Buffer[598] & 0x01)==0x01) Sett->SyncPhonebook = true;
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplyGetSyncMLName(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SyncMLSettings *Sett = s->Phone.Data.SyncMLSettings;
+
+ smprintf(s, "SyncML names received OK\n");
+
+ CopyUnicodeString(Sett->Name,msg.Buffer+18);
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetSyncMLSettings(GSM_StateMachine *s, GSM_SyncMLSettings *settings)
+{
+ GSM_Error error;
+// unsigned char NameReq[] = {N6110_FRAME_HEADER, 0x05,
+// 0x00, 0x00, 0x00, 0x31, 0x00,
+// 0x06, 0x00, 0x00, 0x00, 0xDE, 0x00, 0x00};
+// unsigned char GetActive[] = {N6110_FRAME_HEADER, 0x05,
+// 0x00, 0x00, 0x00, 0x31, 0x00,
+// 0x05, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00};
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x05,
+ 0x00, 0x00, 0x00, 0x31, 0x00,
+ 0x01, //location
+ 0x00, 0x00, 0x02, 0x46, 0x00, 0x00};
+
+ settings->Connection.Location = settings->Location;
+ error = N6510_GetConnectionSettings(s, &settings->Connection, N6510_SYNCML_SETTINGS);
+ if (error != ERR_NONE) return error;
+
+ settings->Active = settings->Connection.Active;
+
+ settings->Name[0] = 0;
+ settings->Name[1] = 0;
+// s->Phone.Data.SyncMLSettings = settings;
+// smprintf(s, "Getting SyncML settings name\n");
+// error = GSM_WaitFor (s, NameReq, 16, 0x43, 4, ID_GetSyncMLName);
+// if (error != ERR_NONE) return error;
+
+ req[9] = settings->Location - 1;
+ smprintf(s, "Getting additional SyncML settings\n");
+ return GSM_WaitFor (s, req, 16, 0x43, 4, ID_GetSyncMLSettings);
+}
+
+static GSM_Error N6510_ReplyGetChatSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_ChatSettings *Sett = s->Phone.Data.ChatSettings;
+ int i;
+
+ Sett->Name[0] = 0;
+ Sett->Name[1] = 0;
+ Sett->HomePage[0] = 0;
+ Sett->HomePage[1] = 0;
+ Sett->User[0] = 0;
+ Sett->User[1] = 0;
+ Sett->Password[0] = 0;
+ Sett->Password[1] = 0;
+
+ switch(msg.Buffer[3]) {
+ case 0x3B:
+ smprintf(s, "Chat settings received OK\n");
+ memcpy(Sett->Name,msg.Buffer+20,msg.Buffer[12]*2);
+ Sett->Name[msg.Buffer[12]*2] = 0;
+ Sett->Name[msg.Buffer[12]*2+1] = 0;
+ memcpy(Sett->HomePage,msg.Buffer+20+msg.Buffer[12]*2,msg.Buffer[15]*2);
+ Sett->HomePage[msg.Buffer[15]*2] = 0;
+ Sett->HomePage[msg.Buffer[15]*2+1] = 0;
+ i = msg.Buffer[12]*2 + msg.Buffer[15]*2 + 29;
+ memcpy(Sett->User,msg.Buffer+i+3,msg.Buffer[i]*2);
+ Sett->User[msg.Buffer[i]*2] = 0;
+ Sett->User[msg.Buffer[i]*2+1] = 0;
+ memcpy(Sett->Password,msg.Buffer+i+3+msg.Buffer[i]*2,msg.Buffer[i+1]*2);
+ Sett->Password[msg.Buffer[i+1]*2] = 0;
+ Sett->Password[msg.Buffer[i+1]*2+1] = 0;
+ return ERR_NONE;
+ case 0x3C:
+ smprintf(s, "Empty chat settings received\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetChatSettings(GSM_StateMachine *s, GSM_ChatSettings *settings)
+{
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x3a,
+ 0x09, // location
+ 0x01, 0x0e};
+
+ settings->Connection.Location = settings->Location;
+ error = N6510_GetConnectionSettings(s, &settings->Connection, N6510_CHAT_SETTINGS);
+ if (error != ERR_NONE) return error;
+
+ settings->Active = settings->Connection.Active;
+
+ s->Phone.Data.ChatSettings = settings;
+ req[4] = settings->Location - 1;
+ smprintf(s, "Getting additional Chat settings\n");
+ return GSM_WaitFor (s, req, 7, 0x3f, 4, ID_GetChatSettings);
+}
+
+static GSM_Error N6510_ReplySetConnectionSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x19:
+ smprintf(s, "Connection settings cleaned\n");
+ return ERR_NONE;
+ case 0x1a:
+ smprintf(s, "Connection settings setting status\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside phone settings menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x03:
+ smprintf(s, "Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ case 0x05:
+ smprintf(s, "Written OK\n");
+ return ERR_NONE;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ case 0x28:
+ case 0x2B:
+ smprintf(s, "Set OK\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_SetConnectionSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings, N6510_Connection_Settings Type)
+{
+ GSM_Error error;
+ int i, pad = 0, length, pos = 5, loc1=-1,loc2=-1,port;
+ unsigned char *Proxy;
+ unsigned char req[2000] = {N6110_FRAME_HEADER, 0x18,
+ 0x00}; /* Location */
+ unsigned char Lock[5] = {N6110_FRAME_HEADER, 0x27,
+ 0x00}; /* Location */
+ unsigned char UnLock[5] = {N6110_FRAME_HEADER, 0x2A,
+ 0x00}; /* Location */
+
+ error = N6510_EnableConnectionFunctions(s, Type);
+ if (error!=ERR_NONE) return error;
+
+ memset(req + pos, 0, 1000 - pos);
+
+ req[4] = settings->Location-1;
+
+ for (i=0;i<settings->Number;i++) {
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_DATA) loc1=i;
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_GPRS) loc2=i;
+ }
+
+ if (loc1 != -1) {
+ /* Name */
+ length = UnicodeLength(settings->Settings[loc1].Title);
+ if (!(length % 2)) pad = 1;
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].Title, false);
+
+ /* Home */
+ length = UnicodeLength(settings->Settings[loc1].HomePage);
+ if (((length + pad) % 2)) pad = 2; else pad = 0;
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].HomePage, true);
+
+ if (settings->Settings[loc1].IsContinuous) req[pos] = 0x01; pos++;
+ if (settings->Settings[loc1].IsSecurity) req[pos] = 0x01; pos++;
+ } else if (loc2 != -1) {
+ /* Name */
+ length = UnicodeLength(settings->Settings[loc2].Title);
+ if (!(length % 2)) pad = 1;
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].Title, false);
+
+ /* Home */
+ length = UnicodeLength(settings->Settings[loc2].HomePage);
+ if (((length + pad) % 2)) pad = 2; else pad = 0;
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].HomePage, true);
+
+ if (settings->Settings[loc2].IsContinuous) req[pos] = 0x01; pos++;
+ if (settings->Settings[loc2].IsSecurity) req[pos] = 0x01; pos++;
+ } else {
+ /* Name */
+ length = 0;
+ if (!(length % 2)) pad = 1;
+ pos ++;
+
+ /* Home */
+ length = 0;
+ if (((length + pad) % 2)) pad = 2; else pad = 0;
+ pos += 2;
+
+ pos += 2;
+ }
+
+ if (Type == N6510_MMS_SETTINGS || Type == N6510_CHAT_SETTINGS) {
+ req[pos++] = 0x03; //active bearer: GPRS
+ } else {
+ if (settings->ActiveBearer == WAPSETTINGS_BEARER_GPRS && loc2 != -1) {
+ req[pos++] = 0x03; //active bearer: GPRS
+ } else {
+ req[pos++] = 0x01; //active bearer: data set
+ }
+ }
+
+ /* Number of sent bearers */
+ if (Type == N6510_MMS_SETTINGS || Type == N6510_CHAT_SETTINGS) {
+ req[pos] = 0x01;
+ } else {
+ req[pos] = 0x02;
+ }
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_WAPMMSPROXY)) req[pos] += 2;
+ pos++;
+ pos += pad;
+
+ if (Type != N6510_MMS_SETTINGS && Type != N6510_CHAT_SETTINGS) {
+ /* GSM data block */
+ memcpy(req + pos, "\x01\x00", 2); pos += 2;
+
+ if (loc1 != -1) {
+ length = UnicodeLength(settings->Settings[loc1].IPAddress)*2+1;
+ length += UnicodeLength(settings->Settings[loc1].DialUp) *2+2;
+ length += UnicodeLength(settings->Settings[loc1].User) *2+2;
+ length += UnicodeLength(settings->Settings[loc1].Password) *2+2;
+ } else {
+ length = 1 + 2 + 2 + 2;
+ }
+ length += 11;
+ req[pos++] = length / 256;
+ req[pos++] = length % 256;
+
+ if (loc1 != -1) {
+ if (!settings->Settings[loc1].IsNormalAuthentication) req[pos]=0x01; pos++;
+ if (settings->Settings[loc1].IsISDNCall) req[pos]=0x01; pos++;
+ switch (settings->Settings[loc1].Speed) {
+ case WAPSETTINGS_SPEED_AUTO : break;
+ case WAPSETTINGS_SPEED_9600 : req[pos]=0x01; break;
+ case WAPSETTINGS_SPEED_14400 : req[pos]=0x02; break;
+ }
+ pos++;
+ req[pos++]=0x01;
+ if (!settings->Settings[loc1].ManualLogin) req[pos] = 0x01; pos++;
+
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].IPAddress, false);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].DialUp, true);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].User, true);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc1].Password, true);
+ } else {
+ pos += 3;
+ req[pos++]=0x01;
+ pos += 8;
+ }
+
+ /* Padding */
+ pos+=2;
+ }
+
+ /* GPRS block */
+ memcpy(req + pos, "\x03\x00", 2); pos += 2;
+
+ if (loc2 != -1) {
+ length = UnicodeLength(settings->Settings[loc2].DialUp) *2+1;
+ length += UnicodeLength(settings->Settings[loc2].IPAddress)*2+2;
+ length += UnicodeLength(settings->Settings[loc2].User) *2+2;
+ length += UnicodeLength(settings->Settings[loc2].Password) *2+2;
+ } else {
+ length = 7;
+ }
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_WAPMMSPROXY)) length+=2;
+ length += 7;
+ req[pos++] = length / 256;
+ req[pos++] = length % 256;
+
+ if (loc2 != -1) {
+ if (!settings->Settings[loc2].IsNormalAuthentication) req[pos] = 0x01; pos++;
+ if (!settings->Settings[loc2].IsContinuous) req[pos] = 0x01; pos++;
+ if (!settings->Settings[loc2].ManualLogin) req[pos] = 0x01; pos++;
+
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].DialUp, false);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].IPAddress, true);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].User, true);
+ pos += NOKIA_SetUnicodeString(s, req + pos, settings->Settings[loc2].Password, true);
+ } else {
+ pos += 10;
+ }
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_WAPMMSPROXY)) {
+ req[pos++] = 0x00;
+ req[pos++] = 0x00;
+
+ /* Proxy block */
+ req[pos++] = 0x06;
+ req[pos++] = 0x01;
+ if (UnicodeLength(settings->Proxy)!=0 ||
+ UnicodeLength(settings->Proxy2)!=0) {
+ req[pos++] = (UnicodeLength(settings->Proxy)+UnicodeLength(settings->Proxy2)+13)/256;
+ req[pos++] = (UnicodeLength(settings->Proxy)+UnicodeLength(settings->Proxy2)+13)%256;
+ } else {
+ req[pos++] = (UnicodeLength(settings->Proxy)+UnicodeLength(settings->Proxy2)+12)/256;
+ req[pos++] = (UnicodeLength(settings->Proxy)+UnicodeLength(settings->Proxy2)+12)%256;
+ }
+ req[pos++] = UnicodeLength(settings->Proxy);
+ req[pos++] = UnicodeLength(settings->Proxy2);
+ req[pos++] = settings->ProxyPort/256;
+ req[pos++] = settings->ProxyPort%256;
+ req[pos++] = settings->Proxy2Port/256;
+ req[pos++] = settings->Proxy2Port%256;
+ if (UnicodeLength(settings->Proxy)!=0) {
+ sprintf(req+pos,"%s",DecodeUnicodeString(settings->Proxy));
+ pos+=UnicodeLength(settings->Proxy);
+ }
+ if (UnicodeLength(settings->Proxy2)!=0) {
+ sprintf(req+pos,"%s",DecodeUnicodeString(settings->Proxy2));
+ pos+=UnicodeLength(settings->Proxy2);
+ }
+ if (UnicodeLength(settings->Proxy)!=0 ||
+ UnicodeLength(settings->Proxy2)!=0) {
+ req[pos++] = 0x00;
+ }
+ req[pos++] = 0x00; req[pos++] = 0x00;
+ req[pos++] = 0x07; //unknown
+ req[pos++] = 0x00; req[pos++] = 0x00;
+ req[pos++] = 0x80; //unknown
+ req[pos++] = 0x01; //unknown
+ req[pos++] = 0x05; //unknown
+ req[pos++] = 0x00; req[pos++] = 0x00;
+
+ /* Proxy data blocks */
+ for (i=0;i<4;i++) {
+ port = 8080;
+ Proxy = NULL;
+ if (i==0) {
+ port = settings->ProxyPort;
+ Proxy = settings->Proxy;
+ } else if (i==1) {
+ port = settings->Proxy2Port;
+ Proxy = settings->Proxy2;
+ }
+ req[pos++] = 0x08; req[pos++] = 0x00;
+ if (Proxy != NULL && UnicodeLength(Proxy)!=0) {
+ if (UnicodeLength(Proxy)%2 != 0) {
+ req[pos++] = (12 + (UnicodeLength(Proxy)+1)*2)/256;
+ req[pos++] = (12 + (UnicodeLength(Proxy)+1)*2)%256;
+ } else {
+ req[pos++] = (12 + UnicodeLength(Proxy)*2)/256;
+ req[pos++] = (12 + UnicodeLength(Proxy)*2)%256;
+ }
+ } else {
+ req[pos++] = 12/256;
+ req[pos++] = 12%256;
+ }
+ req[pos++] = i+1;
+ if (Proxy != NULL) {
+ req[pos++] = UnicodeLength(Proxy);
+ } else {
+ req[pos++] = 0;
+ }
+ req[pos++] = port/256;
+ req[pos++] = port%256;
+ req[pos++] = 0x00;
+
+ req[pos++] = 0x00;
+ req[pos++] = 0x01;
+
+ req[pos++] = 0x00;
+ if (Proxy != NULL && UnicodeLength(Proxy)!=0) {
+ CopyUnicodeString(req+pos,Proxy);
+ pos+=UnicodeLength(Proxy)*2;
+ if (UnicodeLength(Proxy)%2 != 0) {
+ req[pos++] = 0x00;
+ req[pos++] = 0x00;
+ }
+ }
+ }
+
+ req[pos++] = 0x09; req[pos++] = 0x00; req[pos++] = 0x00;
+ req[pos++] = 0x0C; req[pos++] = 0x02; req[pos++] = 0x00;
+ req[pos++] = 0x00; req[pos++] = 0x02; req[pos++] = 0x00;
+ req[pos++] = 0x00; req[pos++] = 0x00; req[pos++] = 0x00;
+ } else {
+ /* end of blocks ? */
+ memcpy(req + pos, "\x80\x00\x00\x0c", 4); pos += 4;
+ }
+
+ UnLock[4] = settings->Location-1;
+ smprintf(s, "Making Connection settings read-write\n");
+ error = GSM_WaitFor (s, UnLock, 5, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+
+ switch (Type) {
+ case N6510_MMS_SETTINGS:
+ smprintf(s, "Setting MMS settings\n");
+ break;
+ case N6510_CHAT_SETTINGS:
+ smprintf(s, "Setting Chat settings\n");
+ break;
+ case N6510_WAP_SETTINGS:
+ smprintf(s, "Setting WAP settings\n");
+ break;
+ case N6510_SYNCML_SETTINGS:
+ smprintf(s, "Setting SyncML settings\n");
+ break;
+ }
+ error = GSM_WaitFor (s, req, pos, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) {
+ if (error == ERR_INSIDEPHONEMENU || error == ERR_INVALIDLOCATION) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ if (settings->ReadOnly) {
+ Lock[4] = settings->Location-1;
+ smprintf(s, "Making Connection settings readonly\n");
+ error = GSM_WaitFor (s, Lock, 5, 0x3f, 4, ID_SetConnectSet);
+ if (error != ERR_NONE) return error;
+ }
+
+ error = DCT3DCT4_SetActiveConnectSet(s, settings);
+ if (error != ERR_NONE) return error;
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+static GSM_Error N6510_SetWAPSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+ return N6510_SetConnectionSettings(s, settings, N6510_WAP_SETTINGS);
+}
+
+static GSM_Error N6510_SetMMSSettings(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+ return N6510_SetConnectionSettings(s, settings, N6510_MMS_SETTINGS);
+}
+
+static GSM_Error N6510_ReplyGetOriginalIMEI(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ if (msg.Buffer[7] == 0x00) {
+ smprintf(s, "No SIM card\n");
+ return ERR_SECURITYERROR;
+ } else {
+ return NOKIA_ReplyGetPhoneString(msg, s);
+ }
+}
+
+static GSM_Error N6510_GetOriginalIMEI(GSM_StateMachine *s, char *value)
+{
+ return NOKIA_GetPhoneString(s,"\x00\x07\x02\x01\x00\x01",6,0x42,value,ID_GetOriginalIMEI,14);
+}
+
+static GSM_Error N6510_ReplyGetSMSStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x09:
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "Max. in phone memory : %i\n",msg.Buffer[10]*256+msg.Buffer[11]);
+ smprintf(s, "Used in phone memory : %i\n",msg.Buffer[12]*256+msg.Buffer[13]);
+ smprintf(s, "Unread in phone memory : %i\n",msg.Buffer[14]*256+msg.Buffer[15]);
+ smprintf(s, "Max. in SIM : %i\n",msg.Buffer[22]*256+msg.Buffer[23]);
+ smprintf(s, "Used in SIM : %i\n",msg.Buffer[24]*256+msg.Buffer[25]);
+ smprintf(s, "Unread in SIM : %i\n",msg.Buffer[26]*256+msg.Buffer[27]);
+ Data->SMSStatus->PhoneSize = msg.Buffer[10]*256+msg.Buffer[11];
+ Data->SMSStatus->PhoneUsed = msg.Buffer[12]*256+msg.Buffer[13];
+ Data->SMSStatus->PhoneUnRead = msg.Buffer[14]*256+msg.Buffer[15];
+ Data->SMSStatus->SIMSize = msg.Buffer[22]*256+msg.Buffer[23];
+ Data->SMSStatus->SIMUsed = msg.Buffer[24]*256+msg.Buffer[25];
+ Data->SMSStatus->SIMUnRead = msg.Buffer[26]*256+msg.Buffer[27];
+ return ERR_NONE;
+ case 0x0f:
+ smprintf(s, "No PIN\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ case 0x1a:
+ smprintf(s, "Wait a moment. Phone is during power on and busy now\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetSMSStatus(GSM_StateMachine *s, GSM_SMSMemoryStatus *status)
+{
+ GSM_Error error;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x08, 0x00, 0x00};
+
+ s->Phone.Data.SMSStatus=status;
+ smprintf(s, "Getting SMS status\n");
+ error = GSM_WaitFor (s, req, 6, 0x14, 2, ID_GetSMSStatus);
+ if (error != ERR_NONE) return error;
+
+ /* DCT4 family doesn't show in frame with SMS status info
+ * about Templates. We get separately info about this SMS folder.
+ */
+ error = N6510_GetSMSFolderStatus(s, 0x06);
+ if (error != ERR_NONE) return error;
+ status->TemplatesUsed = Priv->LastSMSFolder.Number;
+
+ return error;
+}
+
+static GSM_Error N6510_ReplyDeleteSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x05:
+ smprintf(s, "SMS deleted OK\n");
+ return ERR_NONE;
+ case 0x06:
+ switch (msg.Buffer[4]) {
+ case 0x02:
+ smprintf(s, "Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "Unknown error: %02x\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_DeleteSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ unsigned char folderid;
+ int location;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x04,
+ 0x01, /* 0x01=SM, 0x02=ME */
+ 0x00, /* FolderID */
+ 0x00, 0x02, /* Location */
+ 0x0F, 0x55};
+
+ N6510_GetSMSLocation(s, sms, &folderid, &location);
+
+ switch (folderid) {
+ case 0x01: req[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: req[5] = 0x03; break; /* OUTBOX SIM */
+ default : req[5] = folderid - 1; req[4] = 0x02; break; /* ME folders */
+ }
+ req[6]=location / 256;
+ req[7]=location;
+
+ smprintf(s, "Deleting sms\n");
+ return GSM_WaitFor (s, req, 10, 0x14, 4, ID_DeleteSMSMessage);
+}
+
+static GSM_Error N6510_ReplySendSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[8]) {
+ case 0x00:
+ smprintf(s, "SMS sent OK, TPMR for sent sms is %02x\n",msg.Buffer[10]);
+ if (s->User.SendSMSStatus!=NULL) s->User.SendSMSStatus(s->CurrentConfig->Device,0,msg.Buffer[10]);
+ return ERR_NONE;
+ default:
+ smprintf(s, "SMS not sent OK, error code probably %i\n",msg.Buffer[8]);
+ if (s->User.SendSMSStatus!=NULL) s->User.SendSMSStatus(s->CurrentConfig->Device,msg.Buffer[8],msg.Buffer[10]);
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error N6510_SendSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int length = 11;
+ GSM_Error error;
+ GSM_SMSMessageLayout Layout;
+ unsigned char req [300] = {
+ N6110_FRAME_HEADER, 0x02, 0x00, 0x00, 0x00, 0x55, 0x55};
+
+ if (sms->PDU == SMS_Deliver) sms->PDU = SMS_Submit;
+ memset(req+9,0x00,sizeof(req) - 9);
+ error=N6510_EncodeSMSFrame(s, sms, req + 9, &Layout, &length);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Sending sms\n");
+ return s->Protocol.Functions->WriteMessage(s, req, length + 9, 0x02);
+}
+
+static GSM_Error N6510_ReplyGetSecurityStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Security Code status received: ");
+ switch (msg.Buffer[4]) {
+ case 0x01 : smprintf(s, "waiting for Security Code.\n"); *Data->SecurityStatus = SEC_SecurityCode; break;
+ case 0x07 :
+ case 0x02 : smprintf(s, "waiting for PIN.\n"); *Data->SecurityStatus = SEC_Pin; break;
+ case 0x03 : smprintf(s, "waiting for PUK.\n"); *Data->SecurityStatus = SEC_Puk; break;
+ case 0x05 : smprintf(s, "PIN ok, SIM ok\n"); *Data->SecurityStatus = SEC_None; break;
+ case 0x06 : smprintf(s, "No input status\n"); *Data->SecurityStatus = SEC_None; break;
+ case 0x16 : smprintf(s, "No SIM card\n"); *Data->SecurityStatus = SEC_None; break;
+ case 0x1A : smprintf(s, "SIM card rejected!\n"); *Data->SecurityStatus = SEC_None; break;
+ default : smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetSecurityStatus(GSM_StateMachine *s, GSM_SecurityCodeType *Status)
+{
+ unsigned char req[5] = {N6110_FRAME_HEADER, 0x11, 0x00};
+
+ s->Phone.Data.SecurityStatus=Status;
+ smprintf(s, "Getting security code status\n");
+ return GSM_WaitFor (s, req, 5, 0x08, 2, ID_GetSecurityStatus);
+}
+
+static GSM_Error N6510_ReplyEnterSecurityCode(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x08:
+ smprintf(s, "Security code OK\n");
+ return ERR_NONE;
+ case 0x09:
+ switch (msg.Buffer[4]) {
+ case 0x06:
+ smprintf(s, "Wrong PIN\n");
+ return ERR_SECURITYERROR;
+ case 0x09:
+ smprintf(s, "Wrong PUK\n");
+ return ERR_SECURITYERROR;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_EnterSecurityCode(GSM_StateMachine *s, GSM_SecurityCode Code)
+{
+ int len = 0;
+ unsigned char req[15] = {N6110_FRAME_HEADER, 0x07,
+ 0x00}; /* Code type */
+
+ switch (Code.Type) {
+ case SEC_Pin : req[4] = 0x02; break;
+ case SEC_Puk : req[4] = 0x03; break;/* FIXME */
+ default : return ERR_NOTSUPPORTED;
+ }
+
+ len = strlen(Code.Code);
+ memcpy(req+5,Code.Code,len);
+ req[5+len]=0x00;
+
+ smprintf(s, "Entering security code\n");
+ return GSM_WaitFor (s, req, 6+len, 0x08, 4, ID_EnterSecurityCode);
+}
+
+static GSM_Error N6510_ReplySaveSMSMessage(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char folder;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x01:
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ smprintf(s, "Done OK\n");
+ smprintf(s, "Folder info: %i %i\n",msg.Buffer[5],msg.Buffer[8]);
+ switch (msg.Buffer[8]) {
+ case 0x02 : if (msg.Buffer[5] == 0x02) {
+ folder = 0x03; /* INBOX ME */
+ } else {
+ folder = 0x01; /* INBOX SIM */
+ }
+ break;
+ case 0x03 : if (msg.Buffer[5] == 0x02) {
+ folder = 0x04; /* OUTBOX ME */
+ } else {
+ folder = 0x02; /* OUTBOX SIM */
+ }
+ break;
+ default : folder = msg.Buffer[8] + 1;
+ }
+ N6510_SetSMSLocation(s, Data->SaveSMSMessage,folder,msg.Buffer[6]*256+msg.Buffer[7]);
+ smprintf(s, "Saved in folder %i at location %i\n",folder, msg.Buffer[6]*256+msg.Buffer[7]);
+ Data->SaveSMSMessage->Folder = folder;
+ return ERR_NONE;
+ case 0x02:
+ printf("Incorrect location\n");
+ return ERR_INVALIDLOCATION;
+ case 0x05:
+ printf("Incorrect folder\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ case 0x17:
+ smprintf(s, "SMS name changed\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_PrivSetSMSMessage(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int location, length = 11;
+ unsigned char folderid, folder;
+ GSM_SMSMessageLayout Layout;
+ GSM_Error error;
+ unsigned char req [300] = {
+ N6110_FRAME_HEADER, 0x00,
+ 0x01, /* 1 = SIM, 2 = ME */
+ 0x02, /* Folder */
+ 0x00, 0x01, /* Location */
+ 0x01}; /* SMS state */
+ unsigned char NameReq[200] = {
+ N6110_FRAME_HEADER, 0x16,
+ 0x01, /* 1 = SIM, 2 = ME */
+ 0x02, /* Folder */
+ 0x00, 0x01}; /* Location */
+
+ N6510_GetSMSLocation(s, sms, &folderid, &location);
+ switch (folderid) {
+ case 0x01: req[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: req[5] = 0x03; break; /* OUTBOX SIM */
+ default : req[5] = folderid - 1; req[4] = 0x02; break; /* ME folders */
+ }
+ req[6]=location / 256;
+ req[7]=location;
+
+ switch (sms->PDU) {
+ case SMS_Submit:
+ /* Inbox */
+ if (folderid == 0x01 || folderid == 0x03) sms->PDU = SMS_Deliver;
+ break;
+ case SMS_Deliver:
+ /* SIM Outbox */
+ if (folderid == 0x02) sms->PDU = SMS_Submit;
+ break;
+ default:
+ return ERR_UNKNOWN;
+ }
+ if (sms->PDU == SMS_Deliver) {
+ switch (sms->State) {
+ case SMS_Sent : /* We use GSM_Read, because phone return error */
+ case SMS_Read : req[8] = 0x01; break;
+ case SMS_UnSent : /* We use GSM_UnRead, because phone return error */
+ case SMS_UnRead : req[8] = 0x03; break;
+ }
+ } else {
+ switch (sms->State) {
+ case SMS_Sent : /* We use GSM_Sent, because phone change folder */
+ case SMS_Read : req[8] = 0x05; break;
+ case SMS_UnSent : /* We use GSM_UnSent, because phone change folder */
+ case SMS_UnRead : req[8] = 0x07; break;
+ }
+ }
+ memset(req+9,0x00,sizeof(req) - 9);
+ error=N6510_EncodeSMSFrame(s, sms, req + 9, &Layout, &length);
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.SaveSMSMessage=sms;
+ smprintf(s, "Saving sms\n");
+ error=GSM_WaitFor (s, req, length+9, 0x14, 4, ID_SaveSMSMessage);
+ if (error == ERR_NONE && UnicodeLength(sms->Name)!=0) {
+ folder = sms->Folder;
+ sms->Folder = 0;
+ N6510_GetSMSLocation(s, sms, &folderid, &location);
+ switch (folderid) {
+ case 0x01: NameReq[5] = 0x02; break; /* INBOX SIM */
+ case 0x02: NameReq[5] = 0x03; break; /* OUTBOX SIM */
+ default : NameReq[5] = folderid - 1; NameReq[4] = 0x02; break; /* ME folders */
+ }
+ NameReq[6]=location / 256;
+ NameReq[7]=location;
+ length = 8;
+ CopyUnicodeString(NameReq+length, sms->Name);
+ length = length+UnicodeLength(sms->Name)*2;
+ NameReq[length++] = 0;
+ NameReq[length++] = 0;
+ error=GSM_WaitFor (s, NameReq, length, 0x14, 4, ID_SaveSMSMessage);
+ sms->Folder = folder;
+ }
+ return error;
+}
+
+static GSM_Error N6510_SetSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int location;
+ unsigned char folderid;
+
+ N6510_GetSMSLocation(s, sms, &folderid, &location);
+ if (location == 0) return ERR_INVALIDLOCATION;
+ return N6510_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N6510_AddSMS(GSM_StateMachine *s, GSM_SMSMessage *sms)
+{
+ int location;
+ unsigned char folderid;
+
+ N6510_GetSMSLocation(s, sms, &folderid, &location);
+ location = 0;
+ N6510_SetSMSLocation(s, sms, folderid, location);
+ return N6510_PrivSetSMSMessage(s, sms);
+}
+
+static GSM_Error N6510_ReplyGetDateTime(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Date & time received\n");
+ if (msg.Buffer[4]==0x01) {
+ NOKIA_DecodeDateTime(s, msg.Buffer+10, s->Phone.Data.DateTime);
+ return ERR_NONE;
+ }
+ smprintf(s, "Not set in phone\n");
+ return ERR_EMPTY;
+}
+
+static GSM_Error N6510_GetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0A, 0x00, 0x00};
+
+ s->Phone.Data.DateTime=date_time;
+ smprintf(s, "Getting date & time\n");
+ return GSM_WaitFor (s, req, 6, 0x19, 4, ID_GetDateTime);
+}
+
+static GSM_Error N6510_ReplySetDateTime(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Date & time set\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetDateTime(GSM_StateMachine *s, GSM_DateTime *date_time)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER,
+ 0x01, 0x00, 0x01, 0x01, 0x0c, 0x01, 0x03,
+ 0x00, 0x00, /* Year */
+ 0x08, 0x01, /* Month & Day */
+ 0x15, 0x1f, /* Hours & Minutes */
+ 0x2b, /* Second ? */
+ 0x00};
+
+ NOKIA_EncodeDateTime(s, req+10, date_time);
+ req[16] = date_time->Second;
+ smprintf(s, "Setting date & time\n");
+ return GSM_WaitFor (s, req, 18, 0x19, 4, ID_SetDateTime);
+}
+
+static GSM_Error N6510_ReplyGetManufactureMonth(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ if (msg.Buffer[7] == 0x00) {
+ smprintf(s, "No SIM card\n");
+ return ERR_SECURITYERROR;
+ } else {
+ sprintf(s->Phone.Data.PhoneString,"%02i/%04i",msg.Buffer[13],msg.Buffer[14]*256+msg.Buffer[15]);
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error N6510_GetManufactureMonth(GSM_StateMachine *s, char *value)
+{
+ unsigned char req[6] = {0x00, 0x05, 0x02, 0x01, 0x00, 0x02};
+// unsigned char req[6] = {0x00, 0x03, 0x04, 0x0B, 0x01, 0x00};
+
+ s->Phone.Data.PhoneString=value;
+ smprintf(s, "Getting manufacture month\n");
+ return GSM_WaitFor (s, req, 6, 0x42, 2, ID_GetManufactureMonth);
+// return GSM_WaitFor (s, req, 6, 0x1B, 2, ID_GetManufactureMonth);
+}
+
+static GSM_Error N6510_ReplyGetAlarm(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch(msg.Buffer[3]) {
+ case 0x1A:
+ smprintf(s, " Alarm: %02d:%02d\n", msg.Buffer[14], msg.Buffer[15]);
+ Data->Alarm->Repeating = true;
+ Data->Alarm->Text[0] = 0;
+ Data->Alarm->Text[1] = 0;
+ Data->Alarm->DateTime.Hour = msg.Buffer[14];
+ Data->Alarm->DateTime.Minute = msg.Buffer[15];
+ Data->Alarm->DateTime.Second = 0;
+ return ERR_NONE;
+ case 0x20:
+ smprintf(s, "Alarm state received\n");
+ if (msg.Buffer[37] == 0x01) {
+ smprintf(s, " Not set in phone\n");
+ return ERR_EMPTY;
+ }
+ smprintf(s, "Enabled\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ unsigned char StateReq[] = {N6110_FRAME_HEADER, 0x1f, 0x01, 0x00};
+ unsigned char GetReq [] = {N6110_FRAME_HEADER, 0x19, 0x00, 0x02};
+ GSM_Error error;
+
+ if (alarm->Location != 1) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.Alarm=alarm;
+ smprintf(s, "Getting alarm state\n");
+ error = GSM_WaitFor (s, StateReq, 6, 0x19, 4, ID_GetAlarm);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting alarm\n");
+ return GSM_WaitFor (s, GetReq, 6, 0x19, 4, ID_GetAlarm);
+}
+
+static GSM_Error N6510_ReplySetAlarm(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Alarm set\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetAlarm(GSM_StateMachine *s, GSM_Alarm *alarm)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER,
+ 0x11, 0x00, 0x01, 0x01, 0x0c, 0x02,
+ 0x01, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, /* Hours, Minutes */
+ 0x00, 0x00, 0x00 };
+
+ if (alarm->Location != 1) return ERR_NOTSUPPORTED;
+
+ req[14] = alarm->DateTime.Hour;
+ req[15] = alarm->DateTime.Minute;
+
+ smprintf(s, "Setting alarm\n");
+ return GSM_WaitFor (s, req, 19, 0x19, 4, ID_SetAlarm);
+}
+
+static GSM_Error N6510_ReplyGetRingtonesInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int tmp,i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Ringtones info received\n");
+ memset(Data->RingtonesInfo,0,sizeof(GSM_AllRingtonesInfo));
+ if (msg.Buffer[4] * 256 + msg.Buffer[5] == 0x00) return ERR_EMPTY;
+ Data->RingtonesInfo->Number = msg.Buffer[4] * 256 + msg.Buffer[5];
+ tmp = 6;
+ for (i=0;i<Data->RingtonesInfo->Number;i++) {
+ Data->RingtonesInfo->Ringtone[i].Group = msg.Buffer[tmp+4];
+ Data->RingtonesInfo->Ringtone[i].ID = msg.Buffer[tmp+2] * 256 + msg.Buffer[tmp+3];
+ memcpy(Data->RingtonesInfo->Ringtone[i].Name,msg.Buffer+tmp+8,(msg.Buffer[tmp+6]*256+msg.Buffer[tmp+7])*2);
+ smprintf(s, "%5i (%5i). \"%s\"\n",
+ Data->RingtonesInfo->Ringtone[i].ID,
+ Data->RingtonesInfo->Ringtone[i].Group,
+ DecodeUnicodeString(Data->RingtonesInfo->Ringtone[i].Name));
+ tmp = tmp + (msg.Buffer[tmp]*256+msg.Buffer[tmp+1]);
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_PrivGetRingtonesInfo(GSM_StateMachine *s, GSM_AllRingtonesInfo *Info, bool AllRingtones)
+{
+ GSM_Error error;
+ unsigned char UserReq[8] = {N7110_FRAME_HEADER, 0x07, 0x00, 0x00, 0x00, 0x02};
+// unsigned char All_Req[9] = {N7110_FRAME_HEADER, 0x07, 0x00, 0x00, 0xFE, 0x00, 0x7D};
+ unsigned char All_Req[8] = {N7110_FRAME_HEADER, 0x07, 0x00, 0x00, 0x00, 0x00};
+
+ s->Phone.Data.RingtonesInfo=Info;
+ smprintf(s, "Getting binary ringtones ID\n");
+ if (AllRingtones) {
+// error = GSM_WaitFor (s, All_Req, 9, 0x1f, 4, ID_GetRingtonesInfo);
+ error = GSM_WaitFor (s, All_Req, 8, 0x1f, 4, ID_GetRingtonesInfo);
+ if (error == ERR_EMPTY && Info->Number == 0) return ERR_NOTSUPPORTED;
+ return error;
+ } else {
+ error = GSM_WaitFor (s, UserReq, 8, 0x1f, 4, ID_GetRingtonesInfo);
+ if (error == ERR_EMPTY && Info->Number == 0) return ERR_NOTSUPPORTED;
+ return error;
+ }
+}
+
+static GSM_Error N6510_GetRingtonesInfo(GSM_StateMachine *s, GSM_AllRingtonesInfo *Info)
+{
+ return N6510_PrivGetRingtonesInfo(s, Info, true);
+}
+
+static GSM_Error N6510_ReplyGetRingtone(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int tmp,i;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "Ringtone received\n");
+ memcpy(Data->Ringtone->Name,msg.Buffer+8,msg.Buffer[7]*2);
+ Data->Ringtone->Name[msg.Buffer[7]*2]=0;
+ Data->Ringtone->Name[msg.Buffer[7]*2+1]=0;
+ smprintf(s, "Name \"%s\"\n",DecodeUnicodeString(Data->Ringtone->Name));
+ if (msg.Buffer[msg.Buffer[7]*2+10] == 'M' &&
+ msg.Buffer[msg.Buffer[7]*2+11] == 'T' &&
+ msg.Buffer[msg.Buffer[7]*2+12] == 'h' &&
+ msg.Buffer[msg.Buffer[7]*2+13] == 'd') {
+ smprintf(s,"MIDI\n");
+ tmp = msg.Buffer[7]*2+10;
+ i = msg.Length - 2; /* ?????? */
+ Data->Ringtone->Format = RING_MIDI;
+ } else {
+ /* Looking for end */
+ i=8+msg.Buffer[7]*2+3;
+ tmp = i;
+ while (true) {
+ if (msg.Buffer[i]==0x07 && msg.Buffer[i+1]==0x0b) {
+ i=i+2; break;
+ }
+ i++;
+ if (i==msg.Length) return ERR_EMPTY;
+ }
+ }
+ /* Copying frame */
+ memcpy(Data->Ringtone->NokiaBinary.Frame,msg.Buffer+tmp,i-tmp);
+ Data->Ringtone->NokiaBinary.Length=i-tmp;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetRingtone(GSM_StateMachine *s, GSM_Ringtone *Ringtone, bool PhoneRingtone)
+{
+ GSM_AllRingtonesInfo Info;
+ GSM_Error error;
+ unsigned char req2[6] = {N7110_FRAME_HEADER, 0x12,
+ 0x00, 0xe7}; /* Location */
+
+ if (Ringtone->Format == 0x00) Ringtone->Format = RING_NOKIABINARY;
+
+ switch (Ringtone->Format) {
+ case RING_NOTETONE:
+ /* In the future get binary and convert */
+ return ERR_NOTSUPPORTED;
+ case RING_NOKIABINARY:
+ s->Phone.Data.Ringtone = Ringtone;
+ Info.Number = 0;
+ error=N6510_PrivGetRingtonesInfo(s, &Info, PhoneRingtone);
+ if (error != ERR_NONE) return error;
+ if (Ringtone->Location > Info.Number) return ERR_INVALIDLOCATION;
+ req2[4] = Info.Ringtone[Ringtone->Location-1].ID / 256;
+ req2[5] = Info.Ringtone[Ringtone->Location-1].ID % 256;
+ smprintf(s, "Getting binary ringtone\n");
+ return GSM_WaitFor (s, req2, 6, 0x1f, 4, ID_GetRingtone);
+ case RING_MIDI:
+ return ERR_NOTSUPPORTED;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6510_PlayTone(GSM_StateMachine *s, int Herz, unsigned char Volume, bool start)
+{
+ GSM_Error error;
+ unsigned char reqStart[] = {
+ 0x00,0x06,0x01,0x00,0x07,0x00 };
+ unsigned char reqPlay[] = {
+ 0x00,0x06,0x01,0x14,0x05,0x04,
+ 0x00,0x00,0x00,0x03,0x03,0x08,
+ 0x00,0x00,0x00,0x01,0x00,0x00,
+ 0x03,0x08,0x01,0x00,
+ 0x07,0xd0, /*Frequency */
+ 0x00,0x00,0x03,0x08,0x02,0x00,0x00,
+ 0x05, /*Volume */
+ 0x00,0x00};
+ unsigned char reqOff[] = {
+ 0x00,0x06,0x01,0x14,0x05,0x05,
+ 0x00,0x00,0x00,0x01,0x03,0x08,
+ 0x05,0x00,0x00,0x08,0x00,0x00};
+// unsigned char reqOff2[] = {
+// 0x00,0x06,0x01,0x14,0x05,0x04,
+// 0x00,0x00,0x00,0x01,0x03,0x08,
+// 0x00,0x00,0x00,0x00,0x00,0x00};
+
+ if (start) {
+ smprintf(s, "Enabling sound - part 1\n");
+ error=GSM_WaitFor (s, reqStart, 6, 0x0b, 4, ID_PlayTone);
+ if (error!=ERR_NONE) return error;
+ smprintf(s, "Enabling sound - part 2 (disabling sound command)\n");
+ error=GSM_WaitFor (s, reqOff, 18, 0x0b, 4, ID_PlayTone);
+ if (error!=ERR_NONE) return error;
+ }
+
+ /* For Herz==255*255 we have silent */
+ if (Herz!=255*255) {
+ reqPlay[23] = Herz%256;
+ reqPlay[22] = Herz/256;
+ reqPlay[31] = Volume;
+ smprintf(s, "Playing sound\n");
+ return GSM_WaitFor (s, reqPlay, 34, 0x0b, 4, ID_PlayTone);
+ } else {
+ reqPlay[23] = 0;
+ reqPlay[22] = 0;
+ reqPlay[31] = 0;
+ smprintf(s, "Playing silent sound\n");
+ return GSM_WaitFor (s, reqPlay, 34, 0x0b, 4, ID_PlayTone);
+
+// smprintf(s, "Disabling sound - part 1\n");
+// error=GSM_WaitFor (s, reqOff, 18, 0x0b, 4, ID_PlayTone);
+// if (error!=ERR_NONE) return error;
+// smprintf(s, "Disabling sound - part 2\n");
+// return GSM_WaitFor (s, reqOff2, 18, 0x0b, 4, ID_PlayTone);
+ }
+}
+
+static GSM_Error N6510_ReplyGetPPM(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ int pos = 6,len;
+
+ smprintf(s, "Received phone info\n");
+
+ while(pos < msg.Length) {
+ if (msg.Buffer[pos] == 0x55 && msg.Buffer[pos+1] == 0x55) {
+ while(1) {
+ if (msg.Buffer[pos] != 0x55) break;
+ pos++;
+ }
+ }
+ len = pos;
+ while(1) {
+ if (msg.Buffer[len] == 0x00 && msg.Buffer[len+1] == 0x00) break;
+ len++;
+ }
+ while(1) {
+ if (msg.Buffer[len] != 0x00) break;
+ len++;
+ }
+ len = len-pos;
+ smprintf(s, "Block with ID %02x",msg.Buffer[pos]);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, msg.Buffer+pos, len);
+#endif
+ switch (msg.Buffer[pos]) {
+ case 0x49:
+ smprintf(s, "hardware version\n");
+ break;
+ case 0x58:
+ pos += 3;
+ while (msg.Buffer[pos] != 0x00) pos++;
+ Data->PhoneString[0] = msg.Buffer[pos - 1];
+ Data->PhoneString[1] = 0x00;
+ smprintf(s, "PPM %s\n",Data->PhoneString);
+ return ERR_NONE;
+ default:
+ break;
+ }
+ pos += len;
+ }
+ return ERR_NOTSUPPORTED;
+}
+
+static GSM_Error N6510_GetPPM(GSM_StateMachine *s,char *value)
+{
+// unsigned char req[6] = {N6110_FRAME_HEADER, 0x07, 0x01, 0xff};
+ unsigned char req[6] = {N6110_FRAME_HEADER, 0x07, 0x01, 0x00};
+
+ s->Phone.Data.PhoneString=value;
+ smprintf(s, "Getting PPM\n");
+ return GSM_WaitFor (s, req, 6, 0x1b, 3, ID_GetPPM);
+}
+
+static GSM_Error N6510_GetSpeedDial(GSM_StateMachine *s, GSM_SpeedDial *SpeedDial)
+{
+ GSM_MemoryEntry pbk;
+ GSM_Error error;
+
+ pbk.MemoryType = MEM7110_SP;
+ pbk.Location = SpeedDial->Location;
+ SpeedDial->MemoryLocation = 0;
+ s->Phone.Data.SpeedDial = SpeedDial;
+
+ smprintf(s, "Getting speed dial\n");
+ error=N6510_GetMemory(s,&pbk);
+ switch (error) {
+ case ERR_NOTSUPPORTED:
+ smprintf(s, "No speed dials set in phone\n");
+ return ERR_EMPTY;
+ case ERR_NONE:
+ if (SpeedDial->MemoryLocation == 0) {
+ smprintf(s, "Speed dial not assigned or error in firmware\n");
+ return ERR_EMPTY;
+ }
+ return ERR_NONE;
+ default:
+ return error;
+ }
+}
+
+static GSM_Error N6510_ReplyGetProfile(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char *blockstart;
+ int i,j;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ blockstart = msg.Buffer + 7;
+ for (i = 0; i < 11; i++) {
+ smprintf(s, "Profile feature %02x ",blockstart[1]);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, blockstart, blockstart[0]);
+#endif
+
+ switch (blockstart[1]) {
+ case 0x03:
+ smprintf(s, "Ringtone ID\n");
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_RingtoneID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = blockstart[7];
+ if (blockstart[7] == 0x00) {
+ Data->Profile->FeatureValue[Data->Profile->FeaturesNumber] = blockstart[10];
+ }
+ Data->Profile->FeaturesNumber++;
+ break;
+ case 0x05: /* SMS tone */
+ j = Data->Profile->FeaturesNumber;
+ NOKIA_FindFeatureValue(s, Profile71_65,blockstart[1],blockstart[7],Data,false);
+ if (j == Data->Profile->FeaturesNumber) {
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_MessageTone;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = PROFILE_MESSAGE_PERSONAL;
+ Data->Profile->FeaturesNumber++;
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_MessageToneID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = blockstart[7];
+ Data->Profile->FeaturesNumber++;
+ }
+ break;
+ case 0x08: /* Caller groups */
+ NOKIA_FindFeatureValue(s, Profile71_65,blockstart[1],blockstart[7],Data,true);
+ break;
+ case 0x0c :
+ CopyUnicodeString(Data->Profile->Name,blockstart + 7);
+ smprintf(s, "profile Name: \"%s\"\n", DecodeUnicodeString(Data->Profile->Name));
+ Data->Profile->DefaultName = false;
+ break;
+ default:
+ NOKIA_FindFeatureValue(s, Profile71_65,blockstart[1],blockstart[7],Data,false);
+ }
+ blockstart = blockstart + blockstart[0];
+ }
+ return ERR_NONE;
+ case 0x06:
+ Data->Profile->Active = false;
+ if (Data->Profile->Location == msg.Buffer[5]) Data->Profile->Active = true;
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ unsigned char req[150] = {N6110_FRAME_HEADER, 0x01, 0x01, 0x0C, 0x01};
+ unsigned char reqActive[] = {N6110_FRAME_HEADER, 0x05};
+ int i, length = 7;
+ GSM_Error error;
+
+ /* For now !!! */
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"3510")) {
+ if (s->Phone.Data.VerNum>3.37) return ERR_NOTSUPPORTED;
+ }
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"6230")) {
+ return ERR_NOTSUPPORTED;
+ }
+
+ if (Profile->Location>5) return ERR_INVALIDLOCATION;
+
+ for (i = 0; i < 0x0a; i++) {
+ req[length++] = 0x04;
+ req[length++] = Profile->Location;
+ req[length++] = i;
+ req[length++] = 0x01;
+ }
+
+ req[length++] = 0x04;
+ req[length++] = Profile->Location;
+ req[length++] = 0x0c;
+ req[length++] = 0x01;
+
+ req[length++] = 0x04;
+
+ Profile->CarKitProfile = false;
+ Profile->HeadSetProfile = false;
+
+ Profile->FeaturesNumber = 0;
+
+ s->Phone.Data.Profile=Profile;
+ smprintf(s, "Getting profile\n");
+ error = GSM_WaitFor (s, req, length, 0x39, 4, ID_GetProfile);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Checking, which profile is active\n");
+ return GSM_WaitFor (s, reqActive, 4, 0x39, 4, ID_GetProfile);
+}
+
+static GSM_Error N6510_ReplySetProfile(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char *blockstart;
+ int i;
+
+ smprintf(s, "Response to profile writing received!\n");
+
+ blockstart = msg.Buffer + 6;
+ for (i = 0; i < msg.Buffer[5]; i++) {
+ switch (blockstart[2]) {
+ case 0x00: smprintf(s, "keypad tone level"); break;
+ case 0x02: smprintf(s, "call alert"); break;
+ case 0x03: smprintf(s, "ringtone"); break;
+ case 0x04: smprintf(s, "ringtone volume"); break;
+ case 0x05: smprintf(s, "SMS tone"); break;
+ case 0x06: smprintf(s, "vibration"); break;
+ case 0x07: smprintf(s, "warning tone level"); break;
+ case 0x08: smprintf(s, "caller groups"); break;
+ case 0x09: smprintf(s, "automatic answer"); break;
+ case 0x0c: smprintf(s, "name"); break;
+ default:
+ smprintf(s, "Unknown block type %02x", blockstart[2]);
+ break;
+ }
+ if (msg.Buffer[4] == 0x00) {
+ smprintf(s, ": set OK\n");
+ } else {
+ smprintf(s, ": setting error %i\n", msg.Buffer[4]);
+ }
+ blockstart = blockstart + blockstart[1];
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetProfile(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ int i, length = 7, blocks = 0;
+ bool found;
+ unsigned char ID,Value;
+ unsigned char req[150] = {N6110_FRAME_HEADER, 0x03, 0x01,
+ 0x06, /* Number of blocks */
+ 0x03};
+
+ if (Profile->Location>5) return ERR_INVALIDLOCATION;
+
+ for (i=0;i<Profile->FeaturesNumber;i++) {
+ found = false;
+ switch (Profile->FeatureID[i]) {
+ case Profile_RingtoneID:
+ ID = 0x03;
+ Value = Profile->FeatureValue[i];
+ found = true;
+ break;
+ default:
+ found=NOKIA_FindPhoneFeatureValue(
+ s,
+ Profile71_65,
+ Profile->FeatureID[i],Profile->FeatureValue[i],
+ &ID,&Value);
+ }
+ if (found) {
+ req[length] = 0x09;
+ req[length + 1] = ID;
+ req[length + 2] = Profile->Location;
+ memcpy(req + length + 4, "\x00\x00\x01", 3);
+ req[length + 8] = 0x03;
+ req[length + 3] = req[length + 7] = Value;
+ blocks++;
+ length += 9;
+ }
+ }
+
+ smprintf(s, "Setting profile\n");
+ return GSM_WaitFor (s, req, length, 0x39, 4, ID_SetProfile);
+}
+
+static GSM_Error N6510_ReplyIncomingSMS(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_SMSMessage sms;
+
+#ifdef DEBUG
+ smprintf(s, "SMS message received\n");
+ N6510_DecodeSMSFrame(s, &sms, msg.Buffer+10);
+#endif
+
+ if (s->Phone.Data.EnableIncomingSMS && s->User.IncomingSMS!=NULL) {
+ sms.State = SMS_UnRead;
+ sms.InboxFolder = true;
+
+ N6510_DecodeSMSFrame(s, &sms, msg.Buffer+10);
+
+ s->User.IncomingSMS(s->CurrentConfig->Device,sms);
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_DialVoice(GSM_StateMachine *s, char *number, GSM_CallShowNumber ShowNumber)
+{
+ unsigned int pos = 4;
+ unsigned char req[100] = {N6110_FRAME_HEADER,0x01,
+ 0x0c}; /* Number length */
+
+ req[pos++] = strlen(number);
+ EncodeUnicode(req+pos,number,strlen(number));
+ pos += strlen(number)*2;
+ req[pos++] = 0x05; /* call type: voice - 0x05, data - 0x01 */
+ req[pos++] = 0x01;
+ req[pos++] = 0x05;
+ req[pos++] = 0x00;
+ req[pos++] = 0x02;
+ req[pos++] = 0x00;
+ req[pos++] = 0x00;
+ switch (ShowNumber) {
+ case GSM_CALL_HideNumber:
+ req[pos++] = 0x02;
+ break;
+ case GSM_CALL_ShowNumber:
+ req[pos++] = 0x03;
+ break;
+ case GSM_CALL_DefaultNumberPresence:
+ req[pos++] = 0x01;
+ break;
+ }
+
+ smprintf(s, "Making voice call\n");
+ return GSM_WaitFor (s, req, pos, 0x01, 4, ID_DialVoice);
+}
+
+/* method 3 */
+static GSM_Error N6510_ReplyGetCalendarInfo3(GSM_Protocol_Message msg, GSM_StateMachine *s, GSM_NOKIACalToDoLocations *Last)
+{
+ int i=0,j=0;
+
+ while (Last->Location[j] != 0x00) j++;
+ if (j >= GSM_MAXCALENDARTODONOTES) {
+ smprintf(s, "Increase GSM_MAXCALENDARTODONOTES\n");
+ return ERR_UNKNOWN;
+ }
+ if (j == 0) {
+ Last->Number=msg.Buffer[8]*256+msg.Buffer[9];
+ smprintf(s, "Number of Entries: %i\n",Last->Number);
+ }
+ smprintf(s, "Locations: ");
+ while (14+(i*4) <= msg.Length) {
+ Last->Location[j++]=msg.Buffer[12+i*4]*256+msg.Buffer[13+i*4];
+ smprintf(s, "%i ",Last->Location[j-1]);
+ i++;
+ }
+ smprintf(s, "\nNumber of Entries in frame: %i\n",i);
+ Last->Location[j] = 0;
+ smprintf(s, "\n");
+ if (i == 1 && msg.Buffer[12+0*4]*256+msg.Buffer[13+0*4] == 0) return ERR_EMPTY;
+ if (i == 0) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+/* method 3 */
+static GSM_Error N6510_GetCalendarInfo3(GSM_StateMachine *s, GSM_NOKIACalToDoLocations *Last, bool Calendar)
+{
+ GSM_Error error;
+ int i;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x9E, 0xFF, 0xFF, 0x00, 0x00,
+ 0x00, 0x00, /* First location */
+ 0x00}; /* 0 = calendar, 1 = ToDo in 6610 style */
+
+ Last->Location[0] = 0x00;
+ Last->Number = 0;
+
+ if (Calendar) {
+ smprintf(s, "Getting locations for calendar method 3\n");
+ error = GSM_WaitFor (s, req, 11, 0x13, 4, ID_GetCalendarNotesInfo);
+ } else {
+ req[10] = 0x01;
+ smprintf(s, "Getting locations for ToDo method 2\n");
+ error = GSM_WaitFor (s, req, 11, 0x13, 4, ID_GetToDo);
+ }
+ if (error != ERR_NONE && error != ERR_EMPTY) return error;
+
+ while (1) {
+ i=0;
+ while (Last->Location[i] != 0x00) i++;
+ smprintf(s, "i = %i %i\n",i,Last->Number);
+ if (i == Last->Number) break;
+ if (i != Last->Number && error == ERR_EMPTY) {
+ smprintf(s, "Phone doesn't support some notes with this method. Workaround\n");
+ Last->Number = i;
+ break;
+ }
+ req[8] = Last->Location[i-1] / 256;
+ req[9] = Last->Location[i-1] % 256;
+ if (Calendar) {
+ smprintf(s, "Getting locations for calendar method 3\n");
+ error = GSM_WaitFor (s, req, 11, 0x13, 4, ID_GetCalendarNotesInfo);
+ } else {
+ smprintf(s, "Getting locations for todo method 2\n");
+ error = GSM_WaitFor (s, req, 11, 0x13, 4, ID_GetToDo);
+ }
+ if (error != ERR_NONE && error != ERR_EMPTY) return error;
+ }
+ return ERR_NONE;
+}
+
+/* method 3 */
+GSM_Error N6510_ReplyGetCalendar3(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_CalendarEntry *entry = s->Phone.Data.Cal;
+ GSM_DateTime Date;
+ unsigned long diff;
+ int i;
+ bool found = false;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+
+ smprintf(s, "Calendar note received method 3\n");
+
+ smprintf(s,"Note type %02i: ",msg.Buffer[27]);
+ switch(msg.Buffer[27]) {
+ case 0x00: smprintf(s,"Reminder\n"); entry->Type = GSM_CAL_REMINDER; break;
+ case 0x01: smprintf(s,"Meeting\n"); entry->Type = GSM_CAL_MEETING; break;
+ case 0x02: smprintf(s,"Call\n"); entry->Type = GSM_CAL_CALL; break;
+ case 0x04: smprintf(s,"Birthday\n"); entry->Type = GSM_CAL_BIRTHDAY; break;
+ case 0x08: smprintf(s,"Memo\n"); entry->Type = GSM_CAL_MEMO; break;
+ default : smprintf(s,"unknown\n");
+ }
+
+ smprintf(s,"StartTime: %04i-%02i-%02i %02i:%02i\n",
+ msg.Buffer[28]*256+msg.Buffer[29],
+ msg.Buffer[30],msg.Buffer[31],msg.Buffer[32],
+ msg.Buffer[33]);
+ Date.Year = msg.Buffer[28]*256+msg.Buffer[29];
+ if (entry->Type == GSM_CAL_BIRTHDAY) {
+ Date.Year = entry->Entries[0].Date.Year;
+ smprintf(s,"%i\n",Date.Year);
+ }
+ Date.Month = msg.Buffer[30];
+ Date.Day = msg.Buffer[31];
+ Date.Hour = msg.Buffer[32];
+ Date.Minute = msg.Buffer[33];
+ /* Garbage seen with 3510i 3.51 */
+ if (Date.Month == 0 && Date.Day == 0 && Date.Hour == 0 && Date.Minute == 0) return ERR_EMPTY;
+ Date.Second = 0;
+ entry->Entries[0].EntryType = CAL_START_DATETIME;
+ memcpy(&entry->Entries[0].Date,&Date,sizeof(GSM_DateTime));
+ entry->EntriesNum++;
+
+ if (entry->Type != GSM_CAL_BIRTHDAY) {
+ smprintf(s,"EndTime: %04i-%02i-%02i %02i:%02i\n",
+ msg.Buffer[34]*256+msg.Buffer[35],
+ msg.Buffer[36],msg.Buffer[37],msg.Buffer[38],
+ msg.Buffer[39]);
+ Date.Year = msg.Buffer[34]*256+msg.Buffer[35];
+ Date.Month = msg.Buffer[36];
+ Date.Day = msg.Buffer[37];
+ Date.Hour = msg.Buffer[38];
+ Date.Minute = msg.Buffer[39];
+ Date.Second = 0;
+ entry->Entries[1].EntryType = CAL_END_DATETIME;
+ memcpy(&entry->Entries[1].Date,&Date,sizeof(GSM_DateTime));
+ entry->EntriesNum++;
+ }
+
+ smprintf(s, "Note icon: %02x\n",msg.Buffer[21]);
+ for(i=0;i<Priv->CalendarIconsNum;i++) {
+ if (Priv->CalendarIconsTypes[i] == entry->Type) {
+ found = true;
+ }
+ }
+ if (!found) {
+ Priv->CalendarIconsTypes[Priv->CalendarIconsNum] = entry->Type;
+ Priv->CalendarIcons [Priv->CalendarIconsNum] = msg.Buffer[21];
+ Priv->CalendarIconsNum++;
+ }
+
+ if (msg.Buffer[14] == 0xFF && msg.Buffer[15] == 0xFF && msg.Buffer[16] == 0xff && msg.Buffer[17] == 0xff)
+ {
+ smprintf(s, "No alarm\n");
+ } else {
+ diff = ((unsigned int)msg.Buffer[14]) << 24;
+ diff += ((unsigned int)msg.Buffer[15]) << 16;
+ diff += ((unsigned int)msg.Buffer[16]) << 8;
+ diff += msg.Buffer[17];
+
+ memcpy(&entry->Entries[entry->EntriesNum].Date,&entry->Entries[0].Date,sizeof(GSM_DateTime));
+ GetTimeDifference(diff, &entry->Entries[entry->EntriesNum].Date, false, 60);
+ smprintf(s, "Alarm date : %02i-%02i-%04i %02i:%02i:%02i\n",
+ entry->Entries[entry->EntriesNum].Date.Day, entry->Entries[entry->EntriesNum].Date.Month,
+ entry->Entries[entry->EntriesNum].Date.Year, entry->Entries[entry->EntriesNum].Date.Hour,
+ entry->Entries[entry->EntriesNum].Date.Minute,entry->Entries[entry->EntriesNum].Date.Second);
+
+ entry->Entries[entry->EntriesNum].EntryType = CAL_ALARM_DATETIME;
+ if (msg.Buffer[22]==0x00 && msg.Buffer[23]==0x00 &&
+ msg.Buffer[24]==0x00 && msg.Buffer[25]==0x00)
+ {
+ entry->Entries[entry->EntriesNum].EntryType = CAL_SILENT_ALARM_DATETIME;
+ smprintf(s, "Alarm type : Silent\n");
+ }
+ entry->EntriesNum++;
+ }
+
+ N71_65_GetCalendarRecurrance(s, msg.Buffer+40, entry);
+
+ if (entry->Type == GSM_CAL_BIRTHDAY) {
+ if (msg.Buffer[42] == 0xff && msg.Buffer[43] == 0xff) {
+ entry->Entries[0].Date.Year = 0;
+ } else {
+ entry->Entries[0].Date.Year = msg.Buffer[42]*256+msg.Buffer[43];
+ }
+ }
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+54, msg.Buffer[51]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[51]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[51]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ entry->EntriesNum++;
+ smprintf(s, "Note text: \"%s\"\n",DecodeUnicodeString(entry->Entries[entry->EntriesNum-1].Text));
+
+ if (entry->Type == GSM_CAL_CALL) {
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+(54+msg.Buffer[51]*2), msg.Buffer[52]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[52]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[52]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_PHONE;
+ entry->EntriesNum++;
+ }
+ if (entry->Type == GSM_CAL_MEETING) {
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+(54+msg.Buffer[51]*2), msg.Buffer[52]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[52]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[52]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_LOCATION;
+ entry->EntriesNum++;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_PrivGetCalendar3(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, int *LastCalendarYear)
+{
+ GSM_Error error;
+ GSM_DateTime date_time;
+ unsigned char req[] = {N6110_FRAME_HEADER,0x7D,0x00,0x00,0x00,0x00,
+ 0x00,0x99, /* Location */
+ 0xff,0xff,0xff,0xff,0x01};
+
+ if (start) {
+ /* We have to get current year. It's NOT written in frame for
+ * Birthday
+ */
+ error=s->Phone.Functions->GetDateTime(s,&date_time);
+ switch (error) {
+ case ERR_EMPTY:
+ case ERR_NOTIMPLEMENTED:
+ GSM_GetCurrentDateTime(&date_time);
+ break;
+ case ERR_NONE:
+ break;
+ default:
+ return error;
+ }
+ *LastCalendarYear = date_time.Year;
+ }
+
+ Note->EntriesNum = 0;
+ Note->Entries[0].Date.Year = *LastCalendarYear;
+
+ req[8] = Note->Location / 256;
+ req[9] = Note->Location % 256;
+
+ s->Phone.Data.Cal=Note;
+ smprintf(s, "Getting calendar note method 3\n");
+ return GSM_WaitFor (s, req, 15, 0x13, 4, ID_GetCalendarNote);
+}
+
+/* method 3 */
+GSM_Error N6510_GetNextCalendar3(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, GSM_NOKIACalToDoLocations *LastCalendar, int *LastCalendarYear, int *LastCalendarPos)
+{
+ GSM_Error error;
+ bool start2;
+
+ if (start) {
+ error=N6510_GetCalendarInfo3(s,LastCalendar,true);
+ if (error!=ERR_NONE) return error;
+ if (LastCalendar->Number == 0) return ERR_EMPTY;
+
+ *LastCalendarPos = 0;
+ } else {
+ (*LastCalendarPos)++;
+ }
+
+ error = ERR_EMPTY;
+ start2 = start;
+ while (error == ERR_EMPTY) {
+ if (*LastCalendarPos >= LastCalendar->Number) return ERR_EMPTY;
+
+ Note->Location = LastCalendar->Location[*LastCalendarPos];
+ error=N6510_PrivGetCalendar3(s, Note, start2, LastCalendarYear);
+ if (error == ERR_EMPTY) (*LastCalendarPos)++;
+
+ start2 = false;
+ }
+ return error;
+}
+
+static GSM_Error N6510_ReplyGetCalendarInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x3B:
+ /* Old method 1 for accessing calendar */
+ return N71_65_ReplyGetCalendarInfo1(msg, s, &s->Phone.Data.Priv.N6510.LastCalendar);
+ case 0x9F:
+ smprintf(s, "Info with calendar notes locations received method 3\n");
+ return N6510_ReplyGetCalendarInfo3(msg, s, &s->Phone.Data.Priv.N6510.LastCalendar);
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+/* method 3 */
+GSM_Error N6510_ReplyGetCalendarNotePos3(GSM_Protocol_Message msg, GSM_StateMachine *s,int *FirstCalendarPos)
+{
+ smprintf(s, "First calendar location: %i\n",msg.Buffer[8]*256+msg.Buffer[9]);
+ *FirstCalendarPos = msg.Buffer[8]*256+msg.Buffer[9];
+ return ERR_NONE;
+}
+
+/* method 3 */
+static GSM_Error N6510_GetCalendarNotePos3(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x95, 0x00};
+
+ smprintf(s, "Getting first free calendar note location\n");
+ return GSM_WaitFor (s, req, 5, 0x13, 4, ID_GetCalendarNotePos);
+}
+
+static GSM_Error N6510_ReplyGetCalendarNotePos(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x32:
+ /* Old method 1 for accessing calendar */
+ return N71_65_ReplyGetCalendarNotePos1(msg, s,&s->Phone.Data.Priv.N6510.FirstCalendarPos);
+ case 0x96:
+ return N6510_ReplyGetCalendarNotePos3(msg, s,&s->Phone.Data.Priv.N6510.FirstCalendarPos);
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_FindCalendarIconID3(GSM_StateMachine *s, GSM_CalendarEntry *Entry, unsigned char *ID)
+{
+ int i,j,LastCalendarYear;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ GSM_CalendarEntry Note;
+ GSM_NOKIACalToDoLocations LastCalendar1,LastCalendar2;
+ GSM_Error error;
+ bool found;
+
+ for(i=0;i<Priv->CalendarIconsNum;i++) {
+ if (Priv->CalendarIconsTypes[i] == Entry->Type) {
+ *ID = Priv->CalendarIcons[i];
+ return ERR_NONE;
+ }
+ }
+
+ smprintf(s, "Starting finding note ID\n");
+
+ error=N6510_GetCalendarInfo3(s, &Priv->LastCalendar,true);
+ memcpy(&LastCalendar1,&Priv->LastCalendar,sizeof(GSM_NOKIACalToDoLocations));
+ if (error != ERR_NONE) return error;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL35) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL65) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62)) {
+ error=N71_65_AddCalendar2(s,Entry);
+ } else {
+ if (Entry->Type == GSM_CAL_MEETING) {
+ error=N71_65_AddCalendar1(s, Entry, &s->Phone.Data.Priv.N6510.FirstCalendarPos);
+ } else {
+ error=N71_65_AddCalendar2(s,Entry);
+ }
+ }
+ if (error != ERR_NONE) return error;
+
+ error=N6510_GetCalendarInfo3(s, &Priv->LastCalendar,true);
+ memcpy(&LastCalendar2,&Priv->LastCalendar,sizeof(GSM_NOKIACalToDoLocations));
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Number of entries: %i %i\n",LastCalendar1.Number,LastCalendar2.Number);
+
+ for(i=0;i<LastCalendar2.Number;i++) {
+ found = true;
+ for(j=0;j<LastCalendar1.Number;j++) {
+ if (LastCalendar1.Location[j] == LastCalendar2.Location[i]) {
+ found = false;
+ break;
+ }
+ }
+ if (found) {
+ Note.Location = LastCalendar2.Location[i];
+ error=N6510_PrivGetCalendar3(s, &Note, true, &LastCalendarYear);
+ if (error != ERR_NONE) return error;
+
+ error=N71_65_DelCalendar(s, &Note);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Ending finding note ID\n");
+
+ for(j=0;j<Priv->CalendarIconsNum;j++) {
+ if (Priv->CalendarIconsTypes[j] == Entry->Type) {
+ *ID = Priv->CalendarIcons[j];
+ return ERR_NONE;
+ }
+ }
+ return ERR_UNKNOWN;
+ }
+ }
+
+ return ERR_UNKNOWN;
+}
+
+/* method 3 */
+static GSM_Error N6510_ReplyAddCalendar3(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Calendar note added\n");
+ return ERR_NONE;
+}
+
+/* method 3 */
+GSM_Error N6510_AddCalendar3(GSM_StateMachine *s, GSM_CalendarEntry *Note, int *FirstCalendarPos)
+{
+ GSM_CalendarNoteType NoteType, OldNoteType;
+ time_t t_time1,t_time2;
+ long diff;
+ GSM_Error error;
+ GSM_DateTime DT,date_time;
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location, count=54;
+ unsigned char req[5000] = {
+ N6110_FRAME_HEADER, 0x65,
+ 0x00, /* 0 = calendar, 1 = todo */
+ 0x00, 0x00, 0x00,
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00, 0x00,
+ 0xFF, 0xFF, 0xFF, 0xFF, /* alarm */
+ 0x80, 0x00, 0x00,
+ 0x01, /* note icon */
+ 0xFF, 0xFF, 0xFF, 0xFF, /* alarm type */
+ 0x00, /* 0x02 or 0x00 */
+ 0x01, /* note type */
+ 0x07, 0xD0, 0x01, 0x12, 0x0C, 0x00, /* start date/time */
+ 0x07, 0xD0, 0x01, 0x12, 0x0C, 0x00, /* end date/time */
+ 0x00, 0x00, /* recurrance */
+ 0x00, 0x00, /* birth year */
+ 0x20, /* ToDo priority */
+ 0x00, /* ToDo completed ? */
+ 0x00, 0x00, 0x00,
+ 0x00, /* note text length */
+ 0x00, /* phone length/meeting place */
+ 0x00, 0x00, 0x00};
+
+ error=N6510_GetCalendarNotePos3(s);
+ if (error!=ERR_NONE) return error;
+ req[8] = *FirstCalendarPos/256;
+ req[9] = *FirstCalendarPos%256;
+
+ NoteType = N71_65_FindCalendarType(Note->Type, s->Phone.Data.ModelInfo);
+
+ switch(NoteType) {
+ case GSM_CAL_REMINDER : req[27]=0x00; req[26]=0x02; break;
+ case GSM_CAL_MEETING : req[27]=0x01; break;
+ case GSM_CAL_CALL : req[27]=0x02; break;
+ case GSM_CAL_BIRTHDAY : req[27]=0x04; break;
+ case GSM_CAL_MEMO : req[27]=0x08; break;
+ default : return ERR_UNKNOWN;
+ }
+
+ OldNoteType = Note->Type;
+ Note->Type = NoteType;
+ error=N6510_FindCalendarIconID3(s, Note, &req[21]);
+ Note->Type = OldNoteType;
+ if (error!=ERR_NONE) return error;
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(Note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+
+ if (Time == -1) return ERR_UNKNOWN;
+ memcpy(&DT,&Note->Entries[Time].Date,sizeof(GSM_DateTime));
+ req[28] = DT.Year / 256;
+ req[29] = DT.Year % 256;
+ req[30] = DT.Month;
+ req[31] = DT.Day;
+ req[32] = DT.Hour;
+ req[33] = DT.Minute;
+
+ if (NoteType == GSM_CAL_BIRTHDAY) {
+ error=s->Phone.Functions->GetDateTime(s,&date_time);
+ switch (error) {
+ case ERR_EMPTY:
+ case ERR_NOTIMPLEMENTED:
+ GSM_GetCurrentDateTime(&date_time);
+ break;
+ case ERR_NONE:
+ break;
+ default:
+ return error;
+ }
+ req[28] = date_time.Year / 256;
+ req[29] = date_time.Year % 256;
+ if (DT.Year == 0) {
+ req[42] = 0xff;
+ req[43] = 0xff;
+ } else {
+ req[42] = DT.Year / 256;
+ req[43] = DT.Year % 256;
+ }
+ }
+
+ if (EndTime != -1) memcpy(&DT,&Note->Entries[EndTime].Date,sizeof(GSM_DateTime));
+ req[34] = DT.Year / 256;
+ req[35] = DT.Year % 256;
+ req[36] = DT.Month;
+ req[37] = DT.Day;
+ req[38] = DT.Hour;
+ req[39] = DT.Minute;
+ if (NoteType == GSM_CAL_BIRTHDAY) {
+ req[34] = date_time.Year / 256;
+ req[35] = date_time.Year % 256;
+ }
+
+ if (Recurrance != -1) {
+ /* max. 1 Year = 8760 hours */
+ if (Note->Entries[Recurrance].Number >= 8760) {
+ req[40] = 0xff;
+ req[41] = 0xff;
+ } else {
+ req[40] = Note->Entries[Recurrance].Number / 256;
+ req[41] = Note->Entries[Recurrance].Number % 256;
+ }
+ }
+
+ if (Alarm != -1) {
+ memcpy(&DT,&Note->Entries[Time].Date,sizeof(GSM_DateTime));
+ if (Note->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME) {
+ req[22] = 0x00; req[23] = 0x00; req[24] = 0x00; req[25] = 0x00;
+ }
+ if (NoteType == GSM_CAL_BIRTHDAY) DT.Year = date_time.Year;
+ t_time2 = Fill_Time_T(DT,8);
+ t_time1 = Fill_Time_T(Note->Entries[Alarm].Date,8);
+ diff = (t_time1-t_time2)/60;
+
+ smprintf(s, " Difference : %li seconds or minutes\n", -diff);
+ req[14] = (unsigned char)(-diff >> 24);
+ req[15] = (unsigned char)(-diff >> 16);
+ req[16] = (unsigned char)(-diff >> 8);
+ req[17] = (unsigned char)(-diff);
+ }
+
+ if (Text != -1) {
+ req[49] = UnicodeLength(Note->Entries[Text].Text);
+ CopyUnicodeString(req+54,Note->Entries[Text].Text);
+ count+= req[49]*2;
+ }
+
+ if (Phone != -1 && NoteType == GSM_CAL_CALL) {
+ req[50] = UnicodeLength(Note->Entries[Phone].Text);
+ CopyUnicodeString(req+54+req[49]*2,Note->Entries[Phone].Text);
+ count+= req[50]*2;
+ }
+
+ if (Location != -1 && NoteType == GSM_CAL_MEETING) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL65) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL35)) {
+ } else {
+ req[50] = UnicodeLength(Note->Entries[Location].Text);
+ CopyUnicodeString(req+54+req[49]*2,Note->Entries[Location].Text);
+ count+= req[50]*2;
+ }
+ }
+
+ req[count++] = 0x00;
+
+ smprintf(s, "Writing calendar note method 3\n");
+ return GSM_WaitFor (s, req, count, 0x13, 4, ID_SetCalendarNote);
+}
+
+static GSM_Error N6510_GetNextCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start)
+{
+#ifdef GSM_FORCE_DCT4_CALENDAR_6210
+ /* Method 1. Some features missed. Not working with some notes in 3510 */
+ return N71_65_GetNextCalendar1(s,Note,start,&s->Phone.Data.Priv.N6510.LastCalendar,&s->Phone.Data.Priv.N6510.LastCalendarYear,&s->Phone.Data.Priv.N6510.LastCalendarPos);
+#endif
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62)) {
+ /* Method 1. Some features missed. Not working with some notes in 3510 */
+ return N71_65_GetNextCalendar1(s,Note,start,&s->Phone.Data.Priv.N6510.LastCalendar,&s->Phone.Data.Priv.N6510.LastCalendarYear,&s->Phone.Data.Priv.N6510.LastCalendarPos);
+
+ /* Method 2. In known phones texts of notes cut to 50 chars. Some features missed */
+// return N71_65_GetNextCalendar2(s,Note,start,&s->Phone.Data.Priv.N6510.LastCalendarYear,&s->Phone.Data.Priv.N6510.LastCalendarPos);
+ } else {
+ /* Method 3. All DCT4 features supported. Not supported by 8910 */
+ return N6510_GetNextCalendar3(s,Note,start,&s->Phone.Data.Priv.N6510.LastCalendar,&s->Phone.Data.Priv.N6510.LastCalendarYear,&s->Phone.Data.Priv.N6510.LastCalendarPos);
+ }
+}
+
+static GSM_Error N6510_GetCalendarStatus(GSM_StateMachine *s, GSM_CalendarStatus *Status)
+{
+ GSM_Error error;
+
+#ifdef GSM_FORCE_DCT4_CALENDAR_6210
+ /* Method 1 */
+ error=N71_65_GetCalendarInfo1(s, &s->Phone.Data.Priv.N6510.LastCalendar);
+ if (error!=ERR_NONE) return error;
+ Status->Used = s->Phone.Data.Priv.N6510.LastCalendar.Number;
+ return ERR_NONE;
+#endif
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62)) {
+ /* Method 1 */
+ error=N71_65_GetCalendarInfo1(s, &s->Phone.Data.Priv.N6510.LastCalendar);
+ if (error!=ERR_NONE) return error;
+ Status->Used = s->Phone.Data.Priv.N6510.LastCalendar.Number;
+ return ERR_NONE;
+
+ /* Method 2 */
+// return ERR_NOTSUPPORTED;
+ } else {
+ /* Method 3 */
+ error=N6510_GetCalendarInfo3(s,&s->Phone.Data.Priv.N6510.LastCalendar,true);
+ if (error!=ERR_NONE) return error;
+ Status->Used = s->Phone.Data.Priv.N6510.LastCalendar.Number;
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error N6510_AddCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+#ifdef GSM_FORCE_DCT4_CALENDAR_6210
+ return N71_65_AddCalendar2(s,Note);
+#endif
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62)) {
+ return N71_65_AddCalendar2(s,Note);
+// return N71_65_AddCalendar1(s, Note, &s->Phone.Data.Priv.N6510.FirstCalendarPos);
+ } else {
+ /* Method 3. All DCT4 features supported. Not supported by 8910 */
+ return N6510_AddCalendar3(s, Note, &s->Phone.Data.Priv.N6510.FirstCalendarPos);
+ }
+}
+
+static GSM_Error N6510_ReplyLogIntoNetwork(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Probably phone says: I log into network\n");
+ return ERR_NONE;
+}
+
+void N6510_EncodeFMFrequency(double freq, unsigned char *buff)
+{
+ double freq0;
+ unsigned char buffer[20];
+ unsigned int i,freq2;
+
+ sprintf(buffer,"%.3f",freq);
+ for (i=0;i<strlen(buffer);i++) {
+ if (buffer[i] == ',' || buffer[i] == '.') buffer[i] = ' ';
+ }
+ StringToDouble(buffer, &freq0);
+ freq2 = (unsigned int)freq0;
+ dbgprintf("Frequency: %s %i\n",buffer,freq2);
+ freq2 = freq2 - 0xffff;
+ buff[0] = freq2 / 0x100;
+ buff[1] = freq2 % 0x100;
+}
+
+void N6510_DecodeFMFrequency(double *freq, unsigned char *buff)
+{
+ unsigned char buffer[20];
+
+ sprintf(buffer,"%i.%i",(0xffff + buff[0] * 0x100 + buff[1])/1000,
+ (0xffff + buff[0] * 0x100 + buff[1])%1000);
+ dbgprintf("Frequency: %s\n",buffer);
+ StringToDouble(buffer, freq);
+}
+
+static GSM_Error N6510_ReplyGetFMStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "getting FM status OK\n");
+ memcpy(s->Phone.Data.Priv.N6510.FMStatus,msg.Buffer,msg.Length);
+ s->Phone.Data.Priv.N6510.FMStatusLength = msg.Length;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetFMStatus(GSM_StateMachine *s)
+{
+ unsigned char req[7] = {N6110_FRAME_HEADER, 0x0d, 0x00, 0x00, 0x01};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_RADIO)) return ERR_NOTSUPPORTED;
+ return GSM_WaitFor (s, req, 7, 0x3E, 2, ID_GetFMStation);
+}
+
+static GSM_Error N6510_ReplyGetFMStation(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char name[GSM_MAX_FMSTATION_LENGTH*2+2];
+ int length;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ switch (msg.Buffer[3]) {
+ case 0x06:
+ smprintf(s, "Received FM station\n");
+ length = msg.Buffer[8];
+ memcpy(name,msg.Buffer+18,length*2);
+ name[length*2] = 0x00;
+ name[length*2+1] = 0x00;
+ CopyUnicodeString(Data->FMStation->StationName,name);
+ smprintf(s,"Station name: \"%s\"\n",DecodeUnicodeString(Data->FMStation->StationName));
+ N6510_DecodeFMFrequency(&Data->FMStation->Frequency, msg.Buffer+16);
+ return ERR_NONE;
+ case 0x16:
+ smprintf(s, "Received FM station. Empty ?\n");
+ return ERR_EMPTY;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetFMStation (GSM_StateMachine *s, GSM_FMStation *FMStation)
+{
+ GSM_Error error;
+ int location;
+ unsigned char req[7] = {N6110_FRAME_HEADER, 0x05,
+ 0x00, // location
+ 0x00,0x01};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_RADIO)) return ERR_NOTSUPPORTED;
+ if (FMStation->Location > GSM_MAX_FM_STATION) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.FMStation = FMStation;
+
+ error = N6510_GetFMStatus(s);
+ if (error != ERR_NONE) return error;
+
+ location = FMStation->Location-1;
+ if (s->Phone.Data.Priv.N6510.FMStatus[14+location] == 0xFF) return ERR_EMPTY;
+ req[4] = s->Phone.Data.Priv.N6510.FMStatus[14+location];
+
+ smprintf(s, "Getting FM Station %i\n",FMStation->Location);
+ return GSM_WaitFor (s, req, 7, 0x3E, 2, ID_GetFMStation);
+}
+
+static GSM_Error N6510_ReplySetFMStation(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+#ifdef DEBUG
+ switch (msg.Buffer[4]){
+ case 0x03: smprintf(s, "FM stations cleaned\n"); break;
+ case 0x11: smprintf(s, "Setting FM station status OK\n"); break;
+ case 0x12: smprintf(s, "Setting FM station OK\n"); break;
+ }
+#endif
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ClearFMStations (GSM_StateMachine *s)
+{
+ unsigned char req[7] = {N6110_FRAME_HEADER, 0x03,0x0f,0x00,0x01};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_RADIO)) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Cleaning FM Stations\n");
+ return GSM_WaitFor (s, req, 7, 0x3E, 2, ID_SetFMStation);
+}
+
+static GSM_Error N6510_SetFMStation (GSM_StateMachine *s, GSM_FMStation *FMStation)
+{
+ unsigned int len, location;
+ GSM_Error error;
+ unsigned char setstatus[36] = {N6110_FRAME_HEADER,0x11,0x00,0x01,0x01,
+ 0x00,0x00,0x1c,0x00,0x14,0x00,0x00,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0x01};
+ unsigned char req[64] = {N6110_FRAME_HEADER, 0x12,0x00,0x01,0x00,
+ 0x00, // 0x0e + (strlen(name) * 2)
+ 0x00, // strlen(name)
+ 0x14,0x09,0x00,
+ 0x00, // location
+ 0x00,0x00,0x01,
+ 0x00, // freqHi
+ 0x00, // freqLo
+ 0x01};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_RADIO)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.FMStation = FMStation;
+ location = FMStation->Location-1;
+
+ error = N6510_GetFMStatus(s);
+ if (error != ERR_NONE) return error;
+
+ memcpy(setstatus+14,s->Phone.Data.Priv.N6510.FMStatus+14,20);
+ setstatus [14+location] = location;
+
+ smprintf(s, "Setting FM status %i\n",FMStation->Location);
+ error = GSM_WaitFor (s, setstatus, 36 , 0x3E, 2, ID_SetFMStation);
+ if (error != ERR_NONE) return error;
+
+ req[12] = location;
+
+ /* Name */
+ len = UnicodeLength(FMStation->StationName);
+ req[8] = len;
+ req[7] = 0x0e + len * 2;
+ memcpy (req+18,FMStation->StationName,len*2);
+
+ /* Frequency */
+ N6510_EncodeFMFrequency(FMStation->Frequency, req+16);
+
+ smprintf(s, "Setting FM Station %i\n",FMStation->Location);
+ return GSM_WaitFor (s, req, 0x13+len*2, 0x3E, 2, ID_SetFMStation);
+}
+
+static GSM_Error N6510_ReplySetLight(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Light set\n");
+ return ERR_NONE;
+}
+
+GSM_Error N6510_SetLight(GSM_StateMachine *s, N6510_PHONE_LIGHTS light, bool enable)
+{
+ unsigned char req[14] = {
+ N6110_FRAME_HEADER, 0x05,
+ 0x01, /* 0x01 = Display, 0x03 = keypad */
+ 0x01, /* 0x01 = Enable, 0x02 = disable */
+ 0x00, 0x00, 0x00, 0x01,
+ 0x05, 0x04, 0x02, 0x00};
+
+ req[4] = light;
+ if (!enable) req[5] = 0x02;
+ smprintf(s, "Setting light\n");
+ return GSM_WaitFor (s, req, 14, 0x3A, 4, ID_SetLight);
+}
+
+static GSM_Error N6510_ShowStartInfo(GSM_StateMachine *s, bool enable)
+{
+ GSM_Error error;
+
+ if (enable) {
+ error=N6510_SetLight(s,N6510_LIGHT_DISPLAY,true);
+ if (error != ERR_NONE) return error;
+
+ error=N6510_SetLight(s,N6510_LIGHT_TORCH,true);
+ if (error != ERR_NONE) return error;
+
+ return N6510_SetLight(s,N6510_LIGHT_KEYPAD,true);
+ } else {
+ error=N6510_SetLight(s,N6510_LIGHT_DISPLAY,false);
+ if (error != ERR_NONE) return error;
+
+ error=N6510_SetLight(s,N6510_LIGHT_TORCH,false);
+ if (error != ERR_NONE) return error;
+
+ return N6510_SetLight(s,N6510_LIGHT_KEYPAD,false);
+ }
+}
+
+static int N6510_FindFileCheckSum(unsigned char *ptr, int len)
+{
+ int acc, i, accx;
+
+ accx = 0;
+ acc = 0xffff;
+ while (len--) {
+ accx = (accx & 0xffff00ff) | (acc & 0xff00);
+ acc = (acc & 0xffff00ff) | *ptr++ << 8;
+ for (i = 0; i < 8; i++) {
+ acc <<= 1;
+ if (acc & 0x10000) acc ^= 0x1021;
+ if (accx & 0x80000000) acc ^= 0x1021;
+ accx <<= 1;
+ }
+ }
+ dbgprintf("Checksum from Gammu is %04X\n",(acc & 0xffff));
+ return (acc & 0xffff);
+}
+
+static GSM_Error N6510_ReplyGetFileFolderInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_File *File = s->Phone.Data.FileInfo;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ int i;
+
+ switch (msg.Buffer[3]) {
+ case 0x15:
+ smprintf(s,"File or folder details received\n");
+ CopyUnicodeString(File->Name,msg.Buffer+10);
+ if (!strncmp(DecodeUnicodeString(File->Name),"GMSTemp",7)) return ERR_EMPTY;
+ if (File->Name[0] == 0x00 && File->Name[1] == 0x00) return ERR_UNKNOWN;
+
+ i = msg.Buffer[8]*256+msg.Buffer[9];
+ dbgprintf("%02x %02x %02x %02x %02x %02x %02x %02x %02x\n",
+ msg.Buffer[i-5],msg.Buffer[i-4],msg.Buffer[i-3],
+ msg.Buffer[i-2],msg.Buffer[i-1],msg.Buffer[i],
+ msg.Buffer[i+1],msg.Buffer[i+2],msg.Buffer[i+3]);
+
+ File->Folder = false;
+ if (msg.Buffer[i-5] == 0x00) File->Folder = true;
+
+ File->ReadOnly = false;
+ File->Protected = false;
+ File->System = false;
+ File->Hidden = false;
+ if (msg.Buffer[i+2] == 0x01) File->Protected = true;
+ if (msg.Buffer[i+4] == 0x01) File->ReadOnly = true;
+ if (msg.Buffer[i+5] == 0x01) File->Hidden = true;
+ if (msg.Buffer[i+6] == 0x01) File->System = true;//fixme
+
+ File->ModifiedEmpty = false;
+ NOKIA_DecodeDateTime(s, msg.Buffer+i-22, &File->Modified);
+ if (File->Modified.Year == 0x00) File->ModifiedEmpty = true;
+ dbgprintf("%02x %02x %02x %02x\n",msg.Buffer[i-22],msg.Buffer[i-21],msg.Buffer[i-20],msg.Buffer[i-19]);
+
+ Priv->FileToken = msg.Buffer[i-10]*256+msg.Buffer[i-9];
+ Priv->ParentID = msg.Buffer[i]*256+msg.Buffer[i+1];
+ smprintf(s,"ParentID is %i\n",Priv->ParentID);
+
+ File->Type = GSM_File_Other;
+ if (msg.Length > 240){
+ i = 227;
+ if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x01)
+ File->Type = GSM_File_Image_JPG;
+ else if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x02)
+ File->Type = GSM_File_Image_BMP;
+ else if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x07)
+ File->Type = GSM_File_Image_BMP;
+ else if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x03)
+ File->Type = GSM_File_Image_PNG;
+ else if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x05)
+ File->Type = GSM_File_Image_GIF;
+ else if (msg.Buffer[i]==0x02 && msg.Buffer[i+2]==0x09)
+ File->Type = GSM_File_Image_WBMP;
+ else if (msg.Buffer[i]==0x04 && msg.Buffer[i+2]==0x01)
+ File->Type = GSM_File_Sound_AMR;
+ else if (msg.Buffer[i]==0x04 && msg.Buffer[i+2]==0x02)
+ File->Type = GSM_File_Sound_MIDI;
+ else if (msg.Buffer[i]==0x08 && msg.Buffer[i+2]==0x05)
+ File->Type = GSM_File_Video_3GP;
+ else if (msg.Buffer[i]==0x10 && msg.Buffer[i+2]==0x01)
+ File->Type = GSM_File_Java_JAR;
+#if DEVELOP
+ else if (msg.Buffer[i]==0x00 && msg.Buffer[i+2]==0x01)
+ File->Type = GSM_File_MMS;
+#endif
+ }
+ return ERR_NONE;
+ case 0x2F:
+ smprintf(s,"File or folder used bytes received\n");
+ File->Used = msg.Buffer[6]*256*256*256+
+ msg.Buffer[7]*256*256+
+ msg.Buffer[8]*256+
+ msg.Buffer[9];
+ return ERR_NONE;
+ case 0x33:
+ if (s->Phone.Data.RequestID == ID_GetFileInfo) {
+ i = Priv->FilesLocationsUsed-1;
+ while (1) {
+ if (i==Priv->FilesLocationsCurrent-1) break;
+ dbgprintf("Copying %i to %i, max %i, current %i\n",
+ i,i+msg.Buffer[9],
+ Priv->FilesLocationsUsed,Priv->FilesLocationsCurrent);
+ Priv->FilesLocations[i+msg.Buffer[9]] = Priv->FilesLocations[i];
+ Priv->FilesLevels[i+msg.Buffer[9]] = Priv->FilesLevels[i];
+ i--;
+ }
+ Priv->FilesLocationsUsed += msg.Buffer[9];
+ for (i=0;i<msg.Buffer[9];i++) {
+ Priv->FilesLocations[Priv->FilesLocationsCurrent+i] = msg.Buffer[13+i*4-1]*256 + msg.Buffer[13+i*4];
+ Priv->FilesLevels[Priv->FilesLocationsCurrent+i] = File->Level+1;
+ dbgprintf("%i ",Priv->FilesLocations[Priv->FilesLocationsCurrent+i]);
+ }
+ dbgprintf("\n");
+ }
+ if (msg.Buffer[9] != 0x00) File->Folder = true;
+ return ERR_NONE;
+ case 0x43:
+ Priv->FileCheckSum = msg.Buffer[6] * 256 + msg.Buffer[7];
+ smprintf(s,"File checksum from phone is %04X\n",Priv->FileCheckSum);
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetFileFolderInfo(GSM_StateMachine *s, GSM_File *File, GSM_Phone_RequestID Request)
+{
+ GSM_Error error;
+ unsigned char req[10] = {
+ N7110_FRAME_HEADER,
+ 0x14, /* 0x14 - info, 0x22 - free/total, 0x2E - used, 0x32 - sublocations */
+ 0x01, /* 0x00 for sublocations reverse sorting, 0x01 for free */
+ 0x00, 0x00, 0x01,
+ 0x00, 0x01}; /* Folder or file number */
+ unsigned char GetCRC[] = {
+ N7110_FRAME_HEADER, 0x42, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x1E}; /* file ID */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.FileInfo = File;
+ req[8] = atoi(File->ID_FullName) / 256;
+ req[9] = atoi(File->ID_FullName) % 256;
+
+ req[3] = 0x14;
+ req[4] = 0x01;
+ smprintf(s,"Getting info for file in filesystem\n");
+ error=GSM_WaitFor (s, req, 10, 0x6D, 4, Request);
+ if (error != ERR_NONE) return error;
+
+ if (Request != ID_AddFile) {
+ req[3] = 0x32;
+ req[4] = 0x00;
+ smprintf(s,"Getting subfolders for filesystem\n");
+ error=GSM_WaitFor (s, req, 10, 0x6D, 4, Request);
+ if (error != ERR_NONE) return error;
+
+ if (!File->Folder) {
+ req[3] = 0x2E;
+ req[4] = 0x01;
+ smprintf(s,"Getting used memory for file in filesystem\n");
+ error=GSM_WaitFor (s, req, 10, 0x6D, 4, Request);
+ if (error != ERR_NONE) return error;
+
+ GetCRC[8] = atoi(File->ID_FullName) / 256;
+ GetCRC[9] = atoi(File->ID_FullName) % 256;
+ smprintf(s,"Getting CRC for file in filesystem\n");
+ error=GSM_WaitFor (s, GetCRC, 10, 0x6D, 4, Request);
+ }
+ }
+ return error;
+}
+
+static GSM_Error N6510_GetNextFileFolder(GSM_StateMachine *s, GSM_File *File, bool start)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ GSM_Error error;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ if (start) {
+ Priv->FilesLocationsUsed = 1;
+ Priv->FilesLocationsCurrent = 0;
+ Priv->FilesLocations[0] = 0x01;
+ Priv->FilesLevels[0] = 1;
+ }
+
+ while (1) {
+ if (Priv->FilesLocationsCurrent == Priv->FilesLocationsUsed) return ERR_EMPTY;
+
+ sprintf(File->ID_FullName,"%i",Priv->FilesLocations[Priv->FilesLocationsCurrent]);
+ File->Level = Priv->FilesLevels[Priv->FilesLocationsCurrent];
+ Priv->FilesLocationsCurrent++;
+
+ error = N6510_GetFileFolderInfo(s, File, ID_GetFileInfo);
+ if (error == ERR_EMPTY) continue;
+ return error;
+ }
+}
+
+static GSM_Error N6510_ReplyGetFileSystemStatus(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x23:
+ if (!strcmp(s->Phone.Data.ModelInfo->model,"6310i")) {
+ smprintf(s,"File or folder total bytes received\n");
+ s->Phone.Data.FileSystemStatus->Free =
+ 3*256*256 + msg.Buffer[8]*256 + msg.Buffer[9] -
+ s->Phone.Data.FileSystemStatus->Used;
+ } else {
+ smprintf(s,"File or folder free bytes received\n");
+ s->Phone.Data.FileSystemStatus->Free =
+ msg.Buffer[6]*256*256*256+
+ msg.Buffer[7]*256*256+
+ msg.Buffer[8]*256+
+ msg.Buffer[9];
+ }
+ return ERR_NONE;
+ case 0x2F:
+ smprintf(s,"File or folder used bytes received\n");
+ s->Phone.Data.FileSystemStatus->Used =
+ msg.Buffer[6]*256*256*256+
+ msg.Buffer[7]*256*256+
+ msg.Buffer[8]*256+
+ msg.Buffer[9];
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetFileSystemStatus(GSM_StateMachine *s, GSM_FileSystemStatus *status)
+{
+ GSM_Error error;
+ unsigned char req[10] = {
+ N7110_FRAME_HEADER,
+ 0x22, /* 0x14 - info, 0x22 - free/total, 0x2E - used, 0x32 - sublocations */
+ 0x01, /* 0x00 for sublocations reverse sorting, 0x01 for free */
+ 0x00, 0x00, 0x01,
+ 0x00, 0x01}; /* Folder or file number */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.FileSystemStatus = status;
+
+ status->Free = 0;
+
+ req[3] = 0x2E;
+ req[4] = 0x01;
+ smprintf(s, "Getting used/total memory in filesystem\n");
+ error = GSM_WaitFor (s, req, 10, 0x6D, 4, ID_FileSystemStatus);
+
+ req[3] = 0x22;
+ req[4] = 0x01;
+ smprintf(s, "Getting free memory in filesystem\n");
+ return GSM_WaitFor (s, req, 10, 0x6D, 4, ID_FileSystemStatus);
+}
+
+static GSM_Error N6510_SearchForFileName(GSM_StateMachine *s, GSM_File *File)
+{
+ GSM_File File2;
+ GSM_Error error;
+ int FilesLocations[1000],FilesLocations2[1000];
+ int FilesLevels[1000];
+ int FilesLocationsUsed, FilesLocationsCurrent;
+ int FilesLocationsUsed2, FilesLocationsCurrent2;
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+
+ memcpy(FilesLocations, Priv->FilesLocations, sizeof(FilesLocations));
+ memcpy(FilesLevels, Priv->FilesLevels, sizeof(FilesLevels));
+ FilesLocationsUsed = Priv->FilesLocationsUsed;
+ FilesLocationsCurrent = Priv->FilesLocationsCurrent;
+
+ Priv->FilesLocationsUsed = 1;
+ Priv->FilesLocationsCurrent = 1;
+ Priv->FilesLocations[0] = atoi(File->ID_FullName);
+ Priv->FilesLevels[0] = 1;
+
+ strcpy(File2.ID_FullName,File->ID_FullName);
+ error = N6510_GetFileFolderInfo(s, &File2, ID_GetFileInfo);
+ memcpy(FilesLocations2, Priv->FilesLocations, sizeof(FilesLocations2));
+ FilesLocationsUsed2 = Priv->FilesLocationsUsed;
+ FilesLocationsCurrent2 = Priv->FilesLocationsCurrent;
+
+ memcpy(Priv->FilesLocations, FilesLocations, sizeof(FilesLocations));
+ memcpy(Priv->FilesLevels, FilesLevels, sizeof(FilesLevels));
+ Priv->FilesLocationsUsed = FilesLocationsUsed;
+ Priv->FilesLocationsCurrent = FilesLocationsCurrent;
+ if (error != ERR_NONE) return error;
+
+ while (1) {
+ if (FilesLocationsCurrent2 == FilesLocationsUsed2) return ERR_EMPTY;
+
+ sprintf(File2.ID_FullName,"%i",FilesLocations2[FilesLocationsCurrent2]);
+ dbgprintf("Current is %i\n",FilesLocations2[FilesLocationsCurrent2]);
+ FilesLocationsCurrent2++;
+
+ error = N6510_GetFileFolderInfo(s, &File2, ID_AddFile);
+ if (error == ERR_EMPTY) continue;
+ if (error != ERR_NONE) return error;
+ dbgprintf("%s %s\n",DecodeUnicodeString(File->Name),DecodeUnicodeString(File2.Name));
+ if (mywstrncasecmp(File2.Name,File->Name,0)) return ERR_NONE;
+ }
+ return ERR_EMPTY;
+}
+
+static GSM_Error N6510_ReplyGetFilePart(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int old;
+
+ smprintf(s,"File part received\n");
+ old = s->Phone.Data.File->Used;
+ s->Phone.Data.File->Used += msg.Buffer[6]*256*256*256+
+ msg.Buffer[7]*256*256+
+ msg.Buffer[8]*256+
+ msg.Buffer[9];
+ smprintf(s,"Length of file part: %i\n",
+ msg.Buffer[6]*256*256*256+
+ msg.Buffer[7]*256*256+
+ msg.Buffer[8]*256+
+ msg.Buffer[9]);
+ s->Phone.Data.File->Buffer = (unsigned char *)realloc(s->Phone.Data.File->Buffer,s->Phone.Data.File->Used);
+ memcpy(s->Phone.Data.File->Buffer+old,msg.Buffer+10,s->Phone.Data.File->Used-old);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetFilePart(GSM_StateMachine *s, GSM_File *File)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ int old;
+ GSM_Error error;
+ unsigned char req[] = {
+ N7110_FRAME_HEADER, 0x0E, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x01, /* Folder or file number */
+ 0x00, 0x00, 0x00, 0x00, /* Start from xxx byte */
+ 0x00, 0x00,
+ 0x03, 0xE8}; /* Read xxx bytes */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ if (File->Used == 0x00) {
+ error = N6510_GetFileFolderInfo(s, File, ID_GetFile);
+ if (error != ERR_NONE) return error;
+ File->Used = 0;
+ }
+
+ old = File->Used;
+ req[8] = atoi(File->ID_FullName) / 256;
+ req[9] = atoi(File->ID_FullName) % 256;
+ req[10] = old / (256*256*256);
+ req[11] = old / (256*256);
+ req[12] = old / 256;
+ req[13] = old % 256;
+
+ s->Phone.Data.File = File;
+ smprintf(s, "Getting file part from filesystem\n");
+ error=GSM_WaitFor (s, req, 18, 0x6D, 4, ID_GetFile);
+ if (error != ERR_NONE) return error;
+ if (File->Used - old != (0x03 * 256 + 0xE8)) {
+ if (N6510_FindFileCheckSum(File->Buffer, File->Used) != Priv->FileCheckSum) {
+ smprintf(s,"File2 checksum is %i, File checksum is %i\n",N6510_FindFileCheckSum(File->Buffer, File->Used),Priv->FileCheckSum);
+ return ERR_WRONGCRC;
+ }
+ return ERR_EMPTY;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_SetReadOnly(GSM_StateMachine *s, unsigned char *ID, bool enable)
+{
+ unsigned char SetAttr[] = {
+ N7110_FRAME_HEADER, 0x18, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x20}; /* File ID */
+
+ if (!enable) SetAttr[4] = 0x06;
+
+ SetAttr[8] = atoi(ID) / 256;
+ SetAttr[9] = atoi(ID) % 256;
+ smprintf(s, "Setting readonly attribute\n");
+ return GSM_WaitFor (s, SetAttr, 10, 0x6D, 4, ID_DeleteFile);
+}
+
+static GSM_Error N6510_ReplyAddFileHeader(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x03:
+ smprintf(s,"File header added\n");
+ sprintf(s->Phone.Data.File->ID_FullName,"%i",msg.Buffer[9]);
+ return ERR_NONE;
+ case 0x13:
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_ReplyAddFilePart(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_AddFilePart(GSM_StateMachine *s, GSM_File *File, int *Pos)
+{
+ GSM_Phone_N6510Data *Priv = &s->Phone.Data.Priv.N6510;
+ GSM_File File2;
+ GSM_Error error;
+ int j;
+ unsigned char Header[400] = {
+ N7110_FRAME_HEADER, 0x02, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x0C, /* parent folder ID */
+ 0x00, 0x00, 0x00, 0xE8};
+ unsigned char Add[15000] = {
+ N7110_FRAME_HEADER, 0x40, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x04, /* file ID */
+ 0x00, 0x00,
+ 0x01, 0x28}; /* length */
+ unsigned char end[30] = {
+ N7110_FRAME_HEADER, 0x40, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x04, /* file ID */
+ 0x00, 0x00, 0x00, 0x00};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.File = File;
+
+ if (*Pos == 0) {
+ error = N6510_SearchForFileName(s,File);
+ if (error == ERR_NONE) return ERR_INVALIDLOCATION;
+ if (error != ERR_EMPTY) return error;
+
+ Header[8] = atoi(File->ID_FullName) / 256;
+ Header[9] = atoi(File->ID_FullName) % 256;
+ memset(Header+14, 0x00, 300);
+ CopyUnicodeString(Header+14,File->Name);
+ Header[222] = File->Used / (256*256*256);
+ Header[223] = File->Used / (256*256);
+ Header[224] = File->Used / 256;
+ Header[225] = File->Used % 256;
+ switch(File->Type) {
+ case GSM_File_Image_JPG : Header[231]=0x02; Header[233]=0x01; break;
+ case GSM_File_Image_BMP : Header[231]=0x02; Header[233]=0x02; break;
+ case GSM_File_Image_PNG : Header[231]=0x02; Header[233]=0x03; break;
+ case GSM_File_Image_GIF : Header[231]=0x02; Header[233]=0x05; break;
+ case GSM_File_Image_WBMP : Header[231]=0x02; Header[233]=0x09; break;
+ case GSM_File_Sound_AMR : Header[231]=0x04; Header[233]=0x01; break;
+ case GSM_File_Sound_MIDI : Header[231]=0x04; Header[233]=0x05; break; //Header[238]=0x01;
+ case GSM_File_Sound_NRT : Header[231]=0x04; Header[233]=0x06; break;
+ case GSM_File_Video_3GP : Header[231]=0x08; Header[233]=0x05; break;
+ case GSM_File_Java_JAR : Header[231]=0x10; Header[233]=0x01; break;
+#ifdef DEVELOP
+ case GSM_File_MMS:
+ Header[214]=0x07;
+ Header[215]=0xd3;
+ Header[216]=0x06;
+ Header[217]=0x01;
+ Header[218]=0x12;
+ Header[219]=0x13;
+ Header[220]=0x29;
+ Header[233]=0x01;
+ break;
+#endif
+ default : Header[231]=0x01; Header[233]=0x05;
+ }
+ Header[235] = 0x01;
+ Header[236] = atoi(File->ID_FullName) / 256;
+ Header[237] = atoi(File->ID_FullName) % 256;
+ if (File->Protected) Header[238] = 0x01; //Nokia forward lock
+ if (File->Hidden) Header[241] = 0x01;
+ if (File->System) Header[242] = 0x01; //fixme
+ smprintf(s, "Adding file header\n");
+ error=GSM_WaitFor (s, Header, 246, 0x6D, 4, ID_AddFile);
+ if (error != ERR_NONE) return error;
+ }
+
+ j = 1000;
+ if (File->Used - *Pos < 1000) j = File->Used - *Pos;
+ Add[ 8] = atoi(File->ID_FullName) / 256;
+ Add[ 9] = atoi(File->ID_FullName) % 256;
+ Add[12] = j / 256;
+ Add[13] = j % 256;
+ memcpy(Add+14,File->Buffer+(*Pos),j);
+ smprintf(s, "Adding file part %i %i\n",*Pos,j);
+ error=GSM_WaitFor (s, Add, 14+j, 0x6D, 4, ID_AddFile);
+ if (error != ERR_NONE) return error;
+ *Pos = *Pos + j;
+
+ if (j < 1000) {
+ end[8] = atoi(File->ID_FullName) / 256;
+ end[9] = atoi(File->ID_FullName) % 256;
+ smprintf(s, "Frame for ending adding file\n");
+ error = GSM_WaitFor (s, end, 14, 0x6D, 4, ID_AddFile);
+ if (error != ERR_NONE) return error;
+
+ strcpy(File2.ID_FullName,File->ID_FullName);
+ error = N6510_GetFileFolderInfo(s, &File2, ID_GetFileInfo);
+ if (error != ERR_NONE) return error;
+
+ if (!File->ModifiedEmpty) {
+ Header[3] = 0x12;
+ Header[4] = 0x01;
+ Header[12] = 0x00;
+ Header[13] = 0xE8;
+ Header[8] = atoi(File->ID_FullName) / 256;
+ Header[9] = atoi(File->ID_FullName) % 256;
+ memset(Header+14, 0x00, 300);
+ CopyUnicodeString(Header+14,File->Name);
+ NOKIA_EncodeDateTime(s,Header+214,&File->Modified);
+ /* When you save too big file for phone and it changes
+ * size (some part is cut by firmware), you HAVE to write
+ * here correct file size. In other case filesystem
+ * will be damaged
+ */
+ Header[224] = File2.Used / 256;
+ Header[225] = File2.Used % 256;
+ Header[226] = Priv->FileToken / 256;
+ Header[227] = Priv->FileToken % 256;
+ switch(File->Type) {
+ case GSM_File_Image_JPG : Header[231]=0x02; Header[233]=0x01; break;
+ case GSM_File_Image_BMP : Header[231]=0x02; Header[233]=0x02; break;
+ case GSM_File_Image_PNG : Header[231]=0x02; Header[233]=0x03; break;
+ case GSM_File_Image_GIF : Header[231]=0x02; Header[233]=0x05; break;
+ case GSM_File_Image_WBMP : Header[231]=0x02; Header[233]=0x09; break;
+ case GSM_File_Sound_AMR : Header[231]=0x04; Header[233]=0x01; break;
+ case GSM_File_Sound_MIDI : Header[231]=0x04; Header[233]=0x05; break; //Header[238]=0x01;
+ case GSM_File_Sound_NRT : Header[231]=0x04; Header[233]=0x06; break;
+ case GSM_File_Video_3GP : Header[231]=0x08; Header[233]=0x05; break;
+ case GSM_File_Java_JAR : Header[231]=0x10; Header[233]=0x01; break;
+#ifdef DEVELOP
+ case GSM_File_MMS:
+ Header[214]=0x07;
+ Header[215]=0xd3;
+ Header[216]=0x06;
+ Header[217]=0x01;
+ Header[218]=0x12;
+ Header[219]=0x13;
+ Header[220]=0x29;
+ Header[233]=0x01;
+ break;
+#endif
+ default : Header[231]=0x01; Header[233]=0x05;
+ }
+ Header[235] = 0x01;
+ Header[236] = Priv->ParentID / 256;
+ Header[237] = Priv->ParentID % 256;
+ smprintf(s, "Adding file header\n");
+ error=GSM_WaitFor (s, Header, 246, 0x6D, 4, ID_AddFile);
+ if (error != ERR_NONE) return error;
+ }
+
+ /* Can't delete from phone menu */
+ if (File->ReadOnly) {
+ error = N6510_SetReadOnly(s, File->ID_FullName, true);
+ if (error != ERR_NONE) return error;
+ }
+
+ if (N6510_FindFileCheckSum(File->Buffer, File->Used) != Priv->FileCheckSum) {
+ smprintf(s,"File2 checksum is %i, File checksum is %i\n",N6510_FindFileCheckSum(File->Buffer, File->Used),Priv->FileCheckSum);
+ return ERR_WRONGCRC;
+ }
+
+ return ERR_EMPTY;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplyDeleteFile(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_DeleteFile(GSM_StateMachine *s, unsigned char *ID)
+{
+ GSM_Error error;
+ unsigned char Delete[40] = {
+ N7110_FRAME_HEADER, 0x1E, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x35}; /* File ID */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ error = N6510_SetReadOnly(s, ID, false);
+ if (error != ERR_NONE) return error;
+
+ Delete[8] = atoi(ID) / 256;
+ Delete[9] = atoi(ID) % 256;
+ return GSM_WaitFor (s, Delete, 10, 0x6D, 4, ID_DeleteFile);
+}
+
+static GSM_Error N6510_ReplyAddFolder(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ sprintf(s->Phone.Data.File->ID_FullName,"%i",msg.Buffer[9]);
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_AddFolder(GSM_StateMachine *s, GSM_File *File)
+{
+ GSM_Error error;
+ unsigned char Header[400] = {
+ N7110_FRAME_HEADER, 0x04, 0x00, 0x00, 0x00, 0x01,
+ 0x00, 0x0C, /* parent folder ID */
+ 0x00, 0x00, 0x00, 0xE8};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOFILESYSTEM)) return ERR_NOTSUPPORTED;
+
+ error = N6510_SearchForFileName(s,File);
+ if (error == ERR_NONE) return ERR_INVALIDLOCATION;
+ if (error != ERR_EMPTY) return error;
+
+ Header[8] = atoi(File->ID_FullName) / 256;
+ Header[9] = atoi(File->ID_FullName) % 256;
+ memset(Header+14, 0x00, 300);
+ CopyUnicodeString(Header+14,File->Name);
+ Header[233] = 0x02;
+ Header[235] = 0x01;
+ Header[236] = atoi(File->ID_FullName) / 256;
+ Header[237] = atoi(File->ID_FullName) % 256;
+
+ s->Phone.Data.File = File;
+ smprintf(s, "Adding folder\n");
+ error = GSM_WaitFor (s, Header, 246, 0x6D, 4, ID_AddFolder);
+ if (error != ERR_NONE) return error;
+
+ /* Can't delete from phone menu */
+ if (File->ReadOnly) {
+ error = N6510_SetReadOnly(s, File->ID_FullName, true);
+ if (error != ERR_NONE) return error;
+ }
+
+ return error;
+}
+
+#ifdef DEVELOP
+
+static GSM_Error N6510_ReplyEnableGPRSAccessPoint(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ if (msg.Buffer[13] == 0x02) return ERR_NONE;
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_EnableGPRSAccessPoint(GSM_StateMachine *s)
+{
+ GSM_Error error;
+ int i;
+ unsigned char req[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x04, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOGPRSPOINT)) return ERR_NOTSUPPORTED;
+
+ for (i=0;i<3;i++) {
+ smprintf(s, "Activating full GPRS access point support\n");
+ error = GSM_WaitFor (s, req, 16, 0x43, 4, ID_EnableGPRSPoint);
+ if (error != ERR_NONE) return error;
+ }
+ return error;
+}
+
+#endif
+
+static GSM_Error N6510_ReplyGetGPRSAccessPoint(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_GPRSAccessPoint *point = s->Phone.Data.GPRSPoint;
+
+ switch (msg.Buffer[13]) {
+ case 0x01:
+ smprintf(s,"Active GPRS point received\n");
+ point->Active = false;
+ if (point->Location == msg.Buffer[18]) point->Active = true;
+ return ERR_NONE;
+ case 0xD2:
+ smprintf(s,"Names for GPRS points received\n");
+ CopyUnicodeString(point->Name,msg.Buffer+18+(point->Location-1)*42);
+ smprintf(s,"\"%s\"\n",DecodeUnicodeString(point->Name));
+ return ERR_NONE;
+ case 0xF2:
+ smprintf(s,"URL for GPRS points received\n");
+ CopyUnicodeString(point->URL,msg.Buffer+18+(point->Location-1)*202);
+ smprintf(s,"\"%s\"\n",DecodeUnicodeString(point->URL));
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetGPRSAccessPoint(GSM_StateMachine *s, GSM_GPRSAccessPoint *point)
+{
+ GSM_Error error;
+ unsigned char URL[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x00, 0x00, 0x00, 0x03, 0xF2, 0x00, 0x00};
+ unsigned char Name[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x01, 0x00, 0x00, 0x00, 0xD2, 0x00, 0x00};
+ unsigned char Active[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x02, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOGPRSPOINT)) return ERR_NOTSUPPORTED;
+ if (point->Location < 1) return ERR_UNKNOWN;
+ if (point->Location > 5) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.GPRSPoint = point;
+
+#ifdef DEVELOP
+ error = N6510_EnableGPRSAccessPoint(s);
+ if (error != ERR_NONE) return error;
+#endif
+
+ smprintf(s, "Getting GPRS access point name\n");
+ error=GSM_WaitFor (s, Name, 16, 0x43, 4, ID_GetGPRSPoint);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting GPRS access point URL\n");
+ error=GSM_WaitFor (s, URL, 16, 0x43, 4, ID_GetGPRSPoint);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting number of active GPRS access point\n");
+ error=GSM_WaitFor (s, Active, 16, 0x43, 4, ID_GetGPRSPoint);
+ if (error != ERR_NONE) return error;
+
+ if (UnicodeLength(point->URL)==0 && UnicodeLength(point->Name)==0) return ERR_EMPTY;
+ return error;
+}
+
+static GSM_Error N6510_ReplySetGPRSAccessPoint1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[13]) {
+ case 0x01:
+ case 0xD2:
+ case 0xF2:
+ memcpy(s->Phone.Data.Priv.N6510.GPRSPoints,msg.Buffer,msg.Length);
+ s->Phone.Data.Priv.N6510.GPRSPointsLength = msg.Length;
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_SetGPRSAccessPoint(GSM_StateMachine *s, GSM_GPRSAccessPoint *point)
+{
+ unsigned char *buff = s->Phone.Data.Priv.N6510.GPRSPoints;
+ GSM_Error error;
+ unsigned char URL[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x00, 0x00, 0x00, 0x03, 0xF2, 0x00, 0x00};
+ unsigned char Name[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x01, 0x00, 0x00, 0x00, 0xD2, 0x00, 0x00};
+ unsigned char Active[] = {
+ N7110_FRAME_HEADER, 0x05, 0x00, 0x00, 0x00, 0x2C, 0x00,
+ 0x02, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NOGPRSPOINT)) return ERR_NOTSUPPORTED;
+ if (point->Location < 1) return ERR_UNKNOWN;
+ if (point->Location > 5) return ERR_INVALIDLOCATION;
+
+ s->Phone.Data.GPRSPoint = point;
+
+#ifdef DEVELOP
+ error = N6510_EnableGPRSAccessPoint(s);
+ if (error != ERR_NONE) return error;
+#endif
+
+ smprintf(s, "Getting GPRS access point name\n");
+ error=GSM_WaitFor (s, Name, 16, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+ CopyUnicodeString(buff+18+(point->Location-1)*42,point->Name);
+ buff[0] = 0x00;
+ buff[1] = 0x01;
+ buff[2] = 0x01;
+ buff[3] = 0x07;
+ smprintf(s, "Setting GPRS access point name\n");
+ error=GSM_WaitFor (s, buff, s->Phone.Data.Priv.N6510.GPRSPointsLength, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting GPRS access point URL\n");
+ error=GSM_WaitFor (s, URL, 16, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+ CopyUnicodeString(buff+18+(point->Location-1)*42,point->URL);
+ buff[0] = 0x00;
+ buff[1] = 0x01;
+ buff[2] = 0x01;
+ buff[3] = 0x07;
+ smprintf(s, "Setting GPRS access point URL\n");
+ error=GSM_WaitFor (s, buff, s->Phone.Data.Priv.N6510.GPRSPointsLength, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+
+ if (point->Active) {
+ smprintf(s, "Getting number of active GPRS access point\n");
+ error=GSM_WaitFor (s, Active, 16, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+ buff[0] = 0x00;
+ buff[1] = 0x01;
+ buff[2] = 0x01;
+ buff[3] = 0x07;
+ buff[18]= point->Location;
+ smprintf(s, "Setting number of active GPRS access point\n");
+ error=GSM_WaitFor (s, buff, s->Phone.Data.Priv.N6510.GPRSPointsLength, 0x43, 4, ID_SetGPRSPoint);
+ if (error != ERR_NONE) return error;
+ }
+
+ return error;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_ReplyGetToDoStatus1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int i;
+ GSM_NOKIACalToDoLocations *Last = &s->Phone.Data.Priv.N6510.LastToDo;
+
+ smprintf(s, "TODO locations received\n");
+ Last->Number=msg.Buffer[6]*256+msg.Buffer[7];
+ smprintf(s, "Number of Entries: %i\n",Last->Number);
+ smprintf(s, "Locations: ");
+ for (i=0;i<Last->Number;i++) {
+ Last->Location[i]=msg.Buffer[12+(i*4)]*256+msg.Buffer[(i*4)+13];
+ smprintf(s, "%i ",Last->Location[i]);
+ }
+ smprintf(s, "\n");
+ return ERR_NONE;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_GetToDoStatus1(GSM_StateMachine *s, GSM_ToDoStatus *status)
+{
+ GSM_Error error;
+ GSM_NOKIACalToDoLocations *LastToDo = &s->Phone.Data.Priv.N6510.LastToDo;
+ unsigned char reqLoc[] = {
+ N6110_FRAME_HEADER,
+ 0x15, 0x01, 0x00, 0x00,
+ 0x00, 0x00, 0x00};
+
+ smprintf(s, "Getting ToDo locations\n");
+ error = GSM_WaitFor (s, reqLoc, 10, 0x55, 4, ID_GetToDo);
+ if (error != ERR_NONE) return error;
+
+ status->Used = LastToDo->Number;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetToDoStatus2(GSM_StateMachine *s, GSM_ToDoStatus *status)
+{
+ GSM_NOKIACalToDoLocations *LastToDo = &s->Phone.Data.Priv.N6510.LastToDo;
+ GSM_Error error;
+
+ error = N6510_GetCalendarInfo3(s,LastToDo,false);
+ if (error!=ERR_NONE) return error;
+
+ status->Used = LastToDo->Number;
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_GetToDoStatus(GSM_StateMachine *s, GSM_ToDoStatus *status)
+{
+ status->Used = 0;
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO63)) {
+ return N6510_GetToDoStatus1(s, status);
+ } else if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO66)) {
+ return N6510_GetToDoStatus2(s, status);
+ } else {
+ return ERR_NOTSUPPORTED;
+ }
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_ReplyGetToDo1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_ToDoEntry *Last = s->Phone.Data.ToDo;
+
+ smprintf(s, "TODO received method 1\n");
+
+ switch (msg.Buffer[4]) {
+ case 1 : Last->Priority = GSM_Priority_High; break;
+ case 2 : Last->Priority = GSM_Priority_Medium; break;
+ case 3 : Last->Priority = GSM_Priority_Low; break;
+ default : return ERR_UNKNOWN;
+ }
+ smprintf(s, "Priority: %i\n",msg.Buffer[4]);
+
+ CopyUnicodeString(Last->Entries[0].Text,msg.Buffer+14);
+ Last->Entries[0].EntryType = TODO_TEXT;
+ Last->EntriesNum = 1;
+ smprintf(s, "Text: \"%s\"\n",DecodeUnicodeString(Last->Entries[0].Text));
+
+ return ERR_NONE;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_GetNextToDo1(GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool refresh)
+{
+ GSM_Error error;
+ GSM_ToDoStatus status;
+ GSM_NOKIACalToDoLocations *LastToDo = &s->Phone.Data.Priv.N6510.LastToDo;
+ unsigned char reqGet[] = {
+ N6110_FRAME_HEADER,
+ 0x03, 0x00, 0x00, 0x80, 0x00,
+ 0x00, 0x17}; /* Location */
+
+ if (refresh) {
+ error = N6510_GetToDoStatus(s, &status);
+ if (error != ERR_NONE) return error;
+ ToDo->Location = 1;
+ } else {
+ ToDo->Location++;
+ }
+ if (ToDo->Location > LastToDo->Number) return ERR_EMPTY;
+
+ reqGet[8] = LastToDo->Location[ToDo->Location-1] / 256;
+ reqGet[9] = LastToDo->Location[ToDo->Location-1] % 256;
+ s->Phone.Data.ToDo = ToDo;
+ smprintf(s, "Getting ToDo\n");
+ return GSM_WaitFor (s, reqGet, 10, 0x55, 4, ID_GetToDo);
+}
+
+static GSM_Error N6510_ReplyGetToDoStatus2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ return N6510_ReplyGetCalendarInfo3(msg, s, &s->Phone.Data.Priv.N6510.LastToDo);
+}
+
+/* Similiar to getting calendar method 3 */
+static GSM_Error N6510_ReplyGetToDo2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_ToDoEntry *Last = s->Phone.Data.ToDo;
+ GSM_DateTime Date;
+ unsigned long diff;
+
+ smprintf(s, "ToDo received method 2\n");
+
+ switch (msg.Buffer[44]) {
+ case 0x10: Last->Priority = GSM_Priority_Low; break;
+ case 0x20: Last->Priority = GSM_Priority_Medium; break;
+ case 0x30: Last->Priority = GSM_Priority_High; break;
+ default : return ERR_UNKNOWN;
+ }
+
+ memcpy(Last->Entries[0].Text,msg.Buffer+54,msg.Buffer[51]*2);
+ Last->Entries[0].Text[msg.Buffer[51]*2] = 0;
+ Last->Entries[0].Text[msg.Buffer[51]*2+1] = 0;
+ Last->Entries[0].EntryType = TODO_TEXT;
+ smprintf(s, "Text: \"%s\"\n",DecodeUnicodeString(Last->Entries[0].Text));
+
+ smprintf(s,"EndTime: %04i-%02i-%02i %02i:%02i\n",
+ msg.Buffer[34]*256+msg.Buffer[35],
+ msg.Buffer[36],msg.Buffer[37],msg.Buffer[38],
+ msg.Buffer[39]);
+ Date.Year = msg.Buffer[34]*256+msg.Buffer[35];
+ Date.Month = msg.Buffer[36];
+ Date.Day = msg.Buffer[37];
+ Date.Hour = msg.Buffer[38];
+ Date.Minute = msg.Buffer[39];
+ Date.Second = 0;
+ Last->Entries[1].EntryType = TODO_END_DATETIME;
+ memcpy(&Last->Entries[1].Date,&Date,sizeof(GSM_DateTime));
+
+ smprintf(s,"StartTime: %04i-%02i-%02i %02i:%02i\n",
+ msg.Buffer[28]*256+msg.Buffer[29],
+ msg.Buffer[30],msg.Buffer[31],msg.Buffer[32],
+ msg.Buffer[33]);
+ Date.Year = msg.Buffer[28]*256+msg.Buffer[29];
+ Date.Month = msg.Buffer[30];
+ Date.Day = msg.Buffer[31];
+ Date.Hour = msg.Buffer[32];
+ Date.Minute = msg.Buffer[33];
+ Date.Second = 0;
+
+ Last->EntriesNum = 2;
+
+ if (msg.Buffer[45] == 0x01) {
+ Last->Entries[2].Number = msg.Buffer[45];
+ Last->Entries[2].EntryType = TODO_COMPLETED;
+ Last->EntriesNum++;
+ smprintf(s,"Completed\n");
+ }
+
+ if (msg.Buffer[14] == 0xFF && msg.Buffer[15] == 0xFF && msg.Buffer[16] == 0xff && msg.Buffer[17] == 0xff)
+ {
+ smprintf(s, "No alarm\n");
+ } else {
+ diff = ((unsigned int)msg.Buffer[14]) << 24;
+ diff += ((unsigned int)msg.Buffer[15]) << 16;
+ diff += ((unsigned int)msg.Buffer[16]) << 8;
+ diff += msg.Buffer[17];
+
+ memcpy(&Last->Entries[Last->EntriesNum].Date,&Date,sizeof(GSM_DateTime));
+ GetTimeDifference(diff, &Last->Entries[Last->EntriesNum].Date, false, 60);
+ smprintf(s, "Alarm date : %02i-%02i-%04i %02i:%02i:%02i\n",
+ Last->Entries[Last->EntriesNum].Date.Day, Last->Entries[Last->EntriesNum].Date.Month,
+ Last->Entries[Last->EntriesNum].Date.Year, Last->Entries[Last->EntriesNum].Date.Hour,
+ Last->Entries[Last->EntriesNum].Date.Minute,Last->Entries[Last->EntriesNum].Date.Second);
+
+ Last->Entries[Last->EntriesNum].EntryType = TODO_ALARM_DATETIME;
+ if (msg.Buffer[22]==0x00 && msg.Buffer[23]==0x00 &&
+ msg.Buffer[24]==0x00 && msg.Buffer[25]==0x00)
+ {
+ Last->Entries[Last->EntriesNum].EntryType = TODO_SILENT_ALARM_DATETIME;
+ smprintf(s, "Alarm type : Silent\n");
+ }
+ Last->EntriesNum++;
+ }
+
+ return ERR_NONE;
+}
+
+/* ToDo support - 6610 style */
+static GSM_Error N6510_GetNextToDo2(GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool refresh)
+{
+ GSM_Error error;
+ GSM_NOKIACalToDoLocations *LastToDo = &s->Phone.Data.Priv.N6510.LastToDo;
+ /* The same to getting calendar method 3 */
+ unsigned char req[] = {
+ N6110_FRAME_HEADER,0x7D,0x00,0x00,0x00,0x00,
+ 0x00,0x99, /* Location */
+ 0xff,0xff,0xff,0xff,0x01};
+
+ if (refresh) {
+ error=N6510_GetCalendarInfo3(s,LastToDo,false);
+ if (error!=ERR_NONE) return error;
+ ToDo->Location = 1;
+ } else {
+ ToDo->Location++;
+ }
+
+ if (ToDo->Location > LastToDo->Number) return ERR_EMPTY;
+
+ req[8] = LastToDo->Location[ToDo->Location-1] / 256;
+ req[9] = LastToDo->Location[ToDo->Location-1] % 256;
+
+ s->Phone.Data.ToDo = ToDo;
+ smprintf(s, "Getting todo method 2\n");
+ return GSM_WaitFor (s, req, 15, 0x13, 4, ID_GetToDo);
+}
+
+static GSM_Error N6510_GetNextToDo(GSM_StateMachine *s, GSM_ToDoEntry *ToDo, bool refresh)
+{
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO63)) {
+ return N6510_GetNextToDo1(s, ToDo, refresh);
+ } else if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO66)) {
+ return N6510_GetNextToDo2(s, ToDo, refresh);
+ } else {
+ return ERR_NOTSUPPORTED;
+ }
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_ReplyDeleteAllToDo1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "All TODO deleted\n");
+ return ERR_NONE;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_DeleteAllToDo1(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x11};
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO63)) {
+ return ERR_NOTSUPPORTED;
+ }
+
+ smprintf(s, "Deleting all ToDo method 1\n");
+ return GSM_WaitFor (s, req, 4, 0x55, 4, ID_DeleteAllToDo);
+}
+
+static GSM_Error N6510_DeleteToDo2(GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_Error error;
+ GSM_NOKIACalToDoLocations *LastToDo = &s->Phone.Data.Priv.N6510.LastToDo;
+ GSM_CalendarEntry Note;
+
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO66)) {
+ return ERR_NOTSUPPORTED;
+ }
+
+ error=N6510_GetCalendarInfo3(s,LastToDo,false);
+ if (error!=ERR_NONE) return error;
+
+ smprintf(s, "Deleting ToDo method 2\n");
+
+ if (ToDo->Location > LastToDo->Number || ToDo->Location == 0) return ERR_INVALIDLOCATION;
+
+ Note.Location = LastToDo->Location[ToDo->Location-1];
+ return N71_65_DelCalendar(s,&Note);
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_ReplyGetToDoFirstLoc1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "TODO first location received method 1: %02x\n",msg.Buffer[9]);
+ s->Phone.Data.ToDo->Location = msg.Buffer[9];
+ return ERR_NONE;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_ReplyAddToDo1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "TODO set OK\n");
+ return ERR_NONE;
+}
+
+/* ToDo support - 6310 style */
+static GSM_Error N6510_AddToDo1(GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ int Text, Alarm, EndTime, Completed, ulen, Phone;
+ GSM_Error error;
+ unsigned char reqLoc[] = {N6110_FRAME_HEADER, 0x0F};
+ unsigned char reqSet[500] = {
+ N6110_FRAME_HEADER, 0x01,
+ 0x03, /* Priority */
+ 0x00, /* Length of text */
+ 0x80,0x00,0x00,
+ 0x18}; /* Location */
+
+ s->Phone.Data.ToDo = ToDo;
+
+ smprintf(s, "Getting first ToDo location\n");
+ error = GSM_WaitFor (s, reqLoc, 4, 0x55, 4, ID_SetToDo);
+ if (error != ERR_NONE) return error;
+ reqSet[9] = ToDo->Location;
+
+ switch (ToDo->Priority) {
+ case GSM_Priority_Low : reqSet[4] = 3; break;
+ case GSM_Priority_Medium: reqSet[4] = 2; break;
+ case GSM_Priority_High : reqSet[4] = 1; break;
+ }
+
+ GSM_ToDoFindDefaultTextTimeAlarmCompleted(ToDo, &Text, &Alarm, &Completed, &EndTime, &Phone);
+
+ if (Text == -1) return ERR_NOTSUPPORTED; /* XXX: shouldn't this be handled different way? */
+ ulen = UnicodeLength(ToDo->Entries[Text].Text);
+ reqSet[5] = ulen+1;
+ CopyUnicodeString(reqSet+10,ToDo->Entries[Text].Text);
+ reqSet[10+ulen*2] = 0x00;
+ reqSet[10+ulen*2+1] = 0x00;
+ smprintf(s, "Adding ToDo method 1\n");
+ return GSM_WaitFor (s, reqSet, 12+ulen*2, 0x55, 4, ID_SetToDo);
+}
+
+static GSM_Error N6510_ReplyAddToDo2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "ToDo added method 2\n");
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_ReplyGetToDoFirstLoc2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "First ToDo location method 2: %i\n",msg.Buffer[8]*256+msg.Buffer[9]);
+ s->Phone.Data.ToDo->Location = msg.Buffer[9];
+ return ERR_NONE;
+}
+
+static GSM_Error N6510_AddToDo2(GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ GSM_CalendarEntry Note;
+ time_t t_time1,t_time2;
+ long diff;
+ GSM_Error error;
+ GSM_DateTime DT;
+ int Text, Alarm, EndTime, Completed, count=54, Phone;
+ unsigned char reqLoc[] = {N6110_FRAME_HEADER, 0x95, 0x01};
+ unsigned char req[5000] = {
+ N6110_FRAME_HEADER, 0x65,
+ 0x01, /* 0 = calendar, 1 = todo */
+ 0x00, 0x00, 0x00,
+ 0x00, 0x00, /* location */
+ 0x00, 0x00, 0x00, 0x00,
+ 0xFF, 0xFF, 0xFF, 0xFF, /* alarm */
+ 0x80, 0x00, 0x00,
+ 0x01, /* note icon */
+ 0xFF, 0xFF, 0xFF, 0xFF, /* alarm type */
+ 0x00, /* 0x02 or 0x00 */
+ 0x01, /* note type */
+ 0x07, 0xD0, 0x01, 0x12, 0x0C, 0x00, /* start date/time */
+ 0x07, 0xD0, 0x01, 0x12, 0x0C, 0x00, /* end date/time */
+ 0x00, 0x00, /* recurrance */
+ 0x00, 0x00, /* birth year */
+ 0x20, /* ToDo priority */
+ 0x00, /* ToDo completed ? */
+ 0x00, 0x00, 0x00,
+ 0x00, /* note text length */
+ 0x00, /* phone length/meeting place */
+ 0x00, 0x00, 0x00};
+
+ s->Phone.Data.ToDo = ToDo;
+
+ smprintf(s, "Getting first free ToDo location method 2\n");
+ error = GSM_WaitFor (s, reqLoc, 5, 0x13, 4, ID_SetToDo);
+ if (error!=ERR_NONE) return error;
+ req[8] = ToDo->Location/256;
+ req[9] = ToDo->Location%256;
+
+ Note.Type = GSM_CAL_MEETING;
+ DT.Year = 2004; DT.Month = 1; DT.Day = 1;
+ DT.Hour = 12; DT.Minute = 12; DT.Second = 0;
+ memcpy(&Note.Entries[0].Date,&DT,sizeof(GSM_DateTime));
+ Note.Entries[0].EntryType = CAL_START_DATETIME;
+ memcpy(&Note.Entries[1].Date,&DT,sizeof(GSM_DateTime));
+ Note.Entries[1].EntryType = CAL_END_DATETIME;
+ EncodeUnicode(Note.Entries[2].Text,"ala",3);
+ Note.Entries[2].EntryType = CAL_TEXT;
+ Note.EntriesNum = 3;
+ error=N6510_FindCalendarIconID3(s, &Note, &req[21]);
+ if (error!=ERR_NONE) return error;
+
+ switch (ToDo->Priority) {
+ case GSM_Priority_Low : req[44] = 0x10; break;
+ case GSM_Priority_Medium: req[44] = 0x20; break;
+ case GSM_Priority_High : req[44] = 0x30; break;
+ }
+
+ GSM_ToDoFindDefaultTextTimeAlarmCompleted(ToDo, &Text, &Alarm, &Completed, &EndTime, &Phone);
+
+ if (Completed != -1) req[45] = 0x01;
+
+ if (EndTime == -1) {
+ GSM_GetCurrentDateTime(&DT);
+ } else {
+ memcpy(&DT,&ToDo->Entries[EndTime].Date,sizeof(GSM_DateTime));
+ }
+ /*Start time*/
+ req[28] = DT.Year / 256;
+ req[29] = DT.Year % 256;
+ req[30] = DT.Month;
+ req[31] = DT.Day;
+ req[32] = DT.Hour;
+ req[33] = DT.Minute;
+ /*End time*/
+ req[34] = DT.Year / 256;
+ req[35] = DT.Year % 256;
+ req[36] = DT.Month;
+ req[37] = DT.Day;
+ req[38] = DT.Hour;
+ req[39] = DT.Minute;
+
+ if (Alarm != -1) {
+ if (ToDo->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME)
+ {
+ req[22] = 0x00; req[23] = 0x00; req[24] = 0x00; req[25] = 0x00;
+ }
+ t_time2 = Fill_Time_T(DT,8);
+ t_time1 = Fill_Time_T(ToDo->Entries[Alarm].Date,8);
+ diff = (t_time1-t_time2)/60;
+
+ smprintf(s, " Difference : %li seconds or minutes\n", -diff);
+ req[14] = (unsigned char)(-diff >> 24);
+ req[15] = (unsigned char)(-diff >> 16);
+ req[16] = (unsigned char)(-diff >> 8);
+ req[17] = (unsigned char)(-diff);
+ }
+
+ if (Text != -1) {
+ req[49] = UnicodeLength(ToDo->Entries[Text].Text);
+ CopyUnicodeString(req+54,ToDo->Entries[Text].Text);
+ count+= req[49]*2;
+ }
+
+ req[count++] = 0x00;
+
+ smprintf(s, "Adding ToDo method 2\n");
+ return GSM_WaitFor (s, req, count, 0x13, 4, ID_SetToDo);
+}
+
+static GSM_Error N6510_AddToDo(GSM_StateMachine *s, GSM_ToDoEntry *ToDo)
+{
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO63)) {
+ return N6510_AddToDo1(s, ToDo);
+ } else if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_TODO66)) {
+ return N6510_AddToDo2(s, ToDo);
+ } else {
+ return ERR_NOTSUPPORTED;
+ }
+}
+
+static GSM_Error N6510_ReplyGetLocale(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Locale *locale = s->Phone.Data.Locale;
+
+ switch (msg.Buffer[3]) {
+ case 0x8A:
+ smprintf(s, "Date settings received\n");
+ switch (msg.Buffer[4]) {
+ case 0x00:
+ locale->DateFormat = GSM_Date_DDMMYYYY;
+ locale->DateSeparator = '.';
+ break;
+ case 0x01:
+ locale->DateFormat = GSM_Date_MMDDYYYY;
+ locale->DateSeparator = '.';
+ break;
+ case 0x02:
+ locale->DateFormat = GSM_Date_YYYYMMDD;
+ locale->DateSeparator = '.';
+ break;
+ case 0x04:
+ locale->DateFormat = GSM_Date_DDMMYYYY;
+ locale->DateSeparator = '/';
+ break;
+ case 0x05:
+ locale->DateFormat = GSM_Date_MMDDYYYY;
+ locale->DateSeparator = '/';
+ break;
+ case 0x06:
+ locale->DateFormat = GSM_Date_YYYYMMDD;
+ locale->DateSeparator = '/';
+ break;
+ case 0x08:
+ locale->DateFormat = GSM_Date_DDMMYYYY;
+ locale->DateSeparator = '-';
+ break;
+ case 0x09:
+ locale->DateFormat = GSM_Date_MMDDYYYY;
+ locale->DateSeparator = '-';
+ break;
+ case 0x0A:
+ locale->DateFormat = GSM_Date_YYYYMMDD;
+ locale->DateSeparator = '-';
+ break;
+ default:/* FIXME */
+ locale->DateFormat = GSM_Date_DDMMYYYY;
+ locale->DateSeparator = '/';
+ break;
+ }
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetLocale(GSM_StateMachine *s, GSM_Locale *locale)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x89};
+
+ s->Phone.Data.Locale = locale;
+
+ smprintf(s, "Getting date format\n");
+ return GSM_WaitFor (s, req, 4, 0x13, 4, ID_GetLocale);
+}
+
+static GSM_Error N6510_ReplyGetCalendarSettings(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_CalendarSettings *sett = s->Phone.Data.CalendarSettings;
+
+ switch (msg.Buffer[3]) {
+ case 0x86:
+ smprintf(s, "Auto deleting setting received\n");
+ sett->AutoDelete = msg.Buffer[4];
+ return ERR_NONE;
+ case 0x8E:
+ smprintf(s, "Start day for calendar received\n");
+ switch(msg.Buffer[4]) {
+ case 0x03:
+ sett->StartDay = 6;
+ return ERR_NONE;
+ case 0x02:
+ sett->StartDay = 7;
+ return ERR_NONE;
+ case 0x01:
+ sett->StartDay = 1;
+ return ERR_NONE;
+ }
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error N6510_GetCalendarSettings(GSM_StateMachine *s, GSM_CalendarSettings *settings)
+{
+ GSM_Error error;
+ unsigned char req1[] = {N6110_FRAME_HEADER, 0x85};
+ unsigned char req2[] = {N6110_FRAME_HEADER, 0x8D};
+
+ s->Phone.Data.CalendarSettings = settings;
+
+ smprintf(s, "Getting auto delete\n");
+ error = GSM_WaitFor (s, req1, 4, 0x13, 4, ID_GetCalendarSettings);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s, "Getting start day for week\n");
+ return GSM_WaitFor (s, req2, 4, 0x13, 4, ID_GetCalendarSettings);
+}
+
+GSM_Error N6510_CancelCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (all) return ERR_NOTSUPPORTED;
+ return DCT3DCT4_CancelCall(s,ID);
+}
+
+GSM_Error N6510_AnswerCall(GSM_StateMachine *s, int ID, bool all)
+{
+ if (all) return ERR_NOTSUPPORTED;
+ return DCT3DCT4_AnswerCall(s,ID);
+}
+
+static GSM_Error N6510_ReplyAddSMSFolder(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s,"SMS folder \"%s\" has been added\n",DecodeUnicodeString(msg.Buffer+10));
+ return ERR_NONE;
+}
+
+GSM_Error N6510_AddSMSFolder(GSM_StateMachine *s, unsigned char *name)
+{
+ unsigned char req[200] = {N6110_FRAME_HEADER, 0x10, 0x01, 0x00, 0x01,
+ 0x00, /* Length */
+ 0x00, 0x00};
+
+
+ CopyUnicodeString(req+10,name);
+ req[7] = UnicodeLength(name)*2 + 6;
+
+ smprintf(s, "Adding SMS folder\n");
+ return GSM_WaitFor (s, req, req[7] + 6, 0x14, 4, ID_AddSMSFolder);
+}
+
+static GSM_Error N6510_SetWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+ int count=4, location;
+ unsigned char req[600] = {N6110_FRAME_HEADER, 0x09};
+
+ /* We have to enable WAP frames in phone */
+ error=N6510_EnableConnectionFunctions(s,N6510_WAP_SETTINGS);
+ if (error!=ERR_NONE) return error;
+
+ location = bookmark->Location - 1;
+ if (bookmark->Location == 0) location = 0xffff;
+ req[count++] = (location & 0xff00) >> 8;
+ req[count++] = location & 0x00ff;
+
+ count += NOKIA_SetUnicodeString(s, req+count, bookmark->Title, true);
+ count += NOKIA_SetUnicodeString(s, req+count, bookmark->Address, true);
+
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+
+ smprintf(s, "Setting WAP bookmark\n");
+ error = GSM_WaitFor (s, req, count, 0x3f, 4, ID_SetWAPBookmark);
+ if (error != ERR_NONE) {
+ if (error == ERR_INSIDEPHONEMENU || error == ERR_EMPTY || error == ERR_FULL) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error N6510_DeleteWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+
+ /* We have to enable WAP frames in phone */
+ error=N6510_EnableConnectionFunctions(s,N6510_WAP_SETTINGS);
+ if (error!=ERR_NONE) return error;
+
+ return DCT3DCT4_DeleteWAPBookmarkPart(s,bookmark);
+}
+
+GSM_Error N6510_GetWAPBookmark(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+
+ /* We have to enable WAP frames in phone */
+ error=N6510_EnableConnectionFunctions(s,N6510_WAP_SETTINGS);
+ if (error!=ERR_NONE) return error;
+
+ return DCT3DCT4_GetWAPBookmarkPart(s,bookmark);
+}
+
+static GSM_Reply_Function N6510ReplyFunctions[] = {
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x02,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x03,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x04,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x05,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x07,ID_AnswerCall },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x07,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x09,ID_CancelCall },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x09,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0A,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0B,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0C,ID_DialVoice },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x0C,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x23,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x25,ID_IncomingFrame },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x27,ID_IncomingFrame },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x51,ID_SendDTMF },
+ {N71_65_ReplyCallInfo, "\x01",0x03,0x53,ID_IncomingFrame },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x59,ID_SendDTMF },
+ {N71_65_ReplySendDTMF, "\x01",0x03,0x5E,ID_SendDTMF },
+
+ {N6510_ReplySendSMSMessage, "\x02",0x03,0x03,ID_IncomingFrame },
+ {N6510_ReplyIncomingSMS, "\x02",0x03,0x04,ID_IncomingFrame },
+ {N6510_ReplySetSMSC, "\x02",0x03,0x13,ID_SetSMSC },
+ {N6510_ReplyGetSMSC, "\x02",0x03,0x15,ID_GetSMSC },
+
+ {N6510_ReplyGetMemoryStatus, "\x03",0x03,0x04,ID_GetMemoryStatus },
+ {N6510_ReplyGetMemory, "\x03",0x03,0x08,ID_GetMemory },
+ {N6510_ReplyDeleteMemory, "\x03",0x03,0x10,ID_SetMemory },
+ {N71_65_ReplyWritePhonebook, "\x03",0x03,0x0C,ID_SetBitmap },
+ {N71_65_ReplyWritePhonebook, "\x03",0x03,0x0C,ID_SetMemory },
+
+ {DCT3DCT4_ReplyCallDivert, "\x06",0x03,0x02,ID_Divert },
+ {N71_65_ReplyUSSDInfo, "\x06",0x03,0x03,ID_IncomingFrame },
+ {NoneReply, "\x06",0x03,0x06,ID_IncomingFrame },
+ {NoneReply, "\x06",0x03,0x09,ID_IncomingFrame },
+
+ {N6510_ReplyEnterSecurityCode, "\x08",0x03,0x08,ID_EnterSecurityCode },
+ {N6510_ReplyEnterSecurityCode, "\x08",0x03,0x09,ID_EnterSecurityCode },
+ {N6510_ReplyGetSecurityStatus, "\x08",0x03,0x12,ID_GetSecurityStatus },
+
+ {N6510_ReplyGetNetworkInfo, "\x0A",0x03,0x01,ID_GetNetworkInfo },
+ {N6510_ReplyGetNetworkInfo, "\x0A",0x03,0x01,ID_IncomingFrame },
+ {N6510_ReplyLogIntoNetwork, "\x0A",0x03,0x02,ID_IncomingFrame },
+ {N6510_ReplyGetSignalQuality, "\x0A",0x03,0x0C,ID_GetSignalQuality },
+ {N6510_ReplyGetIncSignalQuality, "\x0A",0x03,0x1E,ID_IncomingFrame },
+ {NoneReply, "\x0A",0x03,0x20,ID_IncomingFrame },
+ {N6510_ReplyGetOperatorLogo, "\x0A",0x03,0x24,ID_GetBitmap },
+ {N6510_ReplySetOperatorLogo, "\x0A",0x03,0x26,ID_SetBitmap },
+
+ {NoneReply, "\x0B",0x03,0x01,ID_PlayTone },
+ {NoneReply, "\x0B",0x03,0x15,ID_PlayTone },
+ {NoneReply, "\x0B",0x03,0x16,ID_PlayTone },
+
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x02,ID_SetCalendarNote },
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x04,ID_SetCalendarNote },
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x06,ID_SetCalendarNote },
+ {N71_65_ReplyAddCalendar1, "\x13",0x03,0x08,ID_SetCalendarNote },
+ {N71_65_ReplyDelCalendar, "\x13",0x03,0x0C,ID_DeleteCalendarNote },
+ {N71_65_ReplyGetNextCalendar1, "\x13",0x03,0x1A,ID_GetCalendarNote },/*method 1*/
+ {N6510_ReplyGetCalendarNotePos, "\x13",0x03,0x32,ID_GetCalendarNotePos },/*method 1*/
+ {N6510_ReplyGetCalendarInfo, "\x13",0x03,0x3B,ID_GetCalendarNotesInfo},/*method 1*/
+#ifdef DEBUG
+ {N71_65_ReplyGetNextCalendar2, "\x13",0x03,0x3F,ID_GetCalendarNote },
+#endif
+ {N71_65_ReplyAddCalendar2, "\x13",0x03,0x41,ID_SetCalendarNote },/*method 2*/
+ {N6510_ReplyAddCalendar3, "\x13",0x03,0x66,ID_SetCalendarNote },/*method 3*/
+ {N6510_ReplyAddToDo2, "\x13",0x03,0x66,ID_SetToDo },
+ {N6510_ReplyGetCalendar3, "\x13",0x03,0x7E,ID_GetCalendarNote },/*method 3*/
+ {N6510_ReplyGetToDo2, "\x13",0x03,0x7E,ID_GetToDo },
+ {N6510_ReplyGetCalendarSettings, "\x13",0x03,0x86,ID_GetCalendarSettings },
+ {N6510_ReplyGetLocale, "\x13",0x03,0x8A,ID_GetLocale },
+ {N6510_ReplyGetCalendarSettings, "\x13",0x03,0x8E,ID_GetCalendarSettings },
+ {N6510_ReplyGetCalendarNotePos, "\x13",0x03,0x96,ID_GetCalendarNotePos },/*method 3*/
+ {N6510_ReplyGetToDoFirstLoc2, "\x13",0x03,0x96,ID_SetToDo },
+ {N6510_ReplyGetCalendarInfo, "\x13",0x03,0x9F,ID_GetCalendarNotesInfo},/*method 3*/
+ {N6510_ReplyGetToDoStatus2, "\x13",0x03,0x9F,ID_GetToDo },
+
+ {N6510_ReplySaveSMSMessage, "\x14",0x03,0x01,ID_SaveSMSMessage },
+ {N6510_ReplySetPicture, "\x14",0x03,0x01,ID_SetBitmap },
+ {N6510_ReplyGetSMSMessage, "\x14",0x03,0x03,ID_GetSMSMessage },
+ {N6510_ReplyDeleteSMSMessage, "\x14",0x03,0x05,ID_DeleteSMSMessage },
+ {N6510_ReplyDeleteSMSMessage, "\x14",0x03,0x06,ID_DeleteSMSMessage },
+ {N6510_ReplyGetSMSStatus, "\x14",0x03,0x09,ID_GetSMSStatus },
+ {N6510_ReplyGetSMSFolderStatus, "\x14",0x03,0x0d,ID_GetSMSFolderStatus },
+ {N6510_ReplyGetSMSMessage, "\x14",0x03,0x0f,ID_GetSMSMessage },
+ {N6510_ReplyAddSMSFolder, "\x14",0x03,0x11,ID_AddSMSFolder },
+ {N6510_ReplyGetSMSFolders, "\x14",0x03,0x13,ID_GetSMSFolders },
+ {N6510_ReplySaveSMSMessage, "\x14",0x03,0x17,ID_SaveSMSMessage },
+ {N6510_ReplyGetSMSStatus, "\x14",0x03,0x1a,ID_GetSMSStatus },
+
+ {DCT4_ReplySetPhoneMode, "\x15",0x03,0x64,ID_Reset },
+ {DCT4_ReplyGetPhoneMode, "\x15",0x03,0x65,ID_Reset },
+ {NoneReply, "\x15",0x03,0x68,ID_Reset },
+
+ {N6510_ReplyGetBatteryCharge, "\x17",0x03,0x0B,ID_GetBatteryCharge },
+
+ {N6510_ReplySetDateTime, "\x19",0x03,0x02,ID_SetDateTime },
+ {N6510_ReplyGetDateTime, "\x19",0x03,0x0B,ID_GetDateTime },
+ {N6510_ReplySetAlarm, "\x19",0x03,0x12,ID_SetAlarm },
+ {N6510_ReplyGetAlarm, "\x19",0x03,0x1A,ID_GetAlarm },
+ {N6510_ReplyGetAlarm, "\x19",0x03,0x20,ID_GetAlarm },
+
+ {DCT4_ReplyGetIMEI, "\x1B",0x03,0x01,ID_GetIMEI },
+ {NOKIA_ReplyGetPhoneString, "\x1B",0x03,0x08,ID_GetHardware },
+ {N6510_ReplyGetPPM, "\x1B",0x03,0x08,ID_GetPPM },
+ {NOKIA_ReplyGetPhoneString, "\x1B",0x03,0x0C,ID_GetProductCode },
+
+ /* 0x1C - vibra */
+
+ {N6510_ReplyGetRingtonesInfo, "\x1f",0x03,0x08,ID_GetRingtonesInfo },
+ {N6510_ReplyDeleteRingtones, "\x1f",0x03,0x11,ID_SetRingtone },
+ {N6510_ReplyGetRingtone, "\x1f",0x03,0x13,ID_GetRingtone },
+ {N6510_ReplySetBinRingtone, "\x1f",0x03,0x0F,ID_SetRingtone },
+
+ /* 0x23 - voice records */
+
+ {N6510_ReplyGetProfile, "\x39",0x03,0x02,ID_GetProfile },
+ {N6510_ReplySetProfile, "\x39",0x03,0x04,ID_SetProfile },
+ {N6510_ReplyGetProfile, "\x39",0x03,0x06,ID_GetProfile },
+
+ {N6510_ReplySetLight, "\x3A",0x03,0x06,ID_SetLight },
+
+ {N6510_ReplyGetFMStation, "\x3E",0x03,0x06,ID_GetFMStation },
+ {N6510_ReplyGetFMStatus, "\x3E",0x03,0x0E,ID_GetFMStation },
+ {N6510_ReplySetFMStation, "\x3E",0x03,0x15,ID_SetFMStation },
+ {N6510_ReplyGetFMStation, "\x3E",0x03,0x16,ID_GetFMStation },
+
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x01,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyEnableConnectFunc, "\x3f",0x03,0x02,ID_EnableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x04,ID_DisableConnectFunc },
+ {DCT3DCT4_ReplyDisableConnectFunc,"\x3f",0x03,0x05,ID_DisableConnectFunc },
+ {N6510_ReplyGetWAPBookmark, "\x3f",0x03,0x07,ID_GetWAPBookmark },
+ {N6510_ReplyGetWAPBookmark, "\x3f",0x03,0x08,ID_GetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0A,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplySetWAPBookmark, "\x3f",0x03,0x0B,ID_SetWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0D,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyDelWAPBookmark, "\x3f",0x03,0x0E,ID_DeleteWAPBookmark },
+ {DCT3DCT4_ReplyGetActiveConnectSet,"\x3f",0x03,0x10,ID_GetConnectSet },
+ {DCT3DCT4_ReplySetActiveConnectSet,"\x3f",0x03,0x13,ID_SetConnectSet },
+ {N6510_ReplyGetConnectionSettings,"\x3f",0x03,0x16,ID_GetConnectSet },
+ {N6510_ReplyGetConnectionSettings,"\x3f",0x03,0x17,ID_GetConnectSet },
+ {N6510_ReplySetConnectionSettings,"\x3f",0x03,0x19,ID_SetConnectSet },
+ {N6510_ReplySetConnectionSettings,"\x3f",0x03,0x1A,ID_SetConnectSet },
+ {N6510_ReplySetConnectionSettings,"\x3f",0x03,0x28,ID_SetConnectSet },
+ {N6510_ReplySetConnectionSettings,"\x3f",0x03,0x2B,ID_SetConnectSet },
+ {N6510_ReplyGetChatSettings, "\x3f",0x03,0x3B,ID_GetChatSettings },
+ {N6510_ReplyGetChatSettings, "\x3f",0x03,0x3C,ID_GetChatSettings },
+
+ {N6510_ReplyGetOriginalIMEI, "\x42",0x07,0x00,ID_GetOriginalIMEI },
+ {N6510_ReplyGetManufactureMonth, "\x42",0x07,0x00,ID_GetManufactureMonth },
+ {N6510_ReplyGetOriginalIMEI, "\x42",0x07,0x01,ID_GetOriginalIMEI },
+ {N6510_ReplyGetManufactureMonth, "\x42",0x07,0x02,ID_GetManufactureMonth },
+
+ {N6510_ReplySetOperatorLogo, "\x43",0x03,0x08,ID_SetBitmap },
+ {N6510_ReplyGetGPRSAccessPoint, "\x43",0x03,0x06,ID_GetGPRSPoint },
+ {N6510_ReplySetGPRSAccessPoint1, "\x43",0x03,0x06,ID_SetGPRSPoint },
+#ifdef DEVELOP
+ {N6510_ReplyEnableGPRSAccessPoint,"\x43",0x03,0x06,ID_EnableGPRSPoint },
+#endif
+ {N6510_ReplyGetSyncMLSettings, "\x43",0x03,0x06,ID_GetSyncMLSettings },
+ {N6510_ReplyGetSyncMLName, "\x43",0x03,0x06,ID_GetSyncMLName },
+ {NoneReply, "\x43",0x03,0x08,ID_SetGPRSPoint },
+
+ /* 0x4A - voice records */
+
+ /* 0x53 - simlock */
+
+ {N6510_ReplyAddToDo1, "\x55",0x03,0x02,ID_SetToDo },
+ {N6510_ReplyGetToDo1, "\x55",0x03,0x04,ID_GetToDo },
+ {N6510_ReplyGetToDoFirstLoc1, "\x55",0x03,0x10,ID_SetToDo },
+ {N6510_ReplyDeleteAllToDo1, "\x55",0x03,0x12,ID_DeleteAllToDo },
+ {N6510_ReplyGetToDoStatus1, "\x55",0x03,0x16,ID_GetToDo },
+
+ {N6510_ReplyAddFileHeader, "\x6D",0x03,0x03,ID_AddFile },
+ {N6510_ReplyAddFolder, "\x6D",0x03,0x05,ID_AddFolder },
+ {N6510_ReplyGetFilePart, "\x6D",0x03,0x0F,ID_GetFile },
+ {N6510_ReplyAddFileHeader, "\x6D",0x03,0x13,ID_AddFile },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x15,ID_GetFileInfo },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x15,ID_GetFile },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x15,ID_AddFile },
+ {N6510_ReplyDeleteFile, "\x6D",0x03,0x19,ID_DeleteFile },
+ {N6510_ReplyDeleteFile, "\x6D",0x03,0x1F,ID_DeleteFile },
+ {N6510_ReplyGetFileSystemStatus, "\x6D",0x03,0x23,ID_FileSystemStatus },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x2F,ID_GetFileInfo },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x2F,ID_GetFile },
+ {N6510_ReplyGetFileSystemStatus, "\x6D",0x03,0x2F,ID_FileSystemStatus },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x33,ID_GetFileInfo },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x33,ID_GetFile },
+ {N6510_ReplyAddFilePart, "\x6D",0x03,0x41,ID_AddFile },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x43,ID_AddFile },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x43,ID_GetFile },
+ {N6510_ReplyGetFileFolderInfo, "\x6D",0x03,0x43,ID_GetFileInfo },
+
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x01,ID_GetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x01,ID_SetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x0F,ID_GetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x0F,ID_SetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x10,ID_GetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x10,ID_SetBitmap },
+ {N6510_ReplyStartupNoteLogo, "\x7A",0x04,0x25,ID_SetBitmap },
+
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware, "\xD2",0x02,0x00,ID_GetFirmware },
+
+ /* 0xD7 - Bluetooth */
+
+ {N6510_ReplyGetRingtoneID, "\xDB",0x03,0x02,ID_SetRingtone },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions N6510Phone = {
+ "1100|1100a|1100b|3100|3100b|3108|3200|3200a|3300|3510|3510i|3530|3589i|3590|3595|5100|6100|6200|6220|6230|6310|6310i|6385|6510|6610|6800|7210|7250|7250i|7600|8310|8390|8910|8910i",
+ N6510ReplyFunctions,
+ N6510_Initialise,
+ NONEFUNCTION, /* Terminate */
+ GSM_DispatchMessage,
+ N6510_ShowStartInfo,
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ DCT4_GetIMEI,
+ N6510_GetOriginalIMEI,
+ N6510_GetManufactureMonth,
+ DCT4_GetProductCode,
+ DCT4_GetHardware,
+ N6510_GetPPM,
+ NOTSUPPORTED, /* GetSIMIMSI */
+ N6510_GetDateTime,
+ N6510_SetDateTime,
+ N6510_GetAlarm,
+ N6510_SetAlarm,
+ N6510_GetLocale,
+ NOTSUPPORTED, /* SetLocale */
+ N6510_PressKey,
+ DCT4_Reset,
+ NOTIMPLEMENTED, /* ResetPhoneSettings */
+ N6510_EnterSecurityCode,
+ N6510_GetSecurityStatus,
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTIMPLEMENTED, /* SetAutoNetworkLogin */
+ N6510_GetBatteryCharge,
+ N6510_GetSignalQuality,
+ N6510_GetNetworkInfo,
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ N6510_GetMemoryStatus,
+ N6510_GetMemory,
+ NOTIMPLEMENTED, /* GetNextMemory */
+ N6510_SetMemory,
+ NOTIMPLEMENTED, /* AddMemory */
+ N6510_DeleteMemory,
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ N6510_GetSpeedDial,
+ NOTIMPLEMENTED, /* SetSpeedDial */
+ N6510_GetSMSC,
+ N6510_SetSMSC,
+ N6510_GetSMSStatus,
+ N6510_GetSMSMessage,
+ N6510_GetNextSMSMessage,
+ N6510_SetSMS,
+ N6510_AddSMS,
+ N6510_DeleteSMSMessage,
+ N6510_SendSMSMessage,
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOKIA_SetIncomingSMS,
+ NOTIMPLEMENTED, /* SetIncomingCB */
+ N6510_GetSMSFolders,
+ N6510_AddSMSFolder,
+ NOTIMPLEMENTED, /* DeleteSMSFolder */
+ N6510_DialVoice,
+ N6510_AnswerCall,
+ N6510_CancelCall,
+ NOTIMPLEMENTED, /* HoldCall */
+ NOTIMPLEMENTED, /* UnholdCall */
+ NOTIMPLEMENTED, /* ConferenceCall */
+ NOTIMPLEMENTED, /* SplitCall */
+ NOTIMPLEMENTED, /* TransferCall */
+ NOTIMPLEMENTED, /* SwitchCall */
+ DCT3DCT4_GetCallDivert,
+ DCT3DCT4_SetCallDivert,
+ DCT3DCT4_CancelAllDiverts,
+ NOKIA_SetIncomingCall,
+ NOKIA_SetIncomingUSSD,
+ DCT3DCT4_SendDTMF,
+ N6510_GetRingtone,
+ N6510_SetRingtone,
+ N6510_GetRingtonesInfo,
+ N6510_DeleteUserRingtones,
+ N6510_PlayTone,
+ N6510_GetWAPBookmark,
+ N6510_SetWAPBookmark,
+ N6510_DeleteWAPBookmark,
+ N6510_GetWAPSettings,
+ N6510_SetWAPSettings,
+ N6510_GetMMSSettings,
+ N6510_SetMMSSettings,
+ N6510_GetSyncMLSettings,
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ N6510_GetChatSettings,
+ NOTSUPPORTED, /* SetChatSettings */
+ N6510_GetBitmap,
+ N6510_SetBitmap,
+ N6510_GetToDoStatus,
+ NOTIMPLEMENTED, /* GetToDo */
+ N6510_GetNextToDo,
+ NOTIMPLEMENTED, /* SetToDo */
+ N6510_AddToDo,
+ N6510_DeleteToDo2,
+ N6510_DeleteAllToDo1,
+ N6510_GetCalendarStatus,
+ NOTIMPLEMENTED, /* GetCalendar */
+ N6510_GetNextCalendar,
+ NOTIMPLEMENTED, /* SetCalendar */
+ N6510_AddCalendar,
+ N71_65_DelCalendar,
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ N6510_GetCalendarSettings,
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTIMPLEMENTED, /* GetNote */
+ N6510_GetProfile,
+ N6510_SetProfile,
+ N6510_GetFMStation,
+ N6510_SetFMStation,
+ N6510_ClearFMStations,
+ N6510_GetNextFileFolder,
+ N6510_GetFilePart,
+ N6510_AddFilePart,
+ N6510_GetFileSystemStatus,
+ N6510_DeleteFile,
+ N6510_AddFolder,
+ N6510_GetGPRSAccessPoint,
+ N6510_SetGPRSAccessPoint
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/dct4/n6510.h b/gammu/emb/common/phone/nokia/dct4/n6510.h
new file mode 100644
index 0000000..4717aeb
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/dct4/n6510.h
@@ -0,0 +1,90 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef n6510_h
+#define n6510_h
+
+#include "../ncommon.h"
+#include "../../../service/sms/gsmsms.h"
+
+typedef enum {
+ N6510_MMS_SETTINGS = 0x01,
+ N6510_CHAT_SETTINGS,
+
+ N6510_WAP_SETTINGS,
+ N6510_SYNCML_SETTINGS
+} N6510_Connection_Settings;
+
+typedef enum {
+ N6510_LIGHT_DISPLAY = 0x01,
+ N6510_LIGHT_KEYPAD = 0x03,
+ N6510_LIGHT_TORCH = 0x10
+} N6510_PHONE_LIGHTS;
+
+typedef struct {
+ int LastCalendarYear;
+ int LastCalendarPos;
+ GSM_NOKIACalToDoLocations LastCalendar;
+ int FirstCalendarPos;
+ unsigned char CalendarIcons[10];
+ GSM_CalendarNoteType CalendarIconsTypes[10];
+ int CalendarIconsNum;
+
+ GSM_NOKIASMSFolder LastSMSFolder;
+ GSM_SMSFolders LastSMSFolders;
+
+ GSM_NOKIACalToDoLocations LastToDo;
+
+ unsigned char RingtoneID; /* When set with preview */
+
+ int FilesLocations[1000];
+ int FilesLevels[1000];
+ int FilesLocationsUsed;
+ int FilesLocationsCurrent;
+ int FileToken;
+ int ParentID;
+ int FileCheckSum;
+
+ unsigned char FMStatus[4000];
+ int FMStatusLength;
+
+ unsigned char GPRSPoints[4000];
+ int GPRSPointsLength;
+
+ int BearerNumber;
+
+ unsigned char PhoneMode;
+} GSM_Phone_N6510Data;
+
+void N6510_EncodeFMFrequency(double freq, unsigned char *buff);
+void N6510_DecodeFMFrequency(double *freq, unsigned char *buff);
+
+#ifndef GSM_USED_MBUS2
+# define GSM_USED_MBUS2
+#endif
+#ifndef GSM_USED_FBUS2
+# define GSM_USED_FBUS2
+#endif
+#ifndef GSM_USED_FBUS2DLR3
+# define GSM_USED_FBUS2DLR3
+#endif
+#ifndef GSM_USED_FBUS2DKU5
+# define GSM_USED_FBUS2DKU5
+#endif
+#ifndef GSM_USED_PHONETBLUE
+# define GSM_USED_PHONETBLUE
+#endif
+#ifndef GSM_USED_IRDAPHONET
+# define GSM_USED_IRDAPHONET
+#endif
+#ifndef GSM_USED_BLUEPHONET
+# define GSM_USED_BLUEPHONET
+#endif
+#ifndef GSM_USED_FBUS2PL2303
+# define GSM_USED_FBUS2PL2303
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/nauto.c b/gammu/emb/common/phone/nokia/nauto.c
new file mode 100644
index 0000000..bf74bc9
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/nauto.c
@@ -0,0 +1,144 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "nfunc.h"
+
+#if defined(GSM_ENABLE_NOKIA_DCT3) || defined(GSM_ENABLE_NOKIA_DCT4)
+
+static GSM_Reply_Function NAUTOReplyFunctions[] = {
+ {DCT3DCT4_ReplyGetModelFirmware,"\xD2",0x02,0x00,ID_GetModel },
+ {DCT3DCT4_ReplyGetModelFirmware,"\xD2",0x02,0x00,ID_GetFirmware },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions NAUTOPhone = {
+ "NAUTO",
+ NAUTOReplyFunctions,
+ NONEFUNCTION, /* Initialise */
+ NONEFUNCTION, /* Terminate */
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NOKIA_GetManufacturer,
+ DCT3DCT4_GetModel,
+ DCT3DCT4_GetFirmware,
+ NOTSUPPORTED, /* GetIMEI */
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ NOTSUPPORTED, /* GetProductCode */
+ NOTSUPPORTED, /* GetHardware */
+ NOTSUPPORTED, /* GetPPM */
+ NOTSUPPORTED, /* GetSIMIMSI */
+ NOTSUPPORTED, /* GetDateTime */
+ NOTSUPPORTED, /* SetDateTime */
+ NOTSUPPORTED, /* GetAlarm */
+ NOTSUPPORTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTSUPPORTED, /* PressKey */
+ NOTSUPPORTED, /* Reset */
+ NOTSUPPORTED, /* ResetPhoneSettings */
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTSUPPORTED, /* SetAutoNetworkLogin */
+ NOTSUPPORTED, /* GetBatteryCharge */
+ NOTSUPPORTED, /* GetSignalQuality */
+ NOTSUPPORTED, /* GetNetworkInfo */
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ NOTSUPPORTED, /* GetMemoryStatus */
+ NOTSUPPORTED, /* GetMemory */
+ NOTSUPPORTED, /* GetNextMemory */
+ NOTSUPPORTED, /* SetMemory */
+ NOTSUPPORTED, /* AddMemory */
+ NOTSUPPORTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ NOTSUPPORTED, /* GetSMSC */
+ NOTSUPPORTED, /* SetSMSC */
+ NOTSUPPORTED, /* GetSMSStatus */
+ NOTSUPPORTED, /* GetSMS */
+ NOTSUPPORTED, /* GetNextSMS */
+ NOTSUPPORTED, /* SetSMS */
+ NOTSUPPORTED, /* AddSMS */
+ NOTSUPPORTED, /* DeleteSMS */
+ NOTSUPPORTED, /* SendSMSMessage */
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOTSUPPORTED, /* SetIncomingSMS */
+ NOTSUPPORTED, /* SetIncomingCB */
+ NOTSUPPORTED, /* GetSMSFolders */
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ NOTSUPPORTED, /* DialVoice */
+ NOTSUPPORTED, /* AnswerCall */
+ NOTSUPPORTED, /* CancelCall */
+ NOTSUPPORTED, /* HoldCall */
+ NOTSUPPORTED, /* UnholdCall */
+ NOTSUPPORTED, /* ConferenceCall */
+ NOTSUPPORTED, /* SplitCall */
+ NOTSUPPORTED, /* TransferCall */
+ NOTSUPPORTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NOTSUPPORTED, /* SetIncomingCall */
+ NOTSUPPORTED, /* SetIncomingUSSD */
+ NOTSUPPORTED, /* SendDTMF */
+ NOTSUPPORTED, /* GetRingtone */
+ NOTSUPPORTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetBitmap */
+ NOTSUPPORTED, /* SetBitmap */
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ NOTSUPPORTED, /* GetCalendarStatus */
+ NOTSUPPORTED, /* GetCalendar */
+ NOTSUPPORTED, /* GetNextCalendar */
+ NOTSUPPORTED, /* SetCalendar */
+ NOTSUPPORTED, /* AddCalendar */
+ NOTSUPPORTED, /* DeleteCalendar */
+ NOTSUPPORTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFilePart */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/ncommon.h b/gammu/emb/common/phone/nokia/ncommon.h
new file mode 100644
index 0000000..b8a36aa
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/ncommon.h
@@ -0,0 +1,66 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+
+#ifndef phone_nokia_common_h
+#define phone_nokia_common_h
+
+#include "../../gsmcomon.h"
+
+#define N6110_FRAME_HEADER 0x00, 0x01, 0x00
+#define N7110_FRAME_HEADER 0x00, 0x01, 0x01
+
+typedef enum {
+ MEM7110_CG = 0xf0, /* Caller groups memory */
+ MEM7110_SP = 0xf1 /* Speed dial memory */
+} GSM_71_65MemoryType;
+
+typedef enum {
+ /* DCT3 and DCT4 */
+ N7110_PBK_SIM_SPEEDDIAL = 0x04, /* Speed dial on SIM */
+ N7110_PBK_NAME = 0x07, /* Text: name (always the only one) */
+ N7110_PBK_EMAIL = 0x08, /* Text: email adress */
+ N7110_PBK_POSTAL = 0x09, /* Text: postal address */
+ N7110_PBK_NOTE = 0x0A, /* Text: note */
+ N7110_PBK_NUMBER = 0x0B, /* Phone number */
+ N7110_PBK_RINGTONE_ID = 0x0C, /* Ringtone ID */
+ N7110_PBK_DATETIME = 0x13, /* Call register: date and time */
+ N7110_PBK_UNKNOWN1 = 0x19, /* Call register: with missed calls */
+ N7110_PBK_SPEEDDIAL = 0x1A, /* Speed dial */
+ N7110_PBK_GROUPLOGO = 0x1B, /* Caller group: logo */
+ N7110_PBK_LOGOON = 0x1C, /* Caller group: is logo on ? */
+ N7110_PBK_GROUP = 0x1E, /* Caller group number in pbk entry */
+
+ /* DCT4 only */
+ N6510_PBK_URL = 0x2C, /* Text: URL address */
+ N6510_PBK_SMSLIST_ID = 0x2E, /* SMS list assigment */
+ N6510_PBK_VOICETAG_ID = 0x2F, /* Voice tag assigment */
+ N6510_PBK_PICTURE_ID = 0x33, /* Picture ID assigment */
+ N6510_PBK_RINGTONEFILE_ID= 0x37, /* Ringtone ID from filesystem/internal*/
+ N6510_PBK_USER_ID = 0x38 /* Text: user ID */
+} GSM_71_65_Phonebook_Entries_Types;
+
+typedef enum {
+ N7110_PBK_NUMBER_HOME = 0x02,
+ N7110_PBK_NUMBER_MOBILE = 0x03,
+ N7110_PBK_NUMBER_FAX = 0x04,
+ N7110_PBK_NUMBER_WORK = 0x06,
+ N7110_PBK_NUMBER_GENERAL = 0x0A
+} GSM_71_65_Phonebook_Number_Types;
+
+typedef struct {
+ unsigned char Location[PHONE_MAXSMSINFOLDER]; /* locations of SMS messages in that folder */
+ int Number; /* number of SMS messages in that folder */
+} GSM_NOKIASMSFolder;
+
+typedef struct {
+ int Location[GSM_MAXCALENDARTODONOTES];
+ int Number;
+} GSM_NOKIACalToDoLocations;
+
+#define NOKIA_PRESSPHONEKEY 0x01
+#define NOKIA_RELEASEPHONEKEY 0x02
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/nfunc.c b/gammu/emb/common/phone/nokia/nfunc.c
new file mode 100644
index 0000000..3acfb10
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/nfunc.c
@@ -0,0 +1,2143 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* based on some work from Ralf Thelen, Gabriele Zappi and MyGnokii */
+
+#include <string.h> /* memcpy only */
+#include <stdio.h>
+#include <ctype.h>
+#include <time.h>
+
+#include "../../gsmstate.h"
+#include "../../misc/coding/coding.h"
+#include "../../service/sms/gsmsms.h"
+#include "../pfunc.h"
+#include "nfunc.h"
+
+unsigned char N71_65_MEMORY_TYPES[] = {
+ MEM_DC, 0x01,
+ MEM_MC, 0x02,
+ MEM_RC, 0x03,
+ MEM_ME, 0x05,
+ MEM_SM, 0x06,
+ MEM_VM, 0x09,
+ MEM7110_SP, 0x0e,
+ MEM7110_CG, 0x10,
+ MEM_ON, 0x17,
+ 0x00, 0x00
+};
+
+int N71_65_PackPBKBlock(GSM_StateMachine *s, int id, int size, int no, unsigned char *buf, unsigned char *block)
+{
+ smprintf(s, "Packing phonebook block with ID = %i, block number = %i, block length = %i\n",id,no+1,size+6);
+
+ block[0] = id;
+ block[1] = 0;
+ block[2] = 0;
+ block[3] = size + 6;
+ block[4] = no + 1;
+ memcpy(block+5, buf, size);
+ block[5+size] = 0;
+
+ return (size + 6);
+}
+
+int N71_65_EncodePhonebookFrame(GSM_StateMachine *s, unsigned char *req, GSM_MemoryEntry entry, int *block2, bool DCT4, bool VoiceTag)
+{
+ int count=0, len, i, block=0, j;
+ char string[500];
+ unsigned char type;
+
+ for (i = 0; i < entry.EntriesNum; i++) {
+ type = 0;
+ if (entry.Entries[i].EntryType == PBK_Number_General) type = N7110_PBK_NUMBER_GENERAL;
+ if (entry.Entries[i].EntryType == PBK_Number_Mobile) type = N7110_PBK_NUMBER_MOBILE;
+ if (entry.Entries[i].EntryType == PBK_Number_Work) type = N7110_PBK_NUMBER_WORK;
+ if (entry.Entries[i].EntryType == PBK_Number_Fax) type = N7110_PBK_NUMBER_FAX;
+ if (entry.Entries[i].EntryType == PBK_Number_Home) type = N7110_PBK_NUMBER_HOME;
+ if (type != 0) {
+ string[0] = type;
+ len = UnicodeLength(entry.Entries[i].Text);
+
+ string[1] = 0;
+ string[2] = 0;
+
+ /* DCT 3 */
+ if (!DCT4) string[2] = entry.Entries[i].VoiceTag;
+
+ string[3] = 0;
+ string[4] = len*2+2;
+ CopyUnicodeString(string+5,entry.Entries[i].Text);
+ string[len * 2 + 5] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_NUMBER, len*2+6, block++, string, req+count);
+
+ /* DCT 4 */
+ if (DCT4 && VoiceTag) {
+ block++;
+ req[count++] = N6510_PBK_VOICETAG_ID;
+ req[count++] = 0;
+ req[count++] = 0;
+ req[count++] = 8;
+ req[count++] = 0x00;
+ req[count++] = i+1;
+ req[count++] = 0x00;
+ req[count++] = entry.Entries[i].VoiceTag;
+ }
+ if (DCT4) {
+ j = 0;
+ while (entry.Entries[i].SMSList[j] != 0) {
+ string[0] = i+1;
+ string[1] = 0x00;
+ string[2] = 0x02;
+ string[3] = 0x00;
+ string[4] = entry.Entries[i].SMSList[j];
+ string[5] = 0x00;
+ count += N71_65_PackPBKBlock(s, N6510_PBK_SMSLIST_ID, 6, block++, string, req+count);
+
+ j++;
+ }
+ }
+ continue;
+ }
+ if (entry.Entries[i].EntryType == PBK_Text_Note) type = N7110_PBK_NOTE;
+ if (entry.Entries[i].EntryType == PBK_Text_Postal) type = N7110_PBK_POSTAL;
+ if (entry.Entries[i].EntryType == PBK_Text_Email) type = N7110_PBK_EMAIL;
+ if (entry.Entries[i].EntryType == PBK_Text_Email2) type = N7110_PBK_EMAIL;
+ if (entry.Entries[i].EntryType == PBK_Text_Name) type = N7110_PBK_NAME;
+ if (entry.Entries[i].EntryType == PBK_Text_URL) {
+ type = N7110_PBK_NOTE;
+ if (DCT4) type = N6510_PBK_URL;
+ }
+ if (type != 0) {
+ len = UnicodeLength(entry.Entries[i].Text);
+ string[0] = len*2+2;
+ CopyUnicodeString(string+1,entry.Entries[i].Text);
+ string[len*2+1] = 0;
+ count += N71_65_PackPBKBlock(s, type, len * 2 + 2, block++, string, req + count);
+ continue;
+
+ }
+ if (entry.Entries[i].EntryType == PBK_Caller_Group) {
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBK35)) {
+ string[0] = entry.Entries[i].Number;
+ string[1] = 0;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_GROUP, 2, block++, string, req + count);
+ }
+ continue;
+ }
+ if (entry.Entries[i].EntryType == PBK_RingtoneID) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBK35)) {
+ string[0] = 0x00;
+ string[1] = 0x00;
+ string[2] = entry.Entries[i].Number;
+ count += N71_65_PackPBKBlock(s, N7110_PBK_RINGTONE_ID, 3, block++, string, req + count);
+ count --;
+ req[count-5] = 8;
+ }
+ continue;
+ }
+ if (entry.Entries[i].EntryType == PBK_PictureID) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBKIMG)) {
+ string[0] = 0x00;
+ string[1] = 0x00;
+ string[2] = 0x00;
+ string[3] = 0x00;
+ string[4] = 0x01;
+ string[5] = entry.Entries[i].Number / 256;
+ string[6] = entry.Entries[i].Number % 256;
+ string[7] = 0x00;
+ string[8] = 0x00;
+ string[9] = 0x00;
+ count += N71_65_PackPBKBlock(s, N6510_PBK_PICTURE_ID, 10, block++, string, req + count);
+ req[count-1] = 0x01;
+ }
+ continue;
+ }
+ if (entry.Entries[i].EntryType == PBK_Text_UserID) {
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_PBKUSER)) {
+ string[0] = UnicodeLength(entry.Entries[i].Text)*2;
+ CopyUnicodeString(string+1,entry.Entries[i].Text);
+ count += N71_65_PackPBKBlock(s, N6510_PBK_USER_ID, string[0]+2, block++, string, req+count);
+ req[count-1]--;
+ }
+ continue;
+ }
+ }
+
+ *block2=block;
+
+ return count;
+}
+
+GSM_Error N71_65_DecodePhonebook(GSM_StateMachine *s,
+ GSM_MemoryEntry *entry,
+ GSM_Bitmap *bitmap,
+ GSM_SpeedDial *speed,
+ unsigned char *MessageBuffer,
+ int MessageLength,
+ bool DayMonthReverse)
+{
+ unsigned char *Block;
+ int length = 0, i;
+ GSM_71_65_Phonebook_Entries_Types Type;
+
+ entry->EntriesNum = 0;
+
+ if (entry->MemoryType==MEM7110_CG) {
+ bitmap->Text[0] = 0x00;
+ bitmap->Text[1] = 0x00;
+ bitmap->DefaultBitmap = true;
+ bitmap->DefaultRingtone = true;
+ }
+
+ Block = &MessageBuffer[0];
+ while (length != MessageLength) {
+#ifdef DEBUG
+ smprintf(s, "Phonebook entry block - length %i", Block[3]-6);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, Block+5, Block[3]-6);
+#endif
+ if (entry->EntriesNum==GSM_PHONEBOOK_ENTRIES) {
+ smprintf(s, "Too many entries\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ Type = 0;
+ if (Block[0] == N7110_PBK_NAME) {
+ Type = PBK_Text_Name; smprintf(s,"Name ");
+ }
+ if (Block[0] == N7110_PBK_EMAIL) {
+ Type = PBK_Text_Email; smprintf(s,"Email ");
+ }
+ if (Block[0] == N7110_PBK_POSTAL) {
+ Type = PBK_Text_Postal; smprintf(s,"Postal ");
+ }
+ if (Block[0] == N7110_PBK_NOTE) {
+ Type = PBK_Text_Note; smprintf(s,"Text note ");
+ }
+ if (Block[0] == N6510_PBK_URL) {
+ Type = PBK_Text_URL; smprintf(s,"URL ");
+ }
+ if (Type != 0) {
+ if (Block[5]/2>GSM_PHONEBOOK_TEXT_LENGTH) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ memcpy(entry->Entries[entry->EntriesNum].Text,Block+6,Block[5]);
+ entry->Entries[entry->EntriesNum].EntryType=Type;
+ smprintf(s, " \"%s\"\n",DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ if (Block[0] == N7110_PBK_NAME) {
+ if (entry->MemoryType==MEM7110_CG) {
+ memcpy(bitmap->Text,Block+6,Block[5]);
+ }
+ }
+ entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+
+ if (Block[0] == N7110_PBK_DATETIME) {
+ entry->Entries[entry->EntriesNum].EntryType=PBK_Date;
+ NOKIA_DecodeDateTime(s, Block+6, &entry->Entries[entry->EntriesNum].Date);
+ if (DayMonthReverse) {
+ i = entry->Entries[entry->EntriesNum].Date.Month;
+ entry->Entries[entry->EntriesNum].Date.Month = entry->Entries[entry->EntriesNum].Date.Day;
+ entry->Entries[entry->EntriesNum].Date.Day = i;
+ }
+ entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N6510_PBK_PICTURE_ID) {
+ entry->Entries[entry->EntriesNum].EntryType=PBK_PictureID;
+ smprintf(s, "Picture ID \"%i\"\n",Block[10]*256+Block[11]);
+ entry->Entries[entry->EntriesNum].Number=Block[10]*256+Block[11];
+ entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+
+ if (Block[0] == N7110_PBK_NUMBER) {
+ if (Block[5] == 0x00) {
+ Type = PBK_Number_General; smprintf(s,"General number ");
+ }
+ /* Not assigned dialed number */
+ if (Block[5] == 0x01) {
+ Type = PBK_Number_General; smprintf(s,"General number ");
+ }
+ if (Block[5] == 0x0B) {
+ Type = PBK_Number_General; smprintf(s,"General number ");
+ }
+ /* In many firmwares 0x55 visible after using
+ * Save from Call Register menu and saving number
+ * to existing phonebook entry */
+ if (Block[5] == 0x55) {
+ Type = PBK_Number_General; smprintf(s,"General number ");
+ }
+ if (Block[5] == N7110_PBK_NUMBER_GENERAL) {
+ Type = PBK_Number_General; smprintf(s,"General number ");
+ }
+ if (Block[5] == N7110_PBK_NUMBER_WORK) {
+ Type = PBK_Number_Work; smprintf(s,"Work number ");
+ }
+ if (Block[5] == N7110_PBK_NUMBER_FAX) {
+ Type = PBK_Number_Fax; smprintf(s,"Fax number ");
+ }
+ if (Block[5] == N7110_PBK_NUMBER_MOBILE) {
+ Type = PBK_Number_Mobile; smprintf(s,"Mobile number ");
+ }
+ if (Block[5] == N7110_PBK_NUMBER_HOME) {
+ Type = PBK_Number_Home; smprintf(s,"Home number ");
+ }
+ if (Type == 0x00) {
+ smprintf(s, "Unknown number type %02x\n",Block[5]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ entry->Entries[entry->EntriesNum].EntryType=Type;
+ if (Block[9]/2>GSM_PHONEBOOK_TEXT_LENGTH) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ memcpy(entry->Entries[entry->EntriesNum].Text,Block+10,Block[9]);
+ smprintf(s, " \"%s\"\n",DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ /* DCT3 phones like 6210 */
+ entry->Entries[entry->EntriesNum].VoiceTag = Block[7];
+#ifdef DEBUG
+ if (entry->Entries[entry->EntriesNum].VoiceTag != 0) smprintf(s, "Voice tag %i assigned\n",Block[7]);
+#endif
+ entry->Entries[entry->EntriesNum].SMSList[0] = 0;
+ entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N7110_PBK_RINGTONE_ID) {
+ if (entry->MemoryType==MEM7110_CG) {
+ bitmap->RingtoneID=Block[5];
+ if (Block[5] == 0x00) bitmap->RingtoneID=Block[7];
+ smprintf(s, "Ringtone ID : %i\n",bitmap->RingtoneID);
+ bitmap->DefaultRingtone = false;
+ bitmap->FileSystemRingtone = false;
+ } else {
+ entry->Entries[entry->EntriesNum].EntryType=PBK_RingtoneID;
+ smprintf(s, "Ringtone ID \"%i\"\n",Block[7]);
+ entry->Entries[entry->EntriesNum].Number=Block[7];
+ entry->EntriesNum ++;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N7110_PBK_LOGOON) {
+ if (entry->MemoryType==MEM7110_CG) {
+ bitmap->BitmapEnabled=(Block[5]==0x00 ? false : true);
+ smprintf(s, "Logo : %s\n", bitmap->BitmapEnabled==true ? "enabled":"disabled");
+ } else {
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N7110_PBK_GROUPLOGO) {
+ if (entry->MemoryType==MEM7110_CG) {
+ smprintf(s, "Caller logo\n");
+ PHONE_DecodeBitmap(GSM_NokiaCallerLogo, Block+10, bitmap);
+ bitmap->DefaultBitmap = false;
+ } else {
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N7110_PBK_GROUP) {
+ entry->Entries[entry->EntriesNum].EntryType=PBK_Caller_Group;
+ smprintf(s, "Caller group \"%i\"\n",Block[5]);
+ entry->Entries[entry->EntriesNum].Number=Block[5];
+ if (Block[5]!=0) entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N6510_PBK_VOICETAG_ID) {
+ smprintf(s, "Entry %i has voice tag %i\n",Block[5]-1,Block[7]);
+ entry->Entries[Block[5]-1].VoiceTag = Block[7];
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+
+ /* 6210 5.56, SIM speed dials or ME with 1 number */
+ if (Block[0] == N7110_PBK_SIM_SPEEDDIAL) {
+ if (entry->MemoryType==MEM7110_SP) {
+#ifdef DEBUG
+ smprintf(s, "location %i\n",(Block[6]*256+Block[7]));
+#endif
+ speed->MemoryType = MEM_ME;
+ if (Block[8] == 0x06) speed->MemoryType = MEM_SM;
+ speed->MemoryLocation = (Block[6]*256+Block[7]);
+ speed->MemoryNumberID = 2;
+ } else {
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+
+ if (Block[0] == N7110_PBK_SPEEDDIAL) {
+ if (entry->MemoryType==MEM7110_SP) {
+#ifdef DEBUG
+ switch (Block[12]) {
+ case 0x05: smprintf(s, "ME\n"); break;
+ case 0x06: smprintf(s, "SM\n"); break;
+ default : smprintf(s, "%02x\n",Block[12]);
+ }
+ smprintf(s, "location %i, number %i in location\n",
+ (Block[6]*256+Block[7])-1,Block[14]);
+#endif
+ switch (Block[12]) {
+ case 0x05: speed->MemoryType = MEM_ME; break;
+ case 0x06: speed->MemoryType = MEM_SM; break;
+ }
+ speed->MemoryLocation = (Block[6]*256+Block[7])-1;
+ speed->MemoryNumberID = Block[14];
+ } else {
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N6510_PBK_RINGTONEFILE_ID) {
+ smprintf(s, "Ringtone ID with possibility of using filesystem\n");
+ if (entry->MemoryType==MEM7110_CG) {
+ if (Block[9] == 0x01) {
+ smprintf(s, "Filesystem ringtone ID: %02x\n",Block[10]*256+Block[11]);
+ bitmap->FileSystemRingtone = true;
+ } else {
+ smprintf(s, "Internal ringtone ID: %02x\n",Block[10]*256+Block[11]);
+ bitmap->FileSystemRingtone = false;
+ }
+ bitmap->RingtoneID = Block[10]*256+Block[11];
+ bitmap->DefaultRingtone = false;
+ } else {
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N6510_PBK_SMSLIST_ID) {
+ smprintf(s, "Entry %i is assigned to SMS list %i\n",Block[5]-1,Block[9]);
+ i = 0;
+ while(entry->Entries[Block[5]-1].SMSList[i] != 0) i++;
+ entry->Entries[Block[5]-1].SMSList[i+1] = 0;
+ entry->Entries[Block[5]-1].SMSList[i] = Block[9];
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N6510_PBK_USER_ID) {
+ smprintf(s, "User ID:");
+ entry->Entries[entry->EntriesNum].EntryType=PBK_Text_UserID;
+ if (Block[5]/2>GSM_PHONEBOOK_TEXT_LENGTH) {
+ smprintf(s, "Too long text\n");
+ return ERR_UNKNOWNRESPONSE;
+ }
+ memcpy(entry->Entries[entry->EntriesNum].Text,Block+6,Block[5]);
+ smprintf(s, " \"%s\"\n",DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum ++;
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+ if (Block[0] == N7110_PBK_UNKNOWN1) {
+ smprintf(s,"Unknown entry\n");
+
+ length = length + Block[3];
+ Block = &Block[(int) Block[3]];
+ continue;
+ }
+
+ smprintf(s, "ERROR: unknown pbk entry 0x%02x\n",Block[0]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+
+ if (entry->EntriesNum == 0) return ERR_EMPTY;
+
+ return ERR_NONE;
+}
+
+void NOKIA_GetDefaultCallerGroupName(GSM_StateMachine *s, GSM_Bitmap *Bitmap)
+{
+ Bitmap->DefaultName = false;
+ if (Bitmap->Text[0]==0x00 && Bitmap->Text[1]==0x00) {
+ Bitmap->DefaultName = true;
+ switch(Bitmap->Location) {
+ case 1: EncodeUnicode(Bitmap->Text,GetMsg(s->msg,"Family"),strlen(GetMsg(s->msg,"Family")));
+ break;
+ case 2: EncodeUnicode(Bitmap->Text,GetMsg(s->msg,"VIP"),strlen(GetMsg(s->msg,"VIP")));
+ break;
+ case 3: EncodeUnicode(Bitmap->Text,GetMsg(s->msg,"Friends"),strlen(GetMsg(s->msg,"Friends")));
+ break;
+ case 4: EncodeUnicode(Bitmap->Text,GetMsg(s->msg,"Colleagues"),strlen(GetMsg(s->msg,"Colleagues")));
+ break;
+ case 5: EncodeUnicode(Bitmap->Text,GetMsg(s->msg,"Other"),strlen(GetMsg(s->msg,"Other")));
+ break;
+ }
+ }
+}
+
+void NOKIA_DecodeDateTime(GSM_StateMachine *s, unsigned char* buffer, GSM_DateTime *datetime)
+{
+ datetime->Year = buffer[0] * 256 + buffer[1];
+ datetime->Month = buffer[2];
+ datetime->Day = buffer[3];
+
+ datetime->Hour = buffer[4];
+ datetime->Minute = buffer[5];
+ datetime->Second = buffer[6];
+
+ smprintf(s, "Decoding date and time\n");
+ smprintf(s, " Time: %02d:%02d:%02d\n",
+ datetime->Hour, datetime->Minute, datetime->Second);
+ smprintf(s, " Date: %4d/%02d/%02d\n",
+ datetime->Year, datetime->Month, datetime->Day);
+}
+
+#if defined(GSM_ENABLE_NOKIA_DCT3) || defined(GSM_ENABLE_NOKIA_DCT4)
+
+/* --------------------- Some general Nokia functions ---------------------- */
+
+void NOKIA_DecodeSMSState(GSM_StateMachine *s, unsigned char state, GSM_SMSMessage *sms)
+{
+ switch (state) {
+ case 0x01 : sms->State = SMS_Read; break;
+ case 0x03 : sms->State = SMS_UnRead; break;
+ case 0x05 : sms->State = SMS_Sent; break;
+ case 0x07 : sms->State = SMS_UnSent; break;
+ default : smprintf(s, "Unknown SMS state: %02x\n",state);
+ }
+}
+
+GSM_Error NOKIA_ReplyGetPhoneString(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ strcpy(s->Phone.Data.PhoneString, msg.Buffer+s->Phone.Data.StartPhoneString);
+ return ERR_NONE;
+}
+
+/* Some strings are very easy. Some header, after it required string and 0x00.
+ * We can get them using this function. We give frame to send (*string),
+ * type of message (type), pointer for buffer for response (*value), request
+ * type (request) and what is start byte in response for our string
+ */
+GSM_Error NOKIA_GetPhoneString(GSM_StateMachine *s, unsigned char *msgframe, int msglen, unsigned char msgtype, char *retvalue, GSM_Phone_RequestID request, int startresponse)
+{
+ retvalue[0] = 0;
+ s->Phone.Data.StartPhoneString = startresponse;
+ s->Phone.Data.PhoneString = retvalue;
+ return GSM_WaitFor (s, msgframe, msglen,msgtype, 4, request);
+}
+
+GSM_Error NOKIA_GetManufacturer(GSM_StateMachine *s)
+{
+ strcpy(s->Phone.Data.Manufacturer,"Nokia");
+ return ERR_NONE;
+}
+
+/* Many functions contains such strings:
+ * (1. length/256) - exist or not
+ * 2. length%256
+ * 3. string (unicode, no termination)
+ * This function read string to output and increases counter
+ */
+void NOKIA_GetUnicodeString(GSM_StateMachine *s, int *current, unsigned char *input, unsigned char *output, bool FullLength)
+{
+ int length;
+
+ if (FullLength) {
+ length = (input[*current]*256+input[*current+1])*2;
+ memcpy(output,input+(*current+2),length);
+ *current = *current + 2 + length;
+ } else {
+ length = (input[*current])*2;
+ memcpy(output,input+(*current+1),length);
+ *current = *current + 1 + length;
+ }
+
+ output[length ] = 0;
+ output[length+1] = 0;
+}
+
+int NOKIA_SetUnicodeString(GSM_StateMachine *s, unsigned char *dest, unsigned char *string, bool FullLength)
+{
+ int length;
+
+ length = UnicodeLength(string);
+ if (FullLength) {
+ dest[0] = length / 256;
+ dest[1] = length % 256;
+ CopyUnicodeString(dest + 2, string);
+ return 2+length*2;
+ } else {
+ dest[0] = length % 256;
+ CopyUnicodeString(dest + 1, string);
+ return 1+length*2;
+ }
+}
+
+/* Returns correct ID for concrete memory type */
+GSM_MemoryType NOKIA_GetMemoryType(GSM_StateMachine *s, GSM_MemoryType memory_type, unsigned char *ID)
+{
+ int i=0;
+
+ while (ID[i+1]!=0x00) {
+ if (ID[i]==memory_type) return ID[i+1];
+ i=i+2;
+ }
+ return 0xff;
+}
+
+void NOKIA_EncodeDateTime(GSM_StateMachine *s, unsigned char* buffer, GSM_DateTime *datetime)
+{
+ buffer[0] = datetime->Year / 256;
+ buffer[1] = datetime->Year % 256;
+ buffer[2] = datetime->Month;
+ buffer[3] = datetime->Day;
+
+ buffer[4] = datetime->Hour;
+ buffer[5] = datetime->Minute;
+}
+
+void NOKIA_SortSMSFolderStatus(GSM_StateMachine *s, GSM_NOKIASMSFolder *Folder)
+{
+ int i,j;
+
+ if (Folder->Number!=0) {
+ /* Bouble sorting */
+ i=0;
+ while (i!=Folder->Number-1) {
+ if (Folder->Location[i]>Folder->Location[i+1]) {
+ j=Folder->Location[i];
+ Folder->Location[i]=Folder->Location[i+1];
+ Folder->Location[i+1]=j;
+ i=0;
+ } else {
+ i++;
+ }
+ }
+#ifdef DEBUG
+ smprintf(s, "Locations: ");
+ for (i=0;i<Folder->Number;i++) {
+ smprintf(s, "%i ",Folder->Location[i]);
+ }
+ smprintf(s, "\n");
+#endif
+ }
+}
+
+void NOKIA_GetDefaultProfileName(GSM_StateMachine *s, GSM_Profile *Profile)
+{
+ if (Profile->DefaultName) {
+ switch(Profile->Location) {
+ case 1: EncodeUnicode(Profile->Name,GetMsg(s->msg,"General"),strlen(GetMsg(s->msg,"General")));
+ break;
+ case 2: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Silent"),strlen(GetMsg(s->msg,"Silent")));
+ break;
+ case 3: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Meeting"),strlen(GetMsg(s->msg,"Meeting")));
+ break;
+ case 4: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Outdoor"),strlen(GetMsg(s->msg,"Outdoor")));
+ break;
+ case 5: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Pager"),strlen(GetMsg(s->msg,"Pager")));
+ break;
+ case 6: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Car"),strlen(GetMsg(s->msg,"Car")));
+ break;
+ case 7: EncodeUnicode(Profile->Name,GetMsg(s->msg,"Headset"),strlen(GetMsg(s->msg,"Headset")));
+ break;
+ }
+ }
+}
+
+/* - Shared for DCT3 (n6110.c, n7110.c, n9110.c) and DCT4 (n6510.c) phones - */
+
+GSM_Error DCT3DCT4_ReplyCallDivert(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_MultiCallDivert *cd = s->Phone.Data.Divert;
+ int i,pos = 11,j;
+
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s,"Message: Call divert status received\n");
+ smprintf(s," Divert type: ");
+ switch (msg.Buffer[6]) {
+ case 0x43: smprintf(s,"when busy"); break;
+ case 0x3d: smprintf(s,"when not answered"); break;
+ case 0x3e: smprintf(s,"when phone off or no coverage"); break;
+ case 0x15: smprintf(s,"all types of diverts"); break;
+ default: smprintf(s,"unknown %i",msg.Buffer[6]); break;
+ }
+ /* 6150 */
+ if (msg.Length == 0x0b) {
+ cd->Response.EntriesNum = 0;
+ return ERR_NONE;
+ }
+ cd->Response.EntriesNum = msg.Buffer[10];
+ for (i=0;i<cd->Response.EntriesNum;i++) {
+ smprintf(s,"\n Calls type : ");
+ switch (msg.Buffer[pos]) {
+ case 0x0b:
+ smprintf(s,"voice");
+ cd->Response.Entries[i].CallType = GSM_DIVERT_VoiceCalls;
+ break;
+ case 0x0d:
+ smprintf(s,"fax");
+ cd->Response.Entries[i].CallType = GSM_DIVERT_FaxCalls;
+ break;
+ case 0x19:
+ smprintf(s,"data");
+ cd->Response.Entries[i].CallType = GSM_DIVERT_DataCalls;
+ break;
+ default:
+ smprintf(s,"unknown %i",msg.Buffer[pos]);
+ /* 6310i */
+ cd->Response.EntriesNum = 0;
+ return ERR_NONE;
+ break;
+ }
+ smprintf(s,"\n");
+ j = pos + 2;
+ while (msg.Buffer[j] != 0x00) j++;
+ msg.Buffer[pos+1] = j - pos - 2;
+ GSM_UnpackSemiOctetNumber(cd->Response.Entries[i].Number,msg.Buffer+(pos+1),false);
+ smprintf(s," Number : %s\n",DecodeUnicodeString(cd->Response.Entries[i].Number));
+ cd->Response.Entries[i].Timeout = msg.Buffer[pos+34];
+ smprintf(s," Timeout : %i seconds\n",msg.Buffer[pos+34]);
+ pos+=35;
+ }
+ return ERR_NONE;
+ case 0x03:
+ smprintf(s,"Message: Call divert status receiving error ?\n");
+ return ERR_UNKNOWN;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error DCT3DCT4_CallDivert(GSM_StateMachine *s, GSM_MultiCallDivert *divert, bool get)
+{
+ int length = 0x09;
+ unsigned char req[55] = {N6110_FRAME_HEADER, 0x01,
+ 0x05, /* operation = Query */
+ 0x00,
+ 0x00, /* divert type */
+ 0x00, /* call type */
+ 0x00};
+
+ if (!get) {
+ if (UnicodeLength(divert->Request.Number) == 0) {
+ req[4] = 0x04;
+ } else {
+ req[4] = 0x03;
+ req[8] = 0x01;
+ req[29] = GSM_PackSemiOctetNumber(divert->Request.Number, req + 9, false);
+ req[52] = divert->Request.Timeout;
+ length = 55;
+ }
+ }
+ switch (divert->Request.DivertType) {
+ case GSM_DIVERT_AllTypes : req[6] = 0x15; break;
+ case GSM_DIVERT_Busy : req[6] = 0x43; break;
+ case GSM_DIVERT_NoAnswer : req[6] = 0x3d; break;
+ case GSM_DIVERT_OutOfReach: req[6] = 0x3e; break;
+ default : return ERR_NOTIMPLEMENTED;
+ }
+
+ switch (divert->Request.CallType) {
+ case GSM_DIVERT_AllCalls : break;
+ case GSM_DIVERT_VoiceCalls: req[7] = 0x0b; break;
+ case GSM_DIVERT_FaxCalls : req[7] = 0x0d; break;
+ case GSM_DIVERT_DataCalls : req[7] = 0x19; break;
+ default : return ERR_NOTIMPLEMENTED;
+ }
+
+ s->Phone.Data.Divert = divert;
+ smprintf(s, "Call divert\n");
+ return GSM_WaitFor (s, req, length, 0x06, 10, ID_Divert);
+}
+
+GSM_Error DCT3DCT4_GetCallDivert(GSM_StateMachine *s, GSM_MultiCallDivert *divert)
+{
+ return DCT3DCT4_CallDivert(s,divert,true);
+}
+
+GSM_Error DCT3DCT4_SetCallDivert(GSM_StateMachine *s, GSM_MultiCallDivert *divert)
+{
+ return DCT3DCT4_CallDivert(s,divert,false);
+}
+
+GSM_Error DCT3DCT4_CancelAllDiverts(GSM_StateMachine *s)
+{
+ GSM_MultiCallDivert divert;
+ unsigned char req[55] = {N6110_FRAME_HEADER, 0x01,
+ 0x04, /* operation = Disable */
+ 0x00,
+ 0x02, /* divert type */
+ 0x00, /* call type */
+ 0x00};
+
+ s->Phone.Data.Divert = &divert;
+ smprintf(s, "Call divert\n");
+ return GSM_WaitFor (s, req, 0x09, 0x06, 10, ID_Divert);
+}
+
+GSM_Error DCT3DCT4_ReplyGetActiveConnectSet(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ Data->WAPSettings->Active = false;
+ if (Data->WAPSettings->Location - 1 == msg.Buffer[4]) {
+ Data->WAPSettings->Active = true;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error DCT3DCT4_GetActiveConnectSet(GSM_StateMachine *s)
+{
+ unsigned char GetSetreq[] = {N6110_FRAME_HEADER, 0x0F};
+
+ smprintf(s, "Checking, if connection settings are active\n");
+ return GSM_WaitFor (s, GetSetreq, 4, 0x3f, 4, ID_GetConnectSet);
+}
+
+GSM_Error DCT3DCT4_ReplySetActiveConnectSet(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Connection settings activated\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3DCT4_SetActiveConnectSet(GSM_StateMachine *s, GSM_MultiWAPSettings *settings)
+{
+ unsigned char reqActivate[] = {N6110_FRAME_HEADER, 0x12,
+ 0x00}; /* Location */
+
+ if (settings->Active) {
+ reqActivate[4] = settings->Location-1;
+ smprintf(s, "Activating connection settings number %i\n",settings->Location);
+ return GSM_WaitFor (s, reqActivate, 5, 0x3f, 4, ID_SetMMSSettings);
+ }
+ return ERR_NONE;
+}
+
+
+GSM_Error DCT3DCT4_SendDTMF(GSM_StateMachine *s, char *DTMFSequence)
+{
+ unsigned char req[100] = {N6110_FRAME_HEADER, 0x50,
+ 0x00}; /* Length */
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_NODTMF)) return ERR_NOTSUPPORTED;
+ if (strlen(DTMFSequence) > 100 - 5) return ERR_NOTSUPPORTED;
+
+ req[4] = strlen(DTMFSequence);
+
+ memcpy(req+5,DTMFSequence,strlen(DTMFSequence));
+
+ smprintf(s, "Sending DTMF\n");
+ return GSM_WaitFor (s, req, 5+strlen(DTMFSequence), 0x01, 4, ID_SendDTMF);
+}
+
+GSM_Error DCT3DCT4_ReplyGetWAPBookmark(GSM_Protocol_Message msg, GSM_StateMachine *s, bool FullLength)
+{
+ int tmp;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ smprintf(s, "WAP bookmark received\n");
+ switch (msg.Buffer[3]) {
+ case 0x07:
+ tmp = 4;
+
+ Data->WAPBookmark->Location = msg.Buffer[tmp] * 256 + msg.Buffer[tmp+1];
+ smprintf(s, "Location: %i\n",Data->WAPBookmark->Location);
+ tmp = tmp + 2;
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPBookmark->Title, FullLength);
+ smprintf(s, "Title : \"%s\"\n",DecodeUnicodeString(Data->WAPBookmark->Title));
+
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer, Data->WAPBookmark->Address, FullLength);
+ smprintf(s, "Address : \"%s\"\n",DecodeUnicodeString(Data->WAPBookmark->Address));
+
+ return ERR_NONE;
+ case 0x08:
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside WAP bookmarks menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x02:
+ smprintf(s, "Invalid or empty\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ break;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3DCT4_ReplySetWAPBookmark(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x0A:
+ smprintf(s, "WAP bookmark set OK\n");
+ return ERR_NONE;
+ case 0x0B:
+ smprintf(s, "WAP bookmark setting error\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside WAP bookmarks menu\n");
+ return ERR_INSIDEPHONEMENU;
+ case 0x02:
+ smprintf(s, "Can't write to empty location ?\n");
+ return ERR_EMPTY;
+ case 0x04:
+ smprintf(s, "Full memory\n");
+ return ERR_FULL;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3DCT4_ReplyEnableConnectFunc(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Connection functions enabled\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3DCT4_EnableWAPFunctions(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x00};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOWAP)) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Enabling WAP\n");
+ return GSM_WaitFor (s, req, 4, 0x3f, 4, ID_EnableConnectFunc);
+}
+
+GSM_Error DCT3DCT4_ReplyDisableConnectFunc(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Connection functions disabled\n");
+ return ERR_NONE;
+}
+
+GSM_Error DCT3DCT4_DisableConnectionFunctions(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x03};
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOWAP)) return ERR_NOTSUPPORTED;
+
+ smprintf(s, "Disabling connection settings\n");
+ return GSM_WaitFor (s, req, 4, 0x3f, 4, ID_DisableConnectFunc);
+}
+
+GSM_Error DCT3DCT4_ReplyDelWAPBookmark(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x0D:
+ smprintf(s, "WAP bookmark deleted OK\n");
+ return ERR_NONE;
+ case 0x0E:
+ smprintf(s, "WAP bookmark deleting error\n");
+ switch (msg.Buffer[4]) {
+ case 0x01:
+ smprintf(s, "Security error. Inside WAP bookmarks menu\n");
+ return ERR_SECURITYERROR;
+ case 0x02:
+ smprintf(s, "Invalid location\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[4]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error DCT3DCT4_DeleteWAPBookmarkPart(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0C,
+ 0x00, 0x00}; /* Location */
+
+ req[5] = bookmark->Location;
+
+ smprintf(s, "Deleting WAP bookmark\n");
+ error = GSM_WaitFor (s, req, 6, 0x3f, 4, ID_DeleteWAPBookmark);
+ if (error != ERR_NONE) {
+ if (error == ERR_INVALIDLOCATION || error == ERR_INSIDEPHONEMENU) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error DCT3DCT4_GetWAPBookmarkPart(GSM_StateMachine *s, GSM_WAPBookmark *bookmark)
+{
+ GSM_Error error;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x06,
+ 0x00, 0x00}; /* Location */
+
+ req[5]=bookmark->Location-1;
+
+ s->Phone.Data.WAPBookmark=bookmark;
+ smprintf(s, "Getting WAP bookmark\n");
+ error = GSM_WaitFor (s, req, 6, 0x3f, 4, ID_GetWAPBookmark);
+ if (error != ERR_NONE) {
+ if (error == ERR_INVALIDLOCATION || error == ERR_INSIDEPHONEMENU) {
+ DCT3DCT4_DisableConnectionFunctions(s);
+ }
+ return error;
+ }
+
+ return DCT3DCT4_DisableConnectionFunctions(s);
+}
+
+GSM_Error DCT3DCT4_CancelCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x08, 0x00, 0x85};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Canceling single call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_CancelCall);
+}
+
+GSM_Error DCT3DCT4_AnswerCall(GSM_StateMachine *s, int ID)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x06, 0x00, 0x00};
+
+ req[4] = (unsigned char)ID;
+ s->Phone.Data.CallID = ID;
+
+ smprintf(s, "Answering single call\n");
+ return GSM_WaitFor (s, req, 6, 0x01, 4, ID_AnswerCall);
+}
+
+GSM_Error DCT3DCT4_ReplyGetModelFirmware(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Lines lines;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+
+ SplitLines(msg.Buffer, msg.Length, &lines, "\x20\x0A", 2, false);
+
+ strcpy(Data->Model,GetLineString(msg.Buffer, lines, 4));
+ smprintf(s, "Received model %s\n",Data->Model);
+ Data->ModelInfo = GetModelData(NULL,Data->Model,NULL);
+
+ strcpy(Data->VerDate,GetLineString(msg.Buffer, lines, 3));
+ smprintf(s, "Received firmware date %s\n",Data->VerDate);
+
+ strcpy(Data->Version,GetLineString(msg.Buffer, lines, 2));
+ smprintf(s, "Received firmware version %s\n",Data->Version);
+ GSM_CreateFirmwareNumber(s);
+
+ return ERR_NONE;
+}
+
+GSM_Error DCT3DCT4_GetModel (GSM_StateMachine *s)
+{
+ unsigned char req[5] = {N6110_FRAME_HEADER, 0x03, 0x00};
+ GSM_Error error;
+
+ if (strlen(s->Phone.Data.Model)>0) return ERR_NONE;
+
+ smprintf(s, "Getting model\n");
+ error=GSM_WaitFor (s, req, 5, 0xd1, 3, ID_GetModel);
+ if (error==ERR_NONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "[Connected model - \"%s\"]\n",s->Phone.Data.Model);
+ smprintf(s, "[Firmware version - \"%s\"]\n",s->Phone.Data.Version);
+ smprintf(s, "[Firmware date - \"%s\"]\n",s->Phone.Data.VerDate);
+ }
+ }
+ return error;
+}
+
+GSM_Error DCT3DCT4_GetFirmware (GSM_StateMachine *s)
+{
+ unsigned char req[5] = {N6110_FRAME_HEADER, 0x03, 0x00};
+ GSM_Error error;
+
+ if (strlen(s->Phone.Data.Version)>0) return ERR_NONE;
+
+ smprintf(s, "Getting firmware version\n");
+ error=GSM_WaitFor (s, req, 5, 0xd1, 3, ID_GetFirmware);
+ if (error==ERR_NONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "[Connected model - \"%s\"]\n",s->Phone.Data.Model);
+ smprintf(s, "[Firmware version - \"%s\"]\n",s->Phone.Data.Version);
+ smprintf(s, "[Firmware date - \"%s\"]\n",s->Phone.Data.VerDate);
+ }
+ }
+ return error;
+}
+
+/* ---------- Shared for n7110.c and n6510.c ------------------------------- */
+
+GSM_Error N71_65_ReplyGetMemoryError(unsigned char error, GSM_StateMachine *s)
+{
+ switch (error) {
+ case 0x27:
+ smprintf(s, "No PIN\n");
+ return ERR_SECURITYERROR;
+ case 0x30:
+ smprintf(s, "Invalid memory type\n");
+ if (s->Phone.Data.Memory->MemoryType == MEM_ME) return ERR_EMPTY;
+ if (s->Phone.Data.Memory->MemoryType == MEM_SM) return ERR_EMPTY;
+ return ERR_NOTSUPPORTED;
+ case 0x33:
+ smprintf(s, "Empty location\n");
+ s->Phone.Data.Memory->EntriesNum = 0;
+ return ERR_EMPTY;
+ case 0x34:
+ smprintf(s, "Too high location ?\n");
+ return ERR_INVALIDLOCATION;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",error);
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+GSM_Error N71_65_ReplyWritePhonebook(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Phonebook entry written ");
+ switch (msg.Buffer[6]) {
+ case 0x0f:
+ smprintf(s, " - error\n");
+ switch (msg.Buffer[10]) {
+ case 0x36:
+ smprintf(s, "Too long name\n");
+ return ERR_NOTSUPPORTED;
+ case 0x3c:
+ smprintf(s, "Can not add entry with 0 subentries\n");
+ return ERR_NOTSUPPORTED;
+ case 0x3d:
+ smprintf(s, "Wrong entry type\n");
+ return ERR_NOTSUPPORTED;
+ case 0x3e:
+ smprintf(s, "Too much entries\n");
+ return ERR_NOTSUPPORTED;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[10]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ default:
+ smprintf(s, " - OK\n");
+ return ERR_NONE;
+ }
+}
+
+bool NOKIA_FindPhoneFeatureValue(GSM_StateMachine *s,
+ GSM_Profile_PhoneTableValue ProfileTable[],
+ GSM_Profile_Feat_ID FeatureID,
+ GSM_Profile_Feat_Value FeatureValue,
+ unsigned char *PhoneID,
+ unsigned char *PhoneValue)
+{
+ int i=0;
+
+ smprintf(s, "Trying to find feature %i with value %i\n",FeatureID,FeatureValue);
+ while (ProfileTable[i].ID != 0x00) {
+ if (ProfileTable[i].ID == FeatureID &&
+ ProfileTable[i].Value == FeatureValue) {
+ *PhoneID = ProfileTable[i].PhoneID;
+ *PhoneValue = ProfileTable[i].PhoneValue;
+ return true;
+ }
+ i++;
+ }
+ return false;
+}
+
+#define PROFILE_CALLERGROUPS_GROUP1 0x01
+#define PROFILE_CALLERGROUPS_GROUP2 0x02
+#define PROFILE_CALLERGROUPS_GROUP3 0x04
+#define PROFILE_CALLERGROUPS_GROUP4 0x08
+#define PROFILE_CALLERGROUPS_GROUP5 0x10
+
+void NOKIA_FindFeatureValue(GSM_StateMachine *s,
+ GSM_Profile_PhoneTableValue ProfileTable[],
+ unsigned char ID,
+ unsigned char Value,
+ GSM_Phone_Data *Data,
+ bool CallerGroups)
+{
+ int i;
+
+ if (CallerGroups) {
+ smprintf(s, "Caller groups: %i\n", Value);
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = Profile_CallerGroups;
+ Data->Profile->FeaturesNumber++;
+ for (i=0;i<5;i++) Data->Profile->CallerGroups[i] = false;
+ if ((Value & PROFILE_CALLERGROUPS_GROUP1)==PROFILE_CALLERGROUPS_GROUP1) Data->Profile->CallerGroups[0] = true;
+ if ((Value & PROFILE_CALLERGROUPS_GROUP2)==PROFILE_CALLERGROUPS_GROUP2) Data->Profile->CallerGroups[1] = true;
+ if ((Value & PROFILE_CALLERGROUPS_GROUP3)==PROFILE_CALLERGROUPS_GROUP3) Data->Profile->CallerGroups[2] = true;
+ if ((Value & PROFILE_CALLERGROUPS_GROUP4)==PROFILE_CALLERGROUPS_GROUP4) Data->Profile->CallerGroups[3] = true;
+ if ((Value & PROFILE_CALLERGROUPS_GROUP5)==PROFILE_CALLERGROUPS_GROUP5) Data->Profile->CallerGroups[4] = true;
+ return;
+ }
+
+ i = 0;
+ while (ProfileTable[i].ID != 0x00) {
+ if (ProfileTable[i].PhoneID == ID &&
+ ProfileTable[i].PhoneValue == Value) {
+#ifdef DEBUG
+ switch (ProfileTable[i].ID) {
+ case Profile_KeypadTone : smprintf(s, "Keypad tones\n"); break;
+ case Profile_CallAlert : smprintf(s, "Call alert\n"); break;
+ case Profile_RingtoneVolume : smprintf(s, "Ringtone volume\n"); break;
+ case Profile_MessageTone : smprintf(s, "SMS message tones\n"); break;
+ case Profile_Vibration : smprintf(s, "Vibration\n"); break;
+ case Profile_WarningTone : smprintf(s, "Warning (ang games) tones\n"); break;
+ case Profile_AutoAnswer : smprintf(s, "Automatic answer\n"); break;
+ case Profile_Lights : smprintf(s, "Lights\n"); break;
+ case Profile_ScreenSaver : smprintf(s, "Screen Saver\n"); break;
+ case Profile_ScreenSaverTime : smprintf(s, "Screen Saver timeout\n"); break;
+ default : break;
+ }
+#endif
+ Data->Profile->FeatureID [Data->Profile->FeaturesNumber] = ProfileTable[i].ID;
+ Data->Profile->FeatureValue [Data->Profile->FeaturesNumber] = ProfileTable[i].Value;
+ Data->Profile->FeaturesNumber++;
+ break;
+ }
+ i++;
+ }
+}
+
+GSM_Profile_PhoneTableValue Profile71_65[] = {
+ {Profile_KeypadTone, PROFILE_KEYPAD_OFF, 0x00,0x00},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL1, 0x00,0x01},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL2, 0x00,0x02},
+ {Profile_KeypadTone, PROFILE_KEYPAD_LEVEL3, 0x00,0x03},
+ /* Lights ? */
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGING, 0x02,0x00},
+ {Profile_CallAlert, PROFILE_CALLALERT_ASCENDING, 0x02,0x01},
+ {Profile_CallAlert, PROFILE_CALLALERT_RINGONCE, 0x02,0x02},
+ {Profile_CallAlert, PROFILE_CALLALERT_BEEPONCE, 0x02,0x03},
+ {Profile_CallAlert, PROFILE_CALLALERT_OFF, 0x02,0x05},
+/* {Profile_CallAlert, PROFILE_CALLALERT_CALLERGROUPS,0x02,0x07}, */
+ /* Ringtone ID */
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL1, 0x04,0x00},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL2, 0x04,0x01},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL3, 0x04,0x02},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL4, 0x04,0x03},
+ {Profile_RingtoneVolume, PROFILE_VOLUME_LEVEL5, 0x04,0x04},
+ {Profile_MessageTone, PROFILE_MESSAGE_NOTONE, 0x05,0x00},
+ {Profile_MessageTone, PROFILE_MESSAGE_STANDARD, 0x05,0x01},
+ {Profile_MessageTone, PROFILE_MESSAGE_SPECIAL, 0x05,0x02},
+ {Profile_MessageTone, PROFILE_MESSAGE_BEEPONCE, 0x05,0x03},
+ {Profile_MessageTone, PROFILE_MESSAGE_ASCENDING, 0x05,0x04},
+ {Profile_Vibration, PROFILE_VIBRATION_OFF, 0x06,0x00},
+ {Profile_Vibration, PROFILE_VIBRATION_ON, 0x06,0x01},
+ {Profile_WarningTone, PROFILE_WARNING_OFF, 0x07,0x00},
+ {Profile_WarningTone, PROFILE_WARNING_ON, 0x07,0x01},
+ /* Caller groups */
+ {Profile_AutoAnswer, PROFILE_AUTOANSWER_OFF, 0x09,0x00},
+ {Profile_AutoAnswer, PROFILE_AUTOANSWER_ON, 0x09,0x01},
+ {0x00, 0x00, 0x00,0x00}
+};
+
+GSM_Error NOKIA_SetIncomingSMS(GSM_StateMachine *s, bool enable)
+{
+ s->Phone.Data.EnableIncomingSMS = enable;
+#ifdef DEBUG
+ if (enable) {
+ smprintf(s, "Enabling incoming SMS\n");
+ } else {
+ smprintf(s, "Disabling incoming SMS\n");
+ }
+#endif
+ return ERR_NONE;
+}
+
+GSM_Error N71_65_ReplyUSSDInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ unsigned char buffer[2000],buffer2[4000];
+
+ if (s->Phone.Data.RequestID == ID_Divert) return ERR_NONE;
+
+ memcpy(buffer,msg.Buffer+8,msg.Buffer[7]);
+ buffer[msg.Buffer[7]] = 0x00;
+
+ smprintf(s, "USSD reply: \"%s\"\n",buffer);
+
+ if (s->Phone.Data.EnableIncomingUSSD && s->User.IncomingUSSD!=NULL) {
+ EncodeUnicode(buffer2,buffer,strlen(buffer));
+ s->User.IncomingUSSD(s->CurrentConfig->Device, buffer2);
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error NOKIA_SetIncomingUSSD(GSM_StateMachine *s, bool enable)
+{
+ s->Phone.Data.EnableIncomingUSSD = enable;
+#ifdef DEBUG
+ if (enable) {
+ smprintf(s, "Enabling incoming USSD\n");
+ } else {
+ smprintf(s, "Disabling incoming USSD\n");
+ }
+#endif
+ return ERR_NONE;
+}
+
+GSM_Error NOKIA_SetIncomingCall(GSM_StateMachine *s, bool enable)
+{
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo,F_NOCALLINFO)) return ERR_NOTSUPPORTED;
+
+ s->Phone.Data.EnableIncomingCall = enable;
+#ifdef DEBUG
+ if (enable) {
+ smprintf(s, "Enabling incoming Call\n");
+ } else {
+ smprintf(s, "Disabling incoming Call\n");
+ }
+#endif
+ return ERR_NONE;
+}
+
+GSM_Error N71_65_ReplyCallInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_Call call;
+ int tmp;
+ unsigned char buffer[200];
+
+ call.Status = 0;
+ call.CallIDAvailable = true;
+ smprintf(s, "Call info, ");
+ switch (msg.Buffer[3]) {
+ case 0x02:
+ smprintf(s, "Call established, waiting for answer\n");
+ call.Status = GSM_CALL_CallEstablished;
+ break;
+ case 0x03:
+ smprintf(s, "Call started\n");
+ smprintf(s, "Call mode : %i\n",msg.Buffer[5]);//such interpretation is in gnokii
+ tmp = 6;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,buffer,false);
+ smprintf(s, "Number : \"%s\"\n",DecodeUnicodeString(buffer));
+ /* FIXME: read name from frame */
+
+ call.Status = GSM_CALL_CallStart;
+ break;
+ case 0x04:
+ smprintf(s, "Remote end hang up\n");
+ smprintf(s, "Cause Type : %i\n",msg.Buffer[5]);//such interpretation is in gnokii
+ smprintf(s, "CC : %i\n",msg.Buffer[6]);
+ smprintf(s, "MM(?) : %i\n",msg.Buffer[7]);
+ smprintf(s, "RR(?) : %i\n",msg.Buffer[8]);
+ call.Status = GSM_CALL_CallRemoteEnd;
+ call.StatusCode = msg.Buffer[6];
+ break;
+ case 0x05:
+ smprintf(s, "Incoming call\n");
+ smprintf(s, "Call mode : %i\n",msg.Buffer[5]);//such interpretation is in gnokii
+ tmp = 6;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,buffer,false);
+ smprintf(s, "Number : \"%s\"\n",DecodeUnicodeString(buffer));
+ /* FIXME: read name from frame */
+ call.Status = GSM_CALL_IncomingCall;
+ tmp = 6;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,call.PhoneNumber,false);
+ break;
+ case 0x07:
+ smprintf(s, "Call answer initiated\n");
+ break;
+ case 0x09:
+ smprintf(s, "Call released\n");
+ call.Status = GSM_CALL_CallLocalEnd;
+ break;
+ case 0x0a:
+ smprintf(s, "Call is being released\n");
+ break;
+ case 0x0b:
+ smprintf(s, "Meaning not known\n");
+ call.CallIDAvailable = false;
+ break;
+ case 0x0c:
+ smprintf(s, "Audio status\n");
+ if (msg.Buffer[4] == 0x01) smprintf(s, "Audio enabled\n");
+ else smprintf(s, "Audio disabled\n");
+ call.CallIDAvailable = false;
+ break;
+ case 0x23:
+ smprintf(s, "Call held\n");
+ call.Status = GSM_CALL_CallHeld;
+ break;
+ case 0x25:
+ smprintf(s, "Call resumed\n");
+ call.Status = GSM_CALL_CallResumed;
+ break;
+ case 0x27:
+ smprintf(s, "Call switched\n");
+ call.Status = GSM_CALL_CallSwitched;
+ break;
+ case 0x53:
+ smprintf(s, "Outgoing call\n");
+ smprintf(s, "Call mode : %i\n",msg.Buffer[5]);//such interpretation is in gnokii
+ tmp = 6;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,buffer,false);
+ smprintf(s, "Number : \"%s\"\n",DecodeUnicodeString(buffer));
+ /* FIXME: read name from frame */
+ call.Status = GSM_CALL_OutgoingCall;
+ tmp = 6;
+ NOKIA_GetUnicodeString(s, &tmp, msg.Buffer,call.PhoneNumber,false);
+ break;
+ }
+ if (call.CallIDAvailable) smprintf(s, "Call ID : %d\n",msg.Buffer[4]);
+ if (s->Phone.Data.EnableIncomingCall && s->User.IncomingCall!=NULL && call.Status != 0) {
+ if (call.CallIDAvailable) call.CallID = msg.Buffer[4];
+ s->User.IncomingCall(s->CurrentConfig->Device, call);
+ }
+ if (s->Phone.Data.RequestID == ID_CancelCall) {
+ if (msg.Buffer[3] == 0x09) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ /* when we canceled call and see frame about other
+ * call releasing, we don't give ERR_NONE for "our"
+ * call release command
+ */
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ if (s->Phone.Data.RequestID == ID_AnswerCall) {
+ if (msg.Buffer[3] == 0x07) {
+ if (s->Phone.Data.CallID == msg.Buffer[4]) return ERR_NONE;
+ return ERR_NEEDANOTHERANSWER;
+ }
+ }
+ return ERR_NONE;
+}
+
+void N71_65_GetCalendarRecurrance(GSM_StateMachine *s, unsigned char *buffer, GSM_CalendarEntry *entry)
+{
+ int Recurrance;
+
+ Recurrance = buffer[0]*256 + buffer[1];
+ /* 8760 hours = 1 year */
+ if (Recurrance == 0xffff) Recurrance=8760;
+ if (Recurrance != 0) {
+ smprintf(s, "Recurrance : %i hours\n",Recurrance);
+ entry->Entries[entry->EntriesNum].EntryType = CAL_RECURRANCE;
+ entry->Entries[entry->EntriesNum].Number = Recurrance;
+ entry->EntriesNum++;
+ }
+}
+
+/* method 2 */
+GSM_Error N71_65_ReplyAddCalendar2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Calendar note added\n");
+ return ERR_NONE;
+}
+
+/* method 2 */
+GSM_Error N71_65_AddCalendar2(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ GSM_CalendarNoteType NoteType;
+ time_t t_time1,t_time2;
+ GSM_DateTime Date,date_time;
+ GSM_Error error;
+ long diff;
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location, length=25;
+ unsigned char req[5000] = {
+ N6110_FRAME_HEADER,
+ 0x40,
+ 0x00, /* frame length - 7 */
+ 0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00, /* start time saved as difference */
+ 0x00,0x00,0xff,0xff, /* alarm saved as difference */
+ 0x00, /* frame length - 7 */
+ 0x00, /* note type */
+ 0x00,0x00, /* recurrance */
+ 0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00}; /* rest depends on note type */
+
+ NoteType = N71_65_FindCalendarType(Note->Type, s->Phone.Data.ModelInfo);
+
+ if (IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL62) ||
+ IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL65)) {
+ switch(NoteType) {
+ case GSM_CAL_MEETING : req[18] = 0x01; length = 25; break;
+ case GSM_CAL_CALL : req[18] = 0x02; length = 27; break;
+ case GSM_CAL_BIRTHDAY: req[18] = 0x04; length = 28; break;
+ case GSM_CAL_MEMO : req[18] = 0x08; length = 25; break;
+ default : return ERR_UNKNOWN;
+ }
+ } else {
+ switch(NoteType) {
+ case GSM_CAL_REMINDER: req[18] = 0x01; length = 25; break;
+ case GSM_CAL_CALL : req[18] = 0x02; length = 27; break;
+ case GSM_CAL_BIRTHDAY: req[18] = 0x04; length = 28; break;
+ case GSM_CAL_MEMO : req[18] = 0x08; length = 25; break;
+ default : return ERR_UNKNOWN;
+ }
+ }
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(Note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+
+ if (Time == -1) return ERR_UNKNOWN;
+ if (NoteType != GSM_CAL_BIRTHDAY) {
+ Date.Year = 2030; Date.Month = 01; Date.Day = 01;
+ Date.Hour = 00; Date.Minute = 00; Date.Second = 00;
+ } else {
+ Date.Year = 2029; Date.Month = 12; Date.Day = 31;
+ Date.Hour = 22; Date.Minute = 59; Date.Second = 58;
+ }
+ t_time1 = Fill_Time_T(Date,8);
+ memcpy(&Date,&Note->Entries[Time].Date,sizeof(GSM_DateTime));
+ if (NoteType != GSM_CAL_BIRTHDAY) {
+ Date.Year -= 20;
+ } else {
+ Date.Year = 1980;
+ Date.Hour = 22; Date.Minute = 58; Date.Second = 58;
+ }
+ t_time2 = Fill_Time_T(Date,8);
+ diff = t_time1-t_time2;
+ smprintf(s, " Difference : %li seconds\n", -diff);
+ req[9] = (unsigned char)(-diff >> 24);
+ req[10] = (unsigned char)(-diff >> 16);
+ req[11] = (unsigned char)(-diff >> 8);
+ req[12] = (unsigned char)(-diff);
+ if (NoteType == GSM_CAL_BIRTHDAY) {
+ req[25] = Note->Entries[Time].Date.Year / 256;
+ req[26] = Note->Entries[Time].Date.Year % 256;
+ /* Recurrance = 1 year */
+ req[19] = 0xff;
+ req[20] = 0xff;
+ }
+
+ if (NoteType == GSM_CAL_CALL && Phone != -1) {
+ req[25] = UnicodeLength(Note->Entries[Phone].Text);
+ CopyUnicodeString(req+length,Note->Entries[Phone].Text);
+ length += UnicodeLength(Note->Entries[Phone].Text)*2;
+ }
+
+ if (Alarm != -1) {
+ if (NoteType == GSM_CAL_BIRTHDAY) {
+ if (Note->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME) req[27] = 0x01;
+ error=s->Phone.Functions->GetDateTime(s,&date_time);
+ switch (error) {
+ case ERR_EMPTY:
+ case ERR_NOTIMPLEMENTED:
+ GSM_GetCurrentDateTime(&date_time);
+ break;
+ case ERR_NONE:
+ break;
+ default:
+ return error;
+ }
+ Date.Year = date_time.Year;
+ Date.Hour = 23;
+ Date.Minute = 59;
+ } else {
+ Date.Year += 20;
+ }
+ t_time2 = Fill_Time_T(Date,8);
+ t_time1 = Fill_Time_T(Note->Entries[Alarm].Date,8);
+ diff = t_time1-t_time2;
+
+ /* Sometimes we have difference in minutes */
+ if (NoteType == GSM_CAL_MEETING) diff = diff / 60;
+ if (!IsPhoneFeatureAvailable(s->Phone.Data.ModelInfo, F_CAL35)) {
+ if (NoteType == GSM_CAL_MEMO || NoteType == GSM_CAL_CALL) {
+ diff = diff / 60;
+ }
+ }
+
+ smprintf(s, " Difference : %li seconds or minutes\n", -diff);
+ req[13] = (unsigned char)(-diff >> 24);
+ req[14] = (unsigned char)(-diff >> 16);
+ req[15] = (unsigned char)(-diff >> 8);
+ req[16] = (unsigned char)(-diff);
+ }
+
+ if (Recurrance != -1) {
+ /* 8760 hours = 1 year */
+ if (Note->Entries[Recurrance].Number >= 8760) {
+ req[19] = 0xff;
+ req[20] = 0xff;
+ } else {
+ req[19] = Note->Entries[Recurrance].Number / 256;
+ req[20] = Note->Entries[Recurrance].Number % 256;
+ }
+ }
+
+ if (Text != -1) {
+ switch (NoteType) {
+ case GSM_CAL_CALL:
+ req[26] = UnicodeLength(Note->Entries[Text].Text);
+ break;
+ default:
+ req[length++] = UnicodeLength(Note->Entries[Text].Text);
+ if (NoteType == GSM_CAL_MEMO || NoteType == GSM_CAL_MEETING) req[length++] = 0x00;
+ }
+ CopyUnicodeString(req+length,Note->Entries[Text].Text);
+ length += UnicodeLength(Note->Entries[Text].Text)*2;
+ }
+
+ req[length++] = 0x00;
+ req[length++] = 0x00;
+
+ req[4] = req[17] = length-7;
+
+ smprintf(s, "Writing calendar note method 2\n");
+ return GSM_WaitFor (s, req, length, 0x13, 4, ID_SetCalendarNote);
+}
+
+/* method 1*/
+GSM_Error N71_65_ReplyGetCalendarNotePos1(GSM_Protocol_Message msg, GSM_StateMachine *s,int *FirstCalendarPos)
+{
+ smprintf(s, "First calendar location: %i\n",msg.Buffer[4]*256+msg.Buffer[5]);
+ *FirstCalendarPos = msg.Buffer[4]*256+msg.Buffer[5];
+ return ERR_NONE;
+}
+
+/* method 1*/
+static GSM_Error N71_65_GetCalendarNotePos1(GSM_StateMachine *s)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x31};
+
+ smprintf(s, "Getting first free calendar note location\n");
+ return GSM_WaitFor (s, req, 4, 0x13, 4, ID_GetCalendarNotePos);
+}
+
+/* method 1 */
+GSM_Error N71_65_ReplyAddCalendar1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+#ifdef DEBUG
+ smprintf(s, "Written calendar note type ");
+ switch ((msg.Buffer[3]/2)-1) {
+ case 0: smprintf(s, "Meeting"); break;
+ case 1: smprintf(s, "Call"); break;
+ case 2: smprintf(s, "Birthday");break;
+ case 3: smprintf(s, "Reminder");break;
+ }
+ smprintf(s, " on location %d\n",msg.Buffer[4]*256+msg.Buffer[5]);
+#endif
+ return ERR_NONE;
+}
+
+/* method 1 */
+GSM_Error N71_65_AddCalendar1(GSM_StateMachine *s, GSM_CalendarEntry *Note, int *FirstCalendarPos)
+{
+ long seconds;
+ GSM_Error error;
+ GSM_DateTime DT;
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location, count=12;
+ unsigned char req[5000] = {
+ N6110_FRAME_HEADER,
+ 0x01, /* note type */
+ 0x00, 0x00, /* location ? */
+ 0x00, /* entry type */
+ 0x00,
+ 0x00, 0x00, /* Year */
+ 0x00, /* Month */
+ 0x00, /* Day */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
+
+ error=N71_65_GetCalendarNotePos1(s);
+ if (error!=ERR_NONE) return error;
+ if (FirstCalendarPos != NULL) {
+ req[4] = *FirstCalendarPos/256;
+ req[5] = *FirstCalendarPos%256;
+ }
+
+ switch(Note->Type) {
+ case GSM_CAL_CALL : req[3]=0x03; req[6]=0x02; break;
+ case GSM_CAL_BIRTHDAY: req[3]=0x05; req[6]=0x04; break;
+ case GSM_CAL_MEMO : req[3]=0x07; req[6]=0x08; break;
+ case GSM_CAL_MEETING :
+ default : req[3]=0x01; req[6]=0x01; break;
+ }
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(Note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+
+ if (Time == -1) return ERR_UNKNOWN;
+ memcpy(&DT,&Note->Entries[Time].Date,sizeof(GSM_DateTime));
+ req[8] = DT.Year / 256;
+ req[9] = DT.Year % 256;
+ req[10] = DT.Month;
+ req[11] = DT.Day;
+
+ switch(Note->Type) {
+ case GSM_CAL_BIRTHDAY:
+ /* byte 12 and 13 */
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+
+ /* Alarm - bytes 14 to 17 */
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ req[count++] = 0xff;
+ req[count++] = 0xff;
+ if (Alarm != -1) {
+#ifndef ENABLE_LGPL
+ /* Comment from original source by Gabriele Zappi:
+ * I try with Time.Year = Alarm.Year. If negative, I increase 1 year,
+ * but only once ! This thing, because I may have Alarm period across
+ * a year. (eg. Birthday on 2001-01-10 and Alarm on 2000-12-27)
+ */
+#endif
+ DT.Year = Note->Entries[Alarm].Date.Year;
+ seconds = Fill_Time_T(DT,8)-Fill_Time_T(Note->Entries[Alarm].Date,8);
+ if (seconds<0L) {
+ DT.Year++;
+ seconds = Fill_Time_T(DT,8)-Fill_Time_T(Note->Entries[Alarm].Date,8);
+ }
+ if (seconds>=0L) {
+ count -= 4;
+ /* bytes 14 to 17 */
+ req[count++] = (unsigned char)(seconds>>24);
+ req[count++] = (unsigned char)((seconds>>16) & 0xff);
+ req[count++] = (unsigned char)((seconds>>8) & 0xff);
+ req[count++] = (unsigned char)(seconds & 0xff);
+ }
+ /* byte 18 */
+ if (Note->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME) req[count++] = 0x01; else req[count++] = 0x00;
+ }
+
+ /* byte 19 and next */
+ if (Text != -1) {
+ req[count++] = UnicodeLength(Note->Entries[Text].Text);
+ CopyUnicodeString(req+count,Note->Entries[Text].Text);
+ count=count+2*UnicodeLength(Note->Entries[Text].Text);
+ } else {
+ req[count++] = 0x00;
+ }
+ break;
+ case GSM_CAL_MEMO:
+ /* byte 12 and 13 */
+ if (Recurrance != -1) {
+ /* 8760 hours = 1 year */
+ if (Note->Entries[Recurrance].Number >= 8760) {
+ req[count++] = 0xff;
+ req[count++] = 0xff;
+ } else {
+ req[count++] = Note->Entries[Recurrance].Number / 256;
+ req[count++] = Note->Entries[Recurrance].Number % 256;
+ }
+ } else {
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ }
+
+ /* byte 14 and next */
+ if (Text != -1) {
+ req[count++] = UnicodeLength(Note->Entries[Text].Text);
+ req[count++] = 0x00;
+ CopyUnicodeString(req+count,Note->Entries[Text].Text);
+ count=count+2*UnicodeLength(Note->Entries[Text].Text);
+ } else {
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ }
+ break;
+ case GSM_CAL_MEETING:
+ case GSM_CAL_CALL:
+ default:
+ /* byte 12 and 13 */
+ req[count++] = DT.Hour;
+ req[count++] = DT.Minute;
+
+ /* Alarm - byte 14 and 15 */
+ req[count++] = 0xff;
+ req[count++] = 0xff;
+ if (Alarm != -1) {
+ seconds=Fill_Time_T(DT,8)-Fill_Time_T(Note->Entries[Alarm].Date,8);
+ if (seconds>=0L) {
+ count -= 2;
+ req[count++] = ((unsigned char)(seconds/60L)>>8);
+ req[count++] = (unsigned char)((seconds/60L)&0xff);
+ }
+ }
+
+ /* byte 16 and 17 */
+ if (Recurrance != -1) {
+ /* 8760 hours = 1 year */
+ if (Note->Entries[Recurrance].Number >= 8760) {
+ req[count++] = 0xff;
+ req[count++] = 0xff;
+ } else {
+ req[count++] = Note->Entries[Recurrance].Number / 256;
+ req[count++] = Note->Entries[Recurrance].Number % 256;
+ }
+ } else {
+ req[count++] = 0x00;
+ req[count++] = 0x00;
+ }
+
+ /* byte 18 */
+ if (Text != -1) {
+ req[count++] = UnicodeLength(Note->Entries[Text].Text);
+ } else {
+ req[count++] = 0x00;
+ }
+ /* byte 19 */
+ if (Note->Type == GSM_CAL_CALL && Phone != -1) {
+ req[count++] = UnicodeLength(Note->Entries[Phone].Text);
+ } else {
+ req[count++] = 0x00;
+ }
+ if (Text != -1) {
+ CopyUnicodeString(req+count,Note->Entries[Text].Text);
+ count=count+2*UnicodeLength(Note->Entries[Text].Text);
+ }
+ if (Note->Type == GSM_CAL_CALL && Phone != -1) {
+ CopyUnicodeString(req+count,Note->Entries[Phone].Text);
+ count=count+2*UnicodeLength(Note->Entries[Phone].Text);
+ }
+ break;
+ }
+ req[count] = 0x00;
+ smprintf(s, "Writing calendar note method 1\n");
+ return GSM_WaitFor (s, req, count, 0x13, 4, ID_SetCalendarNote);
+}
+
+GSM_Error N71_65_ReplyDelCalendar(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s, "Deleted calendar note on location %d\n",msg.Buffer[4]*256+msg.Buffer[5]);
+ return ERR_NONE;
+}
+
+GSM_Error N71_65_DelCalendar(GSM_StateMachine *s, GSM_CalendarEntry *Note)
+{
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x0b,
+ 0x00, 0x00}; /* location */
+
+ req[4] = Note->Location / 256;
+ req[5] = Note->Location % 256;
+
+ smprintf(s, "Deleting calendar note\n");
+ return GSM_WaitFor (s, req, 6, 0x13, 4, ID_DeleteCalendarNote);
+}
+
+/* method 1 */
+GSM_Error N71_65_ReplyGetCalendarInfo1(GSM_Protocol_Message msg, GSM_StateMachine *s, GSM_NOKIACalToDoLocations *LastCalendar)
+{
+ int i,j=0;
+
+ smprintf(s, "Info with calendar notes locations received method 1\n");
+ while (LastCalendar->Location[j] != 0x00) j++;
+ if (j >= GSM_MAXCALENDARTODONOTES) {
+ smprintf(s, "Increase GSM_MAXCALENDARNOTES\n");
+ return ERR_UNKNOWN;
+ }
+ if (j == 0) {
+ LastCalendar->Number=msg.Buffer[4]*256+msg.Buffer[5];
+ smprintf(s, "Number of Entries: %i\n",LastCalendar->Number);
+ }
+ smprintf(s, "Locations: ");
+ i = 0;
+ while (9+(i*2) <= msg.Length) {
+ LastCalendar->Location[j++]=msg.Buffer[8+(i*2)]*256+msg.Buffer[9+(i*2)];
+ smprintf(s, "%i ",LastCalendar->Location[j-1]);
+ i++;
+ }
+ smprintf(s, "\nNumber of Entries in frame: %i\n",i);
+ smprintf(s, "\n");
+ LastCalendar->Location[j] = 0;
+ if (i == 1 && msg.Buffer[8+(0*2)]*256+msg.Buffer[9+(0*2)] == 0) return ERR_EMPTY;
+ if (i == 0) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+/* method 1 */
+GSM_Error N71_65_GetCalendarInfo1(GSM_StateMachine *s, GSM_NOKIACalToDoLocations *LastCalendar)
+{
+ GSM_Error error;
+ int i;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x3a,
+ 0xFF, 0xFE}; /* First location number */
+
+ LastCalendar->Location[0] = 0x00;
+ LastCalendar->Number = 0;
+
+ smprintf(s, "Getting locations for calendar method 1\n");
+ error = GSM_WaitFor (s, req, 6, 0x13, 4, ID_GetCalendarNotesInfo);
+ if (error != ERR_NONE && error != ERR_EMPTY) return error;
+
+ while (1) {
+ i=0;
+ while (LastCalendar->Location[i] != 0x00) i++;
+ if (i == LastCalendar->Number) break;
+ if (i != LastCalendar->Number && error == ERR_EMPTY) {
+ smprintf(s, "Phone doesn't support some notes with this method. Workaround\n");
+ LastCalendar->Number = i;
+ break;
+ }
+ smprintf(s, "i = %i %i\n",i,LastCalendar->Number);
+ req[4] = LastCalendar->Location[i-1] / 256;
+ req[5] = LastCalendar->Location[i-1] % 256;
+ smprintf(s, "Getting locations for calendar\n");
+ error = GSM_WaitFor (s, req, 6, 0x13, 4, ID_GetCalendarNotesInfo);
+ if (error != ERR_NONE && error != ERR_EMPTY) return error;
+ }
+ return ERR_NONE;
+}
+
+/* method 1 */
+GSM_Error N71_65_ReplyGetNextCalendar1(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int alarm,i;
+ GSM_CalendarEntry *entry = s->Phone.Data.Cal;
+
+ smprintf(s, "Calendar note received method 1\n");
+
+ /* Later these values can change */
+ if (msg.Buffer[6]!=0x04) { /* Here not birthday */
+ entry->Entries[0].Date.Year = msg.Buffer[8]*256+msg.Buffer[9];
+ }
+ entry->Entries[0].Date.Month = msg.Buffer[10];
+ entry->Entries[0].Date.Day = msg.Buffer[11];
+ entry->Entries[0].Date.Hour = msg.Buffer[12];
+ entry->Entries[0].Date.Minute = msg.Buffer[13];
+ entry->Entries[0].Date.Second = 0;
+ entry->Entries[0].EntryType = CAL_START_DATETIME;
+ entry->EntriesNum++;
+
+ switch (msg.Buffer[6]) {
+ case 0x01:
+ smprintf(s, "Meeting\n");
+ entry->Type = GSM_CAL_MEETING;
+
+ alarm=msg.Buffer[14]*256+msg.Buffer[15];
+ if (alarm != 0xffff) {
+ smprintf(s, " Difference : %i seconds\n", alarm);
+ memcpy(&entry->Entries[1].Date,&entry->Entries[0].Date,sizeof(GSM_DateTime));
+ GetTimeDifference(alarm, &entry->Entries[1].Date, false, 60);
+ entry->Entries[1].EntryType = CAL_ALARM_DATETIME;
+ entry->EntriesNum++;
+ }
+ N71_65_GetCalendarRecurrance(s, msg.Buffer + 16, entry);
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+20, msg.Buffer[18]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[18]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[18]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ smprintf(s, "Text : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ return ERR_NONE;
+ case 0x02:
+ smprintf(s, "Call\n");
+ entry->Type = GSM_CAL_CALL;
+
+ alarm=msg.Buffer[14]*256+msg.Buffer[15];
+ if (alarm != 0xffff) {
+ smprintf(s, " Difference : %i seconds\n", alarm);
+ memcpy(&entry->Entries[1].Date,&entry->Entries[0].Date,sizeof(GSM_DateTime));
+ GetTimeDifference(alarm, &entry->Entries[1].Date, false, 60);
+ entry->Entries[1].EntryType = CAL_ALARM_DATETIME;
+ entry->EntriesNum++;
+ }
+ N71_65_GetCalendarRecurrance(s, msg.Buffer + 16, entry);
+
+ i = msg.Buffer[18] * 2;
+ if (i!=0) {
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+20, i);
+ entry->Entries[entry->EntriesNum].Text[i] = 0;
+ entry->Entries[entry->EntriesNum].Text[i+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ smprintf(s, "Text : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ }
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+20+i, msg.Buffer[19]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[19]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[19]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_PHONE;
+ smprintf(s, "Phone : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ return ERR_NONE;
+ case 0x04:
+ smprintf(s, "Birthday\n");
+ entry->Type = GSM_CAL_BIRTHDAY;
+
+ entry->Entries[0].Date.Hour = 23;
+ entry->Entries[0].Date.Minute = 59;
+ entry->Entries[0].Date.Second = 58;
+
+ alarm = ((unsigned int)msg.Buffer[14]) << 24;
+ alarm += ((unsigned int)msg.Buffer[15]) << 16;
+ alarm += ((unsigned int)msg.Buffer[16]) << 8;
+ alarm += msg.Buffer[17];
+ if (alarm != 0xffff) {
+ smprintf(s, " Difference : %i seconds\n", alarm);
+ memcpy(&entry->Entries[1].Date,&entry->Entries[0].Date,sizeof(GSM_DateTime));
+ GetTimeDifference(alarm, &entry->Entries[1].Date, false, 1);
+ entry->Entries[1].EntryType = CAL_ALARM_DATETIME;
+ if (msg.Buffer[20]!=0x00) {
+ entry->Entries[1].EntryType = CAL_SILENT_ALARM_DATETIME;
+ smprintf(s, "Alarm type : Silent\n");
+ }
+ entry->EntriesNum++;
+ }
+
+ entry->Entries[0].Date.Year = msg.Buffer[18]*256 + msg.Buffer[19];
+ if (entry->Entries[0].Date.Year == 65535) entry->Entries[0].Date.Year = 0;
+ smprintf(s, "Age : %i\n",entry->Entries[0].Date.Year);
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+22, msg.Buffer[21]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[21]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[21]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ smprintf(s, "Text : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ return ERR_NONE;
+ case 0x08:
+ smprintf(s, "Memo\n");
+ entry->Type = GSM_CAL_MEMO;
+
+ entry->Entries[0].Date.Hour = 0;
+ entry->Entries[0].Date.Minute = 0;
+
+ N71_65_GetCalendarRecurrance(s, msg.Buffer + 12, entry);
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+16, msg.Buffer[14]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[14]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[14]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ smprintf(s, "Text : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ return ERR_NONE;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[6]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+}
+
+/* method 1 */
+GSM_Error N71_65_GetNextCalendar1(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, GSM_NOKIACalToDoLocations *LastCalendar, int *LastCalendarYear, int *LastCalendarPos)
+{
+ GSM_Error error;
+ GSM_DateTime date_time;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x19,
+ 0x00, 0x00}; /* Location */
+
+ if (start) {
+ error=N71_65_GetCalendarInfo1(s, LastCalendar);
+ if (error!=ERR_NONE) return error;
+ if (LastCalendar->Number == 0) return ERR_EMPTY;
+
+ /* We have to get current year. It's NOT written in frame for
+ * Birthday
+ */
+ error=s->Phone.Functions->GetDateTime(s,&date_time);
+ switch (error) {
+ case ERR_EMPTY:
+ case ERR_NOTIMPLEMENTED:
+ GSM_GetCurrentDateTime(&date_time);
+ break;
+ case ERR_NONE:
+ break;
+ default:
+ return error;
+ }
+ *LastCalendarYear = date_time.Year;
+ *LastCalendarPos = 0;
+ } else {
+ (*LastCalendarPos)++;
+ }
+
+ if (*LastCalendarPos >= LastCalendar->Number) return ERR_EMPTY;
+
+ req[4] = LastCalendar->Location[*LastCalendarPos] / 256;
+ req[5] = LastCalendar->Location[*LastCalendarPos] % 256;
+
+ Note->EntriesNum = 0;
+ Note->Entries[0].Date.Year = *LastCalendarYear;
+ Note->Location = LastCalendar->Location[*LastCalendarPos];
+
+ s->Phone.Data.Cal=Note;
+ smprintf(s, "Getting calendar note method 1\n");
+ return GSM_WaitFor (s, req, 6, 0x13, 4, ID_GetCalendarNote);
+}
+
+GSM_Error N71_65_EnableFunctions(GSM_StateMachine *s,char *buff,int len)
+{
+ unsigned char buffer[50] = {N6110_FRAME_HEADER, 0x10,
+ 0x07}; /* Length */
+
+ buffer[4] = len;
+ memcpy(buffer+5,buff,len);
+
+ /* Enables various things like incoming SMS, call info, etc. */
+ return s->Protocol.Functions->WriteMessage(s, buffer, 5+len, 0x10);
+}
+
+GSM_Error N71_65_ReplySendDTMF(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Buffer[3]) {
+ case 0x51:
+ smprintf(s, "DTMF sent OK\n");
+ return ERR_NONE;
+ case 0x59:
+ case 0x5E:
+ smprintf(s, "meaning unknown - during sending DTMF\n");
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_CalendarNoteType N71_65_FindCalendarType(GSM_CalendarNoteType Type, OnePhoneModel *model)
+{
+ switch (Type) {
+ case GSM_CAL_CALL:
+ return GSM_CAL_CALL;
+ case GSM_CAL_BIRTHDAY:
+ return GSM_CAL_BIRTHDAY;
+ case GSM_CAL_MEETING:
+ if (IsPhoneFeatureAvailable(model, F_CAL35)) {
+ return GSM_CAL_REMINDER;
+ } else return GSM_CAL_MEETING;
+ case GSM_CAL_MEMO:
+ if (IsPhoneFeatureAvailable(model, F_CAL35)) {
+ return GSM_CAL_REMINDER;
+ } else return GSM_CAL_MEMO;
+ case GSM_CAL_REMINDER:
+ if (IsPhoneFeatureAvailable(model, F_CAL62) ||
+ IsPhoneFeatureAvailable(model, F_CAL65)) {
+ return GSM_CAL_CALL;
+ } else return GSM_CAL_REMINDER;
+ default:
+ return GSM_CAL_CALL;
+ }
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/nfunc.h b/gammu/emb/common/phone/nokia/nfunc.h
new file mode 100644
index 0000000..5ba3df0
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/nfunc.h
@@ -0,0 +1,99 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef phone_nokia_h
+#define phone_nokia_h
+
+#include "ncommon.h"
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+
+extern unsigned char N71_65_MEMORY_TYPES[];
+extern GSM_Profile_PhoneTableValue Profile71_65[];
+
+GSM_Error NOKIA_ReplyGetPhoneString (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyGetMemoryError (unsigned char error , GSM_StateMachine *s);
+GSM_Error N71_65_ReplyWritePhonebook (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyGetCalendarInfo1 (GSM_Protocol_Message msg, GSM_StateMachine *s, GSM_NOKIACalToDoLocations *LastCalendar);
+GSM_Error N71_65_ReplyGetNextCalendar1 (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyGetCalendarNotePos1 (GSM_Protocol_Message msg, GSM_StateMachine *s,int *FirstCalendarPos);
+GSM_Error N71_65_ReplyAddCalendar1 (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyAddCalendar2 (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyDelCalendar (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyCallInfo (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplyUSSDInfo (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyCallDivert (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyGetActiveConnectSet (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplySetActiveConnectSet (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyGetWAPBookmark (GSM_Protocol_Message msg, GSM_StateMachine *s, bool FullLength);
+GSM_Error DCT3DCT4_ReplySetWAPBookmark (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyDelWAPBookmark (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyEnableConnectFunc (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyDisableConnectFunc (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error DCT3DCT4_ReplyGetModelFirmware (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_ReplySendDTMF (GSM_Protocol_Message msg, GSM_StateMachine *s);
+
+GSM_Error NOKIA_GetManufacturer (GSM_StateMachine *s);
+GSM_Error NOKIA_GetPhoneString (GSM_StateMachine *s, unsigned char *msgframe, int msglen, unsigned char msgtype, char *retvalue, GSM_Phone_RequestID request, int startresponse);
+GSM_Error NOKIA_SetIncomingSMS (GSM_StateMachine *s, bool enable);
+GSM_Error NOKIA_SetIncomingCall (GSM_StateMachine *s, bool enable);
+GSM_Error NOKIA_SetIncomingUSSD (GSM_StateMachine *s, bool enable);
+GSM_Error N71_65_EnableFunctions (GSM_StateMachine *s, char *buff,int len);
+GSM_Error N71_65_GetNextCalendar1 (GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, GSM_NOKIACalToDoLocations *LastCalendar, int *LastCalendarYear, int *LastCalendarPos);
+GSM_Error N71_65_AddCalendar2 (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+GSM_Error N71_65_AddCalendar1 (GSM_StateMachine *s, GSM_CalendarEntry *Note, int *FirstCalendarPos);
+GSM_Error N71_65_DelCalendar (GSM_StateMachine *s, GSM_CalendarEntry *Note);
+GSM_Error N71_65_GetCalendarInfo1 (GSM_StateMachine *s, GSM_NOKIACalToDoLocations *LastCalendar);
+GSM_Error DCT3DCT4_EnableWAPFunctions (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_SendDTMF (GSM_StateMachine *s, char *sequence);
+GSM_Error DCT3DCT4_DeleteWAPBookmarkPart (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+GSM_Error DCT3DCT4_GetWAPBookmarkPart (GSM_StateMachine *s, GSM_WAPBookmark *bookmark);
+GSM_Error DCT3DCT4_DisableConnectionFunctions (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_GetModel (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_GetFirmware (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_AnswerCall (GSM_StateMachine *s, int ID);
+GSM_Error DCT3DCT4_CancelCall (GSM_StateMachine *s, int ID);
+GSM_Error DCT3DCT4_GetActiveConnectSet (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_SetActiveConnectSet (GSM_StateMachine *s, GSM_MultiWAPSettings *settings);
+GSM_Error DCT3DCT4_CancelAllDiverts (GSM_StateMachine *s);
+GSM_Error DCT3DCT4_SetCallDivert (GSM_StateMachine *s, GSM_MultiCallDivert *divert);
+GSM_Error DCT3DCT4_GetCallDivert (GSM_StateMachine *s, GSM_MultiCallDivert *divert);
+
+GSM_CalendarNoteType N71_65_FindCalendarType(GSM_CalendarNoteType Type, OnePhoneModel *model);
+void N71_65_GetCalendarRecurrance (GSM_StateMachine *s, unsigned char *buffer, GSM_CalendarEntry *entry);
+int NOKIA_SetUnicodeString (GSM_StateMachine *s, unsigned char *dest, unsigned char *string, bool FullLength);
+void NOKIA_GetUnicodeString (GSM_StateMachine *s, int *current, unsigned char *input, unsigned char *output, bool FullLength);
+GSM_MemoryType NOKIA_GetMemoryType (GSM_StateMachine *s, GSM_MemoryType memory_type, unsigned char *ID);
+void NOKIA_DecodeSMSState (GSM_StateMachine *s, unsigned char state, GSM_SMSMessage *sms);
+void NOKIA_EncodeDateTime (GSM_StateMachine *s, unsigned char* buffer, GSM_DateTime *datetime);
+void NOKIA_DecodeDateTime (GSM_StateMachine *s, unsigned char* buffer, GSM_DateTime *datetime);
+void NOKIA_SortSMSFolderStatus (GSM_StateMachine *s, GSM_NOKIASMSFolder *Folder);
+void NOKIA_GetDefaultCallerGroupName (GSM_StateMachine *s, GSM_Bitmap *Bitmap);
+void NOKIA_GetDefaultProfileName (GSM_StateMachine *s, GSM_Profile *Profile);
+void NOKIA_FindFeatureValue(GSM_StateMachine *s,
+ GSM_Profile_PhoneTableValue ProfileTable[],
+ unsigned char ID,
+ unsigned char Value,
+ GSM_Phone_Data *Data,
+ bool CallerGroups);
+bool NOKIA_FindPhoneFeatureValue(GSM_StateMachine *s,
+ GSM_Profile_PhoneTableValue ProfileTable[],
+ GSM_Profile_Feat_ID FeatureID,
+ GSM_Profile_Feat_Value FeatureValue,
+ unsigned char *PhoneID,
+ unsigned char *PhoneValue);
+
+GSM_Error N71_65_DecodePhonebook (GSM_StateMachine *s,
+ GSM_MemoryEntry *entry,
+ GSM_Bitmap *bitmap,
+ GSM_SpeedDial *speed,
+ unsigned char *MessageBuffer,
+ int MessageLength,
+ bool DayMonthReverse);
+int N71_65_EncodePhonebookFrame (GSM_StateMachine *s, unsigned char *req, GSM_MemoryEntry entry, int *block2, bool DCT4, bool VoiceTag);
+int N71_65_PackPBKBlock (GSM_StateMachine *s, int id, int size, int no, unsigned char *buf, unsigned char *block);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/nfuncold.c b/gammu/emb/common/phone/nokia/nfuncold.c
new file mode 100644
index 0000000..0e40bc6
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/nfuncold.c
@@ -0,0 +1,226 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include <string.h> /* memcpy only */
+#include <stdio.h>
+#include <ctype.h>
+#include <time.h>
+
+#include "../../misc/coding/coding.h"
+#include "../../gsmstate.h"
+#include "../../service/sms/gsmsms.h"
+#include "../pfunc.h"
+#include "nfunc.h"
+
+#ifdef DEBUG
+static void N71_65_GetCalendarAlarm(GSM_StateMachine *s, unsigned char *buffer, GSM_CalendarEntry *entry, int DT, GSM_Phone_Data *Data)
+{
+ unsigned long diff;
+
+ if (buffer[0] == 0x00 && buffer[1] == 0x00 && buffer[2] == 0xff && buffer[3] == 0xff) {
+ smprintf(s, "No alarm\n");
+ } else {
+ memcpy(&entry->Entries[entry->EntriesNum].Date,&entry->Entries[DT].Date,sizeof(GSM_DateTime));
+
+ diff = ((unsigned int)buffer[0]) << 24;
+ diff += ((unsigned int)buffer[1]) << 16;
+ diff += ((unsigned int)buffer[2]) << 8;
+ diff += buffer[3];
+ smprintf(s, " Difference : %li seconds\n", diff);
+
+ switch (entry->Type) {
+ case GSM_CAL_MEETING:
+ GetTimeDifference(diff, &entry->Entries[entry->EntriesNum].Date, false, 60);
+ break;
+ case GSM_CAL_MEMO:
+ if (!IsPhoneFeatureAvailable(Data->ModelInfo, F_CAL35)) {
+ GetTimeDifference(diff, &entry->Entries[entry->EntriesNum].Date, false, 60);
+ break;
+ }
+ case GSM_CAL_CALL:
+ if (!IsPhoneFeatureAvailable(Data->ModelInfo, F_CAL35)) {
+ GetTimeDifference(diff, &entry->Entries[entry->EntriesNum].Date, false, 60);
+ break;
+ }
+ default:
+ GetTimeDifference(diff, &entry->Entries[entry->EntriesNum].Date, false, 1);
+ }
+ smprintf(s, "Alarm date : %02i-%02i-%04i %02i:%02i:%02i\n",
+ entry->Entries[entry->EntriesNum].Date.Day, entry->Entries[entry->EntriesNum].Date.Month,
+ entry->Entries[entry->EntriesNum].Date.Year, entry->Entries[entry->EntriesNum].Date.Hour,
+ entry->Entries[entry->EntriesNum].Date.Minute,entry->Entries[entry->EntriesNum].Date.Second);
+
+ entry->Entries[entry->EntriesNum].EntryType = CAL_ALARM_DATETIME;
+ if (entry->Type == GSM_CAL_BIRTHDAY) {
+ if (buffer[14]!=0x00) entry->Entries[entry->EntriesNum].EntryType = CAL_SILENT_ALARM_DATETIME;
+ smprintf(s, "Alarm type : Silent\n");
+ }
+
+ entry->EntriesNum++;
+ }
+}
+
+/* method 2 */
+GSM_Error N71_65_ReplyGetNextCalendar2(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ GSM_DateTime Date;
+ GSM_CalendarEntry *entry = s->Phone.Data.Cal;
+ GSM_Phone_Data *Data = &s->Phone.Data;
+ int i;
+ unsigned long diff;
+
+ smprintf(s, "Calendar note received method 2\n");
+
+ if (msg.Length < 10) return ERR_EMPTY;
+
+ entry->Location = msg.Buffer[4]*256 + msg.Buffer[5];
+ smprintf(s, "Location: %i\n",entry->Location);
+
+ /* Not birthday */
+ if (msg.Buffer[21] != 0x04) {
+ Date.Year = 2030; Date.Month = 01; Date.Day = 01;
+ Date.Hour = 00; Date.Minute = 00; Date.Second = 00;
+ } else {
+ Date.Year = 2029; Date.Month = 12; Date.Day = 31;
+ Date.Hour = 22; Date.Minute = 59; Date.Second = 58;
+ }
+ diff = ((unsigned int)msg.Buffer[12]) << 24;
+ diff += ((unsigned int)msg.Buffer[13]) << 16;
+ diff += ((unsigned int)msg.Buffer[14]) << 8;
+ diff += msg.Buffer[15];
+ smprintf(s, " Difference : %li seconds\n", diff);
+ GetTimeDifference(diff, &Date, true, 1);
+ Date.Year += 20;
+ entry->Entries[0].EntryType = CAL_START_DATETIME;
+
+ smprintf(s, "Note type %02x: ",msg.Buffer[21]);
+ switch (msg.Buffer[21]) {
+ case 0x01:
+ case 0x08:
+ if (msg.Buffer[21] == 0x01) {
+ smprintf(s, "Meeting or Reminder\n");
+ entry->Type = GSM_CAL_MEETING;
+ } else {
+ smprintf(s, "Memo\n");
+ Data->Cal->Type = GSM_CAL_MEMO;
+ }
+
+ memcpy(&entry->Entries[0].Date,&Date,sizeof(GSM_DateTime));
+ entry->EntriesNum++;
+
+ N71_65_GetCalendarAlarm(s, msg.Buffer+16, entry, 0, Data);
+ N71_65_GetCalendarRecurrance(s, msg.Buffer+22, entry);
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+30, msg.Buffer[28]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[28]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[28]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ break;
+ case 0x02:
+ smprintf(s, "Call\n");
+ entry->Type = GSM_CAL_CALL;
+
+ memcpy(&entry->Entries[0].Date,&Date,sizeof(GSM_DateTime));
+ entry->EntriesNum++;
+
+ N71_65_GetCalendarAlarm(s, msg.Buffer+16, entry, 0, Data);
+ N71_65_GetCalendarRecurrance(s, msg.Buffer+22, entry);
+
+ i = msg.Buffer[28] * 2;
+ if (i!=0) {
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+30, i);
+ entry->Entries[entry->EntriesNum].Text[i] = 0;
+ entry->Entries[entry->EntriesNum].Text[i+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_PHONE;
+ smprintf(s, "Phone : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ }
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+30+i, msg.Buffer[29]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[29]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[29]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ break;
+ case 0x04:
+ smprintf(s, "Birthday\n");
+ Data->Cal->Type = GSM_CAL_BIRTHDAY;
+
+ /* Year was set earlier */
+ entry->Entries[0].Date.Month = Date.Month;
+ entry->Entries[0].Date.Day = Date.Day;
+ entry->Entries[0].Date.Hour = 23;
+ entry->Entries[0].Date.Minute = 59;
+ entry->Entries[0].Date.Second = 58;
+ entry->EntriesNum++;
+
+ N71_65_GetCalendarAlarm(s, msg.Buffer+16, entry, 0, Data);
+ N71_65_GetCalendarRecurrance(s, msg.Buffer+22, entry);
+
+ /* Birthday year */
+ entry->Entries[0].Date.Year = msg.Buffer[28]*256 + msg.Buffer[29];
+ if (msg.Buffer[28] == 0xff && msg.Buffer[29] == 0xff) entry->Entries[0].Date.Year = 0;
+ smprintf(s, "Birthday date: %02i-%02i-%04i\n",
+ entry->Entries[0].Date.Day,entry->Entries[0].Date.Month,
+ entry->Entries[0].Date.Year);
+
+ memcpy(entry->Entries[entry->EntriesNum].Text, msg.Buffer+32, msg.Buffer[31]*2);
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[31]*2] = 0;
+ entry->Entries[entry->EntriesNum].Text[msg.Buffer[31]*2+1] = 0;
+ entry->Entries[entry->EntriesNum].EntryType = CAL_TEXT;
+ break;
+ default:
+ smprintf(s, "ERROR: unknown %i\n",msg.Buffer[6]);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ smprintf(s, "Text : \"%s\"\n", DecodeUnicodeString(entry->Entries[entry->EntriesNum].Text));
+ entry->EntriesNum++;
+ return ERR_NONE;
+}
+
+/* method 2 */
+/* Note: in known phones texts of notes cut to 50 chars */
+GSM_Error N71_65_GetNextCalendar2(GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, int *LastCalendarYear, int *LastCalendarPos)
+{
+ GSM_Error error;
+ GSM_DateTime date_time;
+ unsigned char req[] = {N6110_FRAME_HEADER, 0x3e,
+ 0xFF, 0xFE}; /* Location */
+
+ if (start) {
+ /* We have to get current year. It's NOT written in frame for
+ * Birthday
+ */
+ error=s->Phone.Functions->GetDateTime(s,&date_time);
+ switch (error) {
+ case ERR_EMPTY:
+ case ERR_NOTIMPLEMENTED:
+ GSM_GetCurrentDateTime(&date_time);
+ break;
+ case ERR_NONE:
+ break;
+ default:
+ return error;
+ }
+ *LastCalendarYear = date_time.Year;
+
+ /* First location at all */
+ req[4] = 0xFF;
+ req[5] = 0xFE;
+ } else {
+ req[4] = *LastCalendarPos / 256;
+ req[5] = *LastCalendarPos % 256;
+ }
+ Note->EntriesNum = 0;
+ Note->Entries[0].Date.Year = *LastCalendarYear;
+
+ s->Phone.Data.Cal = Note;
+ smprintf(s, "Getting calendar note method 2\n");
+ error=GSM_WaitFor (s, req, 6, 0x13, 4, ID_GetCalendarNote);
+ *LastCalendarPos = Note->Location;
+ return error;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/nokia/nfuncold.h b/gammu/emb/common/phone/nokia/nfuncold.h
new file mode 100644
index 0000000..236189c
--- a/dev/null
+++ b/gammu/emb/common/phone/nokia/nfuncold.h
@@ -0,0 +1,19 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef phone_nokia2_h
+#define phone_nokia2_h
+
+#include "ncommon.h"
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+
+#ifdef DEBUG
+GSM_Error N71_65_ReplyGetNextCalendar2 (GSM_Protocol_Message msg, GSM_StateMachine *s);
+GSM_Error N71_65_GetNextCalendar2 (GSM_StateMachine *s, GSM_CalendarEntry *Note, bool start, int *LastCalendarYear, int *LastCalendarPos);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/obex/obexgen.c b/gammu/emb/common/phone/obex/obexgen.c
new file mode 100644
index 0000000..dd14f8e
--- a/dev/null
+++ b/gammu/emb/common/phone/obex/obexgen.c
@@ -0,0 +1,851 @@
+/* (c) 2003 by Marcin Wiacek */
+/* www.irda.org OBEX specs 1.3 */
+
+/* Module connects to F9EC7BC4-953c-11d2-984E-525400DC9E09 UUID and in the
+ * future there will required implementing reconnecting. See "ifdef xxxx"
+ */
+
+#include <string.h>
+#include <time.h>
+
+#include "../../misc/coding/coding.h"
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "../../service/gsmmisc.h"
+#include "../../protocol/obex/obex.h"
+
+#ifdef GSM_ENABLE_OBEXGEN
+
+static GSM_Error OBEXGEN_GetNextFileFolder(GSM_StateMachine *s, GSM_File *File, bool start);
+
+static void OBEXGEN_FindNextDir(unsigned char *Path, int *Pos, unsigned char *Return)
+{
+ unsigned char buff[200];
+
+ buff[0] = 0;
+ while(1) {
+ if (Path[*Pos] == 0x00) break;
+ if (Path[*Pos] == '\\') {
+ (*Pos)++;
+ break;
+ }
+ buff[strlen(buff)+1] = 0;
+ buff[strlen(buff)] = Path[(*Pos)];
+ (*Pos)++;
+ }
+ EncodeUnicode(Return,buff,strlen(buff));
+}
+
+static GSM_Error OBEXGEN_ReplyConnect(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Type) {
+ case 0xA0:
+ smprintf(s,"Connected/disconnected OK\n");
+ if (msg.Length != 0) {
+ s->Phone.Data.Priv.OBEXGEN.FrameSize = msg.Buffer[2]*256+msg.Buffer[3];
+ smprintf(s,"Maximal size of frame is %i 0x%x\n",s->Phone.Data.Priv.OBEXGEN.FrameSize,s->Phone.Data.Priv.OBEXGEN.FrameSize);
+ }
+ return ERR_NONE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error OBEXGEN_Disconnect(GSM_StateMachine *s)
+{
+ smprintf(s, "Disconnecting\n");
+ return GSM_WaitFor (s, NULL, 0, 0x81, 2, ID_Initialise);
+}
+
+GSM_Error OBEXGEN_Connect(GSM_StateMachine *s, OBEX_Service service)
+{
+ int Current=4;
+ unsigned char req2[200];
+ unsigned char req[200] = {
+ 0x10, /* Version 1.0 */
+ 0x00, /* no flags */
+ 0x20,0x00}; /* 0x2000 max size of packet */
+
+ if (service == s->Phone.Data.Priv.OBEXGEN.Service) return ERR_NONE;
+
+ switch (service) {
+ case OBEX_None:
+ break;
+ case OBEX_BrowsingFolders:
+ /* Server ID */
+ req2[0] = 0xF9; req2[1] = 0xEC; req2[2] = 0x7B;
+ req2[3] = 0xC4; req2[4] = 0x95; req2[5] = 0x3C;
+ req2[6] = 0x11; req2[7] = 0xD2; req2[8] = 0x98;
+ req2[9] = 0x4E; req2[10]= 0x52; req2[11]= 0x54;
+ req2[12]= 0x00; req2[13]= 0xDC; req2[14]= 0x9E;
+ req2[15]= 0x09;
+
+ /* Target block */
+ OBEXAddBlock(req, &Current, 0x46, req2, 16);
+ }
+
+#ifndef xxxx
+ //disconnect old service
+#else
+ if (s->Phone.Data.Priv.OBEXGEN.Service != 0) return ERR_NONE;
+#endif
+
+ s->Phone.Data.Priv.OBEXGEN.Service = service;
+
+ smprintf(s, "Connecting\n");
+ return GSM_WaitFor (s, req, Current, 0x80, 2, ID_Initialise);
+}
+
+
+GSM_Error OBEXGEN_Initialise(GSM_StateMachine *s)
+{
+// GSM_File File;
+// GSM_Error error = ERR_NONE;
+
+ s->Phone.Data.Priv.OBEXGEN.Service = 0;
+
+ strcpy(s->Phone.Data.Model,"obex");
+
+ s->Phone.Data.VerNum = 0;
+ s->Phone.Data.Version[0] = 0;
+ s->Phone.Data.Manufacturer[0] = 0;
+
+// File.Used = 0;
+// File.ID_FullName[0] = 0;
+// File.Buffer = NULL;
+// while (error == ERR_NONE) error = OBEXGEN_GetFilePart(s,&File);
+
+ return ERR_NONE;
+}
+
+static GSM_Error OBEXGEN_ReplyChangePath(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Type) {
+ case 0xA0:
+ smprintf(s,"Path set OK\n");
+ return ERR_NONE;
+ case 0xA1:
+ smprintf(s,"Folder created\n");
+ return ERR_NONE;
+ case 0xC3:
+ smprintf(s,"Security error\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error OBEXGEN_ChangePath(GSM_StateMachine *s, char *Name, unsigned char Flag1)
+{
+ unsigned char req[400];
+ int Current = 2;
+
+ /* Flags */
+ req[0] = Flag1;
+ req[1] = 0x00;
+
+ /* Name block */
+ if (Name != NULL && UnicodeLength(Name) != 0) {
+ OBEXAddBlock(req, &Current, 0x01, Name, UnicodeLength(Name)*2+2);
+ } else {
+ OBEXAddBlock(req, &Current, 0x01, NULL, 0);
+ }
+
+ /* connection ID block */
+ req[Current++] = 0xCB; // ID
+ req[Current++] = 0x00; req[Current++] = 0x00;
+ req[Current++] = 0x00; req[Current++] = 0x01;
+
+ return GSM_WaitFor (s, req, Current, 0x85, 4, ID_SetPath);
+}
+
+static GSM_Error OBEXGEN_ReplyAddFilePart(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ switch (msg.Type) {
+ case 0x90:
+ smprintf(s,"Last part of file added OK\n");
+ return ERR_NONE;
+ case 0xA0:
+ smprintf(s,"Part of file added OK\n");
+ return ERR_NONE;
+ case 0xC0:
+ smprintf(s,"Not understand. Probably not supported\n");
+ return ERR_NOTSUPPORTED;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+GSM_Error OBEXGEN_AddFilePart(GSM_StateMachine *s, GSM_File *File, int *Pos)
+{
+ GSM_Error error;
+ int j;
+ unsigned int Pos2, Current = 0;
+ unsigned char req[2000],req2[200];
+
+ s->Phone.Data.File = File;
+
+ if (*Pos == 0) {
+ if (!strcmp(File->ID_FullName,"")) {
+#ifndef xxxx
+ error = OBEXGEN_Connect(s,OBEX_None);
+#else
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+#endif
+ if (error != ERR_NONE) return error;
+ } else {
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+ if (error != ERR_NONE) return error;
+
+ if (strcmp(s->CurrentConfig->Model,"seobex")) {
+ smprintf(s,"Changing to root\n");
+ error = OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+
+ Pos2 = 0;
+ do {
+ OBEXGEN_FindNextDir(File->ID_FullName, &Pos2, req2);
+ smprintf(s,"%s %i %i\n",DecodeUnicodeString(req2),Pos2,strlen(File->ID_FullName));
+ smprintf(s,"Changing path down\n");
+ error=OBEXGEN_ChangePath(s, req2, 2);
+ if (error != ERR_NONE) return error;
+ if (Pos2 == strlen(File->ID_FullName)) break;
+ } while (1);
+ }
+ }
+
+ /* Name block */
+ OBEXAddBlock(req, &Current, 0x01, File->Name, UnicodeLength(File->Name)*2+2);
+
+ /* File size block */
+ req[Current++] = 0xC3; // ID
+ req[Current++] = 0;
+ req[Current++] = 0;
+ req[Current++] = File->Used / 256;
+ req[Current++] = File->Used % 256;
+ }
+
+ if (s->Phone.Data.Priv.OBEXGEN.Service == OBEX_BrowsingFolders) {
+ /* connection ID block */
+ req[Current++] = 0xCB; // ID
+ req[Current++] = 0x00; req[Current++] = 0x00;
+ req[Current++] = 0x00; req[Current++] = 0x01;
+ }
+
+ j = s->Phone.Data.Priv.OBEXGEN.FrameSize - Current - 20;
+ if (j > 1000) j = 1000;
+
+ if (File->Used - *Pos < j) {
+ j = File->Used - *Pos;
+ /* End of file body block */
+ OBEXAddBlock(req, &Current, 0x49, File->Buffer+(*Pos), j);
+ smprintf(s, "Adding file part %i %i\n",*Pos,j);
+ *Pos = *Pos + j;
+ error = GSM_WaitFor (s, req, Current, 0x82, 4, ID_AddFile);
+ if (error != ERR_NONE) return error;
+ return ERR_EMPTY;
+ } else {
+ /* File body block */
+ OBEXAddBlock(req, &Current, 0x48, File->Buffer+(*Pos), j);
+ smprintf(s, "Adding file part %i %i\n",*Pos,j);
+ *Pos = *Pos + j;
+ error=GSM_WaitFor (s, req, Current, 0x02, 4, ID_AddFile);
+ }
+ return error;
+}
+
+static GSM_Error OBEXGEN_ReplyGetFilePart(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int old,Pos=0;
+
+ switch (msg.Type) {
+ case 0xA0:
+ smprintf(s,"File part received\n");
+ s->Phone.Data.Priv.OBEXGEN.FileLastPart = true;
+ case 0x90:
+// if (msg.Length < 11) return ERR_NONE;
+ if (msg.Type == 0x90) smprintf(s,"Last file part received\n");
+ while(1) {
+ if (Pos >= msg.Length) break;
+ switch (msg.Buffer[Pos]) {
+ case 0x48:
+ case 0x49:
+ smprintf(s,"File part received\n");
+ old = s->Phone.Data.File->Used;
+ s->Phone.Data.File->Used += msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2]-3;
+ smprintf(s,"Length of file part: %i\n",
+ msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2]-3);
+ s->Phone.Data.File->Buffer = (unsigned char *)realloc(s->Phone.Data.File->Buffer,s->Phone.Data.File->Used);
+ memcpy(s->Phone.Data.File->Buffer+old,msg.Buffer+Pos+3,s->Phone.Data.File->Used-old);
+ return ERR_NONE;
+ default:
+ break;
+ }
+ Pos+=msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2];
+ }
+ return ERR_UNKNOWNRESPONSE;
+ case 0xC3:
+ return ERR_NOTSUPPORTED;
+ case 0xC4:
+ smprintf(s,"Not found\n");
+ return ERR_SECURITYERROR;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error OBEXGEN_ReplyGetFileInfo(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ int old,Pos=0;
+
+ switch (msg.Type) {
+ case 0x83:
+ smprintf(s,"Not available ?\n");
+ return ERR_NONE;
+ case 0x90:
+ smprintf(s,"Last part of file info received\n");
+ return ERR_NONE;
+ case 0xA0:
+ while(1) {
+ if (Pos >= msg.Length) break;
+ switch (msg.Buffer[Pos]) {
+ case 0x48:
+ case 0x49:
+ /* SE T310 */
+ smprintf(s,"File part received\n");
+ old = s->Phone.Data.File->Used;
+ s->Phone.Data.File->Used += msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2]-3;
+ smprintf(s,"Length of file part: %i\n",
+ msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2]-3);
+ s->Phone.Data.File->Buffer = (unsigned char *)realloc(s->Phone.Data.File->Buffer,s->Phone.Data.File->Used);
+ memcpy(s->Phone.Data.File->Buffer+old,msg.Buffer+Pos+3,s->Phone.Data.File->Used-old);
+ return ERR_EMPTY;
+ default:
+ break;
+ }
+ Pos+=msg.Buffer[Pos+1]*256+msg.Buffer[Pos+2];
+ }
+ return ERR_UNKNOWNRESPONSE;
+ }
+ return ERR_UNKNOWNRESPONSE;
+}
+
+static GSM_Error OBEXGEN_PrivGetFilePart(GSM_StateMachine *s, GSM_File *File, bool FolderList)
+{
+ unsigned int Current = 0, Pos;
+ GSM_Error error;
+ unsigned char req[2000], req2[200];
+
+ s->Phone.Data.File = File;
+ File->ReadOnly = false;
+ File->Protected = false;
+ File->Hidden = false;
+ File->System = false;
+
+ if (File->Used == 0x00) {
+ if (FolderList) {
+ /* Type block */
+ strcpy(req2,"x-obex/folder-listing");
+ OBEXAddBlock(req, &Current, 0x42, req2, strlen(req2)+1);
+
+ /* Name block */
+ if (UnicodeLength(File->Name) == 0x00) {
+ OBEXAddBlock(req, &Current, 0x01, NULL, 0);
+ } else {
+ CopyUnicodeString(req2,File->Name);
+ OBEXAddBlock(req, &Current, 0x01, req2, UnicodeLength(req2)*2+2);
+ }
+ } else {
+ File->Folder = false;
+
+ if (File->ID_FullName[0] == 0x00) {
+#ifndef xxxx
+ error = OBEXGEN_Connect(s,OBEX_None);
+#else
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+#endif
+ if (error != ERR_NONE) return error;
+
+ EncodeUnicode(File->Name,"one",3);
+
+ if (strcmp(s->CurrentConfig->Model,"seobex")) {
+ strcpy(req2,"x-obex/capability");
+// strcpy(req2,"x-obex/object-profile");
+
+ /* Type block */
+ OBEXAddBlock(req, &Current, 0x42, req2, strlen(req2)+1);
+ } else {
+ EncodeUnicode(req2,"telecom/devinfo.txt",19);
+
+ /* Name block */
+ OBEXAddBlock(req, &Current, 0x01, req2, UnicodeLength(req2)*2+2);
+ }
+ } else {
+// error = OBEXGEN_Connect(s,OBEX_None);
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+ if (error != ERR_NONE) return error;
+
+ if (strcmp(s->CurrentConfig->Model,"seobex")) {
+ smprintf(s,"Changing to root\n");
+ error = OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ do {
+ OBEXGEN_FindNextDir(File->ID_FullName, &Pos, req2);
+ smprintf(s,"%s %i %i\n",DecodeUnicodeString(req2),Pos,strlen(File->ID_FullName));
+ if (Pos == strlen(File->ID_FullName)) break;
+ smprintf(s,"Changing path down\n");
+ error=OBEXGEN_ChangePath(s, req2, 2);
+ if (error != ERR_NONE) return error;
+ } while (1);
+ } else {
+ EncodeUnicode(req2,File->ID_FullName,strlen(File->ID_FullName));
+ }
+ CopyUnicodeString(File->Name,req2);
+
+ s->Phone.Data.File = File;
+
+ Current = 0;
+ /* Name block */
+ OBEXAddBlock(req, &Current, 0x01, req2, UnicodeLength(req2)*2+2);
+ }
+ }
+ }
+
+ if (s->Phone.Data.Priv.OBEXGEN.Service == OBEX_BrowsingFolders) {
+ /* connection ID block */
+ req[Current++] = 0xCB; // ID
+ req[Current++] = 0x00; req[Current++] = 0x00;
+ req[Current++] = 0x00; req[Current++] = 0x01;
+ }
+
+ smprintf(s, "Getting file info from filesystem\n");
+ error=GSM_WaitFor (s, req, Current, 0x03, 4, ID_GetFileInfo);
+ if (error != ERR_NONE) return error;
+
+ s->Phone.Data.Priv.OBEXGEN.FileLastPart = false;
+
+ while (!s->Phone.Data.Priv.OBEXGEN.FileLastPart) {
+ Current = 0;
+ if (s->Phone.Data.Priv.OBEXGEN.Service == OBEX_BrowsingFolders) {
+ /* connection ID block */
+ req[Current++] = 0xCB; // ID
+ req[Current++] = 0x00; req[Current++] = 0x00;
+ req[Current++] = 0x00; req[Current++] = 0x01;
+ }
+ smprintf(s, "Getting file part from filesystem\n");
+ error=GSM_WaitFor (s, req, Current, 0x83, 4, ID_GetFile);
+ if (error != ERR_NONE) return error;
+ }
+ return ERR_EMPTY;
+}
+
+GSM_Error OBEXGEN_GetFilePart(GSM_StateMachine *s, GSM_File *File)
+{
+ return OBEXGEN_PrivGetFilePart(s,File,false);
+}
+
+static GSM_Error OBEXGEN_GetNextFileFolder(GSM_StateMachine *s, GSM_File *File, bool start)
+{
+ GSM_Phone_OBEXGENData *Priv = &s->Phone.Data.Priv.OBEXGEN;
+ GSM_Error error;
+ unsigned char Line[500],Line2[500],*name,*size;
+ int Pos,i,j,num,pos2,Current,z;
+
+ if (start) {
+ if (!strcmp(s->CurrentConfig->Model,"seobex")) return ERR_NOTSUPPORTED;
+
+ Priv->Files[0].Folder = true;
+ Priv->Files[0].Level = 1;
+ Priv->Files[0].Name[0] = 0;
+ Priv->Files[0].Name[1] = 0;
+ Priv->Files[0].ID_FullName[0] = 0;
+ Priv->Files[0].ID_FullName[1] = 0;
+
+ Priv->FilesLocationsUsed = 1;
+ Priv->FilesLocationsCurrent = 0;
+ Priv->FileLev = 1;
+
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Changing to root\n");
+ error = OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+
+ Current = 0;
+ }
+
+ while (1) {
+ if (Priv->FilesLocationsCurrent == Priv->FilesLocationsUsed) {
+ dbgprintf("Last file\n");
+ return ERR_EMPTY;
+ }
+
+ strcpy(File->ID_FullName,Priv->Files[Priv->FilesLocationsCurrent].ID_FullName);
+ File->Level = Priv->Files[Priv->FilesLocationsCurrent].Level;
+ File->Folder = Priv->Files[Priv->FilesLocationsCurrent].Folder;
+ CopyUnicodeString(File->Name,Priv->Files[Priv->FilesLocationsCurrent].Name);
+ Priv->FilesLocationsCurrent++;
+
+ if (File->Folder) {
+ if (File->Level < Priv->FileLev) {
+ for (i=0;i<File->Level;i++) {
+ smprintf(s,"Changing path up\n");
+ error=OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+ }
+ }
+
+ smprintf(s,"Level %i %i\n",File->Level,Priv->FileLev);
+
+ File->Buffer = NULL;
+ File->Used = 0;
+ OBEXGEN_PrivGetFilePart(s, File,true);
+
+ num = 0;
+ Pos = 0;
+ while (1) {
+ MyGetLine(File->Buffer, &Pos, Line, File->Used);
+ if (strlen(Line) == 0) break;
+ name = strstr(Line,"folder name=\"");
+ if (name != NULL) {
+ name += 13;
+ j = 0;
+ while(1) {
+ if (name[j] == '"') break;
+ j++;
+ }
+ name[j] = 0;
+
+ if (strcmp(name,".")) num++;
+ }
+ name = strstr(Line,"file name=\"");
+ if (name != NULL) num++;
+ }
+ if (num != 0) {
+ i = Priv->FilesLocationsUsed-1;
+ while (1) {
+ if (i==Priv->FilesLocationsCurrent-1) break;
+ memcpy(&Priv->Files[i+num],&Priv->Files[i],sizeof(GSM_File));
+ i--;
+ }
+ }
+
+ Pos = 0;
+ pos2 = 0;
+ while (1) {
+ MyGetLine(File->Buffer, &Pos, Line, File->Used);
+ if (strlen(Line) == 0) break;
+ strcpy(Line2,Line);
+ name = strstr(Line2,"folder name=\"");
+ if (name != NULL) {
+ name += 13;
+ j = 0;
+ while(1) {
+ if (name[j] == '"') break;
+ j++;
+ }
+ name[j] = 0;
+ if (strcmp(name,".")) {
+ dbgprintf("copying folder %s to %i parent %i\n",name,Priv->FilesLocationsCurrent+pos2,Priv->FilesLocationsCurrent);
+ strcpy(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,File->ID_FullName);
+ if (strlen(File->ID_FullName) != 0) strcat(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,"\\");
+ strcat(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,name);
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].Level = File->Level+1;
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].Folder = true;
+ EncodeUnicode(Priv->Files[Priv->FilesLocationsCurrent+pos2].Name,name,strlen(name));
+ Priv->FilesLocationsUsed++;
+ pos2++;
+ }
+ }
+ strcpy(Line2,Line);
+ name = strstr(Line2,"file name=\"");
+ if (name != NULL) {
+ name += 11;
+ j = 0;
+ while(1) {
+ if (name[j] == '"') break;
+ j++;
+ }
+ name[j] = 0;
+ dbgprintf("copying file %s to %i\n",name,Priv->FilesLocationsCurrent+pos2);
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].Level = File->Level+1;
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].Folder = false;
+ strcpy(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,File->ID_FullName);
+ if (strlen(File->ID_FullName) != 0) strcat(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,"\\");
+ strcat(Priv->Files[Priv->FilesLocationsCurrent+pos2].ID_FullName,name);
+ EncodeUnicode(Priv->Files[Priv->FilesLocationsCurrent+pos2].Name,name,strlen(name));
+
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].Used = 0;
+ strcpy(Line2,Line);
+ size = strstr(Line2,"size=\"");
+ if (size != NULL) Priv->Files[Priv->FilesLocationsCurrent+pos2].Used = atoi(size+6);
+
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].ModifiedEmpty = true;
+ strcpy(Line2,Line);
+ size = strstr(Line2,"modified=\"");
+ if (size != NULL) {
+ Priv->Files[Priv->FilesLocationsCurrent+pos2].ModifiedEmpty = false;
+ ReadVCALDateTime(size+10, &Priv->Files[Priv->FilesLocationsCurrent+pos2].Modified);
+ }
+ Priv->FilesLocationsUsed++;
+ pos2++;
+ }
+ }
+
+ z = Priv->FilesLocationsCurrent;
+ if (z != 1) {
+ while (1) {
+ if (z == Priv->FilesLocationsUsed) break;
+ if (Priv->Files[z].Folder) {
+ if (Priv->Files[z].Level > File->Level) {
+ smprintf(s,"Changing path down\n");
+ error=OBEXGEN_ChangePath(s, File->Name, 2);
+ if (error != ERR_NONE) return error;
+ }
+ break;
+ }
+ z++;
+ }
+ }
+
+ Priv->FileLev = File->Level;
+ free(File->Buffer);
+ } else {
+ File->Used = Priv->Files[Priv->FilesLocationsCurrent-1].Used;
+ File->ModifiedEmpty = Priv->Files[Priv->FilesLocationsCurrent-1].ModifiedEmpty;
+ if (!File->ModifiedEmpty) {
+ memcpy(&File->Modified,&Priv->Files[Priv->FilesLocationsCurrent-1].Modified,sizeof(GSM_DateTime));
+ }
+ File->ReadOnly = false;
+ File->Protected = false;
+ File->Hidden = false;
+ File->System = false;
+
+ }
+ return ERR_NONE;
+ }
+}
+
+static GSM_Error OBEXGEN_DeleteFile(GSM_StateMachine *s, unsigned char *ID)
+{
+ GSM_Error error;
+ unsigned int Current = 0, Pos;
+ unsigned char req[200],req2[200];
+
+ if (!strcmp(s->CurrentConfig->Model,"seobex")) return ERR_NOTSUPPORTED;
+
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Changing to root\n");
+ error = OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ do {
+ OBEXGEN_FindNextDir(ID, &Pos, req2);
+ smprintf(s,"%s %i %i\n",DecodeUnicodeString(req2),Pos,strlen(ID));
+ if (Pos == strlen(ID)) break;
+ smprintf(s,"Changing path down\n");
+ error=OBEXGEN_ChangePath(s, req2, 2);
+ if (error != ERR_NONE) return error;
+ } while (1);
+
+ /* Name block */
+ OBEXAddBlock(req, &Current, 0x01, req2, UnicodeLength(req2)*2+2);
+
+ /* connection ID block */
+ req[Current++] = 0xCB; // ID
+ req[Current++] = 0x00; req[Current++] = 0x00;
+ req[Current++] = 0x00; req[Current++] = 0x01;
+
+ return GSM_WaitFor (s, req, Current, 0x82, 4, ID_AddFile);
+}
+
+static GSM_Error OBEXGEN_AddFolder(GSM_StateMachine *s, GSM_File *File)
+{
+ GSM_Error error;
+ unsigned char req2[200];
+ unsigned int Pos;
+
+ if (!strcmp(s->CurrentConfig->Model,"seobex")) return ERR_NOTSUPPORTED;
+
+ error = OBEXGEN_Connect(s,OBEX_BrowsingFolders);
+ if (error != ERR_NONE) return error;
+
+ smprintf(s,"Changing to root\n");
+ error = OBEXGEN_ChangePath(s, NULL, 2);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ do {
+ OBEXGEN_FindNextDir(File->ID_FullName, &Pos, req2);
+ smprintf(s,"%s %i %i\n",DecodeUnicodeString(req2),Pos,strlen(File->ID_FullName));
+ smprintf(s,"Changing path down\n");
+ error=OBEXGEN_ChangePath(s, req2, 2);
+ if (error != ERR_NONE) return error;
+ if (Pos == strlen(File->ID_FullName)) break;
+ } while (1);
+
+ smprintf(s,"Adding directory\n");
+ return OBEXGEN_ChangePath(s, File->Name, 0);
+}
+
+GSM_Reply_Function OBEXGENReplyFunctions[] = {
+ /* CONTINUE block */
+ {OBEXGEN_ReplyAddFilePart, "\x90",0x00,0x00,ID_AddFile },
+ {OBEXGEN_ReplyGetFilePart, "\x90",0x00,0x00,ID_GetFile },
+ {OBEXGEN_ReplyGetFileInfo, "\x90",0x00,0x00,ID_GetFileInfo },
+
+ /* OK block */
+ {OBEXGEN_ReplyChangePath, "\xA0",0x00,0x00,ID_SetPath },
+ {OBEXGEN_ReplyConnect, "\xA0",0x00,0x00,ID_Initialise },
+ {OBEXGEN_ReplyAddFilePart, "\xA0",0x00,0x00,ID_AddFile },
+ {OBEXGEN_ReplyGetFilePart, "\xA0",0x00,0x00,ID_GetFile },
+ {OBEXGEN_ReplyGetFileInfo, "\xA0",0x00,0x00,ID_GetFileInfo },
+
+ /* FOLDER CREATED block */
+ {OBEXGEN_ReplyChangePath, "\xA1",0x00,0x00,ID_SetPath },
+
+ /* NOT UNDERSTAND block */
+ {OBEXGEN_ReplyAddFilePart, "\xC0",0x00,0x00,ID_AddFile },
+
+ /* FORBIDDEN block */
+ {OBEXGEN_ReplyChangePath, "\xC3",0x00,0x00,ID_SetPath },
+ {OBEXGEN_ReplyGetFilePart, "\xC3",0x00,0x00,ID_GetFile },
+
+ /* NOT FOUND block */
+ {OBEXGEN_ReplyGetFilePart, "\xC4",0x00,0x00,ID_GetFile },
+
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions OBEXGENPhone = {
+ "obex|seobex",
+ OBEXGENReplyFunctions,
+ OBEXGEN_Initialise,
+ NONEFUNCTION, /* Terminate */
+ GSM_DispatchMessage,
+ NOTIMPLEMENTED, /* ShowStartInfo */
+ NONEFUNCTION, /* GetManufacturer */
+ NONEFUNCTION, /* GetModel */
+ NONEFUNCTION, /* GetFirmware */
+ NOTIMPLEMENTED, /* GetIMEI */
+ NOTIMPLEMENTED, /* GetOriginalIMEI */
+ NOTIMPLEMENTED, /* GetManufactureMonth */
+ NOTIMPLEMENTED, /* GetProductCode */
+ NOTIMPLEMENTED, /* GetHardware */
+ NOTIMPLEMENTED, /* GetPPM */
+ NOTIMPLEMENTED, /* GetSIMIMSI */
+ NOTIMPLEMENTED, /* GetDateTime */
+ NOTIMPLEMENTED, /* SetDateTime */
+ NOTIMPLEMENTED, /* GetAlarm */
+ NOTIMPLEMENTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTIMPLEMENTED, /* PressKey */
+ NOTIMPLEMENTED, /* Reset */
+ NOTIMPLEMENTED, /* ResetPhoneSettings */
+ NOTIMPLEMENTED, /* EnterSecurityCode */
+ NOTIMPLEMENTED, /* GetSecurityStatus */
+ NOTIMPLEMENTED, /* GetDisplayStatus */
+ NOTIMPLEMENTED, /* SetAutoNetworkLogin */
+ NOTIMPLEMENTED, /* GetBatteryCharge */
+ NOTIMPLEMENTED, /* GetSignalQuality */
+ NOTIMPLEMENTED, /* GetNetworkInfo */
+ NOTIMPLEMENTED, /* GetCategory */
+ NOTSUPPORTED, /* AddCategory */
+ NOTIMPLEMENTED, /* GetCategoryStatus */
+ NOTIMPLEMENTED, /* GetMemoryStatus */
+ NOTIMPLEMENTED, /* GetMemory */
+ NOTIMPLEMENTED, /* GetNextMemory */
+ NOTIMPLEMENTED, /* SetMemory */
+ NOTIMPLEMENTED, /* AddMemory */
+ NOTIMPLEMENTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTIMPLEMENTED, /* GetSpeedDial */
+ NOTIMPLEMENTED, /* SetSpeedDial */
+ NOTIMPLEMENTED, /* GetSMSC */
+ NOTIMPLEMENTED, /* SetSMSC */
+ NOTIMPLEMENTED, /* GetSMSStatus */
+ NOTIMPLEMENTED, /* GetSMS */
+ NOTIMPLEMENTED, /* GetNextSMS */
+ NOTIMPLEMENTED, /* SetSMS */
+ NOTIMPLEMENTED, /* AddSMS */
+ NOTIMPLEMENTED, /* DeleteSMS */
+ NOTIMPLEMENTED, /* SendSMSMessage */
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOTIMPLEMENTED, /* SetIncomingSMS */
+ NOTIMPLEMENTED, /* SetIncomingCB */
+ NOTIMPLEMENTED, /* GetSMSFolders */
+ NOTIMPLEMENTED, /* AddSMSFolder */
+ NOTIMPLEMENTED, /* DeleteSMSFolder */
+ NOTIMPLEMENTED, /* DialVoice */
+ NOTIMPLEMENTED, /* AnswerCall */
+ NOTIMPLEMENTED, /* CancelCall */
+ NOTIMPLEMENTED, /* HoldCall */
+ NOTIMPLEMENTED, /* UnholdCall */
+ NOTIMPLEMENTED, /* ConferenceCall */
+ NOTIMPLEMENTED, /* SplitCall */
+ NOTIMPLEMENTED, /* TransferCall */
+ NOTIMPLEMENTED, /* SwitchCall */
+ NOTIMPLEMENTED, /* GetCallDivert */
+ NOTIMPLEMENTED, /* SetCallDivert */
+ NOTIMPLEMENTED, /* CancelAllDiverts */
+ NOTIMPLEMENTED, /* SetIncomingCall */
+ NOTIMPLEMENTED, /* SetIncomingUSSD */
+ NOTIMPLEMENTED, /* SendDTMF */
+ NOTIMPLEMENTED, /* GetRingtone */
+ NOTIMPLEMENTED, /* SetRingtone */
+ NOTIMPLEMENTED, /* GetRingtonesInfo */
+ NOTIMPLEMENTED, /* DeleteUserRingtones */
+ NOTIMPLEMENTED, /* PlayTone */
+ NOTIMPLEMENTED, /* GetWAPBookmark */
+ NOTIMPLEMENTED, /* SetWAPBookmark */
+ NOTIMPLEMENTED, /* DeleteWAPBookmark */
+ NOTIMPLEMENTED, /* GetWAPSettings */
+ NOTIMPLEMENTED, /* SetWAPSettings */
+ NOTIMPLEMENTED, /* GetMMSSettings */
+ NOTIMPLEMENTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ NOTIMPLEMENTED, /* GetBitmap */
+ NOTIMPLEMENTED, /* SetBitmap */
+ NOTIMPLEMENTED, /* GetToDoStatus */
+ NOTIMPLEMENTED, /* GetToDo */
+ NOTIMPLEMENTED, /* GetNextToDo */
+ NOTIMPLEMENTED, /* SetToDo */
+ NOTIMPLEMENTED, /* AddToDo */
+ NOTIMPLEMENTED, /* DeleteToDo */
+ NOTIMPLEMENTED, /* DeleteAllToDo */
+ NOTIMPLEMENTED, /* GetCalendarStatus */
+ NOTIMPLEMENTED, /* GetCalendar */
+ NOTIMPLEMENTED, /* GetNextCalendar */
+ NOTIMPLEMENTED, /* SetCalendar */
+ NOTIMPLEMENTED, /* AddCalendar */
+ NOTIMPLEMENTED, /* DeleteCalendar */
+ NOTIMPLEMENTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTIMPLEMENTED, /* GetProfile */
+ NOTIMPLEMENTED, /* SetProfile */
+ NOTIMPLEMENTED, /* GetFMStation */
+ NOTIMPLEMENTED, /* SetFMStation */
+ NOTIMPLEMENTED, /* ClearFMStations */
+ OBEXGEN_GetNextFileFolder,
+ OBEXGEN_GetFilePart,
+ OBEXGEN_AddFilePart,
+ NOTIMPLEMENTED, /* GetFileSystemStatus */
+ OBEXGEN_DeleteFile,
+ OBEXGEN_AddFolder,
+ NOTIMPLEMENTED, /* GetGPRSAccessPoint */
+ NOTIMPLEMENTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/obex/obexgen.h b/gammu/emb/common/phone/obex/obexgen.h
new file mode 100644
index 0000000..466fef5
--- a/dev/null
+++ b/gammu/emb/common/phone/obex/obexgen.h
@@ -0,0 +1,38 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef obexgen_h
+#define obexgen_h
+
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "../../service/gsmmisc.h"
+#include "../../service/sms/gsmsms.h"
+
+#ifndef GSM_USED_IRDAOBEX
+# define GSM_USED_IRDAOBEX
+#endif
+#ifndef GSM_USED_BLUEOBEX
+# define GSM_USED_BLUEOBEX
+#endif
+
+typedef enum {
+ OBEX_None = 1,
+ OBEX_BrowsingFolders
+} OBEX_Service;
+
+typedef struct {
+ int FileLev;
+ int FilesLocationsUsed;
+ int FilesLocationsCurrent;
+ GSM_File Files[500];
+ bool FileLastPart;
+
+ int FrameSize;
+ OBEX_Service Service;
+} GSM_Phone_OBEXGENData;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/pfunc.c b/gammu/emb/common/phone/pfunc.c
new file mode 100644
index 0000000..a03a81d
--- a/dev/null
+++ b/gammu/emb/common/phone/pfunc.c
@@ -0,0 +1,138 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../gsmstate.h"
+#include "../service/sms/gsmsms.h"
+#include "../misc/coding/coding.h"
+
+/* These SMS layouts are used exactly as written in Nokia DCT3 phones.
+ * In AT module(s) we have to use some layouts to convert AT frame to format
+ * understod by SMS module. To share source the same layouts are used */
+GSM_SMSMessageLayout PHONE_SMSDeliver = {
+ 35 /* SMS Text */, 16 /* Phone number */,
+ 0 /* SMSC Number */, 14 /* TPDCS */,
+ 28 /* SendingDateTime */, 255 /* SMSCDateTime */,
+ 255 /* TPStatus */, 15 /* TPUDL */,
+ 255 /* TPVP */, 12 /* firstbyte */,
+ 255 /* TPMR */, 13 /* TPPID */};
+GSM_SMSMessageLayout PHONE_SMSSubmit = {
+ 36 /* SMS Text */, 17 /* Phone number */,
+ 0 /* SMSC Number */, 15 /* TPDCS */,
+ 255 /* SendingDateTime */, 255 /* SMSCDateTime */,
+ 255 /* TPStatus */, 16 /* TPUDL */,
+ 29 /* TPVP */, 12 /* firstbyte */,
+ 13 /* TPMR */, 14 /* TPPID */};
+GSM_SMSMessageLayout PHONE_SMSStatusReport = {
+ 255 /* SMS Text */, 15 /* Phone number */,
+ 0 /* SMSC Number */, 255 /* TPDCS */,
+ 27 /* SendingDateTime */, 34 /* SMSCDateTime */,
+ 14 /* TPStatus */, 255 /* TPUDL */,
+ 255 /* TPVP */, 12 /* firstbyte */,
+ 13 /* TPMR */, 255 /* TPPID?? */};
+
+GSM_Error PHONE_GetSMSFolders(GSM_StateMachine *s, GSM_SMSFolders *folders)
+{
+ folders->Number=2;
+ EncodeUnicode(folders->Folder[0].Name,GetMsg(s->msg,"Inbox"),strlen(GetMsg(s->msg,"Inbox")));
+ EncodeUnicode(folders->Folder[1].Name,GetMsg(s->msg,"Outbox"),strlen(GetMsg(s->msg,"Outbox")));
+ folders->Folder[0].InboxFolder = true;
+ folders->Folder[1].InboxFolder = false;
+ folders->Folder[0].Memory = MEM_SM;
+ folders->Folder[1].Memory = MEM_SM;
+ return ERR_NONE;
+}
+
+void GSM_CreateFirmwareNumber(GSM_StateMachine *s)
+{
+ StringToDouble(s->Phone.Data.Version, &s->Phone.Data.VerNum);
+ dbgprintf("Number version is \"%f\"\n", s->Phone.Data.VerNum);
+}
+
+GSM_Error PHONE_EncodeSMSFrame(GSM_StateMachine *s, GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout, int *length, bool clear)
+{
+ GSM_Error error;
+
+ if (SMS->SMSC.Location!=0) {
+ error = s->Phone.Functions->GetSMSC(s, &SMS->SMSC);
+ if (error != ERR_NONE) return error;
+ SMS->SMSC.Location = 0;
+ }
+ if (SMS->PDU == SMS_Deliver) {
+ if (SMS->SMSC.Number[0] == 0x00 && SMS->SMSC.Number[1] == 0x00) {
+ return ERR_EMPTYSMSC;
+ }
+ }
+ return GSM_EncodeSMSFrame(SMS, buffer, Layout, length, clear);
+}
+
+GSM_Error PHONE_Terminate(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ if (s->Phone.Data.EnableIncomingCB==true) {
+ error=s->Phone.Functions->SetIncomingCB(s,false);
+ if (error!=ERR_NONE) return error;
+ }
+ if (s->Phone.Data.EnableIncomingSMS==true) {
+ error=s->Phone.Functions->SetIncomingSMS(s,false);
+ if (error!=ERR_NONE) return error;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error PHONE_RTTLPlayOneNote(GSM_StateMachine *s, GSM_RingNote note, bool first)
+{
+ int duration, Hz;
+ GSM_Error error;
+
+ Hz=GSM_RingNoteGetFrequency(note);
+
+ error=s->Phone.Functions->PlayTone(s,Hz,5,first);
+ if (error!=ERR_NONE) return error;
+
+ duration = GSM_RingNoteGetFullDuration(note);
+
+ /* Is it correct ? Experimental values here */
+ switch (note.Style) {
+ case StaccatoStyle:
+ my_sleep (7500);
+ error=s->Phone.Functions->PlayTone(s,0,0,false);
+ if (error != ERR_NONE) return error;
+ my_sleep ((1400000/note.Tempo*duration)-(7500));
+ break;
+ case ContinuousStyle:
+ my_sleep (1400000/note.Tempo*duration);
+ break;
+ case NaturalStyle:
+ my_sleep (1400000/note.Tempo*duration-50);
+ error=s->Phone.Functions->PlayTone(s,0,0,false);
+ if (error != ERR_NONE) return error;
+ my_sleep (50);
+ break;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error PHONE_Beep(GSM_StateMachine *s)
+{
+ GSM_Error error;
+
+ error=s->Phone.Functions->PlayTone(s, 4000, 5,true);
+ if (error!=ERR_NONE) return error;
+
+ my_sleep(500);
+
+ return s->Phone.Functions->PlayTone(s,255*255,0,false);
+}
+
+GSM_Error NoneReply(GSM_Protocol_Message msg, GSM_StateMachine *s)
+{
+ smprintf(s,"None answer\n");
+ return ERR_NONE;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/pfunc.h b/gammu/emb/common/phone/pfunc.h
new file mode 100644
index 0000000..859085c
--- a/dev/null
+++ b/gammu/emb/common/phone/pfunc.h
@@ -0,0 +1,30 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef phone_common2_h
+#define phone_common2_h
+
+#include "../service/sms/gsmsms.h"
+
+extern GSM_SMSMessageLayout PHONE_SMSSubmit;
+extern GSM_SMSMessageLayout PHONE_SMSDeliver;
+extern GSM_SMSMessageLayout PHONE_SMSStatusReport;
+
+GSM_Error PHONE_GetSMSFolders (GSM_StateMachine *s, GSM_SMSFolders *folders);
+
+void GSM_CreateFirmwareNumber (GSM_StateMachine *s);
+
+GSM_Error PHONE_EncodeSMSFrame (GSM_StateMachine *s, GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout, int *length, bool clear);
+
+GSM_Error PHONE_Terminate (GSM_StateMachine *s);
+
+GSM_Error PHONE_RTTLPlayOneNote (GSM_StateMachine *s, GSM_RingNote note, bool first);
+
+GSM_Error PHONE_Beep (GSM_StateMachine *s);
+
+GSM_Error NoneReply(GSM_Protocol_Message msg, GSM_StateMachine *s);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/symbian/mroutgen.c b/gammu/emb/common/phone/symbian/mroutgen.c
new file mode 100644
index 0000000..2c339be
--- a/dev/null
+++ b/gammu/emb/common/phone/symbian/mroutgen.c
@@ -0,0 +1,220 @@
+/* (c) 2003 by Marcin Wiacek */
+/* EXPERIMENTAL. NOT FUNCTIONAL */
+
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "../../misc/coding/coding.h"
+#include "../../service/gsmmisc.h"
+
+#ifdef GSM_ENABLE_MROUTERGEN
+
+GSM_Error MROUTERGEN_StartModem(GSM_StateMachine *s)
+{
+ return ERR_NONE;
+}
+
+GSM_Error MROUTERGEN_Initialise(GSM_StateMachine *s)
+{
+
+unsigned char req2[]={0xFF,0x03,0x80,0xFD,0x04,0x01,0x00,0x09,0x11,
+0x05,0x00,0x01,0x03,0x77,0xBC};
+
+unsigned char req0[]={
+0xFF,0x7D,0x23,0xC0,0x21,0x7D,0x21,0x7D,0x24,0x7D,
+0x20,0x7D,0x2E,0x7D,0x22,0x7D,0x26,0x7D,0x20,0x7D,
+0x20,0x7D,0x20,0x7D,0x20,0x7D,0x27,0x7D,0x22,0x7D,
+0x28,0x7D,0x22,0x69,0x27,
+0x7E,
+
+0x7E,
+0xFF,0x7D,0x23,0xC0,0x21,0x7D,0x22,0x7D,0x21,0x7D,
+0x20,0x7D,0x34,0x7D,0x25,0x7D,0x26,0x87,0xB9,0x7D,
+0x28,0xCF,0x7D,0x22,0x7D,0x26,0x7D,0x20,0x7D,0x20,
+0x7D,0x20,0x7D,0x20,0x7D,0x27,0x7D,0x22,0x7D,0x28,
+0x7D,0x22,0x82,0xB0,
+0x7E,
+
+0x7E,
+0xFF,0x7D,0x23,0xC0,0x21,0x7D,0x21,0x7D,0x24,0x7D,
+0x20,0x7D,0x2E,0x7D,0x22,0x7D,0x26,0x7D,0x20,0x7D,
+0x20,0x7D,0x20,0x7D,0x20,0x7D,0x27,0x7D,0x22,0x7D,
+0x28,0x7D,0x22,0x69,0x27,
+0x7E,
+
+0x7E,
+0xFF,0x7D,0x23,0xC0,0x21,0x7D,0x22,0x7D,0x21,0x7D,
+0x20,0x7D,0x34,0x7D,0x25,0x7D,0x26,0x87,0xB9,0x7D,
+0x28,0xCF,0x7D,0x22,0x7D,0x26,0x7D,0x20,0x7D,0x20,
+0x7D,0x20,0x7D,0x20,0x7D,0x27,0x7D,0x22,0x7D,0x28,
+0x7D,0x22,0x82,0xB0};
+
+unsigned char req[]={0x2F,0x45,0x00,0x00,0x34,0x00,0xC6,
+ 0x00,0x00,0x0A,0x0D,0x59,0xB4,0xA9,0xFE,0x01,
+ 0x44,0xA9,0xFE,0x01,0x0A,0x04,0x09,0x0B,0xB8,
+ 0x10,0x00,0x02,0x79,0x69,0x81,0x74,0x99,0x50,
+ 0x18,0x60,0x00,0x9D,0x0C,0x00,0x00,0x08,0x00,
+ 0x00,0x00,0x12,0x00,0x41,0x00,0x01,0x14,0x00,
+ 0x00,0xBA,0x4A};
+
+unsigned char req3[]={0x2D,0x5C,0x0D,0xEC,0x4C,0x10,0x26,0x08,0x00,
+0x00,0x00,0x12,0x00,0x15,0x00,0x01,0x08,0x00,0x00,
+0xAD,0xDE};
+smprintf(s,"writing\n");
+ GSM_WaitFor (s, req0, 164, 0x00, 200, ID_Initialise);
+smprintf(s,"writing\n");
+ GSM_WaitFor (s, req3, 21, 0x00, 200, ID_Initialise);
+ GSM_WaitFor (s, req, 55, 0x00, 200, ID_Initialise);
+
+ GSM_WaitFor (s, req2, 15, 0x00, 200, ID_Initialise);
+
+
+ while (1) {
+ GSM_ReadDevice(s,false);
+ }
+
+
+
+ while (1) {
+ GSM_ReadDevice(s,false);
+ }
+
+
+return ERR_UNKNOWN;
+}
+
+static GSM_Reply_Function MROUTERGENReplyFunctions[] = {
+ {NULL, "\x00",0x00,0x00,ID_None }
+};
+
+GSM_Phone_Functions MROUTERGENPhone = {
+ "mrouter",
+ MROUTERGENReplyFunctions,
+ MROUTERGEN_Initialise,
+ NONEFUNCTION, /* Terminate */
+ GSM_DispatchMessage,
+ NOTSUPPORTED, /* ShowStartInfo */
+ NONEFUNCTION, /* GetManufacturer */
+ NONEFUNCTION, /* GetModel */
+ NONEFUNCTION, /* GetFirmware */
+ NOTIMPLEMENTED, /* GetIMEI */
+ NOTSUPPORTED, /* GetOriginalIMEI */
+ NOTSUPPORTED, /* GetManufactureMonth */
+ NOTSUPPORTED, /* GetProductCode */
+ NOTSUPPORTED, /* GetHardware */
+ NOTSUPPORTED, /* GetPPM */
+ NOTSUPPORTED, /* GetSIMIMSI */
+ NOTSUPPORTED, /* GetDateTime */
+ NOTSUPPORTED, /* SetDateTime */
+ NOTSUPPORTED, /* GetAlarm */
+ NOTSUPPORTED, /* SetAlarm */
+ NOTSUPPORTED, /* GetLocale */
+ NOTSUPPORTED, /* SetLocale */
+ NOTSUPPORTED, /* PressKey */
+ NOTSUPPORTED, /* Reset */
+ NOTSUPPORTED, /* ResetPhoneSettings */
+ NOTSUPPORTED, /* EnterSecurityCode */
+ NOTSUPPORTED, /* GetSecurityStatus */
+ NOTSUPPORTED, /* GetDisplayStatus */
+ NOTSUPPORTED, /* SetAutoNetworkLogin */
+ NOTSUPPORTED, /* GetBatteryCharge */
+ NOTSUPPORTED, /* GetSignalQuality */
+ NOTSUPPORTED, /* GetNetworkInfo */
+ NOTSUPPORTED, /* GetCategory */
+ NOTSUPPORTED, /* GetCategoryStatus */
+ NOTSUPPORTED, /* GetMemoryStatus */
+ NOTSUPPORTED, /* GetMemory */
+ NOTSUPPORTED, /* GetNextMemory */
+ NOTSUPPORTED, /* SetMemory */
+ NOTSUPPORTED, /* AddMemory */
+ NOTSUPPORTED, /* DeleteMemory */
+ NOTIMPLEMENTED, /* DeleteAllMemory */
+ NOTSUPPORTED, /* GetSpeedDial */
+ NOTSUPPORTED, /* SetSpeedDial */
+ NOTSUPPORTED, /* GetSMSC */
+ NOTSUPPORTED, /* SetSMSC */
+ NOTSUPPORTED, /* GetSMSStatus */
+ NOTSUPPORTED, /* GetSMS */
+ NOTSUPPORTED, /* GetNextSMS */
+ NOTSUPPORTED, /* SetSMS */
+ NOTSUPPORTED, /* AddSMS */
+ NOTSUPPORTED, /* DeleteSMS */
+ NOTSUPPORTED, /* SendSMSMessage */
+ NOTSUPPORTED, /* SendSavedSMS */
+ NOTSUPPORTED, /* SetIncomingSMS */
+ NOTSUPPORTED, /* SetIncomingCB */
+ NOTSUPPORTED, /* GetSMSFolders */
+ NOTSUPPORTED, /* AddSMSFolder */
+ NOTSUPPORTED, /* DeleteSMSFolder */
+ NOTSUPPORTED, /* DialVoice */
+ NOTSUPPORTED, /* AnswerCall */
+ NOTSUPPORTED, /* CancelCall */
+ NOTSUPPORTED, /* HoldCall */
+ NOTSUPPORTED, /* UnholdCall */
+ NOTSUPPORTED, /* ConferenceCall */
+ NOTSUPPORTED, /* SplitCall */
+ NOTSUPPORTED, /* TransferCall */
+ NOTSUPPORTED, /* SwitchCall */
+ NOTSUPPORTED, /* GetCallDivert */
+ NOTSUPPORTED, /* SetCallDivert */
+ NOTSUPPORTED, /* CancelAllDiverts */
+ NOTSUPPORTED, /* SetIncomingCall */
+ NOTSUPPORTED, /* SetIncomingUSSD */
+ NOTSUPPORTED, /* SendDTMF */
+ NOTSUPPORTED, /* GetRingtone */
+ NOTSUPPORTED, /* SetRingtone */
+ NOTSUPPORTED, /* GetRingtonesInfo */
+ NOTSUPPORTED, /* DeleteUserRingtones */
+ NOTSUPPORTED, /* PlayTone */
+ NOTSUPPORTED, /* GetWAPBookmark */
+ NOTSUPPORTED, /* SetWAPBookmark */
+ NOTSUPPORTED, /* DeleteWAPBookmark */
+ NOTSUPPORTED, /* GetWAPSettings */
+ NOTSUPPORTED, /* SetWAPSettings */
+ NOTSUPPORTED, /* GetMMSSettings */
+ NOTSUPPORTED, /* SetMMSSettings */
+ NOTSUPPORTED, /* GetSyncMLSettings */
+ NOTSUPPORTED, /* SetSyncMLSettings */
+ NOTSUPPORTED, /* GetChatSettings */
+ NOTSUPPORTED, /* SetChatSettings */
+ NOTSUPPORTED, /* GetBitmap */
+ NOTSUPPORTED, /* SetBitmap */
+ NOTSUPPORTED, /* GetToDoStatus */
+ NOTSUPPORTED, /* GetToDo */
+ NOTSUPPORTED, /* GetNextToDo */
+ NOTSUPPORTED, /* SetToDo */
+ NOTSUPPORTED, /* AddToDo */
+ NOTSUPPORTED, /* DeleteToDo */
+ NOTSUPPORTED, /* DeleteAllToDo */
+ NOTSUPPORTED, /* GetCalendarStatus */
+ NOTSUPPORTED, /* GetCalendar */
+ NOTSUPPORTED, /* GetNextCalendar */
+ NOTSUPPORTED, /* SetCalendar */
+ NOTSUPPORTED, /* AddCalendar */
+ NOTSUPPORTED, /* DeleteCalendar */
+ NOTSUPPORTED, /* DeleteAllCalendar */
+ NOTSUPPORTED, /* GetCalendarSettings */
+ NOTSUPPORTED, /* SetCalendarSettings */
+ NOTSUPPORTED, /* GetNote */
+ NOTSUPPORTED, /* GetProfile */
+ NOTSUPPORTED, /* SetProfile */
+ NOTSUPPORTED, /* GetFMStation */
+ NOTSUPPORTED, /* SetFMStation */
+ NOTSUPPORTED, /* ClearFMStations */
+ NOTSUPPORTED, /* GetNextFileFolder */
+ NOTSUPPORTED, /* GetFilePart */
+ NOTSUPPORTED, /* AddFilePart */
+ NOTSUPPORTED, /* GetFileSystemStatus */
+ NOTSUPPORTED, /* DeleteFile */
+ NOTSUPPORTED, /* AddFolder */
+ NOTSUPPORTED, /* GetGPRSAccessPoint */
+ NOTSUPPORTED /* SetGPRSAccessPoint */
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/phone/symbian/mroutgen.h b/gammu/emb/common/phone/symbian/mroutgen.h
new file mode 100644
index 0000000..af38fe5
--- a/dev/null
+++ b/gammu/emb/common/phone/symbian/mroutgen.h
@@ -0,0 +1,29 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef mroutgen_h
+#define mroutgen_h
+
+#include "../../gsmcomon.h"
+#include "../../gsmstate.h"
+#include "../../service/gsmmisc.h"
+#include "../../service/sms/gsmsms.h"
+
+#ifndef GSM_USED_MROUTERBLUE
+# define GSM_USED_MROUTERBLUE
+#endif
+#ifndef GSM_USED_IRDAMROUTER
+# define GSM_USED_IRDAMROUTER
+#endif
+#ifndef GSM_USED_BLUEMROUTER
+# define GSM_USED_BLUEMROUTER
+#endif
+
+typedef struct {
+ int fake;
+} GSM_Phone_MROUTERGENData;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/alcatel/alcabus.c b/gammu/emb/common/protocol/alcatel/alcabus.c
new file mode 100644
index 0000000..b5b5a30
--- a/dev/null
+++ b/gammu/emb/common/protocol/alcatel/alcabus.c
@@ -0,0 +1,255 @@
+/* (c) 2002-2003 by Michal Cihar
+ *
+ * Low level functions for communication with Alcatel One Touch phones.
+ *
+ * This code implements the protocol used for synchronisation with PC.
+ */
+#include "../../gsmstate.h"
+
+#if defined(GSM_ENABLE_ALCABUS)
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "alcabus.h"
+
+static GSM_Error ALCABUS_WriteMessage (GSM_StateMachine *s, unsigned char *data, int len, unsigned char type)
+{
+ GSM_Protocol_ALCABUSData *d = &s->Protocol.Data.ALCABUS;
+ unsigned char buffer[1024];
+ int size = 0;
+ int sent = 0;
+ int i = 0, checksum = 0;
+
+ if ((type == 0) && (len == 0)) return ERR_NONE;
+
+ buffer[0] = ALCATEL_HEADER;
+ buffer[1] = type;
+ switch (type) {
+ case ALCATEL_CONNECT:
+ buffer[2] = 0x0A;
+ buffer[3] = 0x04;
+ buffer[4] = 0x00;
+ size = 5;
+ d->next_frame = ALCATEL_CONNECT_ACK;
+ d->busy = true;
+ break;
+ case ALCATEL_DISCONNECT:
+ size = 2;
+ d->next_frame = ALCATEL_DISCONNECT_ACK;
+ d->busy = true;
+ break;
+ case ALCATEL_DATA:
+ buffer[2] = d->out_counter;
+
+ /* Increase outgoing packet counter */
+ if (d->out_counter == ALCATEL_MAX_COUNTER) d->out_counter = 0;
+ else d->out_counter++;
+
+ buffer[3] = '\0';
+ buffer[4] = len;
+ memcpy(buffer+5, data, len);
+ size = 5 + len;
+ d->next_frame = ALCATEL_ACK;
+ d->busy = true;
+ break;
+ case ALCATEL_ACK:
+ buffer[2] = d->in_counter;
+ if (d->in_counter == 0) d->in_counter = 1;
+ size = 3;
+ d->next_frame = ALCATEL_DATA;
+ break;
+ default:
+ /* In fact, other types probably can came just from mobile... */
+ smprintf(s,"WARNING: Wanted to send some unknown packet (%02X)\n", type);
+ return ERR_NOTIMPLEMENTED;
+ }
+
+ /* Calculate packet checksum */
+ for (i=0; i<size; i++) checksum ^= buffer[i];
+
+ buffer[size] = checksum;
+ size ++;
+
+ GSM_DumpMessageLevel2(s, buffer, size, type);
+ GSM_DumpMessageLevel3(s, buffer, size, type);
+ while (sent != size ) {
+ if ((i = s->Device.Functions->WriteDevice(s,buffer + sent, size - sent)) == 0) {
+ return ERR_DEVICEWRITEERROR;
+ }
+ sent += i;
+ }
+
+ if (type == ALCATEL_CONNECT || type == ALCATEL_DISCONNECT) {
+ /* For connect and disconnect we need a bit larger delay */
+// my_sleep(10);
+ while (d->busy) {
+ GSM_ReadDevice(s,true);
+ my_sleep(1);
+ i++;
+ if (i == 10) return ERR_TIMEOUT;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error ALCABUS_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Protocol_ALCABUSData *d = &s->Protocol.Data.ALCABUS;
+ int i;
+ int checksum = 0;
+
+ if (d->Msg.BufferUsed < d->Msg.Length + 1) {
+ d->Msg.BufferUsed = d->Msg.Length + 1;
+ d->Msg.Buffer = (unsigned char *)realloc(d->Msg.Buffer,d->Msg.BufferUsed);
+ }
+
+ /* Check for header */
+ if ((d->Msg.Length == 0) && (rx_char != ALCATEL_HEADER)) {
+ smprintf(s,"WARNING: Expecting alcatel header (%02X) but got (%02X)\n", ALCATEL_HEADER, rx_char);
+ return ERR_UNKNOWNRESPONSE;
+ /* Check for packet type */
+ } else if (d->Msg.Length == 1){
+ d->Msg.Type = rx_char;
+ /* Was it unexpected packet? */
+ if ((rx_char != d->next_frame) && (rx_char != ALCATEL_CONTROL)) {
+ smprintf(s,"WARNING: Expecting alcatel packet type (%02X) but got (%02X)\n", d->next_frame, rx_char);
+ }
+ /* Determine packet size */
+ switch (rx_char) {
+ case ALCATEL_ACK:
+ d->expected_size = 4;
+ break;
+ case ALCATEL_DATA:
+ /* Packet length is in it's header */
+ d->expected_size = -1;
+ break;
+ case ALCATEL_CONTROL:
+ d->expected_size = 4;
+ break;
+ case ALCATEL_CONNECT_ACK:
+ d->expected_size = 6;
+ break;
+ case ALCATEL_DISCONNECT_ACK:
+ d->expected_size = 3;
+ break;
+ default:
+ smprintf(s,"WARNING: Something went wrong, unknown packet received (%02X)\n", rx_char);
+ return ERR_UNKNOWNRESPONSE;
+ }
+ /* Check counter, we can probably ignore error here ;-) */
+ } else if ((d->Msg.Length == 2) && (d->Msg.Type == ALCATEL_DATA)) {
+ if (rx_char != d->in_counter) {
+ smprintf(s,"WARNING: Unexpected packet number, ignoring (expected %02X, received %02X)\n", d->in_counter, rx_char);
+ d->in_counter = rx_char;
+ }
+ /* Increase incoming packet counter */
+ if (d->in_counter == ALCATEL_MAX_COUNTER) d->in_counter = 0;
+ else d->in_counter++;
+ /* Read size for data packet */
+ } else if ((d->Msg.Length == 4) && (d->Msg.Type == ALCATEL_DATA)) {
+ /* Header till now + checksum */
+ d->expected_size = (int)rx_char + 6;
+ }
+
+ /* Write received byte into buffer */
+ d->Msg.Buffer[d->Msg.Length++] = rx_char;
+
+ /* Did we received whole packet? */
+ if (d->expected_size == d->Msg.Length) {
+ /* Check checksum */
+ for (i=0; i< (d->Msg.Length - 1); i++) checksum ^= d->Msg.Buffer[i];
+ if (checksum != d->Msg.Buffer[d->Msg.Length - 1]) {
+ /* We can only warn, as we don't know what should happend now... */
+ smprintf(s,"WARNING: Ignoring incorrect packet checksum!\n");
+ }
+
+ /* Was it data? */
+ if (d->Msg.Type == ALCATEL_DATA) {
+ /* Dispatch message */
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ /* Send ack */
+ ALCABUS_WriteMessage (s, 0, 0, ALCATEL_ACK);
+ /* Reset message length */
+ d->Msg.Length = 0;
+ /* Was it ack? */
+ } else if ((d->Msg.Type == ALCATEL_ACK) ||
+ (d->Msg.Type == ALCATEL_CONTROL) ||
+ (d->Msg.Type == ALCATEL_CONNECT_ACK) ||
+ (d->Msg.Type == ALCATEL_DISCONNECT_ACK)) {
+ /* TODO: check counter of ack? */
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "Received %s ack ",
+ (d->Msg.Type == ALCATEL_ACK) ? "normal" :
+ (d->Msg.Type == ALCATEL_CONTROL) ? "control" :
+ (d->Msg.Type == ALCATEL_CONNECT_ACK) ? "connect" :
+ (d->Msg.Type == ALCATEL_DISCONNECT_ACK) ? "disconnect" :
+ "BUG");
+ smprintf(s, "0x%02x / 0x%04x", d->Msg.Type, d->Msg.Length);
+ DumpMessage(s->di.df, s->di.dl, d->Msg.Buffer, d->Msg.Length);
+ fflush(s->di.df);
+ }
+ if (s->di.dl==DL_BINARY) {
+ smprintf(s,"%c",0x02); /* Receiving */
+ smprintf(s,"%c",d->Msg.Type);
+ smprintf(s,"%c",d->Msg.Length/256);
+ smprintf(s,"%c",d->Msg.Length%256);
+ for (i=0;i<d->Msg.Length;i++) smprintf(s,"%c",d->Msg.Buffer[i]);
+ }
+ if (d->Msg.Type != ALCATEL_CONTROL) {
+ d->next_frame = ALCATEL_DATA;
+ d->busy = false;
+ }
+ /* Reset message length */
+ d->Msg.Length = 0;
+ }
+
+ /* Was it unexpected type? */
+ if ((d->Msg.Type != d->next_frame) && (d->Msg.Type != ALCATEL_CONTROL)) {
+ return ERR_FRAMENOTREQUESTED;
+ }
+ } /* Last byte of packet */
+
+ return ERR_NONE;
+}
+
+static GSM_Error ALCABUS_Initialise(GSM_StateMachine *s)
+{
+ GSM_Protocol_ALCABUSData *d = &s->Protocol.Data.ALCABUS;
+
+ /* Initialise some variables */
+ d->Msg.BufferUsed = 0;
+ d->Msg.Buffer = NULL;
+ d->Msg.Length = 0;
+ d->Msg.Type = 0;
+ d->in_counter = 1;
+ d->out_counter = 0;
+ d->busy = false;
+
+ /* Initialise protocol */
+ dbgprintf ("Initializing binary mode\n");
+ return ALCABUS_WriteMessage (s, 0, 0, ALCATEL_CONNECT);
+}
+
+static GSM_Error ALCABUS_Terminate(GSM_StateMachine *s)
+{
+ /* Terminate protocol */
+ dbgprintf ("Closing binary mode\n");
+ return ALCABUS_WriteMessage (s, 0, 0, ALCATEL_DISCONNECT);
+}
+
+GSM_Protocol_Functions ALCABUSProtocol = {
+ ALCABUS_WriteMessage,
+ ALCABUS_StateMachine,
+ ALCABUS_Initialise,
+ ALCABUS_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/alcatel/alcabus.h b/gammu/emb/common/protocol/alcatel/alcabus.h
new file mode 100644
index 0000000..3b9b44e
--- a/dev/null
+++ b/gammu/emb/common/protocol/alcatel/alcabus.h
@@ -0,0 +1,61 @@
+/* (c) 2002-2003 by Michal Cihar */
+/*
+ * Low level functions for communication with Alcatel One Touch phones.
+ *
+ * This code implements the protocol used for synchronisation with PC.
+ */
+
+#ifndef alcabus_h
+#define alcabus_h
+
+#include "../protocol.h"
+
+#define ALCATEL_HEADER 0x7E
+
+/* packet types: */
+/* used for starting binary connection (must be preceeded by
+ * AT+CPROT=16,"V1.0",16 and phone should response to it by CONNECT_ACK)
+ */
+#define ALCATEL_CONNECT 0x0A
+/* received when connect suceeded */
+#define ALCATEL_CONNECT_ACK 0x0C
+/* used for stopping binary connection */
+#define ALCATEL_DISCONNECT 0x0D
+/* received when binnary connection ends */
+#define ALCATEL_DISCONNECT_ACK 0x0E
+/* some control ack, I really don't know what should it do, so currently it
+ * is just ignored. It comes time to time, and communication continues OK also
+ * if no reply was made. */
+#define ALCATEL_CONTROL 0x0F
+/* sending/recieving data */
+#define ALCATEL_DATA 0x02
+/* acknowledge to data */
+#define ALCATEL_ACK 0x06
+
+/* Maximal value for packet counter */
+#define ALCATEL_MAX_COUNTER 0x3D
+
+typedef struct {
+ GSM_Protocol_Message Msg;
+ /* Incoming packets ID counter */
+ int in_counter;
+ /* Outgoing packets ID counter */
+ int out_counter;
+ /* Expected size of incoming packet */
+ int expected_size;
+ /* What is type of frame we expect next */
+ unsigned char next_frame;
+ /* State of mobile, if we expect something (generally some ack) we set
+ * this to true and no other action can be performed until it is false. */
+ bool busy;
+} GSM_Protocol_ALCABUSData;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/at/at.c b/gammu/emb/common/protocol/at/at.c
new file mode 100644
index 0000000..f4a75b7
--- a/dev/null
+++ b/gammu/emb/common/protocol/at/at.c
@@ -0,0 +1,229 @@
+/* (c) 2002-2004 by Marcin Wiacek and Michal Cihar */
+
+#include "../../gsmstate.h"
+
+#if defined(GSM_ENABLE_AT) || defined(GSM_ENABLE_BLUEAT) || defined(GSM_ENABLE_IRDAAT)
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "at.h"
+
+static GSM_Error AT_WriteMessage (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type)
+{
+ int i,sent = 0;
+
+ GSM_DumpMessageLevel2(s, buffer, length, type);
+ GSM_DumpMessageLevel3(s, buffer, length, type);
+ if (s->Protocol.Data.AT.FastWrite) {
+ while (sent != length) {
+ if ((i = s->Device.Functions->WriteDevice(s,buffer + sent, length - sent)) == 0) {
+ return ERR_DEVICEWRITEERROR;
+ }
+ sent += i;
+ }
+ } else {
+ for (i=0;i<length;i++) {
+ if (s->Device.Functions->WriteDevice(s,buffer+i,1)!=1) return ERR_DEVICEWRITEERROR;
+ /* For some phones like Siemens M20 we need to wait a little
+ * after writing each char. Possible reason: these phones
+ * can't receive so fast chars or there is bug here in Gammu */
+ my_sleep(1);
+ }
+ my_sleep(400);
+ }
+
+ return ERR_NONE;
+}
+
+typedef struct {
+ char *text;
+ int lines;
+} SpecialAnswersStruct;
+
+static GSM_Error AT_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Protocol_Message Msg2;
+ GSM_Protocol_ATData *d = &s->Protocol.Data.AT;
+ int i;
+
+ /* These are lines with end of "normal" answers */
+ static char *StartStrings[] = {
+ "OK" , "ERROR" ,
+ "+CME ERROR:" , "+CMS ERROR:" ,
+
+ "+CPIN: " , /*A2D issue*/
+
+ NULL};
+
+ /* Some info from phone can be inside "normal" answers
+ * It starts with strings written here
+ */
+ static SpecialAnswersStruct SpecialAnswers[] = {
+ {"_OSIGQ:" ,1}, {"_OBS:" ,1},
+ {"^SCN:" ,1}, {"+CGREG:" ,1},
+ {"+CBM:" ,1}, {"+CMT:" ,2},
+ {"+CMTI:" ,1}, {"+CDS:" ,2},
+ {"+CREG:" ,1},
+
+ {"RING" ,1}, {"NO CARRIER" ,1},
+ {"NO ANSWER" ,1}, {"+COLP" ,1},
+ {"+CLIP" ,1},
+
+ {NULL ,1}};
+
+ /* Ignore leading CR, LF and ESC */
+ if (d->Msg.Length == 0) {
+ if (rx_char == 10 || rx_char == 13 || rx_char == 27) return ERR_NONE;
+ d->LineStart = d->Msg.Length;
+ }
+
+ if (d->Msg.BufferUsed < d->Msg.Length + 2) {
+ d->Msg.BufferUsed = d->Msg.Length + 2;
+ d->Msg.Buffer = (unsigned char *)realloc(d->Msg.Buffer,d->Msg.BufferUsed);
+ }
+ d->Msg.Buffer[d->Msg.Length++] = rx_char;
+ d->Msg.Buffer[d->Msg.Length ] = 0;
+
+ switch (rx_char) {
+ case 0:
+ break;
+ case 10:
+ case 13:
+ if (!d->wascrlf) d->LineEnd = d->Msg.Length-1;
+ d->wascrlf = true;
+ if (d->Msg.Length > 0 && rx_char == 10 && d->Msg.Buffer[d->Msg.Length-2]==13) {
+ i = 0;
+ while (StartStrings[i] != NULL) {
+ if (strncmp(StartStrings[i],d->Msg.Buffer+d->LineStart,strlen(StartStrings[i])) == 0) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ d->Msg.Length = 0;
+ break;
+ }
+ i++;
+ }
+ if (d->Msg.Length == 0) break;
+
+ i = 0;
+ while (SpecialAnswers[i].text != NULL) {
+ if (strncmp(SpecialAnswers[i].text,d->Msg.Buffer+d->LineStart,strlen(SpecialAnswers[i].text)) == 0) {
+ /* We need something better here */
+ if (s->Phone.Data.RequestID == ID_GetNetworkInfo && strncmp(SpecialAnswers[i].text,"+CREG:",6) == 0) {
+ i++;
+ continue;
+ }
+ d->SpecialAnswerStart = d->LineStart;
+ d->SpecialAnswerLines = SpecialAnswers[i].lines;
+ }
+ i++;
+ }
+
+
+ if (d->SpecialAnswerLines == 1) {
+ /* This is end of special answer. We copy it and send to phone module */
+ Msg2.Buffer = malloc(d->LineEnd - d->SpecialAnswerStart + 3);
+ memcpy(Msg2.Buffer,d->Msg.Buffer+d->SpecialAnswerStart,d->LineEnd - d->SpecialAnswerStart + 2);
+ Msg2.Length = d->LineEnd - d->SpecialAnswerStart + 2;
+ Msg2.Buffer[Msg2.Length] = 0;
+
+ s->Phone.Data.RequestMsg = &Msg2;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ free(Msg2.Buffer);
+
+ /* We cut special answer from main buffer */
+ d->Msg.Length = d->SpecialAnswerStart;
+ if (d->Msg.Length != 0) d->Msg.Length = d->Msg.Length - 2;
+
+ /* We need to find earlier values of all variables */
+ d->wascrlf = false;
+ d->LineStart = 0;
+ for (i=0;i<d->Msg.Length;i++) {
+ switch(d->Msg.Buffer[i]) {
+ case 0:
+ break;
+ case 10:
+ case 13:
+ if (!d->wascrlf) d->LineEnd = d->Msg.Length-1;
+ d->wascrlf = true;
+ break;
+ default:
+ if (d->wascrlf) {
+ d->LineStart = d->Msg.Length-1;
+ d->wascrlf = false;
+ }
+ }
+ }
+ d->Msg.Buffer[d->Msg.Length] = 0;
+ }
+ if (d->SpecialAnswerLines > 0) d->SpecialAnswerLines--;
+ }
+ break;
+ case 'T':
+ /* When CONNECT string received, we know there will not follow
+ * anything AT related, after CONNECT can follow ppp data, alcabus
+ * data and also other things.
+ */
+ if (strncmp(d->Msg.Buffer+d->LineStart, "CONNECT", 7) == 0) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ d->LineStart = -1;
+ d->Msg.Length = 0;
+ break;
+ }
+ default:
+ if (d->wascrlf) {
+ d->LineStart = d->Msg.Length-1;
+ d->wascrlf = false;
+ }
+ if (d->EditMode) {
+ if (strlen(d->Msg.Buffer+d->LineStart) == 2 && strncmp(d->Msg.Buffer+d->LineStart,"> ",2)==0) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ }
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error AT_Initialise(GSM_StateMachine *s)
+{
+ GSM_Protocol_ATData *d = &s->Protocol.Data.AT;
+
+ d->Msg.Buffer = NULL;
+ d->Msg.BufferUsed = 0;
+ d->Msg.Length = 0;
+ d->Msg.Type = 0;
+
+ d->SpecialAnswerLines = 0;
+ d->LineStart = -1;
+ d->LineEnd = -1;
+ d->wascrlf = false;
+ d->EditMode = false;
+ d->FastWrite = false;
+
+ s->Device.Functions->DeviceSetDtrRts(s,true,true);
+
+ return s->Device.Functions->DeviceSetSpeed(s,s->Speed);
+}
+
+static GSM_Error AT_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.AT.Msg.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions ATProtocol = {
+ AT_WriteMessage,
+ AT_StateMachine,
+ AT_Initialise,
+ AT_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/at/at.h b/gammu/emb/common/protocol/at/at.h
new file mode 100644
index 0000000..e200646
--- a/dev/null
+++ b/gammu/emb/common/protocol/at/at.h
@@ -0,0 +1,36 @@
+/* (c) 2002-2003 by Marcin Wiacek and Michal Cihar */
+
+#ifndef at_h
+#define at_h
+
+#include "../protocol.h"
+
+typedef struct {
+ GSM_Protocol_Message Msg;
+ bool wascrlf;
+ int LineStart,LineEnd;
+ int SpecialAnswerLines,SpecialAnswerStart;
+
+ bool EditMode; /* wait for modem answer or not */
+ bool FastWrite;
+} GSM_Protocol_ATData;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+#if defined(GSM_ENABLE_BLUEAT)
+# ifndef GSM_USED_BLUETOOTHDEVICE
+# define GSM_USED_BLUETOOTHDEVICE
+# endif
+#endif
+#if defined(GSM_ENABLE_IRDAAT)
+# ifndef GSM_USED_IRDADEVICE
+# define GSM_USED_IRDADEVICE
+# endif
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/fbus2.c b/gammu/emb/common/protocol/nokia/fbus2.c
new file mode 100644
index 0000000..8b3e024
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/fbus2.c
@@ -0,0 +1,444 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+/* based on some work from Gnokii and MyGnokii */
+
+#include "../../gsmstate.h"
+
+#if defined(GSM_ENABLE_FBUS2) || defined(GSM_ENABLE_FBUS2IRDA) || defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2BLUE) || defined(GSM_ENABLE_BLUEFBUS2) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2PL2303)
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "fbus2.h"
+
+static GSM_Error FBUS2_WriteFrame(GSM_StateMachine *s,
+ unsigned char *MsgBuffer,
+ int MsgLength,
+ unsigned char MsgType)
+{
+ unsigned char buffer2[FBUS2_MAX_TRANSMIT_LENGTH + 10];
+ unsigned char checksum=0;
+ int i, len, sent;
+
+ buffer2[0] = FBUS2_FRAME_ID;
+ if (s->ConnectionType==GCT_FBUS2IRDA) buffer2[0] = FBUS2_IRDA_FRAME_ID;
+
+ buffer2[1] = FBUS2_DEVICE_PHONE; //destination
+ buffer2[2] = FBUS2_DEVICE_PC; //source
+ buffer2[3] = MsgType;
+ buffer2[4] = MsgLength / 256;
+ buffer2[5] = MsgLength % 256;
+
+ memcpy(buffer2 + 6, MsgBuffer, MsgLength);
+ len = MsgLength + 6;
+
+ /* Odd messages require additional 0x00 byte */
+ if (MsgLength % 2) buffer2[len++] = 0x00;
+
+ checksum = 0;
+ for (i = 0; i < len; i+=2) checksum ^= buffer2[i];
+ buffer2[len++] = checksum;
+
+ checksum = 0;
+ for (i = 1; i < len; i+=2) checksum ^= buffer2[i];
+ buffer2[len++] = checksum;
+
+ /* Sending to phone */
+ sent=s->Device.Functions->WriteDevice(s,buffer2,len);
+ if (sent!=len) return ERR_DEVICEWRITEERROR;
+
+ return ERR_NONE;
+}
+
+static GSM_Error FBUS2_WriteMessage (GSM_StateMachine *s,
+ unsigned char *MsgBuffer,
+ int MsgLength,
+ unsigned char MsgType)
+{
+ int i, nom, togo, thislength; /* number of messages, ... */
+ unsigned char buffer2[FBUS2_MAX_TRANSMIT_LENGTH + 2], seqnum;
+ GSM_Protocol_FBUS2Data *d = &s->Protocol.Data.FBUS2;
+ GSM_Error error;
+
+ GSM_DumpMessageLevel3(s, MsgBuffer, MsgLength, MsgType);
+
+ nom = (MsgLength + FBUS2_MAX_TRANSMIT_LENGTH - 1) / FBUS2_MAX_TRANSMIT_LENGTH;
+ togo = MsgLength;
+
+ for (i = 0; i < nom; i++) {
+ seqnum = d->MsgSequenceNumber;
+ if (i==0) seqnum = seqnum + 0x40;
+ d->MsgSequenceNumber = (d->MsgSequenceNumber + 1) & 0x07;
+
+ thislength = togo;
+ if (togo > FBUS2_MAX_TRANSMIT_LENGTH) thislength = FBUS2_MAX_TRANSMIT_LENGTH;
+ memcpy(buffer2, MsgBuffer + (MsgLength - togo), thislength);
+ buffer2[thislength] = nom - i;
+ buffer2[thislength + 1] = seqnum;
+ togo = togo - thislength;
+
+ GSM_DumpMessageLevel2(s, buffer2, thislength, MsgType);
+
+ error=FBUS2_WriteFrame(s, buffer2, thislength + 2, MsgType);
+ if (error!=ERR_NONE) return error;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error FBUS2_SendAck(GSM_StateMachine *s,
+ unsigned char MsgType,
+ unsigned char MsgSequence)
+{
+ unsigned char buffer2[2];
+
+ buffer2[0] = MsgType;
+ buffer2[1] = MsgSequence;
+
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s,"[Sending Ack of type %02x, seq %x]\n",buffer2[0],buffer2[1]);
+ }
+
+ /* Sending to phone */
+ return FBUS2_WriteFrame(s, buffer2, 2, FBUS2_ACK_BYTE);
+}
+
+static GSM_Error FBUS2_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Protocol_FBUS2Data *d = &s->Protocol.Data.FBUS2;
+ unsigned char frm_num, seq_num;
+ bool correct = false;
+
+ /* XOR the byte with the earlier checksum */
+ d->Msg.CheckSum[d->Msg.Count & 1] ^= rx_char;
+
+ if (d->MsgRXState == RX_GetMessage) {
+ d->Msg.Buffer[d->Msg.Count] = rx_char;
+ d->Msg.Count++;
+
+ /* This is not last byte in frame */
+ if (d->Msg.Count != d->Msg.Length+(d->Msg.Length%2)+2) return ERR_NONE;
+
+ /* Checksum is incorrect */
+ if (d->Msg.CheckSum[0] != d->Msg.CheckSum[1]) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: checksum]\n");
+ }
+ free(d->Msg.Buffer);
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ seq_num = d->Msg.Buffer[d->Msg.Length-1];
+
+ if (d->Msg.Type == FBUS2_ACK_BYTE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s, "[Received Ack of type %02x, seq %02x]\n",d->Msg.Buffer[0],seq_num);
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ frm_num = d->Msg.Buffer[d->Msg.Length-2];
+
+ if ((seq_num & 0x40) == 0x40) {
+ d->FramesToGo = frm_num;
+ d->MultiMsg.Length = 0;
+ d->MultiMsg.Type = d->Msg.Type;
+ d->MultiMsg.Destination = d->Msg.Destination;
+ d->MultiMsg.Source = d->Msg.Source;
+ }
+
+ if ((seq_num & 0x40) != 0x40 && d->FramesToGo != frm_num) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s, "[ERROR: Missed part of multiframe msg]\n");
+ }
+
+ free(d->Msg.Buffer);
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ if ((seq_num & 0x40) != 0x40 && d->Msg.Type != d->MultiMsg.Type) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s, "[ERROR: Multiframe msg in multiframe msg]\n");
+ }
+
+ free(d->Msg.Buffer);
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ if (d->MultiMsg.BufferUsed < d->MultiMsg.Length+d->Msg.Length-2) {
+ d->MultiMsg.BufferUsed = d->MultiMsg.Length+d->Msg.Length-2;
+ d->MultiMsg.Buffer = (unsigned char *)realloc(d->MultiMsg.Buffer,d->MultiMsg.BufferUsed);
+ }
+ memcpy(d->MultiMsg.Buffer+d->MultiMsg.Length,d->Msg.Buffer,d->Msg.Length-2);
+ d->MultiMsg.Length = d->MultiMsg.Length+d->Msg.Length-2;
+
+ free(d->Msg.Buffer);
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+
+ d->FramesToGo--;
+
+ /* do not ack debug trace, as this could generate a
+ * (feedback loop) flood of which even Noah would be scared.
+ */
+ if (d->Msg.Type != 0) {
+ FBUS2_SendAck(s,d->Msg.Type,((unsigned char)(seq_num & 0x0f)));
+ }
+
+ if (d->FramesToGo == 0) {
+ s->Phone.Data.RequestMsg = &d->MultiMsg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+ }
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetLength2) {
+ d->Msg.Length = d->Msg.Length + rx_char;
+ d->Msg.Buffer = (unsigned char *)malloc(d->Msg.Length+3);
+ d->MsgRXState = RX_GetMessage;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetLength1) {
+ d->Msg.Length = rx_char * 256;
+ d->MsgRXState = RX_GetLength2;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetType) {
+ d->Msg.Type = rx_char;
+ d->MsgRXState = RX_GetLength1;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetSource) {
+ if (rx_char != FBUS2_DEVICE_PHONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, FBUS2_DEVICE_PHONE);
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Source = rx_char;
+
+ d->MsgRXState = RX_GetType;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetDestination) {
+ if (rx_char != FBUS2_DEVICE_PC) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, FBUS2_DEVICE_PC);
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Destination = rx_char;
+
+ d->MsgRXState = RX_GetSource;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_Sync) {
+ switch (s->ConnectionType) {
+ case GCT_FBUS2:
+ case GCT_FBUS2DLR3:
+ case GCT_FBUS2DKU5:
+ case GCT_FBUS2PL2303:
+ case GCT_FBUS2BLUE:
+ case GCT_BLUEFBUS2:
+ if (rx_char == FBUS2_FRAME_ID) correct = true;
+ break;
+ case GCT_FBUS2IRDA:
+ if (rx_char == FBUS2_IRDA_FRAME_ID) correct = true;
+ break;
+ default:
+ break;
+ }
+ if (!correct) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ if (s->ConnectionType==GCT_FBUS2IRDA) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, FBUS2_IRDA_FRAME_ID);
+ } else {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, FBUS2_FRAME_ID);
+ }
+ }
+ return ERR_NONE;
+ }
+
+ d->Msg.CheckSum[0] = rx_char;
+ d->Msg.CheckSum[1] = 0;
+ d->Msg.Count = 0;
+
+ d->MsgRXState = RX_GetDestination;
+ return ERR_NONE;
+ }
+ return ERR_NONE;
+}
+
+#if defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2BLUE) || defined(GSM_ENABLE_BLUEFBUS2) || defined(GSM_ENABLE_FBUS2PL2303)
+static void FBUS2_WriteDLR3(GSM_StateMachine *s, char *command, int length, int timeout)
+{
+ unsigned char buff[300];
+ int w = 0;
+ bool wassomething = false;
+
+ s->Device.Functions->WriteDevice(s,command,length);
+
+ for (w=0;w<timeout;w++) {
+ if (wassomething) {
+ if (s->Device.Functions->ReadDevice(s, buff, 255)==0) return;
+ } else {
+ if (s->Device.Functions->ReadDevice(s, buff, 255)>0) wassomething = true;
+ }
+ my_sleep(50);
+ }
+}
+#endif
+
+static GSM_Error FBUS2_Initialise(GSM_StateMachine *s)
+{
+ unsigned char init_char = 0x55;
+#ifdef GSM_ENABLE_FBUS2IRDA
+ unsigned char end_init_char = 0xc1;
+#endif
+
+ GSM_Protocol_FBUS2Data *d = &s->Protocol.Data.FBUS2;
+ GSM_Device_Functions *Device = s->Device.Functions;
+ GSM_Error error;
+ int count;
+
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+ d->MultiMsg.BufferUsed = 0;
+ d->MultiMsg.Length = 0;
+ d->MultiMsg.Buffer = NULL;
+
+ d->MsgSequenceNumber = 0;
+ d->FramesToGo = 0;
+ d->MsgRXState = RX_Sync;
+
+ error=Device->DeviceSetParity(s,false);
+ if (error!=ERR_NONE) return error;
+
+ switch (s->ConnectionType) {
+#if defined(GSM_ENABLE_BLUEFBUS2) || defined(GSM_ENABLE_FBUS2BLUE)
+ case GCT_FBUS2BLUE:
+ case GCT_BLUEFBUS2:
+ FBUS2_WriteDLR3(s,"AT\r\n", 4,10);
+ FBUS2_WriteDLR3(s,"AT&F\r\n", 6,10);
+ FBUS2_WriteDLR3(s,"AT*NOKIAFBUS\r\n", 14,10);
+ break;
+#endif
+#if defined(GSM_ENABLE_FBUS2DLR3) || defined(GSM_ENABLE_FBUS2DKU5) || defined(GSM_ENABLE_FBUS2PL2303)
+ case GCT_FBUS2DKU5:
+ case GCT_FBUS2PL2303:
+ case GCT_FBUS2DLR3:
+ error=Device->DeviceSetDtrRts(s,false,false);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+
+ error=Device->DeviceSetDtrRts(s,true,true);
+ if (error!=ERR_NONE) return error;
+ error=Device->DeviceSetSpeed(s,19200);
+ if (error!=ERR_NONE) return error;
+
+ FBUS2_WriteDLR3(s,"AT\r\n", 4,10);
+ FBUS2_WriteDLR3(s,"AT&F\r\n", 6,10);
+ FBUS2_WriteDLR3(s,"AT*NOKIAFBUS\r\n", 14,10);
+
+ error=Device->CloseDevice(s);
+ if (error!=ERR_NONE) return error;
+ my_sleep(1000);
+
+ error=Device->OpenDevice(s);
+ if (error!=ERR_NONE) return error;
+ error=Device->DeviceSetParity(s,false);
+ if (error!=ERR_NONE) return error;
+ error=Device->DeviceSetSpeed(s,115200);
+ if (error!=ERR_NONE) return error;
+ error=Device->DeviceSetDtrRts(s,false,false);
+ if (error!=ERR_NONE) return error;
+
+ for (count = 0; count < 55; count ++) {
+ if (Device->WriteDevice(s,&init_char,1)!=1) return ERR_DEVICEWRITEERROR;
+ }
+ break;
+#endif
+ case GCT_FBUS2:
+ error=Device->DeviceSetSpeed(s,115200);
+ if (error!=ERR_NONE) return error;
+
+ error=Device->DeviceSetDtrRts(s,true,false); /*DTR high,RTS low*/
+ if (error!=ERR_NONE) return error;
+
+ for (count = 0; count < 55; count ++) {
+ if (Device->WriteDevice(s,&init_char,1)!=1) return ERR_DEVICEWRITEERROR;
+ my_sleep(10);
+ }
+ break;
+#ifdef GSM_ENABLE_FBUS2IRDA
+ case GCT_FBUS2IRDA:
+ error=Device->DeviceSetSpeed(s,9600);
+ if (error!=ERR_NONE) return error;
+
+ for (count = 0; count < 55; count ++) {
+ if (Device->WriteDevice(s,&init_char,1)!=1) return ERR_DEVICEWRITEERROR;
+ my_sleep(10);
+ }
+
+ if (Device->WriteDevice(s,&end_init_char,1)!=1) return ERR_DEVICEWRITEERROR;
+ my_sleep(20);
+
+ error=Device->DeviceSetSpeed(s,115200);
+ if (error!=ERR_NONE) return error;
+
+ break;
+#endif
+ default:
+ break;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error FBUS2_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.FBUS2.Msg.Buffer);
+ free(s->Protocol.Data.FBUS2.MultiMsg.Buffer);
+
+ my_sleep(200);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions FBUS2Protocol = {
+ FBUS2_WriteMessage,
+ FBUS2_StateMachine,
+ FBUS2_Initialise,
+ FBUS2_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/fbus2.h b/gammu/emb/common/protocol/nokia/fbus2.h
new file mode 100644
index 0000000..5dd45d7
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/fbus2.h
@@ -0,0 +1,38 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+/* based on some work from Gnokii and MyGnokii */
+
+#ifndef fbus2_h
+#define fbus2_h
+
+#include "../protocol.h"
+
+#define FBUS2_FRAME_ID 0x1e
+#define FBUS2_IRDA_FRAME_ID 0x1c
+#define FBUS2_DEVICE_PHONE 0x00 /* Nokia mobile phone */
+#define FBUS2_DEVICE_PC 0x0c /* Our PC */
+#define FBUS2_ACK_BYTE 0x7f /* Acknowledge of the received frame */
+
+#define FBUS2_MAX_TRANSMIT_LENGTH 120
+
+typedef struct {
+ int MsgSequenceNumber;
+ int MsgRXState;
+ int FramesToGo;
+ GSM_Protocol_Message MultiMsg;
+ GSM_Protocol_Message Msg;
+} GSM_Protocol_FBUS2Data;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+#if defined(GSM_ENABLE_BLUEFBUS2)
+# ifndef GSM_USED_BLUETOOTHDEVICE
+# define GSM_USED_BLUETOOTHDEVICE
+# endif
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/mbus2.c b/gammu/emb/common/protocol/nokia/mbus2.c
new file mode 100644
index 0000000..f07d6c5
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/mbus2.c
@@ -0,0 +1,252 @@
+/* (c) 2001-2003 by Marcin Wiacek */
+/* based on some work from MyGnokii */
+
+#include "../../gsmstate.h"
+
+#ifdef GSM_ENABLE_MBUS2
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "mbus2.h"
+
+static GSM_Error MBUS2_WriteMessage (GSM_StateMachine *s,
+ unsigned char *MsgBuffer,
+ int MsgLength,
+ unsigned char MsgType)
+{
+ unsigned char *buffer2, checksum = 0;
+ GSM_Protocol_MBUS2Data *d = &s->Protocol.Data.MBUS2;
+ int i, sent, len;
+
+ GSM_DumpMessageLevel3(s, MsgBuffer, MsgLength, MsgType);
+
+ buffer2 = (unsigned char *)malloc(MsgLength + 8);
+
+ buffer2[0] = MBUS2_FRAME_ID;
+ buffer2[1] = MBUS2_DEVICE_PHONE; // destination
+ buffer2[2] = MBUS2_DEVICE_PC; // source
+ buffer2[3] = MsgType;
+ buffer2[4] = MsgLength / 256;
+ buffer2[5] = MsgLength % 256;
+
+ memcpy(buffer2 + 6, MsgBuffer, MsgLength);
+ len = 6 + MsgLength;
+
+ /* According to http://www.flosys.com/tdma/n5160.html some phones
+ * can have problems with checksum equal 0x1F. Phones can recognize
+ * received frame, but won't send ACK for it. When checksum is 0x1F,
+ * we increment the sequence number
+ */
+ do {
+ d->MsgSequenceNumber++;
+
+ buffer2[len] = d->MsgSequenceNumber;
+
+ /* Calculating checksum */
+ checksum = 0;
+ for (i = 0; i < len + 1; i++) checksum ^= buffer2[i];
+ } while (checksum == 0x1f);
+
+ buffer2[len++] = d->MsgSequenceNumber;
+ buffer2[len++] = checksum;
+
+ GSM_DumpMessageLevel2(s, buffer2+6, MsgLength, MsgType);
+
+ /* Sending to phone */
+ my_sleep(10);
+ sent=s->Device.Functions->WriteDevice(s,buffer2,len);
+
+ free(buffer2);
+
+ if (sent!=len) return ERR_DEVICEWRITEERROR;
+ return ERR_NONE;
+}
+
+static GSM_Error MBUS2_SendAck(GSM_StateMachine *s,
+ unsigned char type,
+ unsigned char sequence)
+{
+ GSM_Device_Functions *Device = s->Device.Functions;
+ unsigned char buffer2[6];
+ int i;
+
+ buffer2[0] = MBUS2_FRAME_ID;
+ buffer2[1] = MBUS2_DEVICE_PHONE; //destination
+ buffer2[2] = MBUS2_DEVICE_PC; //source
+ buffer2[3] = MBUS2_ACK_BYTE;
+ buffer2[4] = sequence;
+ buffer2[5] = 0;
+
+ /* Calculating checksum */
+ for (i = 0; i < 5; i++) buffer2[5] ^= buffer2[i];
+
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s,"[Sending Ack of type %02x, seq: %x]\n",type,sequence);
+ }
+
+ /* Sending to phone */
+ my_sleep(10);
+ if (Device->WriteDevice(s,buffer2,6)!=6) return ERR_DEVICEWRITEERROR;
+
+ return ERR_NONE;
+}
+
+static GSM_Error MBUS2_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Phone_Functions *Phone = s->Phone.Functions;
+ GSM_Protocol_MBUS2Data *d = &s->Protocol.Data.MBUS2;
+
+ d->Msg.CheckSum[0] = d->Msg.CheckSum[1];
+ d->Msg.CheckSum[1] ^= rx_char;
+
+ if (d->MsgRXState == RX_GetMessage) {
+ d->Msg.Buffer[d->Msg.Count] = rx_char;
+ d->Msg.Count++;
+
+ /* This is not last byte in frame */
+ if (d->Msg.Count != d->Msg.Length+2) return ERR_NONE;
+
+ /* Checksum is incorrect */
+ if (d->Msg.CheckSum[0] != rx_char) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: checksum]\n");
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ if (d->Msg.Destination != MBUS2_DEVICE_PHONE) {
+ MBUS2_SendAck(s, d->Msg.Type, d->Msg.Buffer[d->Msg.Count-2]);
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = Phone->DispatchMessage(s);
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetLength2) {
+ if (d->Msg.Type == MBUS2_ACK_BYTE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE) {
+ smprintf(s,"[Received Ack]\n");
+ }
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+
+ d->Msg.Length = d->Msg.Length + rx_char;
+ if (d->Msg.BufferUsed < d->Msg.Length+2) {
+ d->Msg.BufferUsed = d->Msg.Length+2;
+ d->Msg.Buffer = (unsigned char *)realloc(d->Msg.Buffer,d->Msg.BufferUsed);
+ }
+
+ d->MsgRXState = RX_GetMessage;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetLength1) {
+ d->Msg.Length = rx_char * 256;
+
+ d->MsgRXState = RX_GetLength2;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetType) {
+ d->Msg.Type = rx_char;
+
+ d->MsgRXState = RX_GetLength1;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetSource) {
+ if (rx_char != MBUS2_DEVICE_PHONE && rx_char != MBUS2_DEVICE_PC) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x and %02x]\n", rx_char, MBUS2_DEVICE_PHONE, MBUS2_DEVICE_PC);
+ }
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Source = rx_char;
+
+ d->MsgRXState = RX_GetType;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_GetDestination) {
+ if (rx_char != MBUS2_DEVICE_PC && rx_char != MBUS2_DEVICE_PHONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x and %02x]\n", rx_char, MBUS2_DEVICE_PHONE, MBUS2_DEVICE_PC);
+ }
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Destination = rx_char;
+
+ d->MsgRXState = RX_GetSource;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState == RX_Sync) {
+ if (rx_char != MBUS2_FRAME_ID) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, MBUS2_FRAME_ID);
+ }
+ return ERR_NONE;
+ }
+ d->Msg.CheckSum[1] = MBUS2_FRAME_ID;
+ d->Msg.Count = 0;
+
+ d->MsgRXState = RX_GetDestination;
+ return ERR_NONE;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error MBUS2_Initialise(GSM_StateMachine *s)
+{
+ GSM_Device_Functions *Device = s->Device.Functions;
+ GSM_Protocol_MBUS2Data *d = &s->Protocol.Data.MBUS2;
+ GSM_Error error;
+
+ d->Msg.Length = 0;
+ d->Msg.BufferUsed = 0;
+ d->Msg.Buffer = NULL;
+
+ d->MsgSequenceNumber = 0;
+ d->MsgRXState = RX_Sync;
+
+ error=Device->DeviceSetSpeed(s,9600);
+ if (error!=ERR_NONE) return error;
+
+ error=Device->DeviceSetParity(s,true);
+ if (error!=ERR_NONE) return error;
+
+ error=Device->DeviceSetDtrRts(s,false,true); /*DTR low,RTS high*/
+ if (error!=ERR_NONE) return error;
+ my_sleep(200);
+
+ return ERR_NONE;
+}
+
+static GSM_Error MBUS2_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.MBUS2.Msg.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions MBUS2Protocol = {
+ MBUS2_WriteMessage,
+ MBUS2_StateMachine,
+ MBUS2_Initialise,
+ MBUS2_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/mbus2.h b/gammu/emb/common/protocol/nokia/mbus2.h
new file mode 100644
index 0000000..86fcab6
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/mbus2.h
@@ -0,0 +1,28 @@
+/* (c) 2001-2003 by Marcin Wiacek */
+/* based on some work from MyGnokii */
+
+#ifndef mbus2_h
+#define mbus2_h
+
+#include "../protocol.h"
+
+#define MBUS2_FRAME_ID 0x1f
+#define MBUS2_DEVICE_PHONE 0x00 /* Nokia mobile phone */
+#define MBUS2_DEVICE_PC 0x10 /* Our PC (MBUS) */
+#define MBUS2_ACK_BYTE 0x7f /* Acknowledge of the received frame */
+
+typedef struct {
+ int MsgSequenceNumber;
+ int MsgRXState;
+ GSM_Protocol_Message Msg;
+} GSM_Protocol_MBUS2Data;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/phonet.c b/gammu/emb/common/protocol/nokia/phonet.c
new file mode 100644
index 0000000..bc5717d
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/phonet.c
@@ -0,0 +1,217 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+/* based on some work from Gnokii */
+
+#include "../../gsmstate.h"
+
+#if defined(GSM_ENABLE_IRDA) || defined(GSM_ENABLE_PHONETBLUE) || defined(GSM_ENABLE_BLUEPHONET)
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "phonet.h"
+
+static GSM_Error PHONET_WriteMessage (GSM_StateMachine *s,
+ unsigned char *MsgBuffer,
+ int MsgLength,
+ unsigned char MsgType)
+{
+ unsigned char *buffer2;
+ int sent;
+
+ GSM_DumpMessageLevel3(s, MsgBuffer, MsgLength, MsgType);
+
+ buffer2 = (unsigned char *)malloc(MsgLength + 6);
+
+ buffer2[0] = PHONET_FRAME_ID,
+ buffer2[1] = PHONET_DEVICE_PHONE; //destination
+ buffer2[2] = PHONET_DEVICE_PC; //source
+
+ if (s->ConnectionType==GCT_PHONETBLUE || s->ConnectionType==GCT_BLUEPHONET) {
+ buffer2[0] = PHONET_BLUE_FRAME_ID;
+ buffer2[1] = PHONET_DEVICE_PHONE; //destination
+ buffer2[2] = PHONET_BLUE_DEVICE_PC; //source
+ }
+
+ buffer2[3] = MsgType;
+ buffer2[4] = MsgLength / 256;
+ buffer2[5] = MsgLength % 256;
+
+ memcpy(buffer2 + 6, MsgBuffer, MsgLength);
+
+ GSM_DumpMessageLevel2(s, buffer2+6, MsgLength, MsgType);
+
+ /* Sending to phone */
+ sent = s->Device.Functions->WriteDevice(s,buffer2,MsgLength+6);
+
+ free(buffer2);
+
+ if (sent!=MsgLength+6) return ERR_DEVICEWRITEERROR;
+ return ERR_NONE;
+}
+
+static GSM_Error PHONET_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Protocol_PHONETData *d = &s->Protocol.Data.PHONET;
+ bool correct = false;
+
+ if (d->MsgRXState==RX_GetMessage) {
+ d->Msg.Buffer[d->Msg.Count] = rx_char;
+ d->Msg.Count++;
+
+ /* This is not last byte in frame */
+ if (d->Msg.Count != d->Msg.Length) return ERR_NONE;
+
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = s->Phone.Functions->DispatchMessage(s);
+
+ free(d->Msg.Buffer);
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_GetLength2) {
+ d->Msg.Length = d->Msg.Length + rx_char;
+ d->Msg.Buffer = (unsigned char *)malloc(d->Msg.Length);
+
+ d->MsgRXState = RX_GetMessage;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_GetLength1) {
+ d->Msg.Length = rx_char * 256;
+
+ d->MsgRXState = RX_GetLength2;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_GetType) {
+ d->Msg.Type = rx_char;
+
+ d->MsgRXState = RX_GetLength1;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_GetSource) {
+ if (rx_char != PHONET_DEVICE_PHONE) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, PHONET_DEVICE_PHONE);
+ }
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Source = rx_char;
+
+ d->MsgRXState = RX_GetType;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_GetDestination) {
+ switch (s->ConnectionType) {
+ case GCT_IRDAPHONET:
+ if (rx_char == PHONET_DEVICE_PC) correct = true;
+ break;
+ case GCT_PHONETBLUE:
+ case GCT_BLUEPHONET:
+ if (rx_char == PHONET_BLUE_DEVICE_PC) correct = true;
+ break;
+ default:
+ break;
+ }
+ if (!correct) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, PHONET_DEVICE_PC);
+ }
+ d->MsgRXState = RX_Sync;
+ return ERR_NONE;
+ }
+ d->Msg.Destination = rx_char;
+
+ d->MsgRXState = RX_GetSource;
+ return ERR_NONE;
+ }
+ if (d->MsgRXState==RX_Sync) {
+ switch (s->ConnectionType) {
+ case GCT_IRDAPHONET:
+ if (rx_char == PHONET_FRAME_ID) correct = true;
+ break;
+ case GCT_PHONETBLUE:
+ case GCT_BLUEPHONET:
+ if (rx_char == PHONET_BLUE_FRAME_ID) correct = true;
+ break;
+ default:
+ break;
+ }
+ if (!correct) {
+ if (s->di.dl==DL_TEXT || s->di.dl==DL_TEXTALL || s->di.dl==DL_TEXTERROR ||
+ s->di.dl==DL_TEXTDATE || s->di.dl==DL_TEXTALLDATE || s->di.dl==DL_TEXTERRORDATE) {
+ smprintf(s,"[ERROR: incorrect char - %02x, not %02x]\n", rx_char, PHONET_FRAME_ID);
+ }
+ return ERR_NONE;
+ }
+ d->Msg.Count = 0;
+
+ d->MsgRXState = RX_GetDestination;
+ return ERR_NONE;
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error PHONET_Initialise(GSM_StateMachine *s)
+{
+ int total = 0, i, n;
+ GSM_Protocol_PHONETData *d = &s->Protocol.Data.PHONET;
+ unsigned char req[50];
+
+ d->Msg.Length = 0;
+ d->Msg.Buffer = NULL;
+ d->MsgRXState = RX_Sync;
+
+ if (s->ConnectionType == GCT_PHONETBLUE || s->ConnectionType == GCT_BLUEPHONET) {
+ /* Send frame in PHONET style */
+ req[0] = PHONET_BLUE_FRAME_ID; req[1] = PHONET_DEVICE_PHONE;
+ req[2] = PHONET_BLUE_DEVICE_PC; req[3] = 0xD0;
+ req[4] = 0x00; req[5] = 0x01;
+ req[6] = 0x04;
+ if (s->Device.Functions->WriteDevice(s,req,7) != 7) return ERR_DEVICEWRITEERROR;
+
+ while (total < 7) {
+ n = s->Device.Functions->ReadDevice(s, req + total, 50 - total);
+ total += n;
+ }
+
+ /* Answer frame in PHONET style */
+ req[10] = PHONET_BLUE_FRAME_ID; req[11] = PHONET_BLUE_DEVICE_PC;
+ req[12] = PHONET_DEVICE_PHONE; req[13] = 0xD0;
+ req[14] = 0x00; req[15] = 0x01;
+ req[16] = 0x05;
+
+ for (i = 0; i < 7; i++) {
+ if (req[i] != req[10+i]) {
+ smprintf(s,"Incorrect byte in the answer\n");
+ return ERR_UNKNOWN;
+ }
+ }
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error PHONET_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.PHONET.Msg.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions PHONETProtocol = {
+ PHONET_WriteMessage,
+ PHONET_StateMachine,
+ PHONET_Initialise,
+ PHONET_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/nokia/phonet.h b/gammu/emb/common/protocol/nokia/phonet.h
new file mode 100644
index 0000000..2f6e836
--- a/dev/null
+++ b/gammu/emb/common/protocol/nokia/phonet.h
@@ -0,0 +1,35 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+/* based on some work from Gnokii */
+
+#ifndef PHONET_h
+#define PHONET_h
+
+#include "../protocol.h"
+
+#define PHONET_FRAME_ID 0x14
+#define PHONET_BLUE_FRAME_ID 0x19
+#define PHONET_DEVICE_PHONE 0x00 /* Nokia mobile phone */
+#define PHONET_DEVICE_PC 0x0c /* Our PC */
+#define PHONET_BLUE_DEVICE_PC 0x10 /* Our PC */
+
+typedef struct {
+ int MsgRXState;
+ GSM_Protocol_Message Msg;
+} GSM_Protocol_PHONETData;
+
+#if defined(GSM_ENABLE_IRDAPHONET)
+# ifndef GSM_USED_IRDADEVICE
+# define GSM_USED_IRDADEVICE
+# endif
+#endif
+#if defined(GSM_ENABLE_BLUEPHONET)
+# ifndef GSM_USED_BLUETOOTHDEVICE
+# define GSM_USED_BLUETOOTHDEVICE
+# endif
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/obex/obex.c b/gammu/emb/common/protocol/obex/obex.c
new file mode 100644
index 0000000..942c084
--- a/dev/null
+++ b/gammu/emb/common/protocol/obex/obex.c
@@ -0,0 +1,120 @@
+/* (c) 2003 by Marcin Wiacek */
+/* www.irda.org OBEX specs 1.3 */
+
+#include "../../gsmstate.h"
+
+#include <stdio.h>
+#include <string.h>
+
+#if defined(GSM_ENABLE_BLUEOBEX) || defined(GSM_ENABLE_IRDAOBEX)
+
+#include "../../gsmcomon.h"
+#include "obex.h"
+
+static GSM_Error OBEX_WriteMessage (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type)
+{
+ unsigned char *out_buffer;
+ int current=0,sent;
+
+ out_buffer = (unsigned char *)malloc(length + 3);
+
+ OBEXAddBlock(out_buffer, &current, type, buffer, length);
+
+ GSM_DumpMessageLevel2(s, out_buffer+3, length, type);
+ GSM_DumpMessageLevel3(s, out_buffer+3, length, type);
+
+ /* Send it out... */
+ sent = s->Device.Functions->WriteDevice(s,out_buffer,current);
+
+ free(out_buffer);
+
+ if (sent!=current) return ERR_DEVICEWRITEERROR;
+ return ERR_NONE;
+}
+
+static GSM_Error OBEX_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Phone_Functions *Phone = s->Phone.Functions;
+ GSM_Protocol_OBEXData *d = &s->Protocol.Data.OBEX;
+
+ switch (d->MsgRXState) {
+ case RX_Sync:
+ d->Msg.Type = rx_char;
+ d->MsgRXState = RX_GetLength1;
+ break;
+ case RX_GetLength1:
+ d->Msg.Length = rx_char * 256;
+ d->MsgRXState = RX_GetLength2;
+ break;
+ case RX_GetLength2:
+ d->Msg.Length = d->Msg.Length + rx_char - 3;
+ d->Msg.Count = 0;
+ if (d->Msg.Count == d->Msg.Length) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = Phone->DispatchMessage(s);
+ d->MsgRXState = RX_Sync;
+ } else {
+ if (d->Msg.BufferUsed < d->Msg.Length) {
+ d->Msg.BufferUsed = d->Msg.Length;
+ d->Msg.Buffer = (unsigned char *)realloc(d->Msg.Buffer,d->Msg.BufferUsed);
+ }
+ d->MsgRXState = RX_GetMessage;
+ }
+ break;
+ case RX_GetMessage:
+ d->Msg.Buffer[d->Msg.Count] = rx_char;
+ d->Msg.Count++;
+ if (d->Msg.Count == d->Msg.Length) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = Phone->DispatchMessage(s);
+ d->MsgRXState = RX_Sync;
+ }
+ break;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error OBEX_Initialise(GSM_StateMachine *s)
+{
+ GSM_Protocol_OBEXData *d = &s->Protocol.Data.OBEX;
+
+ d->Msg.BufferUsed = 0;
+ d->Msg.Buffer = NULL;
+ d->Msg.Length = 0;
+
+ d->MsgRXState = RX_Sync;
+
+ return ERR_NONE;
+}
+
+static GSM_Error OBEX_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.OBEX.Msg.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions OBEXProtocol = {
+ OBEX_WriteMessage,
+ OBEX_StateMachine,
+ OBEX_Initialise,
+ OBEX_Terminate
+};
+
+#endif
+
+void OBEXAddBlock(char *Buffer, int *Pos, unsigned char ID, char *AddBuffer, int AddLength)
+{
+ Buffer[(*Pos)++] = ID;
+ Buffer[(*Pos)++] = (AddLength+3)/256;
+ Buffer[(*Pos)++] = (AddLength+3)%256;
+ if (AddBuffer != NULL) {
+ memcpy(Buffer+(*Pos),AddBuffer,AddLength);
+ (*Pos) += AddLength;
+ }
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/obex/obex.h b/gammu/emb/common/protocol/obex/obex.h
new file mode 100644
index 0000000..0e927c7
--- a/dev/null
+++ b/gammu/emb/common/protocol/obex/obex.h
@@ -0,0 +1,33 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef obex_h
+#define obex_h
+
+#include "../protocol.h"
+
+typedef struct {
+ int MsgRXState;
+ GSM_Protocol_Message Msg;
+} GSM_Protocol_OBEXData;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+#if defined(GSM_ENABLE_BLUEOBEX)
+# ifndef GSM_USED_BLUETOOTHDEVICE
+# define GSM_USED_BLUETOOTHDEVICE
+# endif
+#endif
+#if defined(GSM_ENABLE_IRDAOBEX)
+# ifndef GSM_USED_IRDADEVICE
+# define GSM_USED_IRDADEVICE
+# endif
+#endif
+
+void OBEXAddBlock(char *Buffer, int *Pos, unsigned char ID, char *AddBuffer, int AddLength);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/protocol.h b/gammu/emb/common/protocol/protocol.h
new file mode 100644
index 0000000..f8e1fe5
--- a/dev/null
+++ b/gammu/emb/common/protocol/protocol.h
@@ -0,0 +1,31 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef protocol_common_h
+#define protocol_common_h
+
+typedef enum {
+ RX_Sync,
+ RX_GetDestination,
+ RX_GetSource,
+ RX_GetType,
+ RX_GetLength1,
+ RX_GetLength2,
+ RX_GetMessage
+} GSM_Protocol_RXState;
+
+typedef struct {
+ int Length;
+ int Count;
+ unsigned char Type;
+ unsigned char Source;
+ unsigned char Destination;
+ unsigned char *Buffer;
+ int BufferUsed;
+ unsigned char CheckSum[2];
+} GSM_Protocol_Message;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/symbian/mrouter.c b/gammu/emb/common/protocol/symbian/mrouter.c
new file mode 100644
index 0000000..2ca7af1
--- a/dev/null
+++ b/gammu/emb/common/protocol/symbian/mrouter.c
@@ -0,0 +1,110 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include "../../gsmstate.h"
+
+#if defined(GSM_ENABLE_MROUTERBLUE)
+
+#include <stdio.h>
+#include <string.h>
+
+#include "../../gsmcomon.h"
+#include "mrouter.h"
+
+static GSM_Error MROUTER_WriteMessage (GSM_StateMachine *s, unsigned char *buffer,
+ int length, unsigned char type)
+{
+ unsigned char *out_buffer;
+ int current=0,sent;
+
+ out_buffer = (unsigned char *)malloc(length+1);
+
+ memcpy(out_buffer,buffer,length);
+ out_buffer[length]=0x7E;
+
+ GSM_DumpMessageLevel2(s, out_buffer, length, type);
+ GSM_DumpMessageLevel3(s, out_buffer, length, type);
+
+ /* Send it out... */
+ sent = s->Device.Functions->WriteDevice(s,out_buffer,length+1);
+
+ free(out_buffer);
+
+ if (sent!=current) return ERR_DEVICEWRITEERROR;
+ return ERR_NONE;
+}
+
+static GSM_Error MROUTER_StateMachine(GSM_StateMachine *s, unsigned char rx_char)
+{
+ GSM_Phone_Functions *Phone = s->Phone.Functions;
+ GSM_Protocol_MROUTERData *d = &s->Protocol.Data.MROUTER;
+
+ switch (d->MsgRXState) {
+ case RX_Sync:
+ if (rx_char == 0x7E) {
+ d->MsgRXState = RX_GetMessage;
+ d->Msg.Count = 0;
+ d->Msg.Length = 0;
+ } else smprintf(s,"Sync error: %02x\n",rx_char);
+ break;
+ case RX_GetMessage:
+ if (rx_char == 0x7E) {
+ s->Phone.Data.RequestMsg = &d->Msg;
+ s->Phone.Data.DispatchError = Phone->DispatchMessage(s);
+ d->Msg.Count = 0;
+ d->Msg.Length = 0;
+ } else {
+ d->Msg.BufferUsed = d->Msg.Length+1;
+ d->Msg.Buffer = (unsigned char *)realloc(d->Msg.Buffer,d->Msg.BufferUsed);
+
+ d->Msg.Buffer[d->Msg.Count] = rx_char;
+ d->Msg.Count++;
+ d->Msg.Length++;
+ }
+ break;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error MROUTER_Initialise(GSM_StateMachine *s)
+{
+ GSM_Protocol_MROUTERData *d = &s->Protocol.Data.MROUTER;
+GSM_Error error;
+
+ d->Msg.BufferUsed = 0;
+ d->Msg.Buffer = NULL;
+ d->Msg.Length = 0;
+
+ d->MsgRXState = RX_Sync;
+
+// error=s->Device.Functions->DeviceSetDtrRts(s,false,false);
+// if (error!=ERR_NONE) return error;
+
+ error=s->Device.Functions->DeviceSetSpeed(s,115200);
+ if (error!=ERR_NONE) return error;
+
+
+// error=s->Device.Functions->DeviceSetSpeed(s,115200);
+// if (error!=ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+static GSM_Error MROUTER_Terminate(GSM_StateMachine *s)
+{
+ free(s->Protocol.Data.MROUTER.Msg.Buffer);
+ return ERR_NONE;
+}
+
+GSM_Protocol_Functions MROUTERProtocol = {
+ MROUTER_WriteMessage,
+ MROUTER_StateMachine,
+ MROUTER_Initialise,
+ MROUTER_Terminate
+};
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/protocol/symbian/mrouter.h b/gammu/emb/common/protocol/symbian/mrouter.h
new file mode 100644
index 0000000..7d72cdd
--- a/dev/null
+++ b/gammu/emb/common/protocol/symbian/mrouter.h
@@ -0,0 +1,31 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef mrouter_h
+#define mrouter_h
+
+#include "../protocol.h"
+
+typedef struct {
+ int MsgRXState;
+ GSM_Protocol_Message Msg;
+} GSM_Protocol_MROUTERData;
+
+#ifndef GSM_USED_SERIALDEVICE
+# define GSM_USED_SERIALDEVICE
+#endif
+//#if defined(GSM_ENABLE_BLUEOBEX)
+//# ifndef GSM_USED_BLUETOOTHDEVICE
+//# define GSM_USED_BLUETOOTHDEVICE
+//# endif
+//#endif
+//#if defined(GSM_ENABLE_IRDAOBEX)
+//# ifndef GSM_USED_IRDADEVICE
+//# define GSM_USED_IRDADEVICE
+//# endif
+//#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backgen.h b/gammu/emb/common/service/backup/backgen.h
new file mode 100644
index 0000000..9d7d973
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backgen.h
@@ -0,0 +1,69 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef __gsm_backgen_h
+#define __gsm_backgen_h
+
+#include "../../config.h"
+#include "../../misc/misc.h"
+#include "../gsmpbk.h"
+#include "../gsmcal.h"
+#include "../gsmlogo.h"
+#include "../gsmring.h"
+#include "../gsmdata.h"
+#include "../gsmprof.h"
+#include "../gsmmisc.h"
+#include "../sms/gsmsms.h"
+
+#define GSM_BACKUP_MAX_PHONEPHONEBOOK 501
+#define GSM_BACKUP_MAX_SIMPHONEBOOK 251
+#define GSM_BACKUP_MAX_CALLER 6
+#define GSM_BACKUP_MAX_SMSC 10
+#define GSM_BACKUP_MAX_WAPBOOKMARK 40
+#define GSM_BACKUP_MAX_WAPSETTINGS 30
+#define GSM_BACKUP_MAX_MMSSETTINGS 30
+#define GSM_BACKUP_MAX_RINGTONES 30
+#define GSM_BACKUP_MAX_PROFILES 10
+#define GSM_BACKUP_MAX_FMSTATIONS 20
+#define GSM_BACKUP_MAX_GPRSPOINT 10
+#define GSM_BACKUP_MAX_NOTE 10 /* FIXME */
+
+typedef struct {
+ char IMEI [MAX_IMEI_LENGTH];
+ char Model [MAX_MODEL_LENGTH+MAX_VERSION_LENGTH];
+ char Creator [80];
+ GSM_DateTime DateTime;
+ bool DateTimeAvailable;
+ char MD5Original [100];
+ char MD5Calculated [100];
+ GSM_MemoryEntry *PhonePhonebook [GSM_BACKUP_MAX_PHONEPHONEBOOK + 1];
+ GSM_MemoryEntry *SIMPhonebook [GSM_BACKUP_MAX_SIMPHONEBOOK + 1];
+ GSM_CalendarEntry *Calendar [GSM_MAXCALENDARTODONOTES + 1];
+ GSM_Bitmap *CallerLogos [GSM_BACKUP_MAX_CALLER + 1];
+ GSM_SMSC *SMSC [GSM_BACKUP_MAX_SMSC + 1];
+ GSM_WAPBookmark *WAPBookmark [GSM_BACKUP_MAX_WAPBOOKMARK + 1];
+ GSM_MultiWAPSettings *WAPSettings [GSM_BACKUP_MAX_WAPSETTINGS + 1];
+ GSM_MultiWAPSettings *MMSSettings [GSM_BACKUP_MAX_MMSSETTINGS + 1];
+ GSM_Ringtone *Ringtone [GSM_BACKUP_MAX_RINGTONES + 1];
+ GSM_ToDoEntry *ToDo [GSM_MAXCALENDARTODONOTES + 1];
+ GSM_Profile *Profiles [GSM_BACKUP_MAX_PROFILES + 1];
+ GSM_FMStation *FMStation [GSM_BACKUP_MAX_FMSTATIONS +1];
+ GSM_GPRSAccessPoint *GPRSPoint [GSM_BACKUP_MAX_GPRSPOINT + 1];
+ GSM_NoteEntry *Note [GSM_BACKUP_MAX_NOTE + 1];
+ GSM_Bitmap *StartupLogo;
+ GSM_Bitmap *OperatorLogo;
+} GSM_Backup;
+
+#define GSM_BACKUP_MAX_SMS 500
+
+typedef struct {
+ GSM_SMSMessage *SMS[GSM_BACKUP_MAX_SMS];
+} GSM_SMS_Backup;
+
+extern GSM_Error GSM_ReadSMSBackupFile(char *FileName, GSM_SMS_Backup *backup);
+extern GSM_Error GSM_SaveSMSBackupFile(char *FileName, GSM_SMS_Backup *backup);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backics.c b/gammu/emb/common/service/backup/backics.c
new file mode 100644
index 0000000..3e6b083
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backics.c
@@ -0,0 +1,42 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backics.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+GSM_Error SaveICS(char *FileName, GSM_Backup *backup)
+{
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ fclose(file);
+ return ERR_NONE;
+}
+
+GSM_Error LoadICS(char *FileName, GSM_Backup *backup)
+{
+ GSM_File File;
+ GSM_Error error;
+
+ File.Buffer = NULL;
+ error = GSM_ReadFile(FileName, &File);
+ if (error != ERR_NONE) return error;
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backics.h b/gammu/emb/common/service/backup/backics.h
new file mode 100644
index 0000000..80e2b15
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backics.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backics_h
+#define __gsm_backics_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error SaveICS(char *FileName, GSM_Backup *backup);
+GSM_Error LoadICS(char *FileName, GSM_Backup *backup);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backldif.c b/gammu/emb/common/service/backup/backldif.c
new file mode 100644
index 0000000..ab16160
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backldif.c
@@ -0,0 +1,297 @@
+/* (c) 2003-2004 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backldif.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+static void SaveLDIFText(FILE *file, unsigned char *Name, unsigned char *Value)
+{
+ unsigned char Buffer[1000],Buffer2[1000];
+
+ if (EncodeUTF8(Buffer, Value)) {
+// dbgprintf("%s\n",Buffer);
+ EncodeBASE64(Buffer, Buffer2, strlen(Buffer));
+ fprintf(file,"%s:: %s%c%c",Name,Buffer2,13,10);
+ } else {
+ fprintf(file,"%s: %s%c%c",Name,DecodeUnicodeString(Value),13,10);
+ }
+}
+
+GSM_Error SaveLDIF(char *FileName, GSM_Backup *backup)
+{
+ int i, j;
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ i=0;
+ while (backup->PhonePhonebook[i]!=NULL) {
+ for (j=0;j<backup->PhonePhonebook[i]->EntriesNum;j++) {
+ switch (backup->PhonePhonebook[i]->Entries[j].EntryType) {
+ case PBK_Text_Name:
+ SaveLDIFText(file, "dn", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ default:
+ break;
+ }
+ }
+ fprintf(file, "objectclass: top%c%c",13,10);
+ fprintf(file, "objectclass: person%c%c",13,10);
+ fprintf(file, "objectclass: organizationalPerson%c%c",13,10);
+ fprintf(file, "objectclass: inetOrgPerson%c%c",13,10);
+ fprintf(file, "objectclass: mozillaAbPersonObsolete%c%c",13,10);
+ for (j=0;j<backup->PhonePhonebook[i]->EntriesNum;j++) {
+ switch (backup->PhonePhonebook[i]->Entries[j].EntryType) {
+ case PBK_Text_Postal:
+ SaveLDIFText(file, "HomePostalAddress", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_URL:
+ SaveLDIFText(file, "homeurl", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Name:
+ SaveLDIFText(file, "givenName", backup->PhonePhonebook[i]->Entries[j].Text);
+ SaveLDIFText(file, "cn", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Note:
+ SaveLDIFText(file, "Description", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Number_Work:
+ SaveLDIFText(file, "workPhone", backup->PhonePhonebook[i]->Entries[j].Text);//not exist in Mozilla 1.4 win32
+ break;
+ case PBK_Number_Mobile:
+ SaveLDIFText(file, "mobile", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Number_Pager:
+ SaveLDIFText(file, "pager", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Number_Fax:
+ SaveLDIFText(file, "fax", backup->PhonePhonebook[i]->Entries[j].Text);//facsimileTelephoneNumber
+ break;
+ case PBK_Number_Home:
+ SaveLDIFText(file, "homePhone", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Number_General:
+ SaveLDIFText(file, "telephoneNumber", backup->PhonePhonebook[i]->Entries[j].Text);//work in Mozilla 1.4 win32
+ break;
+ case PBK_Text_Email:
+ SaveLDIFText(file, "mail", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Email2:
+ SaveLDIFText(file, "mozillaSecondEmail", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Custom1:
+ SaveLDIFText(file, "custom1", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Custom2:
+ SaveLDIFText(file, "custom2", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Custom3:
+ SaveLDIFText(file, "custom3", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Custom4:
+ SaveLDIFText(file, "custom4", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Company:
+ SaveLDIFText(file, "o", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_JobTitle:
+ SaveLDIFText(file, "title", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_StreetAddress:
+ SaveLDIFText(file, "homePostalAddress", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_City:
+ SaveLDIFText(file, "mozillaHomeLocalityName", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_State:
+ SaveLDIFText(file, "mozillaHomeState", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Zip:
+ SaveLDIFText(file, "mozillaHomePostalCode", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Text_Country:
+ SaveLDIFText(file, "mozillaHomeCountryName", backup->PhonePhonebook[i]->Entries[j].Text);
+ break;
+ case PBK_Number_Other:
+ case PBK_Caller_Group:
+ case PBK_RingtoneID:
+ case PBK_PictureID:
+ case PBK_Date:
+ case PBK_RingtoneFileSystemID:
+ case PBK_Text_UserID:
+ case PBK_SMSListID:
+ case PBK_Category:
+ case PBK_Private:
+ case PBK_Text_LastName:
+ case PBK_Text_FirstName:
+ dbgprintf("Feature missed\n");
+ break;
+ }
+ }
+ fprintf(file, "%c%c",13,10);
+ i++;
+ }
+ fclose(file);
+ return ERR_NONE;
+}
+
+static bool ReadLDIFText(char *Buffer, char *Start, char *Value)
+{
+ unsigned char Buffer2[1000],buff[200];
+ int i;
+
+ Value[0] = 0x00;
+
+ strcpy(buff,Start);
+ strcat(buff,":: ");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ i = DecodeBASE64(Buffer+strlen(Start)+3, Buffer2, strlen(Buffer)-(strlen(Start)+3));
+ dbgprintf("Text after DecodeBASE64 is \"%s\"\n",Buffer2);
+ DecodeUTF8(Value, Buffer2, i);
+ dbgprintf("Text after DecodeUTF8 is \"%s\"\n",DecodeUnicodeString(Value));
+ return true;
+ }
+ strcpy(buff,Start);
+ strcat(buff,": ");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ EncodeUnicode(Value,Buffer+strlen(Start)+2,strlen(Buffer)-(strlen(Start)+2));
+ dbgprintf("Text after EncodeUnicode is \"%s\"\n",DecodeUnicodeString(Value));
+ return true;
+ }
+ return false;
+}
+
+static GSM_Error GSM_DecodeLDIFEntry(unsigned char *Buffer, int *Pos, GSM_MemoryEntry *Pbk)
+{
+ unsigned char Line[2000],Buff[2000],Buff2[2000];
+ int Level = 0;
+
+ Buff[0] = 0;
+ Pbk->EntriesNum = 0;
+
+ while (1) {
+ MyGetLine(Buffer, Pos, Line, strlen(Buffer));
+ if (strlen(Line) == 0) break;
+ switch (Level) {
+ case 0:
+ if (ReadLDIFText(Line, "objectclass", Buff)) {
+ sprintf(Buff2,"%s",DecodeUnicodeString(Buff));
+ if (!strcmp("mozillaAbPersonObsolete",Buff2)) {
+ dbgprintf("level1\n");
+ Level = 1;
+ }
+ }
+ break;
+ case 1:
+ if (ReadLDIFText(Line, "dn", Buff)) {
+ dbgprintf("entries num is %i\n",Pbk->EntriesNum);
+ if (Pbk->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+ }
+ if (ReadLDIFText(Line, "givenName", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Name;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "telephoneNumber", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_General;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "mobile", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Mobile;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "workPhone", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Work;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "fax", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Fax;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "homePhone",Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Home;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "Description", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Note;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "HomePostalAddress", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Postal;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "mail", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Email;
+ Pbk->EntriesNum++;
+ }
+ if (ReadLDIFText(Line, "homeurl", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_URL;
+ Pbk->EntriesNum++;
+ }
+ /* FIXME: add rest */
+ break;
+ }
+ }
+
+ dbgprintf("entries num is %i\n",Pbk->EntriesNum);
+ if (Pbk->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+GSM_Error LoadLDIF(char *FileName, GSM_Backup *backup)
+{
+ GSM_File File;
+ GSM_Error error;
+ GSM_MemoryEntry Pbk;
+ int numPbk = 0, Pos;
+
+ File.Buffer = NULL;
+ error = GSM_ReadFile(FileName, &File);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ while (1) {
+ error = GSM_DecodeLDIFEntry(File.Buffer, &Pos, &Pbk);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (numPbk < GSM_BACKUP_MAX_PHONEPHONEBOOK) {
+ backup->PhonePhonebook[numPbk] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->PhonePhonebook[numPbk] == NULL) return ERR_MOREMEMORY;
+ backup->PhonePhonebook[numPbk + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_PHONEPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ memcpy(backup->PhonePhonebook[numPbk],&Pbk,sizeof(GSM_MemoryEntry));
+ backup->PhonePhonebook[numPbk]->Location = numPbk + 1;
+ backup->PhonePhonebook[numPbk]->MemoryType = MEM_ME;
+ numPbk++;
+ }
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backldif.h b/gammu/emb/common/service/backup/backldif.h
new file mode 100644
index 0000000..212c048
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backldif.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backldif_h
+#define __gsm_backldif_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error SaveLDIF(char *FileName, GSM_Backup *backup);
+GSM_Error LoadLDIF(char *FileName, GSM_Backup *backup);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backlmb.c b/gammu/emb/common/service/backup/backlmb.c
new file mode 100644
index 0000000..d7f845a
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backlmb.c
@@ -0,0 +1,413 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backlmb.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+static void SaveLMBStartupEntry(FILE *file, GSM_Bitmap bitmap)
+{
+ int count=13;
+ GSM_Phone_Bitmap_Types Type;
+ /* Welcome note and logo header block */
+ char req[1000] = {
+ 'W','E','L',' ', /*block identifier*/
+ 00,00, /*block data size*/
+ 0x02,00,00,00,00,00,
+ 0x00}; /*number of blocks (like in 6110 frame)*/
+
+ if (bitmap.Type == GSM_StartupLogo) {
+ req[count++] = 0x01;
+ req[count++] = bitmap.BitmapHeight;
+ req[count++] = bitmap.BitmapWidth;
+ Type = GSM_NokiaStartupLogo;
+ switch (bitmap.BitmapHeight) {
+ case 65: Type = GSM_Nokia7110StartupLogo; break;
+ case 60: Type = GSM_Nokia6210StartupLogo; break;
+ }
+ PHONE_EncodeBitmap(Type, req+count, &bitmap);
+ count = count + PHONE_GetBitmapSize(Type, 0, 0);
+
+ req[12]++;
+ }
+ if (bitmap.Type == GSM_WelcomeNote_Text) {
+ req[count++]=0x02;
+ req[count++]=UnicodeLength(bitmap.Text);
+ memcpy(req+count,DecodeUnicodeString(bitmap.Text),UnicodeLength(bitmap.Text));
+ count=count+UnicodeLength(bitmap.Text);
+
+ req[12]++;
+ }
+
+ req[4]=(count-12)%256;
+ req[5]=(count-12)/256;
+
+ fwrite(req, 1, count, file);
+}
+
+static void SaveLMBCallerEntry(FILE *file, GSM_Bitmap bitmap)
+{
+ int count=12, textlen;
+ char req[500] = {
+ 'C','G','R',' ', /*block identifier*/
+ 00,00, /*block data size*/
+ 02,00,
+ 00, /*group number=0,1,etc.*/
+ 00,00,00};
+
+ req[count++] = bitmap.Location - 1;
+ if (bitmap.DefaultName) {
+ req[count++] = 0;
+ } else {
+ textlen = UnicodeLength(bitmap.Text);
+ req[count++] = textlen;
+ memcpy(req+count,DecodeUnicodeString(bitmap.Text),textlen);
+ count += textlen;
+ }
+ if (bitmap.DefaultRingtone) {
+ req[count++] = 0x16;
+ } else {
+ req[count++] = bitmap.RingtoneID;
+ }
+ if (bitmap.BitmapEnabled) req[count++] = 0x01; else req[count++] = 0x00;
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 4) >> 8;
+ req[count++] = (PHONE_GetBitmapSize(GSM_NokiaCallerLogo,0,0) + 4) % 0xff;
+ if (bitmap.DefaultBitmap) {
+ bitmap.BitmapWidth = 72;
+ bitmap.BitmapHeight = 14;
+ GSM_ClearBitmap(&bitmap);
+ }
+ NOKIA_CopyBitmap(GSM_NokiaCallerLogo, &bitmap, req, &count);
+ req[count++]=0;
+
+ req[4]=(count-12)%256;
+ req[5]=(count-12)/256;
+ req[8]=bitmap.Location;
+
+ fwrite(req, 1, count, file);
+}
+
+void SaveLMBPBKEntry(FILE *file, GSM_MemoryEntry *entry)
+{
+ int count = 16, blocks;
+ char req[500] = {
+ 'P','B','E','2', /*block identifier*/
+ 00,00, /*block data size*/
+ 00,00,
+ 00,00, /*position of phonebook entry*/
+ 03, /*memory type. ME=02;SM=03*/
+ 00,
+ 00,00, /*position of phonebook entry*/
+ 03, /*memory type. ME=02;SM=03*/
+ 00};
+
+ count=count+N71_65_EncodePhonebookFrame(NULL, req+16, *entry, &blocks, true, true);
+
+ req[4]=(count-12)%256;
+ req[5]=(count-12)/256;
+ req[8]=req[12] = entry->Location & 0xff;
+ req[9]=req[13] = (entry->Location >> 8);
+ if (entry->MemoryType==MEM_ME) req[10]=req[14]=2;
+
+ fwrite(req, 1, count, file);
+}
+
+GSM_Error SaveLMB(char *FileName, GSM_Backup *backup)
+{
+ FILE *file;
+ int i;
+ char LMBHeader[] = {'L','M','B',' '}; /*file identifier*/
+ char PBKHeader[] = { /*Phonebook header block */
+ 'P','B','K',' ', /*block identifier*/
+ 0x08,00, /*block data size*/
+ 0x02,00,
+ 03, /*memory type. ME=02;SM=03*/
+ 00,00,00,
+ 00,00, /*size of phonebook*/
+ 14, /*max length of each position*/
+ 00,00,00,00,00};
+
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ /* Write the header of the file. */
+ fwrite(LMBHeader, 1, sizeof(LMBHeader), file);
+
+ if (backup->PhonePhonebook[0]!=NULL) {
+ PBKHeader[8] = 2; /* memory type=MEM_ME */
+ PBKHeader[12] = (unsigned char)(500 % 256);
+ PBKHeader[13] = 500 / 256;
+ fwrite(PBKHeader, 1, sizeof(PBKHeader), file);
+ i=0;
+ while (backup->PhonePhonebook[i]!=NULL) {
+ SaveLMBPBKEntry(file, backup->PhonePhonebook[i]);
+ i++;
+ }
+ }
+ if (backup->SIMPhonebook[0]!=NULL) {
+ PBKHeader[8] = 3; /* memory type=MEM_SM */
+ PBKHeader[12] = (unsigned char)(250 % 256);
+ PBKHeader[13] = 250 / 256;
+ PBKHeader[14] = 0x16; /* max size of one entry */
+ fwrite(PBKHeader, 1, sizeof(PBKHeader), file);
+ i=0;
+ while (backup->SIMPhonebook[i]!=NULL) {
+ SaveLMBPBKEntry(file, backup->SIMPhonebook[i]);
+ i++;
+ }
+ }
+ i=0;
+ while (backup->CallerLogos[i]!=NULL) {
+ SaveLMBCallerEntry(file, *backup->CallerLogos[i]);
+ i++;
+ }
+ if (backup->StartupLogo!=NULL) {
+ SaveLMBStartupEntry(file, *backup->StartupLogo);
+ }
+
+ fclose(file);
+ return ERR_NONE;
+}
+
+static GSM_Error LoadLMBCallerEntry(unsigned char *buffer, unsigned char *buffer2, GSM_Backup *backup)
+{
+ GSM_Bitmap bitmap;
+ int num;
+
+#ifdef DEBUG
+ dbgprintf("Number %i, name \"", buffer2[0]+1);
+ for (num=0;num<buffer2[1];num++) dbgprintf("%c", buffer2[num+2]);
+ dbgprintf("\"\n");
+ dbgprintf("Ringtone ID=%i\n", buffer2[num+2]);
+ if (buffer2[num+3]==1) {
+ dbgprintf("Logo enabled\n");
+ } else {
+ dbgprintf("Logo disabled\n");
+ }
+#endif
+
+ bitmap.Location = buffer2[0] + 1;
+ bitmap.Type = GSM_CallerGroupLogo;
+ bitmap.DefaultRingtone = false;
+ bitmap.RingtoneID = buffer2[buffer2[1]+2];
+
+ EncodeUnicode(bitmap.Text,buffer2+2,buffer2[1]);
+ if (bitmap.Text[0] == 0x00 && bitmap.Text[1] == 0x00) {
+ bitmap.DefaultName = true;
+ } else {
+ bitmap.DefaultName = false;
+ }
+
+ bitmap.BitmapEnabled = false;
+ if (buffer2[buffer2[1]+3]==1) bitmap.BitmapEnabled=true;
+
+ bitmap.DefaultBitmap = false;
+ PHONE_DecodeBitmap(GSM_NokiaCallerLogo, buffer2+(buffer2[1]+10), &bitmap);
+
+#ifdef DEBUG
+ dbgprintf("Caller logo\n");
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&bitmap);
+#endif
+
+ num = 0;
+ while (backup->CallerLogos[num] != NULL) num++;
+ if (num < GSM_BACKUP_MAX_CALLER) {
+ backup->CallerLogos[num] = malloc(sizeof(GSM_Bitmap));
+ if (backup->CallerLogos[num] == NULL) return ERR_MOREMEMORY;
+ backup->CallerLogos[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_CALLER\n");
+ return ERR_MOREMEMORY;
+ }
+ *backup->CallerLogos[num] = bitmap;
+
+ return ERR_NONE;
+}
+
+static GSM_Error LoadLMBStartupEntry(unsigned char *buffer, unsigned char *buffer2, GSM_Backup *backup)
+{
+ int i,j;
+#ifdef DEBUG
+ int z;
+#endif
+ GSM_Phone_Bitmap_Types Type;
+
+ j=1;
+ for (i=0;i<buffer2[0];i++) {
+ switch (buffer2[j++]) {
+ case 1:
+ dbgprintf("Block 1 - startup logo\n");
+ backup->StartupLogo = malloc(sizeof(GSM_Bitmap));
+ if (backup->StartupLogo == NULL) return ERR_MOREMEMORY;
+ backup->StartupLogo->Location = 1;
+ backup->StartupLogo->BitmapHeight = buffer2[j++];
+ backup->StartupLogo->BitmapWidth = buffer2[j++];
+ Type = GSM_NokiaStartupLogo;
+ switch (backup->StartupLogo->BitmapHeight) {
+ case 65: Type = GSM_Nokia7110StartupLogo; break;
+ case 60: Type = GSM_Nokia6210StartupLogo; break;
+ }
+ PHONE_DecodeBitmap(Type, buffer2+j, backup->StartupLogo);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,backup->StartupLogo);
+#endif
+ j = j + PHONE_GetBitmapSize(Type,0,0);
+ break;
+ case 2:
+#ifdef DEBUG
+ dbgprintf("Block 2 - welcome note \"");
+ for (z=0;z<buffer2[j];z++) dbgprintf("%c",buffer2[j+z+1]);
+ dbgprintf("\"\n");
+#endif
+ if (backup->StartupLogo == NULL) {
+ backup->StartupLogo = malloc(sizeof(GSM_Bitmap));
+ if (backup->StartupLogo == NULL) return ERR_MOREMEMORY;
+ backup->StartupLogo->Type = GSM_WelcomeNote_Text;
+ EncodeUnicode(backup->StartupLogo->Text,buffer2+j,buffer2[j]);
+ }
+ j = j + buffer2[j];
+ break;
+ default:
+ dbgprintf("Unknown block %02x\n",buffer2[j]);
+ break;
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error LoadLMBPbkEntry(unsigned char *buffer, unsigned char *buffer2, GSM_Backup *backup)
+{
+ GSM_MemoryEntry pbk;
+ int num;
+
+#ifdef DEBUG
+ dbgprintf("Memory : ");
+ switch(buffer[10]) {
+ case 2 : dbgprintf("(internal)\n"); break;
+ case 3 : dbgprintf("(sim)\n"); break;
+ default: dbgprintf("(unknown)\n"); break;
+ }
+ dbgprintf("Location : %i\n",buffer2[0]+buffer2[1]*256);
+#endif
+
+ N71_65_DecodePhonebook(NULL, &pbk, NULL,NULL,buffer2+4,(buffer[4]+buffer[5]*256)-4,false);
+
+ pbk.MemoryType=MEM_SM;
+ if (buffer[10]==2) pbk.MemoryType=MEM_ME;
+
+ pbk.Location=buffer2[0]+256*buffer2[1];
+
+ num = 0;
+ if (buffer[10]==2) {
+ while (backup->PhonePhonebook[num] != NULL) num++;
+ if (num < GSM_BACKUP_MAX_PHONEPHONEBOOK) {
+ backup->PhonePhonebook[num] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->PhonePhonebook[num] == NULL) return ERR_MOREMEMORY;
+ backup->PhonePhonebook[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_PHONEPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ *backup->PhonePhonebook[num] = pbk;
+ } else {
+ while (backup->SIMPhonebook[num] != NULL) num++;
+ if (num < GSM_BACKUP_MAX_SIMPHONEBOOK) {
+ backup->SIMPhonebook[num] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->SIMPhonebook[num] == NULL) return ERR_MOREMEMORY;
+ backup->SIMPhonebook[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_SIMPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ *backup->SIMPhonebook[num] = pbk;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error LoadLMB(char *FileName, GSM_Backup *backup)
+{
+#ifdef DEBUG
+ int i;
+#endif
+ unsigned char buffer[12], buffer2[1000];
+ FILE *file;
+ GSM_Error error;
+
+ file = fopen(FileName, "rb");
+ if (file == NULL) return(ERR_CANTOPENFILE);
+
+ /* Read the header of the file. */
+ fread(buffer, 1, 4, file);
+
+ /* while we have something to read */
+ while (fread(buffer, 1, 12, file)==12) {
+#ifdef DEBUG
+ /* Info about block in the file */
+ dbgprintf("Block \"");
+ for (i=0;i<4;i++) {dbgprintf("%c",buffer[i]);}
+ dbgprintf("\" (");
+ if (memcmp(buffer, "PBK ",4)==0) { dbgprintf("Phonebook");
+ } else if (memcmp(buffer, "PBE2",4)==0) { dbgprintf("Phonebook entry");
+ } else if (memcmp(buffer, "CGR ",4)==0) { dbgprintf("Caller group");
+ } else if (memcmp(buffer, "SPD ",4)==0) { dbgprintf("Speed dial");
+ } else if (memcmp(buffer, "OLG ",4)==0) { dbgprintf("Operator logo");
+ } else if (memcmp(buffer, "WEL ",4)==0) { dbgprintf("Startup logo and welcome text");
+ } else { dbgprintf("unknown - ignored");
+ }
+ dbgprintf(") - length %i\n", buffer[4]+buffer[5]*256);
+#endif
+ /* reading block data */
+ fread(buffer2, 1, buffer[4]+buffer[5]*256, file);
+
+#ifdef DEBUG
+ if (memcmp(buffer, "PBK ",4)==0) {
+ dbgprintf("Size of phonebook %i, type %i ",(buffer2[0]+buffer2[1]*256),buffer[8]);
+ switch(buffer[8]) {
+ case 2 : dbgprintf("(internal)");break;
+ case 3 : dbgprintf("(sim)") ;break;
+ default: dbgprintf("(unknown)") ;break;
+ }
+ dbgprintf(", length of each position - %i\n",buffer2[2]);
+ }
+#endif
+ if (memcmp(buffer, "PBE2",4)==0) {
+ error = LoadLMBPbkEntry(buffer,buffer2,backup);
+ if (error != ERR_NONE) {
+ fclose(file);
+ return error;
+ }
+ }
+ if (memcmp(buffer, "CGR ",4)==0) {
+ error = LoadLMBCallerEntry(buffer, buffer2, backup);
+ if (error != ERR_NONE) {
+ fclose(file);
+ return error;
+ }
+ }
+ if (memcmp(buffer, "WEL ",4)==0) {
+ error = LoadLMBStartupEntry(buffer, buffer2, backup);
+ if (error != ERR_NONE) {
+ fclose(file);
+ return error;
+ }
+ }
+ }
+
+ fclose(file);
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backlmb.h b/gammu/emb/common/service/backup/backlmb.h
new file mode 100644
index 0000000..18bba66
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backlmb.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backlmb_h
+#define __gsm_backlmb_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error SaveLMB(char *FileName, GSM_Backup *backup);
+GSM_Error LoadLMB(char *FileName, GSM_Backup *backup);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backtext.c b/gammu/emb/common/service/backup/backtext.c
new file mode 100644
index 0000000..fee0f73
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backtext.c
@@ -0,0 +1,2908 @@
+/* (c) 2002-2004 by Marcin Wiacek, Walek and Michal Cihar */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/cfg.h"
+#include "../../misc/coding/coding.h"
+#include "../../misc/coding/md5.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backtext.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+GSM_Error FindBackupChecksum(char *FileName, bool UseUnicode, char *checksum)
+{
+ INI_Section *file_info, *h;
+ INI_Entry *e;
+ char *buffer = NULL,buff[100];
+ int len=0;
+
+ //int i;
+
+ file_info = INI_ReadFile(FileName, UseUnicode);
+
+ if (UseUnicode) {
+ for (h = file_info; h != NULL; h = h->Next) {
+ EncodeUnicode(buff,"Checksum",8);
+ if (mywstrncasecmp(buff, h->SectionName, 8)) continue;
+
+ buffer = (unsigned char *)realloc(buffer,len+UnicodeLength(h->SectionName)*2+2);
+ CopyUnicodeString(buffer+len,h->SectionName);
+ len+=UnicodeLength(h->SectionName)*2;
+// dbgprintf("[%s]\n",DecodeUnicodeConsole(h->SectionName));
+
+ for (e = h->SubEntries; e != NULL; e = e->Next) {
+ buffer = (unsigned char *)realloc(buffer,len+UnicodeLength(e->EntryName)*2+2);
+ CopyUnicodeString(buffer+len,e->EntryName);
+ len+=UnicodeLength(e->EntryName)*2;
+ buffer = (unsigned char *)realloc(buffer,len+UnicodeLength(e->EntryValue)*2+2);
+ CopyUnicodeString(buffer+len,e->EntryValue);
+ len+=UnicodeLength(e->EntryValue)*2;
+// dbgprintf("\"%s\"",DecodeUnicodeConsole(e->EntryName));
+// dbgprintf("=\"%s\"\n",DecodeUnicodeConsole(e->EntryValue));
+ }
+ }
+ } else {
+ for (h = file_info; h != NULL; h = h->Next) {
+ if (mystrncasecmp("Checksum", h->SectionName, 8)) continue;
+
+ buffer = (unsigned char *)realloc(buffer,len+strlen(h->SectionName)+1);
+ strcpy(buffer+len,h->SectionName);
+ len+=strlen(h->SectionName);
+
+ for (e = h->SubEntries; e != NULL; e = e->Next) {
+// dbgprintf("%s=%s\n",e->EntryName,e->EntryValue);
+ buffer = (unsigned char *)realloc(buffer,len+strlen(e->EntryName)+1);
+ strcpy(buffer+len,e->EntryName);
+ len+=strlen(e->EntryName);
+ buffer = (unsigned char *)realloc(buffer,len+strlen(e->EntryValue)+1);
+ strcpy(buffer+len,e->EntryValue);
+ len+=strlen(e->EntryValue);
+ }
+ }
+ }
+
+ //for (i=0;i<len;i++) printf("%02x",buffer[i]);
+ CalculateMD5(buffer, len, checksum);
+ free(buffer);
+
+ return ERR_NONE;
+}
+
+static unsigned char *ReadCFGText(INI_Section *cfg, unsigned char *section, unsigned char *key, bool Unicode)
+{
+ unsigned char Buffer[500],Buffer2[500],*retval;
+
+ if (Unicode) {
+ EncodeUnicode(Buffer2,key,strlen(key));
+ retval = INI_GetValue(cfg,section,Buffer2,Unicode);
+ if (retval != NULL) return DecodeUnicodeString(retval);
+ return NULL;
+ } else {
+ strcpy(Buffer,section);
+ strcpy(Buffer2,key);
+ return INI_GetValue(cfg,section,key,Unicode);
+ }
+}
+
+static void SaveLinkedBackupText(FILE *file, char *myname, char *myvalue, bool UseUnicode)
+{
+ int w,current;
+ unsigned char buffer2[1000],buffer3[1000];
+
+ current = strlen(myvalue); w = 0;
+ while (true) {
+ if (current > 200) {
+ memcpy(buffer2,myvalue+(strlen(myvalue)-current),200);
+ buffer2[200] = 0;
+ current = current - 200;
+ } else {
+ memcpy(buffer2,myvalue+(strlen(myvalue)-current),current);
+ buffer2[current] = 0;
+ current = 0;
+ }
+ if (UseUnicode) {
+ sprintf(buffer3,"%s%02i = %s%c%c",myname,w,buffer2,13,10);
+ EncodeUnicode(buffer2,buffer3,strlen(buffer3));
+ fwrite(buffer2,1,strlen(buffer3)*2,file);
+ } else {
+ fprintf(file,"%s%02i = %s%c%c",myname,w,buffer2,13,10);
+ }
+ if (current == 0) break;
+ w++;
+ }
+}
+
+static void ReadLinkedBackupText(INI_Section *file_info, char *section, char *myname, char *myvalue, bool UseUnicode)
+{
+ unsigned char buffer2[300];
+ char *readvalue;
+ int i;
+
+ i=0;
+ myvalue[0] = 0;
+ while (true) {
+ sprintf(buffer2,"%s%02i",myname,i);
+ readvalue = ReadCFGText(file_info, section, buffer2, UseUnicode);
+ if (readvalue!=NULL) {
+ myvalue[strlen(myvalue)+strlen(readvalue)]=0;
+ memcpy(myvalue+strlen(myvalue),readvalue,strlen(readvalue));
+ } else break;
+ i++;
+ }
+}
+
+static void SaveBackupText(FILE *file, char *myname, char *myvalue, bool UseUnicode)
+{
+ unsigned char buffer[10000], buffer2[10000];
+
+ if (myname[0] == 0x00) {
+ if (UseUnicode) {
+ EncodeUnicode(buffer,myvalue,strlen(myvalue));
+ fwrite(buffer,1,strlen(myvalue)*2,file);
+ } else fprintf(file,"%s",myvalue);
+ } else {
+ if (UseUnicode) {
+ sprintf(buffer,"%s = \"",myname);
+ EncodeUnicode(buffer2,buffer,strlen(buffer));
+ fwrite(buffer2,1,strlen(buffer)*2,file);
+
+ fwrite(myvalue,1,UnicodeLength(myvalue)*2,file);
+
+ sprintf(buffer,"\"%c%c",13,10);
+ EncodeUnicode(buffer2,buffer,strlen(buffer));
+ fwrite(buffer2,1,strlen(buffer)*2,file);
+ } else {
+ sprintf(buffer,"%s = \"%s\"%c%c",myname,DecodeUnicodeString(myvalue),13,10);
+ fprintf(file,"%s",buffer);
+
+ EncodeHexBin(buffer,myvalue,UnicodeLength(myvalue)*2);
+ fprintf(file,"%sUnicode = %s%c%c",myname,buffer,13,10);
+ }
+ }
+}
+
+static bool ReadBackupText(INI_Section *file_info, char *section, char *myname, char *myvalue, bool UseUnicode)
+{
+ unsigned char paramname[10000],*readvalue;
+
+ if (UseUnicode) {
+ EncodeUnicode(paramname,myname,strlen(myname));
+ readvalue = INI_GetValue(file_info, section, paramname, UseUnicode);
+ if (readvalue!=NULL) {
+ CopyUnicodeString(myvalue,readvalue+2);
+ myvalue[UnicodeLength(readvalue)*2-4]=0;
+ myvalue[UnicodeLength(readvalue)*2-3]=0;
+ dbgprintf("%s\n",DecodeUnicodeString(readvalue));
+ } else {
+ myvalue[0]=0;
+ myvalue[1]=0;
+ return false;
+ }
+ } else {
+ strcpy(paramname,myname);
+ strcat(paramname,"Unicode");
+ readvalue = ReadCFGText(file_info, section, paramname, UseUnicode);
+ if (readvalue!=NULL) {
+ dbgprintf("%s %i\n",readvalue,strlen(readvalue));
+ DecodeHexBin (myvalue, readvalue, strlen(readvalue));
+ myvalue[strlen(readvalue)/2]=0;
+ myvalue[strlen(readvalue)/2+1]=0;
+ dbgprintf("%s\n",DecodeUnicodeString(myvalue));
+ } else {
+ strcpy(paramname,myname);
+ readvalue = ReadCFGText(file_info, section, paramname, UseUnicode);
+ if (readvalue!=NULL) {
+ EncodeUnicode(myvalue,readvalue+1,strlen(readvalue)-2);
+ } else {
+ myvalue[0]=0;
+ myvalue[1]=0;
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+static void SaveVCalDateTime(FILE *file, GSM_DateTime *dt, bool UseUnicode)
+{
+ unsigned char buffer[100];
+ int Length = 3;
+
+ sprintf(buffer, " = ");
+ SaveVCALDateTime(buffer, &Length, dt, NULL);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveVCalDate(FILE *file, GSM_DateTime *dt, bool UseUnicode)
+{
+ unsigned char buffer[100];
+
+ sprintf(buffer, " = %04d%02d%02d%c%c", dt->Year, dt->Month, dt->Day,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+/* ---------------------- backup files ------------------------------------- */
+
+static void SavePbkEntry(FILE *file, GSM_MemoryEntry *Pbk, bool UseUnicode)
+{
+ bool text;
+ char buffer[1000];
+ int j, i;
+
+ sprintf(buffer,"Location = %03i%c%c",Pbk->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ for (j=0;j<Pbk->EntriesNum;j++) {
+ text = true;
+ switch (Pbk->Entries[j].EntryType) {
+ case PBK_Number_General:
+ sprintf(buffer,"Entry%02iType = NumberGeneral%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Mobile:
+ sprintf(buffer,"Entry%02iType = NumberMobile%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Work:
+ sprintf(buffer,"Entry%02iType = NumberWork%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Fax:
+ sprintf(buffer,"Entry%02iType = NumberFax%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Home:
+ sprintf(buffer,"Entry%02iType = NumberHome%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Pager:
+ sprintf(buffer,"Entry%02iType = NumberPager%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Number_Other:
+ sprintf(buffer,"Entry%02iType = NumberOther%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Note:
+ sprintf(buffer,"Entry%02iType = Note%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Postal:
+ sprintf(buffer,"Entry%02iType = Postal%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Email:
+ sprintf(buffer,"Entry%02iType = Email%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Email2:
+ sprintf(buffer,"Entry%02iType = Email2%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_URL:
+ sprintf(buffer,"Entry%02iType = URL%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Name:
+ sprintf(buffer,"Entry%02iType = Name%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Caller_Group:
+ sprintf(buffer,"Entry%02iType = CallerGroup%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Entry%02iNumber = %i%c%c",j,Pbk->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ text = false;
+ break;
+ case PBK_RingtoneID:
+ sprintf(buffer,"Entry%02iType = RingtoneID%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Entry%02iNumber = %i%c%c",j,Pbk->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ text = false;
+ break;
+ case PBK_PictureID:
+ sprintf(buffer,"Entry%02iType = PictureID%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Entry%02iNumber = %i%c%c",j,Pbk->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ text = false;
+ break;
+ case PBK_Text_UserID:
+ sprintf(buffer,"Entry%02iType = UserID%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Category:
+ sprintf(buffer,"Entry%02iType = Category%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Entry%02iNumber = %i%c%c",j,Pbk->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ text = false;
+ break;
+ case PBK_Private:
+ sprintf(buffer,"Entry%02iType = Private%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Entry%02iNumber = %i%c%c",j,Pbk->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ text = false;
+ break;
+ case PBK_Text_LastName:
+ sprintf(buffer,"Entry%02iType = LastName%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_FirstName:
+ sprintf(buffer,"Entry%02iType = FirstName%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Company:
+ sprintf(buffer,"Entry%02iType = Company%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_JobTitle:
+ sprintf(buffer,"Entry%02iType = JobTitle%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_StreetAddress:
+ sprintf(buffer,"Entry%02iType = Address%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_City:
+ sprintf(buffer,"Entry%02iType = City%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_State:
+ sprintf(buffer,"Entry%02iType = State%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Zip:
+ sprintf(buffer,"Entry%02iType = Zip%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Country:
+ sprintf(buffer,"Entry%02iType = Country%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Custom1:
+ sprintf(buffer,"Entry%02iType = Custom1%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Custom2:
+ sprintf(buffer,"Entry%02iType = Custom2%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Custom3:
+ sprintf(buffer,"Entry%02iType = Custom3%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_Text_Custom4:
+ sprintf(buffer,"Entry%02iType = Custom4%c%c",j,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case PBK_SMSListID:
+ case PBK_RingtoneFileSystemID:
+ case PBK_Date:
+ break;
+ }
+ if (text) {
+ sprintf(buffer,"Entry%02iText",j);
+ SaveBackupText(file,buffer,Pbk->Entries[j].Text, UseUnicode);
+ }
+ switch (Pbk->Entries[j].EntryType) {
+ case PBK_Number_General:
+ case PBK_Number_Mobile:
+ case PBK_Number_Work:
+ case PBK_Number_Fax:
+ case PBK_Number_Home:
+ case PBK_Number_Other:
+ case PBK_Number_Pager:
+ if (Pbk->Entries[j].VoiceTag!=0) {
+ sprintf(buffer,"Entry%02iVoiceTag = %i%c%c",j,Pbk->Entries[j].VoiceTag,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ i = 0;
+ while (Pbk->Entries[j].SMSList[i]!=0) {
+ sprintf(buffer,"Entry%02iSMSList%02i = %i%c%c",j,i,Pbk->Entries[j].SMSList[i],13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ i++;
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveCalendarEntry(FILE *file, GSM_CalendarEntry *Note, bool UseUnicode)
+{
+ int i;
+ char buffer[1000];
+
+ sprintf(buffer,"Location = %d%c%c", Note->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "", "Type = ", UseUnicode);
+ switch (Note->Type) {
+ case GSM_CAL_REMINDER : sprintf(buffer,"Reminder%c%c", 13,10); break;
+ case GSM_CAL_CALL : sprintf(buffer,"Call%c%c", 13,10); break;
+ case GSM_CAL_MEETING : sprintf(buffer,"Meeting%c%c", 13,10); break;
+ case GSM_CAL_BIRTHDAY : sprintf(buffer,"Birthday%c%c", 13,10); break;
+ case GSM_CAL_TRAVEL : sprintf(buffer,"Travel%c%c", 13,10); break;
+ case GSM_CAL_VACATION : sprintf(buffer,"Vacation%c%c", 13,10); break;
+ case GSM_CAL_MEMO : sprintf(buffer,"Memo%c%c", 13,10); break;
+ case GSM_CAL_ALARM : sprintf(buffer,"Alarm%c%c", 13,10); break;
+ case GSM_CAL_DAILY_ALARM : sprintf(buffer,"DailyAlarm%c%c", 13,10); break;
+ case GSM_CAL_T_ATHL : sprintf(buffer,"Training/Athletism%c%c", 13,10); break;
+ case GSM_CAL_T_BALL : sprintf(buffer,"Training/BallGames%c%c", 13,10); break;
+ case GSM_CAL_T_CYCL : sprintf(buffer,"Training/Cycling%c%c", 13,10); break;
+ case GSM_CAL_T_BUDO : sprintf(buffer,"Training/Budo%c%c", 13,10); break;
+ case GSM_CAL_T_DANC : sprintf(buffer,"Training/Dance%c%c", 13,10); break;
+ case GSM_CAL_T_EXTR : sprintf(buffer,"Training/ExtremeSports%c%c", 13,10); break;
+ case GSM_CAL_T_FOOT : sprintf(buffer,"Training/Football%c%c", 13,10); break;
+ case GSM_CAL_T_GOLF : sprintf(buffer,"Training/Golf%c%c", 13,10); break;
+ case GSM_CAL_T_GYM : sprintf(buffer,"Training/Gym%c%c", 13,10); break;
+ case GSM_CAL_T_HORS : sprintf(buffer,"Training/HorseRaces%c%c", 13,10); break;
+ case GSM_CAL_T_HOCK : sprintf(buffer,"Training/Hockey%c%c", 13,10); break;
+ case GSM_CAL_T_RACE : sprintf(buffer,"Training/Races%c%c", 13,10); break;
+ case GSM_CAL_T_RUGB : sprintf(buffer,"Training/Rugby%c%c", 13,10); break;
+ case GSM_CAL_T_SAIL : sprintf(buffer,"Training/Sailing%c%c", 13,10); break;
+ case GSM_CAL_T_STRE : sprintf(buffer,"Training/StreetGames%c%c", 13,10); break;
+ case GSM_CAL_T_SWIM : sprintf(buffer,"Training/Swimming%c%c", 13,10); break;
+ case GSM_CAL_T_TENN : sprintf(buffer,"Training/Tennis%c%c", 13,10); break;
+ case GSM_CAL_T_TRAV : sprintf(buffer,"Training/Travels%c%c", 13,10); break;
+ case GSM_CAL_T_WINT : sprintf(buffer,"Training/WinterGames%c%c", 13,10); break;
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ for (i=0;i<Note->EntriesNum;i++) {
+ switch (Note->Entries[i].EntryType) {
+ case CAL_START_DATETIME:
+ SaveBackupText(file, "", "StartTime", UseUnicode);
+ SaveVCalDateTime(file, &Note->Entries[i].Date, UseUnicode);
+ break;
+ case CAL_END_DATETIME:
+ SaveBackupText(file, "", "StopTime", UseUnicode);
+ SaveVCalDateTime(file, &Note->Entries[i].Date, UseUnicode);
+ break;
+ case CAL_ALARM_DATETIME:
+ SaveBackupText(file, "", "Alarm", UseUnicode);
+ SaveVCalDateTime(file, &Note->Entries[i].Date, UseUnicode);
+ sprintf(buffer,"AlarmType = Tone%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_SILENT_ALARM_DATETIME:
+ SaveBackupText(file, "", "Alarm", UseUnicode);
+ SaveVCalDateTime(file, &Note->Entries[i].Date, UseUnicode);
+ sprintf(buffer,"AlarmType = Silent%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_PRIVATE:
+ sprintf(buffer, "Private = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_LOCATION:
+ SaveBackupText(file, "EventLocation", Note->Entries[i].Text, UseUnicode);
+ break;
+ case CAL_CONTACTID:
+ sprintf(buffer, "ContactID = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_RECURRANCE:
+ sprintf(buffer, "Recurrance = %d%c%c",Note->Entries[i].Number/24,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_TEXT:
+ SaveBackupText(file, "Text", Note->Entries[i].Text, UseUnicode);
+ break;
+ case CAL_PHONE:
+ SaveBackupText(file, "Phone", Note->Entries[i].Text, UseUnicode);
+ break;
+ case CAL_REPEAT_STOPDATE:
+ SaveBackupText(file, "", "RepeatStopDate", UseUnicode);
+ SaveVCalDate(file, &Note->Entries[i].Date, UseUnicode);
+ break;
+ case CAL_REPEAT_STARTDATE:
+ SaveBackupText(file, "", "RepeatStartDate", UseUnicode);
+ SaveVCalDate(file, &Note->Entries[i].Date, UseUnicode);
+ break;
+ case CAL_REPEAT_DAYOFWEEK:
+ sprintf(buffer, "RepeatDayOfWeek = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_REPEAT_DAY:
+ sprintf(buffer, "RepeatDay = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_REPEAT_WEEKOFMONTH:
+ sprintf(buffer, "RepeatWeekOfMonth = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_REPEAT_MONTH:
+ sprintf(buffer, "RepeatMonth = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case CAL_REPEAT_FREQUENCY:
+ sprintf(buffer, "RepeatFrequency = %d%c%c",Note->Entries[i].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ }
+ }
+ sprintf(buffer, "%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveWAPSettingsEntry(FILE *file, GSM_MultiWAPSettings *settings, bool UseUnicode)
+{
+ int i;
+ char buffer[10000];
+
+ if (settings->Active) {
+ sprintf(buffer,"Active = Yes%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ switch (settings->ActiveBearer) {
+ case WAPSETTINGS_BEARER_SMS : sprintf(buffer,"Bearer = SMS%c%c",13,10); break;
+ case WAPSETTINGS_BEARER_GPRS: sprintf(buffer,"Bearer = GPRS%c%c",13,10); break;
+ case WAPSETTINGS_BEARER_DATA: sprintf(buffer,"Bearer = Data%c%c",13,10); break;
+ case WAPSETTINGS_BEARER_USSD: sprintf(buffer,"Bearer = USSD%c%c",13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (settings->ReadOnly) {
+ sprintf(buffer,"ReadOnly = Yes%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ sprintf(buffer,"Proxy");
+ SaveBackupText(file, buffer, settings->Proxy, UseUnicode);
+ sprintf(buffer,"ProxyPort = %i%c%c",settings->ProxyPort,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Proxy2");
+ SaveBackupText(file, buffer, settings->Proxy2, UseUnicode);
+ sprintf(buffer,"Proxy2Port = %i%c%c",settings->Proxy2Port,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ for (i=0;i<settings->Number;i++) {
+ sprintf(buffer,"Title%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].Title, UseUnicode);
+ sprintf(buffer,"HomePage%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].HomePage, UseUnicode);
+ if (settings->Settings[i].IsContinuous) {
+ sprintf(buffer,"Type%02i = Continuous%c%c",i,13,10);
+ } else {
+ sprintf(buffer,"Type%02i = Temporary%c%c",i,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (settings->Settings[i].IsSecurity) {
+ sprintf(buffer,"Security%02i = On%c%c",i,13,10);
+ } else {
+ sprintf(buffer,"Security%02i = Off%c%c",i,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ switch (settings->Settings[i].Bearer) {
+ case WAPSETTINGS_BEARER_SMS:
+ sprintf(buffer,"Bearer%02i = SMS%c%c",i,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Server%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].Server, UseUnicode);
+ sprintf(buffer,"Service%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].Service, UseUnicode);
+ break;
+ case WAPSETTINGS_BEARER_GPRS:
+ sprintf(buffer,"Bearer%02i = GPRS%c%c",i,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"IP%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].IPAddress, UseUnicode);
+ case WAPSETTINGS_BEARER_DATA:
+ if (settings->Settings[i].Bearer == WAPSETTINGS_BEARER_DATA) {
+ sprintf(buffer,"Bearer%02i = Data%c%c",i,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (settings->Settings[i].IsISDNCall) {
+ sprintf(buffer,"CallType%02i = ISDN%c%c",i,13,10);
+ } else {
+ sprintf(buffer,"CallType%02i = Analogue%c%c",i,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"IP%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].IPAddress, UseUnicode);
+ }
+ sprintf(buffer,"Number%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].DialUp, UseUnicode);
+ if (settings->Settings[i].ManualLogin) {
+ sprintf(buffer,"Login%02i = Manual%c%c",i,13,10);
+ } else {
+ sprintf(buffer,"Login%02i = Automatic%c%c",i,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (settings->Settings[i].IsNormalAuthentication) {
+ sprintf(buffer,"Authentication%02i = Normal%c%c",i,13,10);
+ } else {
+ sprintf(buffer,"Authentication%02i = Secure%c%c",i,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ switch (settings->Settings[i].Speed) {
+ case WAPSETTINGS_SPEED_9600 : sprintf(buffer,"CallSpeed%02i = 9600%c%c" ,i,13,10); break;
+ case WAPSETTINGS_SPEED_14400: sprintf(buffer,"CallSpeed%02i = 14400%c%c",i,13,10); break;
+ case WAPSETTINGS_SPEED_AUTO : sprintf(buffer,"CallSpeed%02i = auto%c%c" ,i,13,10); break;
+ }
+ switch (settings->Settings[i].Speed) {
+ case WAPSETTINGS_SPEED_9600 :
+ case WAPSETTINGS_SPEED_14400:
+ case WAPSETTINGS_SPEED_AUTO :
+ SaveBackupText(file, "", buffer, UseUnicode);
+ default:
+ break;
+ }
+ sprintf(buffer,"User%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].User, UseUnicode);
+ sprintf(buffer,"Password%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].Password, UseUnicode);
+ break;
+ case WAPSETTINGS_BEARER_USSD:
+ sprintf(buffer,"Bearer%02i = USSD%c%c",i,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"ServiceCode%02i",i);
+ SaveBackupText(file, buffer, settings->Settings[i].Code, UseUnicode);
+ if (settings->Settings[i].IsIP) {
+ sprintf(buffer,"IP%02i",i);
+ } else {
+ sprintf(buffer,"Number%02i",i);
+ }
+ SaveBackupText(file, buffer, settings->Settings[i].Service, UseUnicode);
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+}
+
+static void SaveBitmapEntry(FILE *file, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[10000],buffer2[10000];
+ int x,y;
+
+ sprintf(buffer,"Width = %i%c%c",bitmap->BitmapWidth,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Height = %i%c%c",bitmap->BitmapHeight,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ for (y=0;y<bitmap->BitmapHeight;y++) {
+ for (x=0;x<bitmap->BitmapWidth;x++) {
+ buffer[x] = ' ';
+ if (GSM_IsPointBitmap(bitmap,x,y)) buffer[x]='#';
+ }
+ buffer[bitmap->BitmapWidth] = 0;
+ sprintf(buffer2,"Bitmap%02i = \"%s\"%c%c",y,buffer,13,10);
+ SaveBackupText(file, "", buffer2, UseUnicode);
+ }
+}
+
+static void SaveCallerEntry(FILE *file, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"Location = %03i%c%c",bitmap->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (!bitmap->DefaultName) SaveBackupText(file, "Name", bitmap->Text, UseUnicode);
+ if (!bitmap->DefaultRingtone) {
+ if (bitmap->FileSystemRingtone) {
+ sprintf(buffer,"FileRingtone = %02x%c%c",bitmap->RingtoneID,13,10);
+ } else {
+ sprintf(buffer,"Ringtone = %02x%c%c",bitmap->RingtoneID,13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ if (bitmap->BitmapEnabled) {
+ sprintf(buffer,"Enabled = True%c%c",13,10);
+ } else {
+ sprintf(buffer,"Enabled = False%c%c",13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (!bitmap->DefaultBitmap) SaveBitmapEntry(file, bitmap, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveWAPBookmarkEntry(FILE *file, GSM_WAPBookmark *bookmark, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ SaveBackupText(file, "URL", bookmark->Address, UseUnicode);
+ SaveBackupText(file, "Title", bookmark->Title, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveStartupEntry(FILE *file, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"[Startup]%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (bitmap->Type == GSM_WelcomeNote_Text) {
+ SaveBackupText(file, "Text", bitmap->Text, UseUnicode);
+ }
+ if (bitmap->Type == GSM_StartupLogo) {
+ SaveBitmapEntry(file, bitmap, UseUnicode);
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveSMSCEntry(FILE *file, GSM_SMSC *SMSC, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"Location = %03i%c%c",SMSC->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "Name", SMSC->Name, UseUnicode);
+ SaveBackupText(file, "Number", SMSC->Number, UseUnicode);
+ SaveBackupText(file, "DefaultNumber", SMSC->DefaultNumber, UseUnicode);
+ SaveBackupText(file, "", "Format = ", UseUnicode);
+ switch (SMSC->Format) {
+ case SMS_FORMAT_Text : sprintf(buffer,"Text"); break;
+ case SMS_FORMAT_Fax : sprintf(buffer,"Fax"); break;
+ case SMS_FORMAT_Email : sprintf(buffer,"Email"); break;
+ case SMS_FORMAT_Pager : sprintf(buffer,"Pager"); break;
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"%c%cValidity = ",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ switch (SMSC->Validity.Relative) {
+ case SMS_VALID_1_Hour : sprintf(buffer, "1hour" ); break;
+ case SMS_VALID_6_Hours : sprintf(buffer, "6hours" ); break;
+ case SMS_VALID_1_Day : sprintf(buffer, "24hours" ); break;
+ case SMS_VALID_3_Days : sprintf(buffer, "72hours" ); break;
+ case SMS_VALID_1_Week : sprintf(buffer, "1week" ); break;
+ case SMS_VALID_Max_Time :
+ default : sprintf(buffer,"MaximumTime" ); break;
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"%c%c%c%c",13,10,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveRingtoneEntry(FILE *file, GSM_Ringtone *ringtone, bool UseUnicode)
+{
+ unsigned char buffer[45000];
+ int i,j;
+
+ sprintf(buffer,"Location = %i%c%c",ringtone->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "Name", ringtone->Name, UseUnicode);
+ switch (ringtone->Format) {
+ case RING_NOKIABINARY:
+ j = 0; i = 0;
+ EncodeHexBin(buffer,ringtone->NokiaBinary.Frame,ringtone->NokiaBinary.Length);
+ SaveLinkedBackupText(file, "NokiaBinary", buffer, UseUnicode);
+ break;
+ case RING_MIDI:
+ j = 0; i = 0;
+ EncodeHexBin(buffer,ringtone->NokiaBinary.Frame,ringtone->NokiaBinary.Length);
+ SaveLinkedBackupText(file, "Pure Midi", buffer, UseUnicode);
+ break;
+ case RING_NOTETONE:
+ break;
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveOperatorEntry(FILE *file, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"[Operator]%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Network = \"%s\"%c%c", bitmap->NetworkCode,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBitmapEntry(file, bitmap, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveToDoEntry(FILE *file, GSM_ToDoEntry *ToDo, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+ int j;
+
+ sprintf(buffer,"Location = %i%c%c",ToDo->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ switch (ToDo->Priority) {
+ case GSM_Priority_High:
+ sprintf(buffer,"Priority = High%c%c",13,10);
+ break;
+ case GSM_Priority_Medium:
+ sprintf(buffer,"Priority = Medium%c%c",13,10);
+ break;
+ case GSM_Priority_Low:
+ sprintf(buffer,"Priority = Low%c%c",13,10);
+ break;
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+
+ for (j=0;j<ToDo->EntriesNum;j++) {
+ switch (ToDo->Entries[j].EntryType) {
+ case TODO_END_DATETIME:
+ SaveBackupText(file, "", "DueTime", UseUnicode);
+ SaveVCalDateTime(file, &ToDo->Entries[j].Date, UseUnicode);
+ break;
+ case TODO_COMPLETED:
+ sprintf(buffer,"Completed = %s%c%c",ToDo->Entries[j].Number == 1 ? "yes" : "no" ,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case TODO_ALARM_DATETIME:
+ SaveBackupText(file, "", "Alarm", UseUnicode);
+ SaveVCalDateTime(file, &ToDo->Entries[j].Date, UseUnicode);
+ break;
+ case TODO_SILENT_ALARM_DATETIME:
+ SaveBackupText(file, "", "SilentAlarm", UseUnicode);
+ SaveVCalDateTime(file, &ToDo->Entries[j].Date, UseUnicode);
+ break;
+ case TODO_TEXT:
+ SaveBackupText(file, "Text", ToDo->Entries[j].Text, UseUnicode);
+ break;
+ case TODO_PRIVATE:
+ sprintf(buffer,"Private = %i%c%c",ToDo->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case TODO_CATEGORY:
+ sprintf(buffer,"Category = %i%c%c",ToDo->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case TODO_CONTACTID:
+ sprintf(buffer,"ContactID = %i%c%c",ToDo->Entries[j].Number,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case TODO_PHONE:
+ SaveBackupText(file, "Phone", ToDo->Entries[j].Text, UseUnicode);
+ break;
+ }
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveProfileEntry(FILE *file, GSM_Profile *Profile, bool UseUnicode)
+{
+ int j,k;
+ bool special;
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"Location = %i%c%c",Profile->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "Name",Profile->Name, UseUnicode);
+
+ if (Profile->DefaultName) {
+ sprintf(buffer,"DefaultName = true%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ if (Profile->HeadSetProfile) {
+ sprintf(buffer,"HeadSetProfile = true%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ if (Profile->CarKitProfile) {
+ sprintf(buffer,"CarKitProfile = true%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+
+ for (j=0;j<Profile->FeaturesNumber;j++) {
+ sprintf(buffer,"Feature%02i = ",j);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ special = false;
+ switch (Profile->FeatureID[j]) {
+ case Profile_MessageToneID:
+ case Profile_RingtoneID:
+ special = true;
+ if (Profile->FeatureID[j] == Profile_RingtoneID) {
+ sprintf(buffer,"RingtoneID%c%c",13,10);
+ } else {
+ sprintf(buffer,"MessageToneID%c%c",13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Value%02i = %i%c%c",j,Profile->FeatureValue[j],13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case Profile_CallerGroups:
+ special = true;
+ sprintf(buffer,"CallerGroups%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Value%02i = ",j);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ for (k=0;k<5;k++) {
+ if (Profile->CallerGroups[k]) {
+ sprintf(buffer,"%i",k);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case Profile_ScreenSaverNumber:
+ special = true;
+ sprintf(buffer,"ScreenSaverNumber%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Value%02i = %i%c%c",j,Profile->FeatureValue[j],13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ break;
+ case Profile_CallAlert : sprintf(buffer,"IncomingCallAlert%c%c",13,10); break;
+ case Profile_RingtoneVolume : sprintf(buffer,"RingtoneVolume%c%c",13,10); break;
+ case Profile_Vibration : sprintf(buffer,"Vibrating%c%c",13,10); break;
+ case Profile_MessageTone : sprintf(buffer,"MessageTone%c%c",13,10); break;
+ case Profile_KeypadTone : sprintf(buffer,"KeypadTones%c%c",13,10); break;
+ case Profile_WarningTone : sprintf(buffer,"WarningTones%c%c",13,10); break;
+ case Profile_ScreenSaver : sprintf(buffer,"ScreenSaver%c%c",13,10); break;
+ case Profile_ScreenSaverTime : sprintf(buffer,"ScreenSaverTimeout%c%c",13,10); break;
+ case Profile_AutoAnswer : sprintf(buffer,"AutomaticAnswer%c%c",13,10); break;
+ case Profile_Lights : sprintf(buffer,"Lights%c%c",13,10); break;
+ default : special = true;
+ }
+ if (!special) {
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Value%02i = ",j);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ switch (Profile->FeatureValue[j]) {
+ case PROFILE_VOLUME_LEVEL1 :
+ case PROFILE_KEYPAD_LEVEL1 : sprintf(buffer,"Level1%c%c",13,10); break;
+ case PROFILE_VOLUME_LEVEL2 :
+ case PROFILE_KEYPAD_LEVEL2 : sprintf(buffer,"Level2%c%c",13,10); break;
+ case PROFILE_VOLUME_LEVEL3 :
+ case PROFILE_KEYPAD_LEVEL3 : sprintf(buffer,"Level3%c%c",13,10); break;
+ case PROFILE_VOLUME_LEVEL4 : sprintf(buffer,"Level4%c%c",13,10); break;
+ case PROFILE_VOLUME_LEVEL5 : sprintf(buffer,"Level5%c%c",13,10); break;
+ case PROFILE_MESSAGE_NOTONE :
+ case PROFILE_AUTOANSWER_OFF :
+ case PROFILE_LIGHTS_OFF :
+ case PROFILE_SAVER_OFF :
+ case PROFILE_WARNING_OFF :
+ case PROFILE_CALLALERT_OFF :
+ case PROFILE_VIBRATION_OFF :
+ case PROFILE_KEYPAD_OFF : sprintf(buffer,"Off%c%c",13,10); break;
+ case PROFILE_CALLALERT_RINGING : sprintf(buffer,"Ringing%c%c",13,10); break;
+ case PROFILE_CALLALERT_RINGONCE : sprintf(buffer,"RingOnce%c%c",13,10); break;
+ case PROFILE_CALLALERT_ASCENDING : sprintf(buffer,"Ascending%c%c",13,10); break;
+ case PROFILE_CALLALERT_CALLERGROUPS : sprintf(buffer,"CallerGroups%c%c",13,10); break;
+ case PROFILE_MESSAGE_STANDARD : sprintf(buffer,"Standard%c%c",13,10); break;
+ case PROFILE_MESSAGE_SPECIAL : sprintf(buffer,"Special%c%c",13,10); break;
+ case PROFILE_MESSAGE_BEEPONCE :
+ case PROFILE_CALLALERT_BEEPONCE : sprintf(buffer,"BeepOnce%c%c",13,10); break;
+ case PROFILE_MESSAGE_ASCENDING : sprintf(buffer,"Ascending%c%c",13,10); break;
+ case PROFILE_MESSAGE_PERSONAL : sprintf(buffer,"Personal%c%c",13,10); break;
+ case PROFILE_AUTOANSWER_ON :
+ case PROFILE_WARNING_ON :
+ case PROFILE_SAVER_ON :
+ case PROFILE_VIBRATION_ON : sprintf(buffer,"On%c%c",13,10); break;
+ case PROFILE_VIBRATION_FIRST : sprintf(buffer,"VibrateFirst%c%c",13,10); break;
+ case PROFILE_LIGHTS_AUTO : sprintf(buffer,"Auto%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_5SEC : sprintf(buffer,"5Seconds%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_20SEC : sprintf(buffer,"20Seconds%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_1MIN : sprintf(buffer,"1Minute%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_2MIN : sprintf(buffer,"2Minutes%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_5MIN : sprintf(buffer,"5Minutes%c%c",13,10); break;
+ case PROFILE_SAVER_TIMEOUT_10MIN : sprintf(buffer,"10Minutes%c%c",13,10); break;
+ default : sprintf(buffer,"UNKNOWN%c%c",13,10);
+ }
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveFMStationEntry(FILE *file, GSM_FMStation *FMStation, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"Location = %i%c%c",FMStation->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "StationName", FMStation->StationName, UseUnicode);
+ sprintf(buffer,"Frequency = %f%c%c",FMStation->Frequency,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+static void SaveGPRSPointEntry(FILE *file, GSM_GPRSAccessPoint *GPRSPoint, bool UseUnicode)
+{
+ unsigned char buffer[1000];
+
+ sprintf(buffer,"Location = %i%c%c",GPRSPoint->Location,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveBackupText(file, "Name", GPRSPoint->Name, UseUnicode);
+ SaveBackupText(file, "URL", GPRSPoint->URL, UseUnicode);
+ if (GPRSPoint->Active) {
+ sprintf(buffer,"Active = Yes%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+}
+
+GSM_Error SaveBackup(char *FileName, GSM_Backup *backup, bool UseUnicode)
+{
+ int i;
+ unsigned char buffer[1000],checksum[200];
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ if (UseUnicode) {
+ sprintf(buffer,"%c%c", 0xFE, 0xFF);
+ SaveBackupText(file, "", buffer, false);
+ }
+
+ sprintf(buffer,"# Format of this file was designed for Gammu (see www.mwiacek.com)%c%c%c%c",13,10,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"[Backup]%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"IMEI = \"%s\"%c%c",backup->IMEI,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"Phone = \"%s\"%c%c",backup->Model,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ if (backup->Creator[0] != 0) {
+ sprintf(buffer,"Creator = \"%s\"%c%c",backup->Creator,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ }
+ if (backup->DateTimeAvailable) {
+ SaveBackupText(file, "", "DateTime", UseUnicode);
+ SaveVCalDateTime(file, &backup->DateTime, UseUnicode);
+ }
+ sprintf(buffer,"Format = 1.03%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+
+ i=0;
+ while (backup->PhonePhonebook[i]!=NULL) {
+ sprintf(buffer,"[PhonePBK%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SavePbkEntry(file, backup->PhonePhonebook[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->SIMPhonebook[i]!=NULL) {
+ sprintf(buffer,"[SIMPBK%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SavePbkEntry(file, backup->SIMPhonebook[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->Calendar[i]!=NULL) {
+ sprintf(buffer,"[Calendar%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveCalendarEntry(file, backup->Calendar[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->CallerLogos[i]!=NULL) {
+ sprintf(buffer,"[Caller%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveCallerEntry(file, backup->CallerLogos[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->SMSC[i]!=NULL) {
+ sprintf(buffer,"[SMSC%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveSMSCEntry(file, backup->SMSC[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->WAPBookmark[i]!=NULL) {
+ sprintf(buffer,"[WAPBookmark%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveWAPBookmarkEntry(file, backup->WAPBookmark[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->WAPSettings[i]!=NULL) {
+ sprintf(buffer,"[WAPSettings%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveWAPSettingsEntry(file, backup->WAPSettings[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->MMSSettings[i]!=NULL) {
+ sprintf(buffer,"[MMSSettings%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveWAPSettingsEntry(file, backup->MMSSettings[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->Ringtone[i]!=NULL) {
+ sprintf(buffer,"[Ringtone%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveRingtoneEntry(file, backup->Ringtone[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->ToDo[i]!=NULL) {
+ sprintf(buffer,"[TODO%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveToDoEntry(file, backup->ToDo[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->Profiles[i]!=NULL) {
+ sprintf(buffer,"[Profile%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveProfileEntry(file, backup->Profiles[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->FMStation[i]!=NULL) {
+ sprintf(buffer,"[FMStation%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveFMStationEntry(file, backup->FMStation[i], UseUnicode);
+ i++;
+ }
+ i=0;
+ while (backup->GPRSPoint[i]!=NULL) {
+ sprintf(buffer,"[GPRSPoint%03i]%c%c",i+1,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ SaveGPRSPointEntry(file, backup->GPRSPoint[i], UseUnicode);
+ i++;
+ }
+
+ if (backup->StartupLogo!=NULL) {
+ SaveStartupEntry(file, backup->StartupLogo, UseUnicode);
+ }
+ if (backup->OperatorLogo!=NULL) {
+ SaveOperatorEntry(file, backup->OperatorLogo, UseUnicode);
+ }
+
+ fclose(file);
+
+ FindBackupChecksum(FileName, UseUnicode, checksum);
+
+ file = fopen(FileName, "ab");
+ if (file == NULL) return ERR_CANTOPENFILE;
+ sprintf(buffer,"[Checksum]%c%c",13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ sprintf(buffer,"MD5=%s%c%c",checksum,13,10);
+ SaveBackupText(file, "", buffer, UseUnicode);
+ fclose(file);
+
+ return ERR_NONE;
+}
+
+static void ReadPbkEntry(INI_Section *file_info, char *section, GSM_MemoryEntry *Pbk, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+ int num,i;
+ INI_Entry *e;
+
+ Pbk->EntriesNum = 0;
+ e = INI_FindLastSectionEntry(file_info, section, UseUnicode);
+
+ while (e != NULL) {
+ num = -1;
+ if (UseUnicode) {
+ sprintf(buffer,"%s",DecodeUnicodeString(e->EntryName));
+ } else {
+ sprintf(buffer,"%s",e->EntryName);
+ }
+ if (strlen(buffer) == 11) {
+ if (mystrncasecmp("Entry", buffer, 5) &&
+ mystrncasecmp("Type", buffer+7, 4)) {
+ num = atoi(buffer+5);
+ }
+ }
+ e = e->Prev;
+ if (num != -1) {
+ sprintf(buffer,"Entry%02iType",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (mystrncasecmp(readvalue,"NumberGeneral",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_General;
+ } else if (mystrncasecmp(readvalue,"NumberMobile",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Mobile;
+ } else if (mystrncasecmp(readvalue,"NumberWork",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Work;
+ } else if (mystrncasecmp(readvalue,"NumberFax",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Fax;
+ } else if (mystrncasecmp(readvalue,"NumberHome",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Home;
+ } else if (mystrncasecmp(readvalue,"NumberOther",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Other;
+ } else if (mystrncasecmp(readvalue,"NumberPager",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Pager;
+ } else if (mystrncasecmp(readvalue,"Note",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Note;
+ } else if (mystrncasecmp(readvalue,"Postal",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Postal;
+ } else if (mystrncasecmp(readvalue,"Email",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Email;
+ } else if (mystrncasecmp(readvalue,"Email2",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Email2;
+ } else if (mystrncasecmp(readvalue,"URL",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_URL;
+ } else if (mystrncasecmp(readvalue,"FirstName",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_FirstName;
+ } else if (mystrncasecmp(readvalue,"LastName",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_LastName;
+ } else if (mystrncasecmp(readvalue,"Company",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Company;
+ } else if (mystrncasecmp(readvalue,"JobTitle",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_JobTitle;
+ } else if (mystrncasecmp(readvalue,"Address",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_StreetAddress;
+ } else if (mystrncasecmp(readvalue,"City",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_City;
+ } else if (mystrncasecmp(readvalue,"State",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_State;
+ } else if (mystrncasecmp(readvalue,"Zip",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Zip;
+ } else if (mystrncasecmp(readvalue,"Country",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Country;
+ } else if (mystrncasecmp(readvalue,"Custom1",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Custom1;
+ } else if (mystrncasecmp(readvalue,"Custom2",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Custom2;
+ } else if (mystrncasecmp(readvalue,"Custom3",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Custom3;
+ } else if (mystrncasecmp(readvalue,"Custom4",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Custom4;
+ } else if (mystrncasecmp(readvalue,"Name",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Name;
+ } else if (mystrncasecmp(readvalue,"Category",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Category;
+ Pbk->Entries[Pbk->EntriesNum].Number = 0;
+ sprintf(buffer,"Entry%02iNumber",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].Number = atoi(readvalue);
+ }
+ Pbk->EntriesNum ++;
+ continue;
+ } else if (mystrncasecmp(readvalue,"Private",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Private;
+ Pbk->Entries[Pbk->EntriesNum].Number = 0;
+ sprintf(buffer,"Entry%02iNumber",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].Number = atoi(readvalue);
+ }
+ Pbk->EntriesNum ++;
+ continue;
+ } else if (mystrncasecmp(readvalue,"CallerGroup",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Caller_Group;
+ Pbk->Entries[Pbk->EntriesNum].Number = 0;
+ sprintf(buffer,"Entry%02iNumber",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].Number = atoi(readvalue);
+ }
+ Pbk->EntriesNum ++;
+ continue;
+ } else if (mystrncasecmp(readvalue,"RingtoneID",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_RingtoneID;
+ Pbk->Entries[Pbk->EntriesNum].Number = 0;
+ sprintf(buffer,"Entry%02iNumber",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].Number = atoi(readvalue);
+ }
+ Pbk->EntriesNum ++;
+ continue;
+ } else if (mystrncasecmp(readvalue,"PictureID",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_PictureID;
+ Pbk->Entries[Pbk->EntriesNum].Number = 0;
+ sprintf(buffer,"Entry%02iNumber",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].Number = atoi(readvalue);
+ }
+ Pbk->EntriesNum ++;
+ continue;
+ } else if (mystrncasecmp(readvalue,"UserID",0)) {
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_UserID;
+ }
+ sprintf(buffer,"Entry%02iText",num);
+ ReadBackupText(file_info, section, buffer, Pbk->Entries[Pbk->EntriesNum].Text,UseUnicode);
+ dbgprintf("text \"%s\", type %i\n",DecodeUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text),Pbk->Entries[Pbk->EntriesNum].EntryType);
+ Pbk->Entries[Pbk->EntriesNum].VoiceTag = 0;
+ sprintf(buffer,"Entry%02iVoiceTag",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ Pbk->Entries[Pbk->EntriesNum].VoiceTag = atoi(readvalue);
+ }
+ i = 0;
+ while (1) {
+ Pbk->Entries[Pbk->EntriesNum].SMSList[i] = 0;
+ sprintf(buffer,"Entry%02iSMSList%02i",num,i);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) break;
+ Pbk->Entries[Pbk->EntriesNum].SMSList[i] = atoi(readvalue);
+ i++;
+ }
+ Pbk->EntriesNum ++;
+ }
+ }
+}
+
+static void ReadCalendarEntry(INI_Section *file_info, char *section, GSM_CalendarEntry *note, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) note->Location = atoi(readvalue);
+
+ sprintf(buffer,"Type");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ note->Type = GSM_CAL_REMINDER;
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Call",0)) {
+ note->Type = GSM_CAL_CALL;
+ } else if (mystrncasecmp(readvalue,"Meeting",0)) {
+ note->Type = GSM_CAL_MEETING;
+ } else if (mystrncasecmp(readvalue,"Birthday",0)) {
+ note->Type = GSM_CAL_BIRTHDAY;
+ } else if (mystrncasecmp(readvalue,"Memo",0)) {
+ note->Type = GSM_CAL_MEMO;
+ } else if (mystrncasecmp(readvalue,"Travel",0)) {
+ note->Type = GSM_CAL_TRAVEL;
+ } else if (mystrncasecmp(readvalue,"Vacation",0)) {
+ note->Type = GSM_CAL_VACATION;
+ } else if (mystrncasecmp(readvalue,"DailyAlarm",0)) {
+ note->Type = GSM_CAL_DAILY_ALARM;
+ } else if (mystrncasecmp(readvalue,"Alarm",0)) {
+ note->Type = GSM_CAL_ALARM;
+ } else if (mystrncasecmp(readvalue,"Training/Athletism",0)) {
+ note->Type = GSM_CAL_T_ATHL;
+ } else if (mystrncasecmp(readvalue,"Training/BallGames",0)) {
+ note->Type = GSM_CAL_T_BALL;
+ } else if (mystrncasecmp(readvalue,"Training/Cycling",0)) {
+ note->Type = GSM_CAL_T_CYCL;
+ } else if (mystrncasecmp(readvalue,"Training/Budo",0)) {
+ note->Type = GSM_CAL_T_BUDO;
+ } else if (mystrncasecmp(readvalue,"Training/Dance",0)) {
+ note->Type = GSM_CAL_T_DANC;
+ } else if (mystrncasecmp(readvalue,"Training/ExtremeSports",0)) {
+ note->Type = GSM_CAL_T_EXTR;
+ } else if (mystrncasecmp(readvalue,"Training/Football",0)) {
+ note->Type = GSM_CAL_T_FOOT;
+ } else if (mystrncasecmp(readvalue,"Training/Golf",0)) {
+ note->Type = GSM_CAL_T_GOLF;
+ } else if (mystrncasecmp(readvalue,"Training/Gym",0)) {
+ note->Type = GSM_CAL_T_GYM;
+ } else if (mystrncasecmp(readvalue,"Training/HorseRaces",0)) {
+ note->Type = GSM_CAL_T_HORS;
+ } else if (mystrncasecmp(readvalue,"Training/Hockey",0)) {
+ note->Type = GSM_CAL_T_HOCK;
+ } else if (mystrncasecmp(readvalue,"Training/Races",0)) {
+ note->Type = GSM_CAL_T_RACE;
+ } else if (mystrncasecmp(readvalue,"Training/Rugby",0)) {
+ note->Type = GSM_CAL_T_RUGB;
+ } else if (mystrncasecmp(readvalue,"Training/Sailing",0)) {
+ note->Type = GSM_CAL_T_SAIL;
+ } else if (mystrncasecmp(readvalue,"Training/StreetGames",0)) {
+ note->Type = GSM_CAL_T_STRE;
+ } else if (mystrncasecmp(readvalue,"Training/Swimming",0)) {
+ note->Type = GSM_CAL_T_SWIM;
+ } else if (mystrncasecmp(readvalue,"Training/Tennis",0)) {
+ note->Type = GSM_CAL_T_TENN;
+ } else if (mystrncasecmp(readvalue,"Training/Travels",0)) {
+ note->Type = GSM_CAL_T_TRAV;
+ } else if (mystrncasecmp(readvalue,"Training/WinterGames",0)) {
+ note->Type = GSM_CAL_T_WINT;
+ }
+ }
+ note->EntriesNum = 0;
+ sprintf(buffer,"Text");
+ if (ReadBackupText(file_info, section, buffer, note->Entries[note->EntriesNum].Text,UseUnicode)) {
+ note->Entries[note->EntriesNum].EntryType = CAL_TEXT;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"Phone");
+ if (ReadBackupText(file_info, section, buffer, note->Entries[note->EntriesNum].Text,UseUnicode)) {
+ note->Entries[note->EntriesNum].EntryType = CAL_PHONE;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"Private");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_PRIVATE;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"EventLocation");
+ if (ReadBackupText(file_info, section, buffer, note->Entries[note->EntriesNum].Text,UseUnicode)) {
+ note->Entries[note->EntriesNum].EntryType = CAL_LOCATION;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"ContactID");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_CONTACTID;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"Recurrance");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue) * 24;
+ note->Entries[note->EntriesNum].EntryType = CAL_RECURRANCE;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"StartTime");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &note->Entries[note->EntriesNum].Date);
+ note->Entries[note->EntriesNum].EntryType = CAL_START_DATETIME;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"StopTime");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &note->Entries[note->EntriesNum].Date);
+ note->Entries[note->EntriesNum].EntryType = CAL_END_DATETIME;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"Alarm");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ ReadVCALDateTime(readvalue, &note->Entries[note->EntriesNum].Date);
+ note->Entries[note->EntriesNum].EntryType = CAL_ALARM_DATETIME;
+ sprintf(buffer,"AlarmType");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Silent",0)) {
+ note->Entries[note->EntriesNum].EntryType = CAL_SILENT_ALARM_DATETIME;
+ }
+ }
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatStartDate");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &note->Entries[note->EntriesNum].Date);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_STARTDATE;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatStopDate");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &note->Entries[note->EntriesNum].Date);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_STOPDATE;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatDayOfWeek");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_DAYOFWEEK;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatDay");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_DAY;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatWeekOfMonth");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_WEEKOFMONTH;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatMonth");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_MONTH;
+ note->EntriesNum++;
+ }
+ sprintf(buffer,"RepeatFrequency");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ note->Entries[note->EntriesNum].Number = atoi(readvalue);
+ note->Entries[note->EntriesNum].EntryType = CAL_REPEAT_FREQUENCY;
+ note->EntriesNum++;
+ }
+}
+
+static void ReadToDoEntry(INI_Section *file_info, char *section, GSM_ToDoEntry *ToDo, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+
+ ToDo->EntriesNum = 0;
+
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) ToDo->Location = atoi(readvalue);
+
+ ToDo->Priority = GSM_Priority_High;
+ sprintf(buffer,"Priority");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (!strcmp(readvalue,"3") || !strcmp(readvalue,"Low")) {
+ ToDo->Priority = GSM_Priority_Low;
+ }
+ if (!strcmp(readvalue,"2") || !strcmp(readvalue,"Medium")) {
+ ToDo->Priority = GSM_Priority_Medium;
+ }
+ }
+
+ sprintf(buffer,"Text");
+ if (ReadBackupText(file_info, section, buffer, ToDo->Entries[ToDo->EntriesNum].Text,UseUnicode)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_TEXT;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"Phone");
+ if (ReadBackupText(file_info, section, buffer, ToDo->Entries[ToDo->EntriesNum].Text,UseUnicode)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_PHONE;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"Private");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ToDo->Entries[ToDo->EntriesNum].Number = atoi(readvalue);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_PRIVATE;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"Completed");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (strncmp(readvalue, "yes", 3) == 0) {
+ ToDo->Entries[ToDo->EntriesNum].Number = 1;
+ } else {
+ ToDo->Entries[ToDo->EntriesNum].Number = 0;
+ }
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_COMPLETED;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"Category");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ToDo->Entries[ToDo->EntriesNum].Number = atoi(readvalue);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_CATEGORY;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"ContactID");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ToDo->Entries[ToDo->EntriesNum].Number = atoi(readvalue);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_CONTACTID;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"DueTime");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_END_DATETIME;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"Alarm");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_ALARM_DATETIME;
+ ToDo->EntriesNum++;
+ }
+
+ sprintf(buffer,"SilentAlarm");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_SILENT_ALARM_DATETIME;
+ ToDo->EntriesNum++;
+ }
+}
+
+static bool ReadBitmapEntry(INI_Section *file_info, char *section, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ char *readvalue;
+ unsigned char buffer[10000];
+ unsigned char Width, Height;
+ int x, y;
+
+ GSM_GetMaxBitmapWidthHeight(bitmap->Type, &Width, &Height);
+ sprintf(buffer,"Width");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) bitmap->BitmapWidth = Width; else bitmap->BitmapWidth = atoi(readvalue);
+ sprintf(buffer,"Height");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) bitmap->BitmapHeight = Height; else bitmap->BitmapHeight = atoi(readvalue);
+ GSM_ClearBitmap(bitmap);
+ for (y=0;y<bitmap->BitmapHeight;y++) {
+ sprintf(buffer,"Bitmap%02i",y);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ for (x=0;x<bitmap->BitmapWidth;x++) {
+ if (readvalue[x+1]=='#') GSM_SetPointBitmap(bitmap,x,y);
+ }
+ } else return false;
+ }
+ return true;
+}
+
+static void ReadCallerEntry(INI_Section *file_info, char *section, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+
+ bitmap->Type = GSM_CallerGroupLogo;
+ bitmap->DefaultBitmap = !ReadBitmapEntry(file_info, section, bitmap, UseUnicode);
+ if (bitmap->DefaultBitmap) {
+ bitmap->BitmapWidth = 72;
+ bitmap->BitmapHeight = 14;
+ GSM_ClearBitmap(bitmap);
+ }
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, bitmap->Text,UseUnicode);
+ if (bitmap->Text[0] == 0x00 && bitmap->Text[1] == 0x00) {
+ bitmap->DefaultName = true;
+ } else {
+ bitmap->DefaultName = false;
+ }
+ sprintf(buffer,"Ringtone");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) {
+ sprintf(buffer,"FileRingtone");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) {
+ bitmap->DefaultRingtone = true;
+ } else {
+ DecodeHexBin (&bitmap->RingtoneID, readvalue, 2);
+ bitmap->DefaultRingtone = false;
+ bitmap->FileSystemRingtone = true;
+ }
+ } else {
+ DecodeHexBin (&bitmap->RingtoneID, readvalue, 2);
+ bitmap->DefaultRingtone = false;
+ bitmap->FileSystemRingtone = false;
+ }
+ sprintf(buffer,"Enabled");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ bitmap->BitmapEnabled = true;
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"False",0)) bitmap->BitmapEnabled = false;
+ }
+}
+
+static void ReadStartupEntry(INI_Section *file_info, char *section, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+
+ sprintf(buffer,"Text");
+ ReadBackupText(file_info, section, buffer, bitmap->Text,UseUnicode);
+ if (bitmap->Text[0]!=0 || bitmap->Text[1]!=0) {
+ bitmap->Type = GSM_WelcomeNote_Text;
+ } else {
+ bitmap->Type = GSM_StartupLogo;
+ bitmap->Location = 1;
+ ReadBitmapEntry(file_info, section, bitmap, UseUnicode);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,bitmap);
+#endif
+ }
+}
+
+static void ReadWAPBookmarkEntry(INI_Section *file_info, char *section, GSM_WAPBookmark *bookmark, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+
+ sprintf(buffer,"URL");
+ ReadBackupText(file_info, section, buffer, bookmark->Address,UseUnicode);
+ sprintf(buffer,"Title");
+ ReadBackupText(file_info, section, buffer, bookmark->Title,UseUnicode);
+}
+
+static void ReadOperatorEntry(INI_Section *file_info, char *section, GSM_Bitmap *bitmap, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+
+ sprintf(buffer,"Network");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ memcpy(bitmap->NetworkCode, readvalue + 1, 6);
+ bitmap->NetworkCode[6] = 0;
+ bitmap->Type = GSM_OperatorLogo;
+ ReadBitmapEntry(file_info, section, bitmap, UseUnicode);
+}
+
+static void ReadSMSCEntry(INI_Section *file_info, char *section, GSM_SMSC *SMSC, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, SMSC->Name,UseUnicode);
+ sprintf(buffer,"Number");
+ ReadBackupText(file_info, section, buffer, SMSC->Number,UseUnicode);
+ sprintf(buffer,"DefaultNumber");
+ ReadBackupText(file_info, section, buffer, SMSC->DefaultNumber,UseUnicode);
+ sprintf(buffer,"Format");
+ SMSC->Format = SMS_FORMAT_Text;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Fax",0)) {
+ SMSC->Format = SMS_FORMAT_Fax;
+ } else if (mystrncasecmp(readvalue,"Email",0)) {
+ SMSC->Format = SMS_FORMAT_Email;
+ } else if (mystrncasecmp(readvalue,"Pager",0)) {
+ SMSC->Format = SMS_FORMAT_Pager;
+ }
+ }
+ sprintf(buffer,"Validity");
+ SMSC->Validity.Relative = SMS_VALID_Max_Time;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"1hour",0)) {
+ SMSC->Validity.Relative = SMS_VALID_1_Hour;
+ } else if (mystrncasecmp(readvalue,"6hours",0)) {
+ SMSC->Validity.Relative = SMS_VALID_6_Hours;
+ } else if (mystrncasecmp(readvalue,"24hours",0)) {
+ SMSC->Validity.Relative = SMS_VALID_1_Day;
+ } else if (mystrncasecmp(readvalue,"72hours",0)) {
+ SMSC->Validity.Relative = SMS_VALID_3_Days;
+ } else if (mystrncasecmp(readvalue,"1week",0)) {
+ SMSC->Validity.Relative = SMS_VALID_1_Week;
+ }
+ }
+}
+
+static void ReadWAPSettingsEntry(INI_Section *file_info, char *section, GSM_MultiWAPSettings *settings, bool UseUnicode)
+{
+ unsigned char buffer[10000], *readvalue;
+ int num;
+ INI_Entry *e;
+
+ settings->ActiveBearer = WAPSETTINGS_BEARER_DATA;
+ sprintf(buffer,"Bearer");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"SMS",0)) {
+ settings->ActiveBearer = WAPSETTINGS_BEARER_SMS;
+ } else if (mystrncasecmp(readvalue,"GPRS",0)) {
+ settings->ActiveBearer = WAPSETTINGS_BEARER_GPRS;
+ } else if (mystrncasecmp(readvalue,"USSD",0)) {
+ settings->ActiveBearer = WAPSETTINGS_BEARER_USSD;
+ }
+ }
+
+ settings->Active = false;
+ sprintf(buffer,"Active");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Yes",0)) settings->Active = true;
+ }
+
+ settings->ReadOnly = false;
+ sprintf(buffer,"ReadOnly");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Yes",0)) settings->ReadOnly = true;
+ }
+
+ sprintf(buffer,"Proxy");
+ ReadBackupText(file_info, section, buffer, settings->Proxy,UseUnicode);
+ sprintf(buffer,"ProxyPort");
+ settings->ProxyPort = 8080;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) settings->ProxyPort = atoi(readvalue);
+ sprintf(buffer,"Proxy2");
+ ReadBackupText(file_info, section, buffer, settings->Proxy2,UseUnicode);
+ sprintf(buffer,"Proxy2Port");
+ settings->Proxy2Port = 8080;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) settings->Proxy2Port = atoi(readvalue);
+
+ settings->Number = 0;
+ e = INI_FindLastSectionEntry(file_info, section, UseUnicode);
+ while (e != NULL) {
+ num = -1;
+ if (UseUnicode) {
+ sprintf(buffer,"%s",DecodeUnicodeString(e->EntryName));
+ } else {
+ sprintf(buffer,"%s",e->EntryName);
+ }
+ if (strlen(buffer) == 7) {
+ if (mystrncasecmp("Title", buffer,5)) num = atoi(buffer+5);
+ }
+ e = e->Prev;
+ if (num != -1) {
+ sprintf(buffer,"Title%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Title,UseUnicode);
+ sprintf(buffer,"HomePage%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].HomePage,UseUnicode);
+ sprintf(buffer,"Type%02i",num);
+ settings->Settings[settings->Number].IsContinuous = true;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Temporary",0)) settings->Settings[settings->Number].IsContinuous = false;
+ }
+ sprintf(buffer,"Security%02i",num);
+ settings->Settings[settings->Number].IsSecurity = true;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Off",0)) settings->Settings[settings->Number].IsSecurity = false;
+ }
+ sprintf(buffer,"Bearer%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"SMS",0)) {
+ settings->Settings[settings->Number].Bearer = WAPSETTINGS_BEARER_SMS;
+ sprintf(buffer,"Server%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Server,UseUnicode);
+ sprintf(buffer,"Service%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Service,UseUnicode);
+ } else if ((mystrncasecmp(readvalue,"Data",0) || mystrncasecmp(readvalue,"GPRS",0))) {
+ settings->Settings[settings->Number].Bearer = WAPSETTINGS_BEARER_DATA;
+ if (mystrncasecmp(readvalue,"GPRS",0)) settings->Settings[settings->Number].Bearer = WAPSETTINGS_BEARER_GPRS;
+ sprintf(buffer,"Number%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].DialUp,UseUnicode);
+ sprintf(buffer,"IP%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].IPAddress,UseUnicode);
+ sprintf(buffer,"User%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].User,UseUnicode);
+ sprintf(buffer,"Password%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Password,UseUnicode);
+ sprintf(buffer,"Authentication%02i",num);
+ settings->Settings[settings->Number].IsNormalAuthentication = true;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Secure",0)) settings->Settings[settings->Number].IsNormalAuthentication = false;
+ }
+ sprintf(buffer,"CallSpeed%02i",num);
+ settings->Settings[settings->Number].Speed = WAPSETTINGS_SPEED_14400;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"9600",0)) settings->Settings[settings->Number].Speed = WAPSETTINGS_SPEED_9600;
+ if (mystrncasecmp(readvalue,"auto",0)) settings->Settings[settings->Number].Speed = WAPSETTINGS_SPEED_AUTO;
+ }
+ sprintf(buffer,"Login%02i",num);
+ settings->Settings[settings->Number].ManualLogin = false;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Manual",0)) settings->Settings[settings->Number].ManualLogin = true;
+ }
+ sprintf(buffer,"CallType%02i",num);
+ settings->Settings[settings->Number].IsISDNCall = true;
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL)
+ {
+ if (mystrncasecmp(readvalue,"Analogue",0)) settings->Settings[settings->Number].IsISDNCall = false;
+ }
+ } else if (mystrncasecmp(readvalue,"USSD",0)) {
+ settings->Settings[settings->Number].Bearer = WAPSETTINGS_BEARER_USSD;
+ sprintf(buffer,"ServiceCode%02i",num);
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Code,UseUnicode);
+ sprintf(buffer,"IP%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ settings->Settings[settings->Number].IsIP = true;
+ sprintf(buffer,"IP%02i",num);
+ } else {
+ settings->Settings[settings->Number].IsIP = false;
+ sprintf(buffer,"Number%02i",num);
+ }
+ ReadBackupText(file_info, section, buffer, settings->Settings[settings->Number].Service,UseUnicode);
+ }
+ }
+ settings->Number++;
+ }
+ }
+}
+
+static void ReadRingtoneEntry(INI_Section *file_info, char *section, GSM_Ringtone *ringtone, bool UseUnicode)
+{
+ unsigned char buffer[10000], buffer2[10000], *readvalue;
+
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, ringtone->Name,UseUnicode);
+ ringtone->Location = 0;
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) ringtone->Location = atoi(readvalue);
+ sprintf(buffer,"NokiaBinary00");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ringtone->Format = RING_NOKIABINARY;
+ ReadLinkedBackupText(file_info, section, "NokiaBinary", buffer2, UseUnicode);
+ DecodeHexBin (ringtone->NokiaBinary.Frame, buffer2, strlen(buffer2));
+ ringtone->NokiaBinary.Length = strlen(buffer2)/2;
+ }
+ sprintf(buffer,"Pure Midi00");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ ringtone->Format = RING_MIDI;
+ ReadLinkedBackupText(file_info, section, "Pure Midi", buffer2, UseUnicode);
+ DecodeHexBin (ringtone->NokiaBinary.Frame, buffer2, strlen(buffer2));
+ ringtone->NokiaBinary.Length = strlen(buffer2)/2;
+ }
+
+}
+
+static void ReadProfileEntry(INI_Section *file_info, char *section, GSM_Profile *Profile, bool UseUnicode)
+{
+ unsigned char buffer[10000];
+ char *readvalue;
+ bool unknown;
+ int num,j;
+ INI_Entry *e;
+
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, Profile->Name,UseUnicode);
+
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ Profile->Location = atoi(readvalue);
+
+ Profile->DefaultName = false;
+ sprintf(buffer,"DefaultName");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL && mystrncasecmp(buffer,"true",0)) Profile->DefaultName = true;
+
+ Profile->HeadSetProfile = false;
+ sprintf(buffer,"HeadSetProfile");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL && mystrncasecmp(buffer,"true",0)) Profile->HeadSetProfile = true;
+
+ Profile->CarKitProfile = false;
+ sprintf(buffer,"CarKitProfile");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL && mystrncasecmp(buffer,"true",0)) Profile->CarKitProfile = true;
+
+ Profile->FeaturesNumber = 0;
+ e = INI_FindLastSectionEntry(file_info, section, UseUnicode);
+ while (e != NULL) {
+ num = -1;
+ if (UseUnicode) {
+ sprintf(buffer,"%s",DecodeUnicodeString(e->EntryName));
+ } else {
+ sprintf(buffer,"%s",e->EntryName);
+ }
+ if (strlen(buffer) == 9) {
+ if (mystrncasecmp("Feature", buffer, 7)) num = atoi(buffer+7);
+ }
+ e = e->Prev;
+ if (num != -1) {
+ sprintf(buffer,"Feature%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue==NULL) break;
+ unknown = true;
+ if (mystrncasecmp(readvalue,"RingtoneID",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_RingtoneID;
+ sprintf(buffer,"Value%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ Profile->FeatureValue[Profile->FeaturesNumber]=atoi(readvalue);
+ Profile->FeaturesNumber++;
+ } else if (mystrncasecmp(readvalue,"MessageToneID",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_MessageToneID;
+ sprintf(buffer,"Value%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ Profile->FeatureValue[Profile->FeaturesNumber]=atoi(readvalue);
+ Profile->FeaturesNumber++;
+ } else if (mystrncasecmp(readvalue,"ScreenSaverNumber",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_ScreenSaverNumber;
+ sprintf(buffer,"Value%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ Profile->FeatureValue[Profile->FeaturesNumber]=atoi(readvalue);
+ Profile->FeaturesNumber++;
+ } else if (mystrncasecmp(readvalue,"CallerGroups",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_CallerGroups;
+ sprintf(buffer,"Value%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ for (j=0;j<5;j++) {
+ Profile->CallerGroups[j]=false;
+ if (strstr(readvalue,"1"+j)!=NULL) Profile->CallerGroups[j]=true;
+ }
+ Profile->FeaturesNumber++;
+ } else if (mystrncasecmp(readvalue,"IncomingCallAlert",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_CallAlert;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"RingtoneVolume",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_RingtoneVolume;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"Vibrating",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_Vibration;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"MessageTone",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_MessageTone;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"KeypadTones",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_KeypadTone;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"WarningTones",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_WarningTone;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"ScreenSaver",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_ScreenSaver;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"ScreenSaverTimeout",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_ScreenSaverTime;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"AutomaticAnswer",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_AutoAnswer;
+ unknown = false;
+ } else if (mystrncasecmp(readvalue,"Lights",0)) {
+ Profile->FeatureID[Profile->FeaturesNumber]=Profile_Lights;
+ unknown = false;
+ }
+ if (!unknown) {
+ sprintf(buffer,"Value%02i",num);
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (mystrncasecmp(readvalue,"Level1",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VOLUME_LEVEL1;
+ if (Profile->FeatureID[Profile->FeaturesNumber]==Profile_KeypadTone) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_KEYPAD_LEVEL1;
+ }
+ } else if (mystrncasecmp(readvalue,"Level2",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VOLUME_LEVEL2;
+ if (Profile->FeatureID[Profile->FeaturesNumber]==Profile_KeypadTone) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_KEYPAD_LEVEL2;
+ }
+ } else if (mystrncasecmp(readvalue,"Level3",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VOLUME_LEVEL3;
+ if (Profile->FeatureID[Profile->FeaturesNumber]==Profile_KeypadTone) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_KEYPAD_LEVEL3;
+ }
+ } else if (mystrncasecmp(readvalue,"Level4",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VOLUME_LEVEL4;
+ } else if (mystrncasecmp(readvalue,"Level5",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VOLUME_LEVEL5;
+ } else if (mystrncasecmp(readvalue,"Off",0)) {
+ switch (Profile->FeatureID[Profile->FeaturesNumber]) {
+ case Profile_MessageTone:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_NOTONE;
+ break;
+ case Profile_AutoAnswer:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_AUTOANSWER_OFF;
+ break;
+ case Profile_Lights:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_LIGHTS_OFF;
+ break;
+ case Profile_ScreenSaver:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_OFF;
+ break;
+ case Profile_WarningTone:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_WARNING_OFF;
+ break;
+ case Profile_CallAlert:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_OFF;
+ break;
+ case Profile_Vibration:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VIBRATION_OFF;
+ break;
+ default:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_KEYPAD_OFF;
+ break;
+ }
+ } else if (mystrncasecmp(readvalue,"Ringing",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_RINGING;
+ } else if (mystrncasecmp(readvalue,"BeepOnce",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_BEEPONCE;
+ if (Profile->FeatureID[Profile->FeaturesNumber]==Profile_MessageTone) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_BEEPONCE;
+ }
+ } else if (mystrncasecmp(readvalue,"RingOnce",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_RINGONCE;
+ } else if (mystrncasecmp(readvalue,"Ascending",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_ASCENDING;
+ } else if (mystrncasecmp(readvalue,"CallerGroups",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_CALLALERT_CALLERGROUPS;
+ } else if (mystrncasecmp(readvalue,"Standard",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_STANDARD;
+ } else if (mystrncasecmp(readvalue,"Special",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_SPECIAL;
+ } else if (mystrncasecmp(readvalue,"Ascending",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_ASCENDING;
+ } else if (mystrncasecmp(readvalue,"Personal",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_MESSAGE_PERSONAL;
+ } else if (mystrncasecmp(readvalue,"VibrateFirst",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VIBRATION_FIRST;
+ } else if (mystrncasecmp(readvalue,"Auto",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_LIGHTS_AUTO;
+ } else if (mystrncasecmp(readvalue,"5Seconds",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_5SEC;
+ } else if (mystrncasecmp(readvalue,"20Seconds",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_20SEC;
+ } else if (mystrncasecmp(readvalue,"1Minute",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_1MIN;
+ } else if (mystrncasecmp(readvalue,"2Minutes",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_2MIN;
+ } else if (mystrncasecmp(readvalue,"5Minutes",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_5MIN;
+ } else if (mystrncasecmp(readvalue,"10Minutes",0)) {
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_TIMEOUT_10MIN;
+ } else if (mystrncasecmp(readvalue,"On",0)) {
+ switch (Profile->FeatureID[Profile->FeaturesNumber]) {
+ case Profile_AutoAnswer:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_AUTOANSWER_ON;
+ break;
+ case Profile_WarningTone:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_WARNING_ON;
+ break;
+ case Profile_ScreenSaver:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_SAVER_ON;
+ break;
+ default:
+ Profile->FeatureValue[Profile->FeaturesNumber]=PROFILE_VIBRATION_ON;
+ break;
+ }
+ } else unknown = true;
+ }
+ if (!unknown) Profile->FeaturesNumber++;
+ }
+ }
+}
+
+static void ReadFMStationEntry(INI_Section *file_info, char *section, GSM_FMStation *FMStation, bool UseUnicode)
+{
+ unsigned char buffer[10000], *readvalue;
+
+ FMStation->Location = 0;
+ FMStation->Frequency = 0;
+
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) FMStation->Location = atoi(readvalue);
+
+ sprintf(buffer,"StationName");
+ ReadBackupText(file_info, section, buffer, FMStation->StationName,UseUnicode);
+
+ sprintf(buffer,"Frequency");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) StringToDouble(readvalue, &FMStation->Frequency);
+}
+
+static void ReadGPRSPointEntry(INI_Section *file_info, char *section, GSM_GPRSAccessPoint *GPRSPoint, bool UseUnicode)
+{
+ unsigned char buffer[10000], *readvalue;
+
+ GPRSPoint->Name[0] = 0;
+ GPRSPoint->Name[1] = 0;
+ GPRSPoint->URL[0] = 0;
+ GPRSPoint->URL[1] = 0;
+ GPRSPoint->Location = 0;
+
+ GPRSPoint->Active = false;
+ sprintf(buffer,"Active");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Yes",0)) GPRSPoint->Active = true;
+ }
+
+ sprintf(buffer,"Location");
+ readvalue = ReadCFGText(file_info, section, buffer, UseUnicode);
+ if (readvalue!=NULL) GPRSPoint->Location = atoi(readvalue);
+
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, GPRSPoint->Name,UseUnicode);
+
+ sprintf(buffer,"URL");
+ ReadBackupText(file_info, section, buffer, GPRSPoint->URL,UseUnicode);
+}
+
+static void ReadNoteEntry(INI_Section *file_info, char *section, GSM_NoteEntry *Note, bool UseUnicode)
+{
+ unsigned char buffer[100];
+
+ sprintf(buffer,"Text");
+ ReadBackupText(file_info, section, buffer, Note->Text,UseUnicode);
+}
+
+GSM_Error LoadBackup(char *FileName, GSM_Backup *backup, bool UseUnicode)
+{
+ INI_Section *file_info, *h;
+ char buffer[100], *readvalue;
+ int num;
+ GSM_MemoryEntry PBK;
+ bool found;
+
+ file_info = INI_ReadFile(FileName, UseUnicode);
+
+ sprintf(buffer,"Backup");
+ if (UseUnicode) EncodeUnicode(buffer,"Backup",6);
+
+ readvalue = ReadCFGText(file_info, buffer, "Format", UseUnicode);
+ /* Did we read anything? */
+ if (readvalue == NULL) return ERR_FILENOTSUPPORTED;
+ /* Is this format version supported ? */
+ if (strcmp(readvalue,"1.01")!=0 && strcmp(readvalue,"1.02")!=0 &&
+ strcmp(readvalue,"1.03")!=0) return ERR_FILENOTSUPPORTED;
+
+ readvalue = ReadCFGText(file_info, buffer, "IMEI", UseUnicode);
+ if (readvalue!=NULL) strcpy(backup->IMEI,readvalue);
+ readvalue = ReadCFGText(file_info, buffer, "Phone", UseUnicode);
+ if (readvalue!=NULL) strcpy(backup->Model,readvalue);
+ readvalue = ReadCFGText(file_info, buffer, "Creator", UseUnicode);
+ if (readvalue!=NULL) strcpy(backup->Creator,readvalue);
+ readvalue = ReadCFGText(file_info, buffer, "DateTime", UseUnicode);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &backup->DateTime);
+ backup->DateTimeAvailable = true;
+ }
+
+ sprintf(buffer,"Checksum");
+ if (UseUnicode) EncodeUnicode(buffer,"Checksum",8);
+ readvalue = ReadCFGText(file_info, buffer, "MD5", UseUnicode);
+ if (readvalue!=NULL) strcpy(backup->MD5Original,readvalue);
+
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"Profile",7);
+ if (mywstrncasecmp(buffer, h->SectionName, 7)) found = true;
+ } else {
+ if (mystrncasecmp("Profile", h->SectionName, 7)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_PROFILES) {
+ backup->Profiles[num] = malloc(sizeof(GSM_Profile));
+ if (backup->Profiles[num] == NULL) return ERR_MOREMEMORY;
+ backup->Profiles[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_PROFILES\n");
+ return ERR_MOREMEMORY;
+ }
+ ReadProfileEntry(file_info, h->SectionName, backup->Profiles[num], UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"PhonePBK",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ } else {
+ if (mystrncasecmp("PhonePBK", h->SectionName, 8)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_PHONEPHONEBOOK) {
+ backup->PhonePhonebook[num] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->PhonePhonebook[num] == NULL) return ERR_MOREMEMORY;
+ backup->PhonePhonebook[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_PHONEPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->PhonePhonebook[num]->Location = atoi (readvalue);
+ backup->PhonePhonebook[num]->MemoryType = MEM_ME;
+ ReadPbkEntry(file_info, h->SectionName, backup->PhonePhonebook[num],UseUnicode);
+ dbgprintf("number of entries = %i\n",backup->PhonePhonebook[num]->EntriesNum);
+ num++;
+ }
+ }
+ num = 0;
+ while (0) {
+ if (backup->PhonePhonebook[num] == NULL) break;
+ if (backup->PhonePhonebook[num+1] != NULL) {
+ if (backup->PhonePhonebook[num+1]->Location < backup->PhonePhonebook[num]->Location) {
+ memcpy(&PBK,backup->PhonePhonebook[num+1],sizeof(GSM_MemoryEntry));
+ memcpy(backup->PhonePhonebook[num+1],backup->PhonePhonebook[num],sizeof(GSM_MemoryEntry));
+ memcpy(backup->PhonePhonebook[num],&PBK,sizeof(GSM_MemoryEntry));
+ num = 0;
+ continue;
+ }
+ }
+ num++;
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"SIMPBK",6);
+ if (mywstrncasecmp(buffer, h->SectionName, 6)) found = true;
+ } else {
+ if (mystrncasecmp("SIMPBK", h->SectionName, 6)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_SIMPHONEBOOK) {
+ backup->SIMPhonebook[num] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->SIMPhonebook[num] == NULL) return ERR_MOREMEMORY;
+ backup->SIMPhonebook[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_SIMPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->SIMPhonebook[num]->Location = atoi (readvalue);
+ backup->SIMPhonebook[num]->MemoryType = MEM_SM;
+ ReadPbkEntry(file_info, h->SectionName, backup->SIMPhonebook[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ while (0) {
+ if (backup->SIMPhonebook[num] == NULL) break;
+ if (backup->SIMPhonebook[num+1] != NULL) {
+ if (backup->SIMPhonebook[num+1]->Location < backup->SIMPhonebook[num]->Location) {
+ memcpy(&PBK,backup->SIMPhonebook[num+1],sizeof(GSM_MemoryEntry));
+ memcpy(backup->SIMPhonebook[num+1],backup->SIMPhonebook[num],sizeof(GSM_MemoryEntry));
+ memcpy(backup->SIMPhonebook[num],&PBK,sizeof(GSM_MemoryEntry));
+ num = 0;
+ continue;
+ }
+ }
+ num++;
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"Calendar",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ } else {
+ if (mystrncasecmp("Calendar", h->SectionName, 8)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Type", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_MAXCALENDARTODONOTES) {
+ backup->Calendar[num] = malloc(sizeof(GSM_CalendarEntry));
+ if (backup->Calendar[num] == NULL) return ERR_MOREMEMORY;
+ backup->Calendar[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_MAXCALENDARTODONOTES\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->Calendar[num]->Location = num + 1;
+ ReadCalendarEntry(file_info, h->SectionName, backup->Calendar[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"Caller",6);
+ if (mywstrncasecmp(buffer, h->SectionName, 6)) found = true;
+ } else {
+ if (mystrncasecmp("Caller", h->SectionName, 6)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_CALLER) {
+ backup->CallerLogos[num] = malloc(sizeof(GSM_Bitmap));
+ if (backup->CallerLogos[num] == NULL) return ERR_MOREMEMORY;
+ backup->CallerLogos[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_CALLER\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->CallerLogos[num]->Location = atoi (readvalue);
+ ReadCallerEntry(file_info, h->SectionName, backup->CallerLogos[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"SMSC",4);
+ if (mywstrncasecmp(buffer, h->SectionName, 4)) found = true;
+ } else {
+ if (mystrncasecmp("SMSC", h->SectionName, 4)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_SMSC) {
+ backup->SMSC[num] = malloc(sizeof(GSM_SMSC));
+ if (backup->SMSC[num] == NULL) return ERR_MOREMEMORY;
+ backup->SMSC[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_SMSC\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->SMSC[num]->Location = atoi (readvalue);
+ ReadSMSCEntry(file_info, h->SectionName, backup->SMSC[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"WAPBookmark",11);
+ if (mywstrncasecmp(buffer, h->SectionName, 11)) found = true;
+ if (!found) {
+ EncodeUnicode(buffer,"Bookmark",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ }
+ } else {
+ if (mystrncasecmp("WAPBookmark", h->SectionName, 11)) found = true;
+ if (!found) {
+ if (mystrncasecmp("Bookmark", h->SectionName, 8)) found = true;
+ }
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "URL", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_WAPBOOKMARK) {
+ backup->WAPBookmark[num] = malloc(sizeof(GSM_WAPBookmark));
+ if (backup->WAPBookmark[num] == NULL) return ERR_MOREMEMORY;
+ backup->WAPBookmark[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_WAPBOOKMARK\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->WAPBookmark[num]->Location = num + 1;
+ ReadWAPBookmarkEntry(file_info, h->SectionName, backup->WAPBookmark[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"WAPSettings",11);
+ if (mywstrncasecmp(buffer, h->SectionName, 11)) found = true;
+ if (!found) {
+ EncodeUnicode(buffer,"Settings",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ }
+ } else {
+ if (mystrncasecmp("WAPSettings", h->SectionName, 11)) found = true;
+ if (!found) {
+ if (mystrncasecmp("Settings", h->SectionName, 8)) found = true;
+ }
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Title00", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_WAPSETTINGS) {
+ backup->WAPSettings[num] = malloc(sizeof(GSM_MultiWAPSettings));
+ if (backup->WAPSettings[num] == NULL) return ERR_MOREMEMORY;
+ backup->WAPSettings[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_WAPSETTINGS\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->WAPSettings[num]->Location = num + 1;
+ dbgprintf("reading wap settings\n");
+ ReadWAPSettingsEntry(file_info, h->SectionName, backup->WAPSettings[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"MMSSettings",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ } else {
+ if (mystrncasecmp("MMSSettings", h->SectionName, 8)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Title00", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_MMSSETTINGS) {
+ backup->MMSSettings[num] = malloc(sizeof(GSM_MultiWAPSettings));
+ if (backup->MMSSettings[num] == NULL) return ERR_MOREMEMORY;
+ backup->MMSSettings[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_MMSSETTINGS\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->MMSSettings[num]->Location = num + 1;
+ dbgprintf("reading mms settings\n");
+ ReadWAPSettingsEntry(file_info, h->SectionName, backup->MMSSettings[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"Ringtone",8);
+ if (mywstrncasecmp(buffer, h->SectionName, 8)) found = true;
+ } else {
+ if (mystrncasecmp("Ringtone", h->SectionName, 8)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_RINGTONES) {
+ backup->Ringtone[num] = malloc(sizeof(GSM_Ringtone));
+ if (backup->Ringtone[num] == NULL) return ERR_MOREMEMORY;
+ backup->Ringtone[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_RINGTONES\n");
+ return ERR_MOREMEMORY;
+ }
+ ReadRingtoneEntry(file_info, h->SectionName, backup->Ringtone[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"TODO",4);
+ if (mywstrncasecmp(buffer, h->SectionName, 4)) found = true;
+ } else {
+ if (mystrncasecmp("TODO", h->SectionName, 4)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_MAXCALENDARTODONOTES) {
+ backup->ToDo[num] = malloc(sizeof(GSM_ToDoEntry));
+ if (backup->ToDo[num] == NULL) return ERR_MOREMEMORY;
+ backup->ToDo[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_MAXCALENDARTODONOTES\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->ToDo[num]->Location = num + 1;
+ ReadToDoEntry(file_info, h->SectionName, backup->ToDo[num],UseUnicode);
+ num++;
+ }
+ }
+ sprintf(buffer,"Startup");
+ readvalue = ReadCFGText(file_info, buffer, "Text", UseUnicode);
+ if (readvalue==NULL) {
+ readvalue = ReadCFGText(file_info, buffer, "Width", UseUnicode);
+ }
+ if (readvalue!=NULL) {
+ backup->StartupLogo = malloc(sizeof(GSM_Bitmap));
+ if (backup->StartupLogo == NULL) return ERR_MOREMEMORY;
+ ReadStartupEntry(file_info, buffer, backup->StartupLogo,UseUnicode);
+ }
+ sprintf(buffer,"Operator");
+ readvalue = ReadCFGText(file_info, buffer, "Network", UseUnicode);
+ if (readvalue!=NULL) {
+ backup->OperatorLogo = malloc(sizeof(GSM_Bitmap));
+ if (backup->OperatorLogo == NULL) return ERR_MOREMEMORY;
+ ReadOperatorEntry(file_info, buffer, backup->OperatorLogo,UseUnicode);
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"FMStation",9);
+ if (mywstrncasecmp(buffer, h->SectionName, 9)) found = true;
+ } else {
+ if (mystrncasecmp("FMStation", h->SectionName, 9)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_FMSTATIONS) {
+ backup->FMStation[num] = malloc(sizeof(GSM_FMStation));
+ if (backup->FMStation[num] == NULL) return ERR_MOREMEMORY;
+ backup->FMStation[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_FMSTATIONS\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->FMStation[num]->Location = num + 1;
+ ReadFMStationEntry(file_info, h->SectionName, backup->FMStation[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"GPRSPoint",9);
+ if (mywstrncasecmp(buffer, h->SectionName, 9)) found = true;
+ } else {
+ if (mystrncasecmp("GPRSPoint", h->SectionName, 9)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Location", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_GPRSPOINT) {
+ backup->GPRSPoint[num] = malloc(sizeof(GSM_GPRSAccessPoint));
+ if (backup->GPRSPoint[num] == NULL) return ERR_MOREMEMORY;
+ backup->GPRSPoint[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_GPRSPOINT\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->GPRSPoint[num]->Location = num + 1;
+ ReadGPRSPointEntry(file_info, h->SectionName, backup->GPRSPoint[num],UseUnicode);
+ num++;
+ }
+ }
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ found = false;
+ if (UseUnicode) {
+ EncodeUnicode(buffer,"Note",4);
+ if (mywstrncasecmp(buffer, h->SectionName, 4)) found = true;
+ } else {
+ if (mystrncasecmp("Note", h->SectionName, 4)) found = true;
+ }
+ if (found) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Text", UseUnicode);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_NOTE) {
+ backup->Note[num] = malloc(sizeof(GSM_NoteEntry));
+ if (backup->Note[num] == NULL) return ERR_MOREMEMORY;
+ backup->Note[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_NOTE\n");
+ return ERR_MOREMEMORY;
+ }
+ ReadNoteEntry(file_info, h->SectionName, backup->Note[num],UseUnicode);
+ num++;
+ }
+ }
+ if (backup->MD5Original[0]!=0) {
+ FindBackupChecksum(FileName, UseUnicode, backup->MD5Calculated);
+ }
+
+ return ERR_NONE;
+}
+
+/* ---------------------- backup files for SMS ----------------------------- */
+
+static void ReadSMSBackupEntry(INI_Section *file_info, char *section, GSM_SMSMessage *SMS)
+{
+ unsigned char buffer[10000], *readvalue;
+
+ GSM_SetDefaultSMSData(SMS);
+
+ SMS->PDU = SMS_Submit;
+ SMS->SMSC.Location = 0;
+ sprintf(buffer,"SMSC");
+ ReadBackupText(file_info, section, buffer, SMS->SMSC.Number, false);
+ sprintf(buffer,"ReplySMSC");
+ SMS->ReplyViaSameSMSC = false;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"True",0)) SMS->ReplyViaSameSMSC = true;
+ }
+ sprintf(buffer,"Class");
+ SMS->Class = -1;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) SMS->Class = atoi(readvalue);
+ sprintf(buffer,"Sent");
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ ReadVCALDateTime(readvalue, &SMS->DateTime);
+ SMS->PDU = SMS_Deliver;
+ }
+ sprintf(buffer,"RejectDuplicates");
+ SMS->RejectDuplicates = false;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"True",0)) SMS->RejectDuplicates = true;
+ }
+ sprintf(buffer,"ReplaceMessage");
+ SMS->ReplaceMessage = 0;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) SMS->ReplaceMessage = atoi(readvalue);
+ sprintf(buffer,"MessageReference");
+ SMS->MessageReference = 0;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) SMS->MessageReference = atoi(readvalue);
+ sprintf(buffer,"State");
+ SMS->State = SMS_UnRead;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Read",0)) SMS->State = SMS_Read;
+ else if (mystrncasecmp(readvalue,"Sent",0)) SMS->State = SMS_Sent;
+ else if (mystrncasecmp(readvalue,"UnSent",0)) SMS->State = SMS_UnSent;
+ }
+ sprintf(buffer,"Number");
+ ReadBackupText(file_info, section, buffer, SMS->Number, false);
+ sprintf(buffer,"Name");
+ ReadBackupText(file_info, section, buffer, SMS->Name, false);
+ sprintf(buffer,"Length");
+ SMS->Length = 0;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) SMS->Length = atoi(readvalue);
+ sprintf(buffer,"Coding");
+ SMS->Coding = SMS_Coding_Default;
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ if (mystrncasecmp(readvalue,"Unicode",0)) {
+ SMS->Coding = SMS_Coding_Unicode;
+ } else if (mystrncasecmp(readvalue,"8bit",0)) {
+ SMS->Coding = SMS_Coding_8bit;
+ }
+ }
+ ReadLinkedBackupText(file_info, section, "Text", buffer, false);
+ DecodeHexBin (SMS->Text, buffer, strlen(buffer));
+ SMS->Text[strlen(buffer)/2] = 0;
+ SMS->Text[strlen(buffer)/2+1] = 0;
+ sprintf(buffer,"Folder");
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) SMS->Folder = atoi(readvalue);
+ SMS->UDH.Type = UDH_NoUDH;
+ SMS->UDH.Length = 0;
+ SMS->UDH.ID8bit = -1;
+ SMS->UDH.ID16bit = -1;
+ SMS->UDH.PartNumber = -1;
+ SMS->UDH.AllParts = -1;
+ sprintf(buffer,"UDH");
+ readvalue = ReadCFGText(file_info, section, buffer, false);
+ if (readvalue!=NULL) {
+ DecodeHexBin (SMS->UDH.Text, readvalue, strlen(readvalue));
+ SMS->UDH.Length = strlen(readvalue)/2;
+ GSM_DecodeUDHHeader(&SMS->UDH);
+ }
+}
+
+static GSM_Error GSM_ReadSMSBackupTextFile(char *FileName, GSM_SMS_Backup *backup)
+{
+ INI_Section *file_info, *h;
+ char *readvalue;
+ int num;
+
+ backup->SMS[0] = NULL;
+
+ file_info = INI_ReadFile(FileName, false);
+
+ num = 0;
+ for (h = file_info; h != NULL; h = h->Next) {
+ if (mystrncasecmp("SMSBackup", h->SectionName, 9)) {
+ readvalue = ReadCFGText(file_info, h->SectionName, "Number", false);
+ if (readvalue==NULL) break;
+ if (num < GSM_BACKUP_MAX_SMS) {
+ backup->SMS[num] = malloc(sizeof(GSM_SMSMessage));
+ if (backup->SMS[num] == NULL) return ERR_MOREMEMORY;
+ backup->SMS[num + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_SMS\n");
+ return ERR_MOREMEMORY;
+ }
+ backup->SMS[num]->Location = num + 1;
+ ReadSMSBackupEntry(file_info, h->SectionName, backup->SMS[num]);
+ num++;
+ }
+ }
+ return ERR_NONE;
+}
+
+GSM_Error GSM_ReadSMSBackupFile(char *FileName, GSM_SMS_Backup *backup)
+{
+ FILE *file;
+
+ backup->SMS[0] = NULL;
+
+ file = fopen(FileName, "rb");
+ if (file == NULL) return(ERR_CANTOPENFILE);
+
+ fclose(file);
+
+ return GSM_ReadSMSBackupTextFile(FileName, backup);
+}
+
+GSM_Error SaveSMSBackupTextFile(FILE *file, GSM_SMS_Backup *backup)
+{
+ int i,w,current;
+ unsigned char buffer[10000];
+ GSM_DateTime DT;
+
+ fprintf(file,"\n# File created by Gammu (www.mwiacek.com) version %s\n",VERSION);
+ GSM_GetCurrentDateTime (&DT);
+ fprintf(file,"# Saved %s\n\n",OSDateTime(DT,false));
+
+ i=0;
+ while (backup->SMS[i]!=NULL) {
+ fprintf(file,"[SMSBackup%03i]\n",i);
+ switch (backup->SMS[i]->Coding) {
+ case SMS_Coding_Unicode:
+ case SMS_Coding_Default:
+ sprintf(buffer,"%s",DecodeUnicodeString(backup->SMS[i]->Text));
+ fprintf(file,"#");
+ current = 0;
+ for (w=0;w<(int)(strlen(buffer));w++) {
+ switch (buffer[w]) {
+ case 10:
+ fprintf(file,"\n#");
+ current = 0;
+ break;
+ case 13:
+ break;
+ default:
+ if (isprint(buffer[w])) {
+ fprintf(file,"%c",buffer[w]);
+ current ++;
+ }
+ if (current == 75) {
+ fprintf(file,"\n#");
+ current = 0;
+ }
+ }
+ }
+ fprintf(file,"\n");
+ break;
+ default:
+ break;
+ }
+ if (backup->SMS[i]->PDU == SMS_Deliver) {
+ SaveBackupText(file, "SMSC", backup->SMS[i]->SMSC.Number, false);
+ if (backup->SMS[i]->ReplyViaSameSMSC) fprintf(file,"SMSCReply = true\n");
+ fprintf(file,"Sent");
+ SaveVCalDateTime(file,&backup->SMS[i]->DateTime, false);
+ }
+ fprintf(file,"State = ");
+ switch (backup->SMS[i]->State) {
+ case SMS_UnRead : fprintf(file,"UnRead\n"); break;
+ case SMS_Read : fprintf(file,"Read\n"); break;
+ case SMS_Sent : fprintf(file,"Sent\n"); break;
+ case SMS_UnSent : fprintf(file,"UnSent\n"); break;
+ }
+ SaveBackupText(file, "Number", backup->SMS[i]->Number, false);
+ SaveBackupText(file, "Name", backup->SMS[i]->Name, false);
+ if (backup->SMS[i]->UDH.Type != UDH_NoUDH) {
+ EncodeHexBin(buffer,backup->SMS[i]->UDH.Text,backup->SMS[i]->UDH.Length);
+ fprintf(file,"UDH = %s\n",buffer);
+ }
+ switch (backup->SMS[i]->Coding) {
+ case SMS_Coding_Unicode:
+ case SMS_Coding_Default:
+ EncodeHexBin(buffer,backup->SMS[i]->Text,backup->SMS[i]->Length*2);
+ break;
+ default:
+ EncodeHexBin(buffer,backup->SMS[i]->Text,backup->SMS[i]->Length);
+ break;
+ }
+ SaveLinkedBackupText(file, "Text", buffer, false);
+ switch (backup->SMS[i]->Coding) {
+ case SMS_Coding_Unicode : fprintf(file,"Coding = Unicode\n"); break;
+ case SMS_Coding_Default : fprintf(file,"Coding = Default\n"); break;
+ case SMS_Coding_8bit : fprintf(file,"Coding = 8bit\n"); break;
+ }
+ fprintf(file,"Folder = %i\n",backup->SMS[i]->Folder);
+ fprintf(file,"Length = %i\n",backup->SMS[i]->Length);
+ fprintf(file,"Class = %i\n",backup->SMS[i]->Class);
+ fprintf(file,"ReplySMSC = ");
+ if (backup->SMS[i]->ReplyViaSameSMSC) fprintf(file,"True\n"); else fprintf(file,"False\n");
+ fprintf(file,"RejectDuplicates = ");
+ if (backup->SMS[i]->RejectDuplicates) fprintf(file,"True\n"); else fprintf(file,"False\n");
+ fprintf(file,"ReplaceMessage = %i\n",backup->SMS[i]->ReplaceMessage);
+ fprintf(file,"MessageReference = %i\n",backup->SMS[i]->MessageReference);
+ fprintf(file,"\n");
+ i++;
+ }
+ return ERR_NONE;
+}
+
+GSM_Error GSM_SaveSMSBackupFile(char *FileName, GSM_SMS_Backup *backup)
+{
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return(ERR_CANTOPENFILE);
+
+ SaveSMSBackupTextFile(file,backup);
+
+ fclose(file);
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backtext.h b/gammu/emb/common/service/backup/backtext.h
new file mode 100644
index 0000000..a95bd89
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backtext.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backtext_h
+#define __gsm_backtext_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error LoadBackup(char *FileName, GSM_Backup *backup, bool UseUnicode);
+GSM_Error SaveBackup(char *FileName, GSM_Backup *backup, bool UseUnicode);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backvcf.c b/gammu/emb/common/service/backup/backvcf.c
new file mode 100644
index 0000000..761a81b
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backvcf.c
@@ -0,0 +1,75 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backvcf.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+GSM_Error SaveVCard(char *FileName, GSM_Backup *backup)
+{
+ int i, Length = 0;
+ unsigned char Buffer[1000];
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ i=0;
+ while (backup->PhonePhonebook[i]!=NULL) {
+ sprintf(Buffer, "%c%c",13,10);
+ fwrite(Buffer,1,2,file);
+ Length = 0;
+ GSM_EncodeVCARD(Buffer,&Length,backup->PhonePhonebook[i],true,Nokia_VCard21);
+ fwrite(Buffer,1,Length,file);
+ i++;
+ }
+
+ fclose(file);
+ return ERR_NONE;
+}
+
+GSM_Error LoadVCard(char *FileName, GSM_Backup *backup)
+{
+ GSM_File File;
+ GSM_Error error;
+ GSM_MemoryEntry Pbk;
+ int numPbk = 0, Pos;
+
+ File.Buffer = NULL;
+ error = GSM_ReadFile(FileName, &File);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ while (1) {
+ error = GSM_DecodeVCARD(File.Buffer, &Pos, &Pbk, Nokia_VCard21);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (numPbk < GSM_BACKUP_MAX_PHONEPHONEBOOK) {
+ backup->PhonePhonebook[numPbk] = malloc(sizeof(GSM_MemoryEntry));
+ if (backup->PhonePhonebook[numPbk] == NULL) return ERR_MOREMEMORY;
+ backup->PhonePhonebook[numPbk + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_BACKUP_MAX_PHONEPHONEBOOK\n");
+ return ERR_MOREMEMORY;
+ }
+ memcpy(backup->PhonePhonebook[numPbk],&Pbk,sizeof(GSM_MemoryEntry));
+ backup->PhonePhonebook[numPbk]->Location = numPbk + 1;
+ backup->PhonePhonebook[numPbk]->MemoryType = MEM_ME;
+ numPbk++;
+ }
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backvcf.h b/gammu/emb/common/service/backup/backvcf.h
new file mode 100644
index 0000000..8cbb6cf
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backvcf.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backvcf_h
+#define __gsm_backvcf_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error SaveVCard(char *FileName, GSM_Backup *backup);
+GSM_Error LoadVCard(char *FileName, GSM_Backup *backup);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backvcs.c b/gammu/emb/common/service/backup/backvcs.c
new file mode 100644
index 0000000..332e718
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backvcs.c
@@ -0,0 +1,106 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "backvcs.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+GSM_Error SaveVCalendar(char *FileName, GSM_Backup *backup)
+{
+ int i, Length = 0;
+ unsigned char Buffer[1000];
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ Length=sprintf(Buffer, "BEGIN:VCALENDAR%c%c",13,10);
+ Length+=sprintf(Buffer+Length, "VERSION:1.0%c%c",13,10);
+ fwrite(Buffer,1,Length,file);
+
+ i=0;
+ while (backup->Calendar[i]!=NULL) {
+ sprintf(Buffer, "%c%c",13,10);
+ fwrite(Buffer,1,2,file);
+ Length = 0;
+ GSM_EncodeVCALENDAR(Buffer,&Length,backup->Calendar[i],false,Nokia_VCalendar);
+ fwrite(Buffer,1,Length,file);
+ i++;
+ }
+ i=0;
+ while (backup->ToDo[i]!=NULL) {
+ sprintf(Buffer, "%c%c",13,10);
+ fwrite(Buffer,1,2,file);
+ Length = 0;
+ GSM_EncodeVTODO(Buffer,&Length,backup->ToDo[i],false,Nokia_VToDo);
+ fwrite(Buffer,1,Length,file);
+ i++;
+ }
+
+ Length=sprintf(Buffer, "%c%cEND:VCALENDAR%c%c",13,10,13,10);
+ fwrite(Buffer,1,Length,file);
+
+ fclose(file);
+ return ERR_NONE;
+}
+
+GSM_Error LoadVCalendar(char *FileName, GSM_Backup *backup)
+{
+ GSM_File File;
+ GSM_Error error;
+ GSM_CalendarEntry Calendar;
+ GSM_ToDoEntry ToDo;
+ int numCal = 0, numToDo = 0, Pos;
+
+ File.Buffer = NULL;
+ error = GSM_ReadFile(FileName, &File);
+ if (error != ERR_NONE) return error;
+
+ Pos = 0;
+ while (1) {
+ error = GSM_DecodeVCALENDAR_VTODO(File.Buffer, &Pos, &Calendar, &ToDo, Nokia_VCalendar, Nokia_VToDo);
+ if (error == ERR_EMPTY) break;
+ if (error != ERR_NONE) return error;
+ if (Calendar.EntriesNum != 0) {
+ if (numCal < GSM_MAXCALENDARTODONOTES) {
+ backup->Calendar[numCal] = malloc(sizeof(GSM_CalendarEntry));
+ if (backup->Calendar[numCal] == NULL) return ERR_MOREMEMORY;
+ backup->Calendar[numCal + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_MAXCALENDARTODONOTES\n");
+ return ERR_MOREMEMORY;
+ }
+ memcpy(backup->Calendar[numCal],&Calendar,sizeof(GSM_CalendarEntry));
+ backup->Calendar[numCal]->Location = numCal + 1;
+ numCal++;
+ }
+ if (ToDo.EntriesNum != 0) {
+ if (numToDo < GSM_MAXCALENDARTODONOTES) {
+ backup->ToDo[numToDo] = malloc(sizeof(GSM_ToDoEntry));
+ if (backup->ToDo[numToDo] == NULL) return ERR_MOREMEMORY;
+ backup->ToDo[numToDo + 1] = NULL;
+ } else {
+ dbgprintf("Increase GSM_MAXCALENDARTODONOTES\n");
+ return ERR_MOREMEMORY;
+ }
+ memcpy(backup->ToDo[numToDo],&ToDo,sizeof(GSM_ToDoEntry));
+ backup->ToDo[numToDo]->Location = numToDo + 1;
+ numToDo++;
+ }
+ }
+
+ return ERR_NONE;
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/backvcs.h b/gammu/emb/common/service/backup/backvcs.h
new file mode 100644
index 0000000..f3331b5
--- a/dev/null
+++ b/gammu/emb/common/service/backup/backvcs.h
@@ -0,0 +1,17 @@
+/* (c) 2003 by Marcin Wiacek */
+
+#ifndef __gsm_backvcs_h
+#define __gsm_backvcs_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+GSM_Error SaveVCalendar(char *FileName, GSM_Backup *backup);
+GSM_Error LoadVCalendar(char *FileName, GSM_Backup *backup);
+#endif
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/gsmback.c b/gammu/emb/common/service/backup/gsmback.c
new file mode 100644
index 0000000..91ac745
--- a/dev/null
+++ b/gammu/emb/common/service/backup/gsmback.c
@@ -0,0 +1,280 @@
+/* (c) 2002-2004 by Marcin Wiacek & Michal Cihar */
+
+#include <string.h>
+#include <ctype.h>
+
+#include "../../phone/nokia/nfunc.h"
+#include "../../phone/nokia/dct3/n7110.h"
+#include "../../misc/coding/coding.h"
+#include "../../misc/cfg.h"
+#include "../gsmlogo.h"
+#include "../gsmmisc.h"
+#include "../gsmcal.h"
+#include "gsmback.h"
+#include "backtext.h"
+#include "backldif.h"
+#include "backlmb.h"
+#include "backvcs.h"
+#include "backvcf.h"
+#include "backics.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+void GSM_FreeBackup(GSM_Backup *backup)
+{
+ int i;
+
+ i=0;
+ while (backup->PhonePhonebook[i]!=NULL) {
+ free(backup->PhonePhonebook[i]);
+ backup->PhonePhonebook[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->SIMPhonebook[i]!=NULL) {
+ free(backup->SIMPhonebook[i]);
+ backup->SIMPhonebook[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->Calendar[i]!=NULL) {
+ free(backup->Calendar[i]);
+ backup->Calendar[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->CallerLogos[i]!=NULL) {
+ free(backup->CallerLogos[i]);
+ backup->CallerLogos[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->SMSC[i]!=NULL) {
+ free(backup->SMSC[i]);
+ backup->SMSC[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->WAPBookmark[i]!=NULL) {
+ free(backup->WAPBookmark[i]);
+ backup->WAPBookmark[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->WAPSettings[i]!=NULL) {
+ free(backup->WAPSettings[i]);
+ backup->WAPSettings[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->MMSSettings[i]!=NULL) {
+ free(backup->MMSSettings[i]);
+ backup->MMSSettings[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->Ringtone[i]!=NULL) {
+ free(backup->Ringtone[i]);
+ backup->Ringtone[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->ToDo[i]!=NULL) {
+ free(backup->ToDo[i]);
+ backup->ToDo[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->Profiles[i]!=NULL) {
+ free(backup->Profiles[i]);
+ backup->Profiles[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->FMStation[i]!=NULL) {
+ free(backup->FMStation[i]);
+ backup->FMStation[i] = NULL;
+ i++;
+ }
+ if (backup->StartupLogo!=NULL) {
+ free(backup->StartupLogo);
+ backup->StartupLogo = NULL;
+ }
+ if (backup->OperatorLogo!=NULL) {
+ free(backup->OperatorLogo);
+ backup->OperatorLogo = NULL;
+ }
+ i=0;
+ while (backup->GPRSPoint[i]!=NULL) {
+ free(backup->GPRSPoint[i]);
+ backup->GPRSPoint[i] = NULL;
+ i++;
+ }
+ i=0;
+ while (backup->Note[i]!=NULL) {
+ free(backup->Note[i]);
+ backup->Note[i] = NULL;
+ i++;
+ }
+}
+
+GSM_Error GSM_SaveBackupFile(char *FileName, GSM_Backup *backup, bool UseUnicode)
+{
+ if (strstr(FileName,".lmb")) {
+ return SaveLMB(FileName,backup);
+ } else if (strstr(FileName,".vcs")) {
+ return SaveVCalendar(FileName,backup);
+ } else if (strstr(FileName,".vcf")) {
+ return SaveVCard(FileName,backup);
+ } else if (strstr(FileName,".ldif")) {
+ return SaveLDIF(FileName,backup);
+ } else if (strstr(FileName,".ics")) {
+ return SaveICS(FileName,backup);
+ } else {
+ return SaveBackup(FileName,backup, UseUnicode);
+ }
+}
+
+GSM_Error GSM_ReadBackupFile(char *FileName, GSM_Backup *backup)
+{
+ FILE *file;
+ unsigned char buffer[300];
+
+ file = fopen(FileName, "rb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+ fread(buffer, 1, 9, file); /* Read the header of the file. */
+ fclose(file);
+
+ GSM_ClearBackup(backup);
+
+ /* Attempt to identify filetype */
+ if (strstr(FileName,".vcs")) {
+ return LoadVCalendar(FileName,backup);
+ } else if (strstr(FileName,".vcf")) {
+ return LoadVCard(FileName,backup);
+ } else if (strstr(FileName,".ldif")) {
+ return LoadLDIF(FileName,backup);
+ } else if (strstr(FileName,".ics")) {
+ return LoadICS(FileName,backup);
+ } else if (memcmp(buffer, "LMB ",4)==0) {
+ return LoadLMB(FileName,backup);
+ } else if (buffer[0] == 0xFE && buffer[1] == 0xFF) {
+ return LoadBackup(FileName,backup,true);
+ } else if (buffer[0] == 0xFF && buffer[1] == 0xFE) {
+ return LoadBackup(FileName,backup,true);
+ } else {
+ return LoadBackup(FileName,backup,false);
+ }
+}
+
+void GSM_ClearBackup(GSM_Backup *backup)
+{
+ backup->PhonePhonebook [0] = NULL;
+ backup->SIMPhonebook [0] = NULL;
+ backup->Calendar [0] = NULL;
+ backup->CallerLogos [0] = NULL;
+ backup->SMSC [0] = NULL;
+ backup->WAPBookmark [0] = NULL;
+ backup->WAPSettings [0] = NULL;
+ backup->MMSSettings [0] = NULL;
+ backup->Ringtone [0] = NULL;
+ backup->Profiles [0] = NULL;
+ backup->ToDo [0] = NULL;
+ backup->GPRSPoint [0] = NULL;
+ backup->FMStation [0] = NULL;
+ backup->Note [0] = NULL;
+ backup->StartupLogo = NULL;
+ backup->OperatorLogo = NULL;
+
+ backup->Creator [0] = 0;
+ backup->IMEI [0] = 0;
+ backup->Model [0] = 0;
+ backup->DateTimeAvailable = false;
+ backup->MD5Original [0] = 0;
+ backup->MD5Calculated [0] = 0;
+}
+
+void GSM_GetBackupFormatFeatures(char *FileName, GSM_Backup_Info *info)
+{
+ info->UseUnicode = false;
+ info->IMEI = false;
+ info->Model = false;
+ info->DateTime = false;
+ info->PhonePhonebook = false;
+ info->SIMPhonebook = false;
+ info->ToDo = false;
+ info->Calendar = false;
+ info->CallerLogos = false;
+ info->SMSC = false;
+ info->WAPBookmark = false;
+ info->WAPSettings = false;
+ info->MMSSettings = false;
+ info->Ringtone = false;
+ info->StartupLogo = false;
+ info->OperatorLogo = false;
+ info->Profiles = false;
+ info->FMStation = false;
+ info->GPRSPoint = false;
+ info->Note = false;
+
+ if (strstr(FileName,".lmb")) {
+ info->PhonePhonebook = true;
+ info->SIMPhonebook = true;
+ info->CallerLogos = true;
+ info->StartupLogo = true;
+ } else if (strstr(FileName,".vcs")) {
+ info->ToDo = true;
+ info->Calendar = true;
+ } else if (strstr(FileName,".vcf")) {
+ info->PhonePhonebook = true;
+ } else if (strstr(FileName,".ics")) {
+ info->ToDo = true;
+ info->Calendar = true;
+ } else if (strstr(FileName,".ldif")) {
+ info->PhonePhonebook = true;
+ } else {
+ info->UseUnicode = true;
+ info->IMEI = true;
+ info->Model = true;
+ info->DateTime = true;
+ info->PhonePhonebook = true;
+ info->SIMPhonebook = true;
+ info->ToDo = true;
+ info->Calendar = true;
+ info->CallerLogos = true;
+ info->SMSC = true;
+ info->WAPBookmark = true;
+ info->WAPSettings = true;
+ info->MMSSettings = true;
+ info->Ringtone = true;
+ info->StartupLogo = true;
+ info->OperatorLogo = true;
+ info->Profiles = true;
+ info->FMStation = true;
+ info->GPRSPoint = true;
+ info->Note = true;
+ }
+}
+
+void GSM_GetBackupFileFeatures(char *FileName, GSM_Backup_Info *info, GSM_Backup *backup)
+{
+ GSM_GetBackupFormatFeatures(FileName, info);
+
+ if (info->PhonePhonebook && backup->PhonePhonebook[0] == NULL) info->PhonePhonebook = false;
+ if (info->SIMPhonebook && backup->SIMPhonebook[0] == NULL) info->SIMPhonebook = false;
+ if (info->Calendar && backup->Calendar[0] == NULL) info->Calendar = false;
+ if (info->ToDo && backup->ToDo[0] == NULL) info->ToDo = false;
+ if (info->WAPBookmark && backup->WAPBookmark[0] == NULL) info->WAPBookmark = false;
+ if (info->WAPSettings && backup->WAPSettings[0] == NULL) info->WAPSettings = false;
+ if (info->MMSSettings && backup->MMSSettings[0] == NULL) info->MMSSettings = false;
+ if (info->FMStation && backup->FMStation[0] == NULL) info->FMStation = false;
+ if (info->GPRSPoint && backup->GPRSPoint[0] == NULL) info->GPRSPoint = false;
+ if (info->Profiles && backup->Profiles[0] == NULL) info->Profiles = false;
+ /* .... */
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/backup/gsmback.h b/gammu/emb/common/service/backup/gsmback.h
new file mode 100644
index 0000000..1fd99b0
--- a/dev/null
+++ b/gammu/emb/common/service/backup/gsmback.h
@@ -0,0 +1,48 @@
+/* (c) 2003-2004 by Marcin Wiacek */
+
+#ifndef __gsm_back_h
+#define __gsm_back_h
+
+#include "backgen.h"
+
+#ifdef GSM_ENABLE_BACKUP
+
+GSM_Error GSM_SaveBackupFile(char *FileName, GSM_Backup *backup, bool UseUnicode);
+GSM_Error GSM_ReadBackupFile(char *FileName, GSM_Backup *backup);
+
+void GSM_ClearBackup (GSM_Backup *backup);
+void GSM_FreeBackup (GSM_Backup *backup);
+
+typedef struct {
+ bool UseUnicode;
+
+ bool IMEI;
+ bool Model;
+ bool DateTime;
+ bool ToDo;
+ bool PhonePhonebook;
+ bool SIMPhonebook;
+ bool Calendar;
+ bool CallerLogos;
+ bool SMSC;
+ bool WAPBookmark;
+ bool Profiles;
+ bool WAPSettings;
+ bool MMSSettings;
+ bool Ringtone;
+ bool StartupLogo;
+ bool OperatorLogo;
+ bool FMStation;
+ bool GPRSPoint;
+ bool Note;
+} GSM_Backup_Info;
+
+void GSM_GetBackupFormatFeatures(char *FileName, GSM_Backup_Info *info);
+void GSM_GetBackupFileFeatures (char *FileName, GSM_Backup_Info *info, GSM_Backup *backup);
+
+#endif
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmcal.c b/gammu/emb/common/service/gsmcal.c
new file mode 100644
index 0000000..950e35e
--- a/dev/null
+++ b/gammu/emb/common/service/gsmcal.c
@@ -0,0 +1,509 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#include <string.h>
+
+#include "gsmcal.h"
+#include "gsmmisc.h"
+#include "../misc/coding/coding.h"
+
+bool IsCalendarNoteFromThePast(GSM_CalendarEntry *note)
+{
+ bool Past = true;
+ int i;
+ GSM_DateTime DT;
+
+ GSM_GetCurrentDateTime (&DT);
+ for (i = 0; i < note->EntriesNum; i++) {
+ switch (note->Entries[i].EntryType) {
+ case CAL_RECURRANCE:
+ Past = false;
+ break;
+ case CAL_START_DATETIME :
+ if (note->Entries[i].Date.Year > DT.Year) Past = false;
+ if (note->Entries[i].Date.Year == DT.Year &&
+ note->Entries[i].Date.Month > DT.Month) Past = false;
+ if (note->Entries[i].Date.Year == DT.Year &&
+ note->Entries[i].Date.Month == DT.Month &&
+ note->Entries[i].Date.Day > DT.Day) Past = false;
+ break;
+ default:
+ break;
+ }
+ if (!Past) break;
+ }
+ switch (note->Type) {
+ case GSM_CAL_BIRTHDAY:
+ Past = false;
+ break;
+ default:
+ break;
+ }
+ return Past;
+}
+
+void GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(GSM_CalendarEntry *entry, int *Text, int *Time, int *Alarm, int *Phone, int *Recurrance, int *EndTime, int *Location)
+{
+ int i;
+
+ *Text = -1;
+ *Time = -1;
+ *Alarm = -1;
+ *Phone = -1;
+ *Recurrance = -1;
+ *EndTime = -1;
+ *Location = -1;
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case CAL_START_DATETIME :
+ if (*Time == -1) *Time = i;
+ break;
+ case CAL_END_DATETIME :
+ if (*EndTime == -1) *EndTime = i;
+ break;
+ case CAL_ALARM_DATETIME :
+ case CAL_SILENT_ALARM_DATETIME:
+ if (*Alarm == -1) *Alarm = i;
+ break;
+ case CAL_RECURRANCE:
+ if (*Recurrance == -1) *Recurrance = i;
+ break;
+ case CAL_TEXT:
+ if (*Text == -1) *Text = i;
+ break;
+ case CAL_PHONE:
+ if (*Phone == -1) *Phone = i;
+ break;
+ case CAL_LOCATION:
+ if (*Location == -1) *Location = i;
+ break;
+ default:
+ break;
+ }
+ }
+}
+
+GSM_Error GSM_EncodeVCALENDAR(char *Buffer, int *Length, GSM_CalendarEntry *note, bool header, GSM_VCalendarVersion Version)
+{
+ int Text, Time, Alarm, Phone, Recurrance, EndTime, Location;
+ char buffer[2000];
+
+ GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(note, &Text, &Time, &Alarm, &Phone, &Recurrance, &EndTime, &Location);
+
+ if (header) {
+ *Length+=sprintf(Buffer, "BEGIN:VCALENDAR%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "VERSION:1.0%c%c",13,10);
+ }
+ *Length+=sprintf(Buffer+(*Length), "BEGIN:VEVENT%c%c",13,10);
+
+ if (Version == Nokia_VCalendar) {
+ *Length+=sprintf(Buffer+(*Length), "CATEGORIES:");
+ switch (note->Type) {
+ case GSM_CAL_REMINDER:
+ *Length+=sprintf(Buffer+(*Length), "REMINDER%c%c",13,10);
+ break;
+ case GSM_CAL_MEMO:
+ *Length+=sprintf(Buffer+(*Length), "MISCELLANEOUS%c%c",13,10);
+ break;
+ case GSM_CAL_CALL:
+ *Length+=sprintf(Buffer+(*Length), "PHONE CALL%c%c",13,10);
+ break;
+ case GSM_CAL_BIRTHDAY:
+ *Length+=sprintf(Buffer+(*Length), "SPECIAL OCCASION%c%c",13,10);
+ break;
+ case GSM_CAL_MEETING:
+ default:
+ *Length+=sprintf(Buffer+(*Length), "MEETING%c%c",13,10);
+ break;
+ }
+ if (note->Type == GSM_CAL_CALL) {
+ buffer[0] = 0;
+ buffer[1] = 0;
+ if (Phone != -1) CopyUnicodeString(buffer,note->Entries[Phone].Text);
+ if (Text != -1) {
+ if (Phone != -1) EncodeUnicode(buffer+UnicodeLength(buffer)*2," ",1);
+ CopyUnicodeString(buffer+UnicodeLength(buffer)*2,note->Entries[Text].Text);
+ }
+ SaveVCALText(Buffer, Length, buffer, "SUMMARY");
+ } else {
+ SaveVCALText(Buffer, Length, note->Entries[Text].Text, "SUMMARY");
+ }
+ if (note->Type == GSM_CAL_MEETING && Location != -1) {
+ SaveVCALText(Buffer, Length, note->Entries[Location].Text, "LOCATION");
+ }
+
+ if (Time == -1) return ERR_UNKNOWN;
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Time].Date, "DTSTART");
+
+ if (EndTime != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[EndTime].Date, "DTEND");
+ }
+
+ if (Alarm != -1) {
+ if (note->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "DALARM");
+ } else {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "AALARM");
+ }
+ }
+
+ /* Birthday is known to be recurranced */
+ if (Recurrance != -1 && note->Type != GSM_CAL_BIRTHDAY) {
+ switch(note->Entries[Recurrance].Number/24) {
+ case 1 : *Length+=sprintf(Buffer+(*Length), "RRULE:D1 #0%c%c",13,10); break;
+ case 7 : *Length+=sprintf(Buffer+(*Length), "RRULE:W1 #0%c%c",13,10); break;
+ case 14 : *Length+=sprintf(Buffer+(*Length), "RRULE:W2 #0%c%c",13,10); break;
+ case 365 : *Length+=sprintf(Buffer+(*Length), "RRULE:YD1 #0%c%c",13,10); break;
+ }
+ }
+ } else if (Version == Siemens_VCalendar) {
+ *Length+=sprintf(Buffer+(*Length), "CATEGORIES:");
+ switch (note->Type) {
+ case GSM_CAL_MEETING:
+ *Length+=sprintf(Buffer+(*Length), "MEETING%c%c",13,10);
+ break;
+ case GSM_CAL_CALL:
+ *Length+=sprintf(Buffer+(*Length), "PHONE CALL%c%c",13,10);
+ break;
+ case GSM_CAL_BIRTHDAY:
+ *Length+=sprintf(Buffer+(*Length), "ANNIVERSARY%c%c",13,10);
+ break;
+ case GSM_CAL_MEMO:
+ default:
+ *Length+=sprintf(Buffer+(*Length), "MISCELLANEOUS%c%c",13,10);
+ break;
+ }
+
+ if (Time == -1) return ERR_UNKNOWN;
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Time].Date, "DTSTART");
+
+ if (Alarm != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "DALARM");
+ }
+
+ if (Recurrance != -1) {
+ switch(note->Entries[Recurrance].Number/24) {
+ case 1 : *Length+=sprintf(Buffer+(*Length), "RRULE:D1%c%c",13,10); break;
+ case 7 : *Length+=sprintf(Buffer+(*Length), "RRULE:D7%c%c",13,10); break;
+ case 30 : *Length+=sprintf(Buffer+(*Length), "RRULE:MD1%c%c",13,10); break;
+ case 365 : *Length+=sprintf(Buffer+(*Length), "RRULE:YD1%c%c",13,10); break;
+ }
+ }
+
+ if (note->Type == GSM_CAL_CALL) {
+ buffer[0] = 0;
+ buffer[1] = 0;
+ if (Phone != -1) CopyUnicodeString(buffer,note->Entries[Phone].Text);
+ if (Text != -1) {
+ if (Phone != -1) EncodeUnicode(buffer+UnicodeLength(buffer)*2," ",1);
+ CopyUnicodeString(buffer+UnicodeLength(buffer)*2,note->Entries[Text].Text);
+ }
+ SaveVCALText(Buffer, Length, buffer, "DESCRIPTION");
+ } else {
+ SaveVCALText(Buffer, Length, note->Entries[Text].Text, "DESCRIPTION");
+ }
+ } else if (Version == SonyEricsson_VCalendar) {
+ *Length+=sprintf(Buffer+(*Length), "CATEGORIES:");
+ switch (note->Type) {
+ case GSM_CAL_MEETING:
+ *Length+=sprintf(Buffer+(*Length), "MEETING%c%c",13,10);
+ break;
+ case GSM_CAL_REMINDER:
+ *Length+=sprintf(Buffer+(*Length), "DATE%c%c",13,10);
+ break;
+ case GSM_CAL_TRAVEL:
+ *Length+=sprintf(Buffer+(*Length), "TRAVEL%c%c",13,10);
+ break;
+ case GSM_CAL_VACATION:
+ *Length+=sprintf(Buffer+(*Length), "VACATION%c%c",13,10);
+ break;
+ case GSM_CAL_BIRTHDAY:
+ *Length+=sprintf(Buffer+(*Length), "ANNIVERSARY%c%c",13,10);
+ break;
+ case GSM_CAL_MEMO:
+ default:
+ *Length+=sprintf(Buffer+(*Length), "MISCELLANEOUS%c%c",13,10);
+ break;
+ }
+
+ if (Time == -1) return ERR_UNKNOWN;
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Time].Date, "DTSTART");
+
+ if (EndTime != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[EndTime].Date, "DTEND");
+ }
+
+ if (Alarm != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "AALARM");
+ }
+
+ SaveVCALText(Buffer, Length, note->Entries[Text].Text, "SUMMARY");
+
+ if (Location != -1) {
+ SaveVCALText(Buffer, Length, note->Entries[Location].Text, "LOCATION");
+ }
+ }
+
+ *Length+=sprintf(Buffer+(*Length), "END:VEVENT%c%c",13,10);
+ if (header) *Length+=sprintf(Buffer+(*Length), "END:VCALENDAR%c%c",13,10);
+
+ return ERR_NONE;
+}
+
+void GSM_ToDoFindDefaultTextTimeAlarmCompleted(GSM_ToDoEntry *entry, int *Text, int *Alarm, int *Completed, int *EndTime, int *Phone)
+{
+ int i;
+
+ *Text = -1;
+ *EndTime = -1;
+ *Alarm = -1;
+ *Completed = -1;
+ *Phone = -1;
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case TODO_END_DATETIME :
+ if (*EndTime == -1) *EndTime = i;
+ break;
+ case TODO_ALARM_DATETIME :
+ case TODO_SILENT_ALARM_DATETIME:
+ if (*Alarm == -1) *Alarm = i;
+ break;
+ case TODO_TEXT:
+ if (*Text == -1) *Text = i;
+ break;
+ case TODO_COMPLETED:
+ if (*Completed == -1) *Completed = i;
+ break;
+ case TODO_PHONE:
+ if (*Phone == -1) *Phone = i;
+ break;
+ default:
+ break;
+ }
+ }
+}
+
+GSM_Error GSM_EncodeVTODO(char *Buffer, int *Length, GSM_ToDoEntry *note, bool header, GSM_VToDoVersion Version)
+{
+ int Text, Alarm, Completed, EndTime, Phone;
+
+ GSM_ToDoFindDefaultTextTimeAlarmCompleted(note, &Text, &Alarm, &Completed, &EndTime, &Phone);
+
+ if (header) {
+ *Length+=sprintf(Buffer, "BEGIN:VCALENDAR%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "VERSION:1.0%c%c",13,10);
+ }
+
+ *Length+=sprintf(Buffer+(*Length), "BEGIN:VTODO%c%c",13,10);
+
+ if (Version == Nokia_VToDo) {
+ if (Text == -1) return ERR_UNKNOWN;
+ SaveVCALText(Buffer, Length, note->Entries[Text].Text, "SUMMARY");
+
+ if (Completed == -1) {
+ *Length+=sprintf(Buffer+(*Length), "STATUS:NEEDS ACTION%c%c",13,10);
+ } else {
+ *Length+=sprintf(Buffer+(*Length), "STATUS:COMPLETED%c%c",13,10);
+ }
+
+ switch (note->Priority) {
+ case GSM_Priority_Low : *Length+=sprintf(Buffer+(*Length), "PRIORITY:1%c%c",13,10); break;
+ case GSM_Priority_Medium: *Length+=sprintf(Buffer+(*Length), "PRIORITY:2%c%c",13,10); break;
+ case GSM_Priority_High : *Length+=sprintf(Buffer+(*Length), "PRIORITY:3%c%c",13,10); break;
+ }
+
+ if (EndTime != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[EndTime].Date, "DUE");
+ }
+
+ if (Alarm != -1) {
+ if (note->Entries[Alarm].EntryType == CAL_SILENT_ALARM_DATETIME) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "DALARM");
+ } else {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "AALARM");
+ }
+ }
+ } else if (Version == SonyEricsson_VToDo) {
+ if (Text == -1) return ERR_UNKNOWN;
+ SaveVCALText(Buffer, Length, note->Entries[Text].Text, "SUMMARY");
+
+ if (Completed == -1) {
+ *Length+=sprintf(Buffer+(*Length), "STATUS:NEEDS ACTION%c%c",13,10);
+ } else {
+ *Length+=sprintf(Buffer+(*Length), "STATUS:COMPLETED%c%c",13,10);
+ }
+
+ switch (note->Priority) {
+ case GSM_Priority_Low : *Length+=sprintf(Buffer+(*Length), "PRIORITY:3%c%c",13,10); break;
+ case GSM_Priority_Medium: *Length+=sprintf(Buffer+(*Length), "PRIORITY:2%c%c",13,10); break;
+ case GSM_Priority_High : *Length+=sprintf(Buffer+(*Length), "PRIORITY:1%c%c",13,10); break;
+ }
+
+ if (Alarm != -1) {
+ SaveVCALDateTime(Buffer, Length, &note->Entries[Alarm].Date, "AALARM");
+ }
+ }
+
+ *Length+=sprintf(Buffer+(*Length), "END:VTODO%c%c",13,10);
+
+ if (header) {
+ *Length+=sprintf(Buffer+(*Length), "END:VCALENDAR%c%c",13,10);
+ }
+ return ERR_NONE;
+}
+
+GSM_Error GSM_DecodeVCALENDAR_VTODO(unsigned char *Buffer, int *Pos, GSM_CalendarEntry *Calendar, GSM_ToDoEntry *ToDo, GSM_VCalendarVersion CalVer, GSM_VToDoVersion ToDoVer)
+{
+ unsigned char Line[2000],Buff[2000];
+ int Level = 0;
+
+ Calendar->EntriesNum = 0;
+ ToDo->EntriesNum = 0;
+
+ while (1) {
+ MyGetLine(Buffer, Pos, Line, strlen(Buffer));
+ if (strlen(Line) == 0) break;
+ switch (Level) {
+ case 0:
+ if (strstr(Line,"BEGIN:VEVENT")) {
+ Calendar->Type = GSM_CAL_MEMO;
+ Level = 1;
+ }
+ if (strstr(Line,"BEGIN:VTODO")) {
+ ToDo->Priority = GSM_Priority_Low;
+ Level = 2;
+ }
+ break;
+ case 1: /* Calendar note */
+ if (strstr(Line,"END:VEVENT")) {
+ if (Calendar->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+ }
+ if (strstr(Line,"CATEGORIES:REMINDER")) Calendar->Type = GSM_CAL_REMINDER;
+ if (strstr(Line,"CATEGORIES:DATE")) Calendar->Type = GSM_CAL_REMINDER;//SE
+ if (strstr(Line,"CATEGORIES:TRAVEL")) Calendar->Type = GSM_CAL_TRAVEL; //SE
+ if (strstr(Line,"CATEGORIES:VACATION")) Calendar->Type = GSM_CAL_VACATION;//SE
+ if (strstr(Line,"CATEGORIES:MISCELLANEOUS")) Calendar->Type = GSM_CAL_MEMO;
+ if (strstr(Line,"CATEGORIES:PHONE CALL")) Calendar->Type = GSM_CAL_CALL;
+ if (strstr(Line,"CATEGORIES:SPECIAL OCCASION")) Calendar->Type = GSM_CAL_BIRTHDAY;
+ if (strstr(Line,"CATEGORIES:ANNIVERSARY")) Calendar->Type = GSM_CAL_BIRTHDAY;
+ if (strstr(Line,"CATEGORIES:MEETING")) Calendar->Type = GSM_CAL_MEETING;
+ if (strstr(Line,"CATEGORIES:APPOINTMENT")) Calendar->Type = GSM_CAL_MEETING;
+ if (strstr(Line,"RRULE:D1")) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_RECURRANCE;
+ Calendar->Entries[Calendar->EntriesNum].Number = 1*24;
+ Calendar->EntriesNum++;
+ }
+ if ((strstr(Line,"RRULE:W1")) || (strstr(Line,"RRULE:D7"))) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_RECURRANCE;
+ Calendar->Entries[Calendar->EntriesNum].Number = 7*24;
+ Calendar->EntriesNum++;
+ }
+ if (strstr(Line,"RRULE:W2")) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_RECURRANCE;
+ Calendar->Entries[Calendar->EntriesNum].Number = 14*24;
+ Calendar->EntriesNum++;
+ }
+ if (strstr(Line,"RRULE:MD1")) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_RECURRANCE;
+ Calendar->Entries[Calendar->EntriesNum].Number = 30*24;
+ Calendar->EntriesNum++;
+ }
+ if (strstr(Line,"RRULE:YD1")) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_RECURRANCE;
+ Calendar->Entries[Calendar->EntriesNum].Number = 365*24;
+ Calendar->EntriesNum++;
+ }
+ if ((ReadVCALText(Line, "SUMMARY", Buff)) || (ReadVCALText(Line, "DESCRIPTION", Buff))) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_TEXT;
+ CopyUnicodeString(Calendar->Entries[Calendar->EntriesNum].Text,Buff);
+ Calendar->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "LOCATION", Buff)) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_LOCATION;
+ CopyUnicodeString(Calendar->Entries[Calendar->EntriesNum].Text,Buff);
+ Calendar->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "DTSTART", Buff)) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_START_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &Calendar->Entries[Calendar->EntriesNum].Date);
+ Calendar->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "DTEND", Buff)) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_END_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &Calendar->Entries[Calendar->EntriesNum].Date);
+ Calendar->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "DALARM", Buff)) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_SILENT_ALARM_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &Calendar->Entries[Calendar->EntriesNum].Date);
+ Calendar->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "AALARM", Buff)) {
+ Calendar->Entries[Calendar->EntriesNum].EntryType = CAL_ALARM_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &Calendar->Entries[Calendar->EntriesNum].Date);
+ Calendar->EntriesNum++;
+ }
+ break;
+ case 2: /* ToDo note */
+ if (strstr(Line,"END:VTODO")) {
+ if (ToDo->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+ }
+ if (ReadVCALText(Line, "DUE", Buff)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_END_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "DALARM", Buff)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_SILENT_ALARM_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "AALARM", Buff)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_ALARM_DATETIME;
+ ReadVCALDateTime(DecodeUnicodeString(Buff), &ToDo->Entries[ToDo->EntriesNum].Date);
+ ToDo->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "SUMMARY", Buff)) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_TEXT;
+ CopyUnicodeString(ToDo->Entries[ToDo->EntriesNum].Text,Buff);
+ ToDo->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "PRIORITY", Buff)) {
+ if (ToDoVer == SonyEricsson_VToDo) {
+ ToDo->Priority = GSM_Priority_Low;
+ if (atoi(DecodeUnicodeString(Buff))==2) ToDo->Priority = GSM_Priority_Medium;
+ if (atoi(DecodeUnicodeString(Buff))==1) ToDo->Priority = GSM_Priority_High;
+ dbgprintf("atoi is %i %s\n",atoi(DecodeUnicodeString(Buff)),DecodeUnicodeString(Buff));
+ } else if (ToDoVer == Nokia_VToDo) {
+ ToDo->Priority = GSM_Priority_Low;
+ if (atoi(DecodeUnicodeString(Buff))==2) ToDo->Priority = GSM_Priority_Medium;
+ if (atoi(DecodeUnicodeString(Buff))==3) ToDo->Priority = GSM_Priority_High;
+ }
+ }
+ if (strstr(Line,"STATUS:COMPLETED")) {
+ ToDo->Entries[ToDo->EntriesNum].EntryType = TODO_COMPLETED;
+ ToDo->Entries[ToDo->EntriesNum].Number = 1;
+ ToDo->EntriesNum++;
+ }
+ break;
+ }
+ }
+
+ if (Calendar->EntriesNum == 0 && ToDo->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+GSM_Error GSM_EncodeVNTFile(unsigned char *Buffer, int *Length, GSM_NoteEntry *Note)
+{
+ *Length+=sprintf(Buffer+(*Length), "BEGIN:VNOTE%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "VERSION:1.1%c%c",13,10);
+ SaveVCALText(Buffer, Length, Note->Text, "BODY");
+ *Length+=sprintf(Buffer+(*Length), "END:VNOTE%c%c",13,10);
+
+ return ERR_NONE;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmcal.h b/gammu/emb/common/service/gsmcal.h
new file mode 100644
index 0000000..067a4a4
--- a/dev/null
+++ b/gammu/emb/common/service/gsmcal.h
@@ -0,0 +1,445 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+/* 5210 calendar IDs by Frederick Ros */
+
+#ifndef __gsm_cal_h
+#define __gsm_cal_h
+
+#include "../gsmcomon.h"
+
+/* ---------------------------- calendar ----------------------------------- */
+
+#define GSM_CALENDAR_ENTRIES 16
+#define MAX_CALENDAR_TEXT_LENGTH 256 /* In 6310 max. 256 chars */
+
+/**
+ * Enum defines types of calendar notes
+ */
+typedef enum {
+ /**
+ * Reminder or Date
+ */
+ GSM_CAL_REMINDER=1,
+ /**
+ * Call
+ */
+ GSM_CAL_CALL,
+ /**
+ * Meeting
+ */
+ GSM_CAL_MEETING,
+ /**
+ * Birthday or Anniversary or Special Occasion
+ */
+ GSM_CAL_BIRTHDAY,
+ /**
+ * Memo or Miscellaneous
+ */
+ GSM_CAL_MEMO,
+ /**
+ * Travel
+ */
+ GSM_CAL_TRAVEL,
+ /**
+ * Vacation
+ */
+ GSM_CAL_VACATION,
+ /**
+ * Training - Athletism
+ */
+ GSM_CAL_T_ATHL,
+ /**
+ * Training - Ball Games
+ */
+ GSM_CAL_T_BALL,
+ /**
+ * Training - Cycling
+ */
+ GSM_CAL_T_CYCL,
+ /**
+ * Training - Budo
+ */
+ GSM_CAL_T_BUDO,
+ /**
+ * Training - Dance
+ */
+ GSM_CAL_T_DANC,
+ /**
+ * Training - Extreme Sports
+ */
+ GSM_CAL_T_EXTR,
+ /**
+ * Training - Football
+ */
+ GSM_CAL_T_FOOT,
+ /**
+ * Training - Golf
+ */
+ GSM_CAL_T_GOLF,
+ /**
+ * Training - Gym
+ */
+ GSM_CAL_T_GYM,
+ /**
+ * Training - Horse Race
+ */
+ GSM_CAL_T_HORS,
+ /**
+ * Training - Hockey
+ */
+ GSM_CAL_T_HOCK,
+ /**
+ * Training - Races
+ */
+ GSM_CAL_T_RACE,
+ /**
+ * Training - Rugby
+ */
+ GSM_CAL_T_RUGB,
+ /**
+ * Training - Sailing
+ */
+ GSM_CAL_T_SAIL,
+ /**
+ * Training - Street Games
+ */
+ GSM_CAL_T_STRE,
+ /**
+ * Training - Swimming
+ */
+ GSM_CAL_T_SWIM,
+ /**
+ * Training - Tennis
+ */
+ GSM_CAL_T_TENN,
+ /**
+ * Training - Travels
+ */
+ GSM_CAL_T_TRAV,
+ /**
+ * Training - Winter Games
+ */
+ GSM_CAL_T_WINT,
+ /**
+ * Alarm
+ */
+ GSM_CAL_ALARM,
+ /**
+ * Alarm repeating each day.
+ */
+ GSM_CAL_DAILY_ALARM
+} GSM_CalendarNoteType;
+
+/**
+ * One value of calendar event.
+ */
+typedef enum {
+ /**
+ * Date and time of event start.
+ */
+ CAL_START_DATETIME = 1,
+ /**
+ * Date and time of event end.
+ */
+ CAL_END_DATETIME,
+ /**
+ * Alarm date and time.
+ */
+ CAL_ALARM_DATETIME,
+ /**
+ * Date and time of silent alarm.
+ */
+ CAL_SILENT_ALARM_DATETIME,
+ /**
+ * Recurrance.
+ */
+ CAL_RECURRANCE,
+ /**
+ * Text.
+ */
+ CAL_TEXT,
+ /**
+ * Location.
+ */
+ CAL_LOCATION,
+ /**
+ * Phone number.
+ */
+ CAL_PHONE,
+ /**
+ * Whether this entry is private.
+ */
+ CAL_PRIVATE,
+ /**
+ * Related contact id.
+ */
+ CAL_CONTACTID,
+ /**
+ * Repeat each x'th day of week.
+ */
+ CAL_REPEAT_DAYOFWEEK,
+ /**
+ * Repeat each x'th day of month.
+ */
+ CAL_REPEAT_DAY,
+ /**
+ * Repeat x'th week of month.
+ */
+ CAL_REPEAT_WEEKOFMONTH,
+ /**
+ * Repeat x'th month.
+ */
+ CAL_REPEAT_MONTH,
+ /**
+ * Repeating frequency.
+ */
+ CAL_REPEAT_FREQUENCY,
+ /**
+ * Repeating start.
+ */
+ CAL_REPEAT_STARTDATE,
+ /**
+ * Repeating end.
+ */
+ CAL_REPEAT_STOPDATE
+} GSM_CalendarType;
+
+/**
+ * One value of calendar event.
+ */
+typedef struct {
+ /**
+ * Type of value.
+ */
+ GSM_CalendarType EntryType;
+ /**
+ * Text of value, if applicable.
+ */
+ unsigned char Text[(MAX_CALENDAR_TEXT_LENGTH + 1)*2];
+ /**
+ * Date and time of value, if applicable.
+ */
+ GSM_DateTime Date;
+ /**
+ * Number of value, if applicable.
+ */
+ unsigned int Number;
+} GSM_SubCalendarEntry;
+
+/**
+ * Calendar note values.
+ */
+typedef struct {
+ /**
+ * Type of calendar note.
+ */
+ GSM_CalendarNoteType Type;
+ /**
+ * Location in memory.
+ */
+ int Location;
+ /**
+ * Number of entries.
+ */
+ int EntriesNum;
+ /**
+ * Values of entries.
+ */
+ GSM_SubCalendarEntry Entries[GSM_CALENDAR_ENTRIES];
+} GSM_CalendarEntry;
+
+void GSM_CalendarFindDefaultTextTimeAlarmPhoneRecurrance(GSM_CalendarEntry *entry, int *Text, int *Time, int *Alarm, int *Phone, int *Recurrance, int *EndTime, int *Location);
+
+typedef enum {
+ Nokia_VCalendar = 1,
+ Siemens_VCalendar,
+ SonyEricsson_VCalendar
+} GSM_VCalendarVersion;
+
+GSM_Error GSM_EncodeVCALENDAR(char *Buffer, int *Length, GSM_CalendarEntry *note, bool header, GSM_VCalendarVersion Version);
+
+bool IsCalendarNoteFromThePast(GSM_CalendarEntry *note);
+
+typedef struct {
+ /**
+ * Monday = 1, Tuesday = 2,...
+ */
+ int StartDay;
+ /**
+ * 0 = no delete, 1 = after day,...
+ */
+ int AutoDelete;
+} GSM_CalendarSettings;
+
+/**
+ * Structure used for returning calendar status.
+ */
+typedef struct {
+ /**
+ * Number of used positions.
+ */
+ int Used;
+} GSM_CalendarStatus;
+
+
+/* ------------------------------ to-do ------------------------------------ */
+
+#define GSM_TODO_ENTRIES 7
+#define MAX_TODO_TEXT_LENGTH 50 /* Alcatel BE5 50 chars */
+
+/**
+ * Types of to do values. In parenthesis is member of @ref GSM_SubToDoEntry,
+ * where value is stored.
+ */
+typedef enum {
+ /**
+ * Due date. (Date)
+ */
+ TODO_END_DATETIME = 1,
+ /**
+ * Whether is completed. (Number)
+ */
+ TODO_COMPLETED,
+ /**
+ * When should alarm be fired (Date).
+ */
+ TODO_ALARM_DATETIME,
+ /**
+ * When should silent alarm be fired (Date).
+ */
+ TODO_SILENT_ALARM_DATETIME,
+ /**
+ * Text of to do (Text).
+ */
+ TODO_TEXT,
+ /**
+ * Whether entry is private (Number).
+ */
+ TODO_PRIVATE,
+ /**
+ * Category of entry (Number).
+ */
+ TODO_CATEGORY,
+ /**
+ * Related contact ID (Number).
+ */
+ TODO_CONTACTID,
+ /**
+ * Number to call (Text).
+ */
+ TODO_PHONE
+} GSM_ToDoType;
+
+/**
+ * Priority of to do.
+ */
+typedef enum {
+ GSM_Priority_High = 1,
+ GSM_Priority_Medium,
+ GSM_Priority_Low
+} GSM_ToDo_Priority;
+
+/**
+ * Value of to do entry.
+ */
+typedef struct {
+ /**
+ * Type of entry.
+ */
+ GSM_ToDoType EntryType;
+ /**
+ * Text of value, if appropriate, see @ref GSM_ToDoType.
+ */
+ unsigned char Text[(MAX_TODO_TEXT_LENGTH + 1)*2];
+ /**
+ * Date of value, if appropriate, see @ref GSM_ToDoType.
+ */
+ GSM_DateTime Date;
+ /**
+ * Number of value, if appropriate, see @ref GSM_ToDoType.
+ */
+ unsigned int Number;
+} GSM_SubToDoEntry;
+
+/**
+ * To do entry.
+ */
+typedef struct {
+ /**
+ * Priority of entry.
+ */
+ GSM_ToDo_Priority Priority;
+ /**
+ * Location in memory.
+ */
+ int Location;
+ /**
+ * Number of entries.
+ */
+ int EntriesNum;
+ /**
+ * Values of current entry.
+ */
+ GSM_SubToDoEntry Entries[GSM_TODO_ENTRIES];
+} GSM_ToDoEntry;
+
+void GSM_ToDoFindDefaultTextTimeAlarmCompleted(GSM_ToDoEntry *entry, int *Text, int *Alarm, int *Completed, int *EndTime, int *Phone);
+
+typedef enum {
+ Nokia_VToDo = 1,
+ SonyEricsson_VToDo
+} GSM_VToDoVersion;
+
+GSM_Error GSM_EncodeVTODO(char *Buffer, int *Length, GSM_ToDoEntry *note, bool header, GSM_VToDoVersion Version);
+
+/**
+ * Status of to do entries.
+ */
+typedef struct {
+ /**
+ * Number of used positions.
+ */
+ int Used;
+} GSM_ToDoStatus;
+
+/* --------------------------- note ---------------------------------------- */
+
+typedef struct {
+ int Location;
+ char Text[100];
+} GSM_NoteEntry;
+
+GSM_Error GSM_EncodeVNTFile(unsigned char *Buffer, int *Length, GSM_NoteEntry *Note);
+
+/* --------------------------- alarm --------------------------------------- */
+
+/**
+ * Alarm values.
+ */
+typedef struct {
+ /**
+ * Location where it is stored.
+ */
+ int Location;
+ /**
+ * Date and time of alarm.
+ */
+ GSM_DateTime DateTime;
+ /**
+ * Whether it repeats each day.
+ */
+ bool Repeating;
+ /**
+ * Text that is shown on display.
+ */
+ char Text[(MAX_CALENDAR_TEXT_LENGTH + 1) * 2];
+} GSM_Alarm;
+
+/* --------------------------- calendar & todo ----------------------------- */
+
+GSM_Error GSM_DecodeVCALENDAR_VTODO(unsigned char *Buffer, int *Pos, GSM_CalendarEntry *Calendar, GSM_ToDoEntry *ToDo, GSM_VCalendarVersion CalVer, GSM_VToDoVersion ToDoVer);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmcall.h b/gammu/emb/common/service/gsmcall.h
new file mode 100644
index 0000000..c5032a5
--- a/dev/null
+++ b/gammu/emb/common/service/gsmcall.h
@@ -0,0 +1,185 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef _gsm_call_h
+#define _gsm_call_h
+
+#include "../misc/misc.h"
+
+/* ------------------ call info -------------------------------------------- */
+
+/**
+ * Enum with status of call.
+ */
+typedef enum {
+ /**
+ * Somebody calls to us
+ */
+ GSM_CALL_IncomingCall=1,
+ /**
+ * We call somewhere
+ */
+ GSM_CALL_OutgoingCall,
+ /**
+ * Call started
+ */
+ GSM_CALL_CallStart,
+ /**
+ * End of call from unknown side
+ */
+ GSM_CALL_CallEnd,
+ /**
+ * End of call from remote side
+ */
+ GSM_CALL_CallRemoteEnd,
+ /**
+ * End of call from our side
+ */
+ GSM_CALL_CallLocalEnd,
+ /**
+ * Call established. Waiting for answer or dropping
+ */
+ GSM_CALL_CallEstablished,
+ /**
+ * Call held
+ */
+ GSM_CALL_CallHeld,
+ /**
+ * Call resumed
+ */
+ GSM_CALL_CallResumed,
+ /**
+ * We switch to call
+ */
+ GSM_CALL_CallSwitched
+} GSM_CallStatus;
+
+/**
+ * Call information.
+ */
+typedef struct {
+ /**
+ * Call status.
+ */
+ GSM_CallStatus Status;
+ /**
+ * Remote phone number.
+ */
+ char PhoneNumber [(GSM_MAX_NUMBER_LENGTH+1)*2];
+ /**
+ * Call ID
+ */
+ int CallID;
+ /**
+ * Whether Call ID is available.
+ */
+ bool CallIDAvailable;
+ /**
+ * Status code.
+ */
+ int StatusCode;
+} GSM_Call;
+
+/* --------------- Data structures for the call divert -------------------- */
+
+/**
+ * Defines when diversion is active.
+ */
+typedef enum {
+ /**
+ * Divert when busy.
+ */
+ GSM_DIVERT_Busy = 0x01,
+ /**
+ * Divert when not answered.
+ */
+ GSM_DIVERT_NoAnswer,
+ /**
+ * Divert when phone off or no coverage.
+ */
+ GSM_DIVERT_OutOfReach,
+ /**
+ * Divert all calls without ringing.
+ */
+ GSM_DIVERT_AllTypes
+} GSM_Divert_DivertTypes;
+
+/**
+ * Which type of calls should be diverted.
+ */
+typedef enum {
+ /**
+ * Voice calls.
+ */
+ GSM_DIVERT_VoiceCalls = 0x01,
+ /**
+ * Fax calls.
+ */
+ GSM_DIVERT_FaxCalls,
+ /**
+ * Data calls.
+ */
+ GSM_DIVERT_DataCalls,
+ /**
+ * All calls.
+ */
+ GSM_DIVERT_AllCalls
+} GSM_Divert_CallTypes;
+
+/**
+ * Call diversion definition.
+ */
+typedef struct {
+ /**
+ * When diversion is active.
+ */
+ GSM_Divert_DivertTypes DivertType;
+ /**
+ * Type of call to divert.
+ */
+ GSM_Divert_CallTypes CallType;
+ /**
+ * Number where to divert.
+ */
+ char Number[(GSM_MAX_NUMBER_LENGTH+1)*2];
+ /**
+ * Timeout for diversion.
+ */
+ unsigned int Timeout;
+} GSM_CallDivert;
+
+/**
+ * Multiple call diversions.
+ */
+typedef struct {
+ GSM_CallDivert Request;
+ struct {
+ int EntriesNum;
+ GSM_CallDivert Entries[10];
+ } Response;
+} GSM_MultiCallDivert;
+
+/* -------------------------------- dial voice ---------------------------- */
+
+/**
+ * How to handle number when initiating voice call.
+ */
+typedef enum {
+ /**
+ * Show number.
+ */
+ GSM_CALL_ShowNumber = 1,
+ /**
+ * Hide number.
+ */
+ GSM_CALL_HideNumber,
+ /**
+ * Keep phone default settings.
+ */
+ GSM_CALL_DefaultNumberPresence
+} GSM_CallShowNumber;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmdata.c b/gammu/emb/common/service/gsmdata.c
new file mode 100644
index 0000000..94e9b7b
--- a/dev/null
+++ b/gammu/emb/common/service/gsmdata.c
@@ -0,0 +1,366 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#include <string.h>
+
+#include "gsmdata.h"
+#include "../misc/coding/coding.h"
+
+/* SNIFFS, specs somewhere in http://www.wapforum.org */
+void GSM_EncodeMMSIndicatorSMSText(unsigned char *Buffer, int *Length, GSM_MMSIndicator Indicator)
+{
+ unsigned char buffer[200];
+ int i;
+
+ strcpy(Buffer+(*Length),"\xE6\x06\"");
+ (*Length)=(*Length)+3;
+ strcpy(Buffer+(*Length),"application/vnd.wap.mms-message");
+ (*Length)=(*Length)+31;
+ Buffer[(*Length)++] = 0x00;
+
+ strcpy(Buffer+(*Length),"\xAF\x84\x8C\x82\x98");
+ (*Length)=(*Length)+5;
+
+ i = strlen(Indicator.Address);
+ while (Indicator.Address[i] != '/' && i!=0) i--;
+ strcpy(Buffer+(*Length),Indicator.Address+i+1);
+ (*Length)=(*Length)+strlen(Indicator.Address+i+1);
+ Buffer[(*Length)++] = 0x00;
+
+ strcpy(Buffer+(*Length),"\x8D\x90\x89");
+ (*Length)=(*Length)+3;
+
+ sprintf(buffer,"%s/TYPE=PLMN",Indicator.Sender);
+ Buffer[(*Length)++] = strlen(buffer);
+ Buffer[(*Length)++] = 0x80;
+ strcpy(Buffer+(*Length),buffer);
+ (*Length)=(*Length)+strlen(buffer);
+ Buffer[(*Length)++] = 0x00;
+
+ Buffer[(*Length)++] = 0x96;
+ strcpy(Buffer+(*Length),Indicator.Title);
+ (*Length)=(*Length)+strlen(Indicator.Title);
+ Buffer[(*Length)++] = 0x00;
+
+ strcpy(Buffer+(*Length),"\x8A\x80\x8E\x02\x47\xBB\x88\x05\x81\x03\x02\xA3");
+ (*Length)=(*Length)+12;
+ Buffer[(*Length)++] = 0x00;
+
+ Buffer[(*Length)++] = 0x83;
+ strcpy(Buffer+(*Length),Indicator.Address);
+ (*Length)=(*Length)+strlen(Indicator.Address);
+ Buffer[(*Length)++] = 0x00;
+}
+
+/* http://forum.nokia.com: OTA MMS Settings 1.0, OTA Settings 7.0 */
+static void AddWAPSMSParameterText(unsigned char *Buffer, int *Length, unsigned char ID, char *Text, int Len)
+{
+ int i;
+
+ Buffer[(*Length)++] = 0x87; //PARM with attributes
+ Buffer[(*Length)++] = ID;
+ Buffer[(*Length)++] = 0x11; //VALUE
+ Buffer[(*Length)++] = 0x03; //Inline string
+ for (i=0;i<Len;i++) {
+ Buffer[(*Length)++] = Text[i]; //Text
+ }
+ Buffer[(*Length)++] = 0x00; //END Inline string
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+}
+
+/* http://forum.nokia.com: OTA MMS Settings 1.0, OTA Settings 7.0 */
+static void AddWAPSMSParameterInt(unsigned char *Buffer, int *Length, unsigned char ID, unsigned char Value)
+{
+ Buffer[(*Length)++] = 0x87; //PARM with attributes
+ Buffer[(*Length)++] = ID;
+ Buffer[(*Length)++] = Value;
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+}
+
+/* http://forum.nokia.com : OTA MMS Settings 1.0, OTA Settings 7.0
+ * http://www.wapforum.org : Wireless Datagram Protocol
+ */
+void NOKIA_EncodeWAPMMSSettingsSMSText(unsigned char *Buffer, int *Length, GSM_WAPSettings *settings, bool MMS)
+{
+ int i;
+ unsigned char buffer[400];
+
+ Buffer[(*Length)++] = 0x01; //Push ID
+ Buffer[(*Length)++] = 0x06; //PDU Type (push)
+ Buffer[(*Length)++] = 0x2C; //Headers length (content type + headers)
+ strcpy(Buffer+(*Length),"\x1F\x2A");
+ (*Length)=(*Length)+2; //Value length
+ strcpy(Buffer+(*Length),"application/x-wap-prov.browser-settings");
+ (*Length)=(*Length)+39; //MIME-Type
+ Buffer[(*Length)++] = 0x00; //end inline string
+ strcpy(Buffer+(*Length),"\x81\xEA");
+ (*Length)=(*Length)+2; //charset UTF-8 short int.
+ strcpy(Buffer+(*Length),"\x01\x01");
+ (*Length)=(*Length)+2; //version WBXML 1.1
+ Buffer[(*Length)++] = 0x6A; //charset UTF-8
+ Buffer[(*Length)++] = 0x00; //string table length
+
+ Buffer[(*Length)++] = 0x45; //CHARACTERISTIC-LIST with content
+ Buffer[(*Length)++] = 0xC6; //CHARACTERISTIC with content and attributes
+ Buffer[(*Length)++] = 0x06; //TYPE=ADDRESS
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+ switch (settings->Bearer) {
+ case WAPSETTINGS_BEARER_GPRS:
+ /* Bearer */
+ AddWAPSMSParameterInt(Buffer, Length, 0x12, 0x49);
+ /* PPP_LOGINTYPE (manual login or not) */
+ if (settings->ManualLogin) {
+ AddWAPSMSParameterInt(Buffer, Length, 0x1D, 0x65);
+ } else {
+ AddWAPSMSParameterInt(Buffer, Length, 0x1D, 0x64);
+ }
+ /* PPP_AUTHTYPE*/
+ if (settings->IsNormalAuthentication) {
+ /* OTA_CSD_AUTHTYPE_PAP */
+ AddWAPSMSParameterInt(Buffer, Length, 0x22, 0x70);
+ } else {
+ /* OTA_CSD_AUTHTYPE_CHAP */
+ AddWAPSMSParameterInt(Buffer, Length, 0x22, 0x71);
+ }
+ /* GPRS_ACCESSPOINTNAME */
+ AddWAPSMSParameterText(Buffer, Length, 0x1C, DecodeUnicodeString(settings->DialUp), UnicodeLength(settings->DialUp));
+ /* PROXY */
+ AddWAPSMSParameterText(Buffer, Length, 0x13, DecodeUnicodeString(settings->IPAddress), UnicodeLength(settings->IPAddress));
+ /* PPP_AUTHNAME (user) */
+ AddWAPSMSParameterText(Buffer, Length, 0x23, DecodeUnicodeString(settings->User), UnicodeLength(settings->User));
+ /* PPP_AUTHSECRET (password) */
+ AddWAPSMSParameterText(Buffer, Length, 0x24, DecodeUnicodeString(settings->Password), UnicodeLength(settings->Password));
+ break;
+ case WAPSETTINGS_BEARER_DATA:
+ /* Bearer */
+ AddWAPSMSParameterInt(Buffer, Length, 0x12, 0x45);
+ /* CSD_DIALSTRING */
+ AddWAPSMSParameterText(Buffer, Length, 0x21, DecodeUnicodeString(settings->DialUp), UnicodeLength(settings->DialUp));
+ /* PROXY */
+ AddWAPSMSParameterText(Buffer, Length, 0x13, DecodeUnicodeString(settings->IPAddress), UnicodeLength(settings->IPAddress));
+ /* PPP_LOGINTYPE (manual login or not) */
+ if (settings->ManualLogin) {
+ AddWAPSMSParameterInt(Buffer, Length, 0x1D, 0x65);
+ } else {
+ AddWAPSMSParameterInt(Buffer, Length, 0x1D, 0x64);
+ }
+ /* PPP_AUTHTYPE*/
+ if (settings->IsNormalAuthentication) {
+ /* OTA_CSD_AUTHTYPE_PAP */
+ AddWAPSMSParameterInt(Buffer, Length, 0x22, 0x70);
+ } else {
+ /* OTA_CSD_AUTHTYPE_CHAP */
+ AddWAPSMSParameterInt(Buffer, Length, 0x22, 0x71);
+ }
+ /* CSD_CALLTYPE (type of call) */
+ if (settings->IsISDNCall) {
+ /* ISDN */
+ AddWAPSMSParameterInt(Buffer, Length, 0x28, 0x73);
+ } else {
+ /* analogue */
+ AddWAPSMSParameterInt(Buffer, Length, 0x28, 0x72);
+ }
+ /* CSD_CALLSPEED (speed of call) */
+ switch (settings->Speed) {
+ case WAPSETTINGS_SPEED_AUTO:
+ AddWAPSMSParameterInt(Buffer, Length, 0x29, 0x6A);
+ break;
+ case WAPSETTINGS_SPEED_9600:
+ AddWAPSMSParameterInt(Buffer, Length, 0x29, 0x6B);
+ break;
+ case WAPSETTINGS_SPEED_14400:
+ AddWAPSMSParameterInt(Buffer, Length, 0x29, 0x6C);
+ }
+ /* PPP_AUTHNAME (user) */
+ AddWAPSMSParameterText(Buffer, Length, 0x23, DecodeUnicodeString(settings->User), UnicodeLength(settings->User));
+ /* PPP_AUTHSECRET (password) */
+ AddWAPSMSParameterText(Buffer, Length, 0x24, DecodeUnicodeString(settings->Password), UnicodeLength(settings->Password));
+ break;
+#ifdef DEVELOP
+ case WAPSETTINGS_BEARER_SMS:
+ /* Bearer */
+ AddWAPSMSParameterInt(Buffer, Length, 0x12, 0x41);
+ /* PROXY */
+ AddWAPSMSParameterText(Buffer, Length, 0x13, DecodeUnicodeString(settings->Server), UnicodeLength(settings->Server));
+ /* SMS_SMSC_ADDRESS */
+ // .....
+ break;
+ case WAPSETTINGS_BEARER_USSD:
+ /* FIXME */
+ /* Bearer */
+ AddWAPSMSParameterInt(Buffer, Length, 0x12, 0x41);
+ /* PROXY */
+ AddWAPSMSParameterText(Buffer, Length, 0x13, DecodeUnicodeString(settings->Service), UnicodeLength(settings->Service));
+ /* USSD_SERVICE_CODE */
+ /* FIXME */
+ AddWAPSMSParameterText(Buffer, Length, 0x13, DecodeUnicodeString(settings->Code), UnicodeLength(settings->Code));
+#else
+ case WAPSETTINGS_BEARER_SMS:
+ case WAPSETTINGS_BEARER_USSD:
+ break;
+#endif
+ }
+ /* PORT */
+ if (settings->IsSecurity) {
+ if (settings->IsContinuous) {
+ /* Port = 9203. Continuous */
+ AddWAPSMSParameterInt(Buffer, Length, 0x14, 0x63);
+ } else {
+ /* Port = 9202. Temporary */
+ AddWAPSMSParameterInt(Buffer, Length, 0x14, 0x62);
+ }
+ } else {
+ if (settings->IsContinuous) {
+ /* Port = 9201. Continuous */
+ AddWAPSMSParameterInt(Buffer, Length, 0x14, 0x61);
+ } else {
+ /* Port = 9200. Temporary */
+ AddWAPSMSParameterInt(Buffer, Length, 0x14, 0x60);
+ }
+ }
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+
+ /* URL */
+ Buffer[(*Length)++] = 0x86; //CHARACTERISTIC-LIST with attributes
+ if (MMS) {
+ Buffer[(*Length)++] = 0x7C; //TYPE = MMSURL
+ } else {
+ Buffer[(*Length)++] = 0x07; //TYPE = URL
+ }
+ Buffer[(*Length)++] = 0x11; //VALUE
+ Buffer[(*Length)++] = 0x03; //Inline string
+ sprintf(buffer,"%s",DecodeUnicodeString(settings->HomePage));
+ for (i=0;i<(int)strlen(buffer);i++) {
+ Buffer[(*Length)++] = buffer[i];//Text
+ }
+ Buffer[(*Length)++] = 0x00; //END Inline string
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+
+ /* ISP_NAME (name) */
+ Buffer[(*Length)++] = 0xC6; //CHARACTERISTIC with content and attributes
+ Buffer[(*Length)++] = 0x08; //TYPE=NAME
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+ /* Settings name */
+ AddWAPSMSParameterText(Buffer, Length, 0x15, DecodeUnicodeString(settings->Title), UnicodeLength(settings->Title));
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+}
+
+/* http://forum.nokia.com: OTA Settings 7.0 */
+void NOKIA_EncodeWAPBookmarkSMSText(unsigned char *Buffer, int *Length, GSM_WAPBookmark *bookmark)
+{
+ unsigned char buffer[100];
+ bool UnicodeCoding = false;
+
+ EncodeUTF8QuotedPrintable(buffer,bookmark->Title);
+ if (UnicodeLength(bookmark->Title)!=strlen(buffer)) UnicodeCoding = true;
+
+ Buffer[(*Length)++] = 0x01; //Push ID
+ Buffer[(*Length)++] = 0x06; //PDU Type (push)
+ Buffer[(*Length)++] = 0x2D; //Headers length (content type + headers)
+ strcpy(Buffer+(*Length),"\x1F\x2B");
+ (*Length)=(*Length)+2; //Value length
+ strcpy(Buffer+(*Length),"application/x-wap-prov.browser-bookmarks");
+ (*Length)=(*Length)+40; //MIME-Type
+ Buffer[(*Length)++] = 0x00; //end inline string
+ strcpy(Buffer+(*Length),"\x81\xEA");
+ (*Length)=(*Length)+2; //charset UTF-8 short int.
+
+ /* Block from sniffs. UNKNOWN */
+ if (!UnicodeCoding) {
+ Buffer[(*Length)++] = 0x00;
+ Buffer[(*Length)++] = 0x01;
+ } else {
+ strcpy(Buffer+(*Length),"\x01\x01\x87\x68");
+ (*Length)=(*Length)+4;
+ }
+ Buffer[(*Length)++] = 0x00;
+
+ Buffer[(*Length)++] = 0x45; //CHARACTERISTIC-LIST with content
+ /* URL */
+ Buffer[(*Length)++] = 0xC6; //CHARACTERISTIC with content and attributes
+ Buffer[(*Length)++] = 0x7F; //TYPE = BOOKMARK
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+ if (!UnicodeCoding) {
+ /* TITLE */
+ AddWAPSMSParameterText(Buffer, Length, 0x15, DecodeUnicodeString(bookmark->Title), UnicodeLength(bookmark->Title));
+ /* URL */
+ AddWAPSMSParameterText(Buffer, Length, 0x17, DecodeUnicodeString(bookmark->Address), UnicodeLength(bookmark->Address));
+ } else {
+ /* TITLE */
+ AddWAPSMSParameterText(Buffer, Length, 0x15, bookmark->Title, UnicodeLength(bookmark->Title)*2+1);
+ /* URL */
+ AddWAPSMSParameterText(Buffer, Length, 0x17, bookmark->Address, UnicodeLength(bookmark->Address)*2+1);
+ }
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+ Buffer[(*Length)++] = 0x01; //END PARMeter
+}
+
+void GSM_EncodeMMSFile(GSM_EncodeMultiPartMMSInfo *Info, unsigned char *Buffer, int *Length)
+{
+ int i;
+
+ strcpy(Buffer+(*Length),"\x8C\x80\x98\x4F");
+ (*Length)=(*Length)+4;
+
+ /* Unique MMS ID ? */
+ strcpy(Buffer+(*Length),"123456789");
+ (*Length)=(*Length)+9;
+ Buffer[(*Length)++] = 0x00;
+
+ strcpy(Buffer+(*Length),"\x8D\x90\x89");
+ (*Length)=(*Length)+3;
+
+ strcpy(Buffer+(*Length),"\x01\x81\x86\x81\x96");
+ (*Length)=(*Length)+5;
+
+ if (UnicodeLength(Info->Subject) != 0) {
+ sprintf(Buffer+(*Length),"%s",DecodeUnicodeString(Info->Subject));
+ (*Length)=(*Length)+UnicodeLength(Info->Subject);
+ Buffer[(*Length)++] = 0x00;
+ }
+
+ for (i=0;i<Info->EntriesNum;i++) {
+ switch(Info->Entries[i].ID) {
+ case MMS_Text:
+ strcpy(Buffer+(*Length),"\x84\xA3\x01\x04\x04\x03\x83\x81\xEA");
+ (*Length)=(*Length)+9;
+
+ sprintf(Buffer+(*Length),"%s",DecodeUnicodeString(Info->Entries[i].Buffer));
+ (*Length)=(*Length)+UnicodeLength(Info->Entries[i].Buffer);
+ break;
+ default:
+ break;
+ }
+ }
+}
+
+void GSM_ClearMultiPartMMSInfo(GSM_EncodeMultiPartMMSInfo *Info)
+{
+ Info->EntriesNum = 0;
+ Info->Subject[0] = 0x00;
+ Info->Subject[1] = 0x00;
+ Info->Source[0] = 0x00;
+ Info->Source[1] = 0x00;
+ Info->Destination[0] = 0x00;
+ Info->Destination[1] = 0x00;
+}
+
+GSM_Error GSM_EncodeURLFile(unsigned char *Buffer, int *Length, GSM_WAPBookmark *bookmark)
+{
+ *Length+=sprintf(Buffer+(*Length), "BEGIN:VBKM%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "VERSION:1.0%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "TITLE:%s%c%c",DecodeUnicodeString(bookmark->Title),13,10);
+ *Length+=sprintf(Buffer+(*Length), "URL:%s%c%c",DecodeUnicodeString(bookmark->Address),13,10);
+ *Length+=sprintf(Buffer+(*Length), "BEGIN:ENV%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "X-IRMC-URL;QUOTED-PRINTABLE:=%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "[InternetShortcut] =%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "URL=%s%c%c",DecodeUnicodeString(bookmark->Address),13,10);
+ *Length+=sprintf(Buffer+(*Length), "END:ENV%c%c",13,10);
+ *Length+=sprintf(Buffer+(*Length), "END:VBKM%c%c",13,10);
+
+ return ERR_NONE;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmdata.h b/gammu/emb/common/service/gsmdata.h
new file mode 100644
index 0000000..f5f8e07
--- a/dev/null
+++ b/gammu/emb/common/service/gsmdata.h
@@ -0,0 +1,152 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+
+#ifndef __gsm_wap_h
+#define __gsm_wap_h
+
+#include "gsmmisc.h"
+#include "../misc/misc.h"
+
+/* --------------------------- WAP or MMS settings ------------------------- */
+
+typedef enum {
+ WAPSETTINGS_SPEED_9600,
+ WAPSETTINGS_SPEED_14400,
+ WAPSETTINGS_SPEED_AUTO
+} WAPSettings_Speed;
+
+typedef enum {
+ WAPSETTINGS_BEARER_SMS = 1,
+ WAPSETTINGS_BEARER_DATA,
+ WAPSETTINGS_BEARER_USSD,
+ WAPSETTINGS_BEARER_GPRS
+} WAPSettings_Bearer;
+
+typedef struct {
+ char Title [(20+1)*2];
+ char HomePage [(100+1)*2];
+ WAPSettings_Bearer Bearer;
+ bool IsSecurity;
+ bool IsContinuous;
+
+ /* for data bearer */
+ bool IsISDNCall;
+ bool IsNormalAuthentication;
+
+ /* for sms bearer */
+ char Server [(21+1)*2];
+
+ /* for sms or ussd bearer */
+ char Service [(20+1)*2];
+ bool IsIP;
+
+ /* for ussd bearer */
+ char Code [(10+1)*2];
+
+ /* for data or gprs */
+ char IPAddress [(20+1)*2];
+ bool ManualLogin;
+ char DialUp [(20+1)*2];
+ char User [(50+1)*2]; /*is length OK ?*/
+ char Password [(50+1)*2]; /*is length OK ?*/
+ WAPSettings_Speed Speed;
+} GSM_WAPSettings;
+
+typedef struct {
+ int Location;
+ unsigned char Number;
+ GSM_WAPSettings Settings[4];
+ bool Active;
+ bool ReadOnly;
+ char Proxy [(100+1)*2];
+ int ProxyPort;
+ char Proxy2 [(100+1)*2];
+ int Proxy2Port;
+
+ WAPSettings_Bearer ActiveBearer;
+} GSM_MultiWAPSettings;
+
+void NOKIA_EncodeWAPMMSSettingsSMSText(unsigned char *Buffer, int *Length, GSM_WAPSettings *settings, bool MMS);
+
+/* -------------------------------- WAP Bookmark --------------------------- */
+
+typedef struct {
+ char Address [(255+1)*2];
+ char Title [(50+1)*2];
+ int Location;
+} GSM_WAPBookmark;
+
+void NOKIA_EncodeWAPBookmarkSMSText (unsigned char *Buffer, int *Length, GSM_WAPBookmark *bookmark);
+GSM_Error GSM_EncodeURLFile (unsigned char *Buffer, int *Length, GSM_WAPBookmark *bookmark);
+
+/* ------------------------------ MMS Indicator ---------------------------- */
+
+typedef struct {
+ unsigned char Address[500];
+ unsigned char Title[200];
+ unsigned char Sender[200];
+} GSM_MMSIndicator;
+
+void GSM_EncodeMMSIndicatorSMSText(unsigned char *Buffer, int *Length, GSM_MMSIndicator Indicator);
+
+/* ------------------------------ MMS file --------------------------------- */
+
+#define MAX_MULTI_MMS 20
+
+typedef enum {
+ MMS_Text = 1,
+ MMS_Bitmap_JPG
+} EncodeMultiPartMMSID;
+
+typedef struct {
+ EncodeMultiPartMMSID ID;
+
+ GSM_File File;
+ unsigned char *Buffer;
+} EncodeMultiPartMMSEntry;
+
+typedef struct {
+ /* Input values */
+ EncodeMultiPartMMSEntry Entries[MAX_MULTI_MMS];
+ int EntriesNum;
+
+ unsigned char Source[200];
+ unsigned char Destination[200];
+ unsigned char Subject[200];
+} GSM_EncodeMultiPartMMSInfo;
+
+void GSM_EncodeMMSFile (GSM_EncodeMultiPartMMSInfo *Info, unsigned char *Buffer, int *Length);
+void GSM_ClearMultiPartMMSInfo (GSM_EncodeMultiPartMMSInfo *Info);
+
+/* ------------------------------------------------------------------------ */
+
+typedef struct {
+ int Location;
+ bool Active;
+ bool SyncPhonebook;
+ bool SyncCalendar;
+ char Name[(20+1)*2];
+ char PhonebookDataBase[(50+1)*2];
+ char CalendarDataBase[(50+1)*2];
+ char User[(30+1)*2];
+ char Password[(20+1)*2];
+ char Server[(128+1)*2];
+ GSM_MultiWAPSettings Connection;
+} GSM_SyncMLSettings;
+
+/* ------------------------------------------------------------------------ */
+
+typedef struct {
+ char Name[(50+1)*2];
+ char HomePage[(200+1)*2];
+ char User[(50+1)*2];
+ char Password[(50+1)*2];
+ int Location;
+ bool Active;
+ GSM_MultiWAPSettings Connection;
+} GSM_ChatSettings;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmlogo.c b/gammu/emb/common/service/gsmlogo.c
new file mode 100644
index 0000000..c992915
--- a/dev/null
+++ b/gammu/emb/common/service/gsmlogo.c
@@ -0,0 +1,1003 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+
+#include <string.h>
+#include <stdlib.h>
+
+#include "../misc/misc.h"
+#include "../misc/coding/coding.h"
+#include "gsmlogo.h"
+#include "gsmnet.h"
+
+void PHONE_GetBitmapWidthHeight(GSM_Phone_Bitmap_Types Type, int *width, int *height)
+{
+ *width = 0;
+ *height = 0;
+ switch (Type) {
+ case GSM_EMSSmallPicture : *width=8; *height=8; break;
+ case GSM_EMSMediumPicture : *width=16; *height=16; break;
+ case GSM_EMSBigPicture : *width=32; *height=32; break;
+ case GSM_NokiaOperatorLogo :
+ case GSM_NokiaCallerLogo : *width=72; *height=14; break;
+ case GSM_NokiaPictureImage : *width=72; *height=28; break;
+ case GSM_Nokia7110OperatorLogo :
+ case GSM_Nokia6510OperatorLogo : *width=78; *height=21; break;
+ case GSM_NokiaStartupLogo : *width=84; *height=48; break;
+ case GSM_Nokia6210StartupLogo : *width=96; *height=60; break;
+ case GSM_Nokia7110StartupLogo : *width=96; *height=65; break;
+ case GSM_EMSVariablePicture : break;
+ case GSM_AlcatelBMMIPicture : break;
+ }
+}
+
+int PHONE_GetBitmapSize(GSM_Phone_Bitmap_Types Type, int Width, int Height)
+{
+ int width, height, x;
+
+ PHONE_GetBitmapWidthHeight(Type, &width, &height);
+ if (width == 0 && height == 0) {
+ width = Width;
+ height = Height;
+ }
+ switch (Type) {
+ case GSM_Nokia6510OperatorLogo:
+ x = width * height;
+ return x/8 + (x%8 > 0);
+ case GSM_Nokia7110OperatorLogo:
+ return (width*height + 7)/8;
+ case GSM_NokiaStartupLogo:
+ case GSM_NokiaOperatorLogo:
+ case GSM_NokiaCallerLogo:
+ case GSM_NokiaPictureImage:
+ case GSM_EMSSmallPicture:
+ case GSM_EMSMediumPicture:
+ case GSM_EMSBigPicture:
+ case GSM_EMSVariablePicture:
+ return height*width/8;
+ case GSM_Nokia7110StartupLogo:
+ case GSM_Nokia6210StartupLogo:
+ return (height+7)/8*width;
+ case GSM_AlcatelBMMIPicture:
+ return width*((height+7)/8);
+ }
+ return 0;
+}
+
+static bool PHONE_IsPointBitmap(GSM_Phone_Bitmap_Types Type, char *buffer, int x, int y, int width, int height)
+{
+ int i=0, pixel;
+
+ switch (Type) {
+ case GSM_NokiaStartupLogo:
+ case GSM_Nokia6210StartupLogo:
+ case GSM_Nokia7110StartupLogo:
+ case GSM_Nokia6510OperatorLogo:
+ i=(buffer[(y/8*width) + x] & 1<<(y%8));
+ break;
+ case GSM_NokiaOperatorLogo:
+ case GSM_Nokia7110OperatorLogo:
+ case GSM_NokiaCallerLogo:
+ case GSM_EMSVariablePicture:
+ case GSM_EMSSmallPicture:
+ case GSM_EMSMediumPicture:
+ case GSM_EMSBigPicture:
+ pixel=width*y + x;
+ i=(buffer[pixel/8] & 1<<(7-(pixel%8)));
+ break;
+ case GSM_NokiaPictureImage:
+ i=(buffer[9*y + x/8] & 1<<(7-(x%8)));
+ break;
+ case GSM_AlcatelBMMIPicture:
+ break;
+ }
+ if (i) return true; else return false;
+}
+
+static void PHONE_SetPointBitmap(GSM_Phone_Bitmap_Types Type, char *buffer, int x, int y, int width, int height)
+{
+ int pixel;
+
+ switch (Type) {
+ case GSM_NokiaStartupLogo:
+ case GSM_Nokia6210StartupLogo:
+ case GSM_Nokia7110StartupLogo:
+ case GSM_Nokia6510OperatorLogo:
+ buffer[(y/8*width)+x] |= 1 << (y%8);
+ break;
+ case GSM_NokiaOperatorLogo:
+ case GSM_Nokia7110OperatorLogo:
+ case GSM_NokiaCallerLogo:
+ case GSM_EMSSmallPicture:
+ case GSM_EMSMediumPicture:
+ case GSM_EMSBigPicture:
+ case GSM_EMSVariablePicture:
+ pixel = width*y + x;
+ buffer[pixel/8] |= 1 << (7-(pixel%8));
+ break;
+ case GSM_NokiaPictureImage:
+ buffer[9*y + x/8] |= 1 << (7-(x%8));
+ break;
+ case GSM_AlcatelBMMIPicture:
+ pixel = height / 8;
+ if ((height % 8) != 0) pixel++;
+ buffer[pixel*x + y/8] |= 1 << (7 - (y%8));
+ break;
+ }
+}
+
+void PHONE_DecodeBitmap(GSM_Phone_Bitmap_Types Type, char *buffer, GSM_Bitmap *Bitmap)
+{
+ int width, height, x,y;
+
+ PHONE_GetBitmapWidthHeight(Type, &width, &height);
+ if (Type != GSM_Nokia6510OperatorLogo && Type != GSM_Nokia7110OperatorLogo && Type != GSM_EMSVariablePicture) {
+ Bitmap->BitmapHeight = height;
+ Bitmap->BitmapWidth = width;
+ }
+ switch (Type) {
+ case GSM_NokiaOperatorLogo :
+ case GSM_Nokia7110OperatorLogo :
+ case GSM_Nokia6510OperatorLogo : Bitmap->Type=GSM_OperatorLogo; break;
+ case GSM_NokiaCallerLogo : Bitmap->Type=GSM_CallerGroupLogo; break;
+ case GSM_AlcatelBMMIPicture :
+ case GSM_NokiaStartupLogo :
+ case GSM_Nokia7110StartupLogo :
+ case GSM_Nokia6210StartupLogo : Bitmap->Type=GSM_StartupLogo; break;
+ case GSM_NokiaPictureImage :
+ case GSM_EMSVariablePicture :
+ case GSM_EMSSmallPicture :
+ case GSM_EMSMediumPicture :
+ case GSM_EMSBigPicture : Bitmap->Type=GSM_PictureImage; break;
+ }
+
+ Bitmap->Location = 0;
+ Bitmap->Text[0] = 0;
+ Bitmap->Text[1] = 0;
+ Bitmap->BitmapEnabled = false;
+ Bitmap->DefaultName = false;
+ Bitmap->DefaultBitmap = false;
+ Bitmap->DefaultRingtone = false;
+ Bitmap->RingtoneID = 0;
+ Bitmap->NetworkCode[0] = 0;
+ Bitmap->Sender[0] = 0;
+ Bitmap->Sender[1] = 0;
+ Bitmap->ID = 0;
+
+ GSM_ClearBitmap(Bitmap);
+ for (x=0;x<Bitmap->BitmapWidth;x++) {
+ for (y=0;y<Bitmap->BitmapHeight;y++) {
+ if (PHONE_IsPointBitmap(Type, buffer, x, y, Bitmap->BitmapWidth, Bitmap->BitmapHeight)) {
+ GSM_SetPointBitmap(Bitmap,x,y);
+ }
+ }
+ }
+}
+
+void PHONE_ClearBitmap(GSM_Phone_Bitmap_Types Type, char *buffer, int width, int height)
+{
+ memset(buffer,0,PHONE_GetBitmapSize(Type,width,height));
+}
+
+void PHONE_EncodeBitmap(GSM_Phone_Bitmap_Types Type, char *buffer, GSM_Bitmap *Bitmap)
+{
+ int width, height, x, y;
+ GSM_Bitmap dest;
+
+ PHONE_GetBitmapWidthHeight(Type, &width, &height);
+ if (width == 0 && height == 0) {
+ width = Bitmap->BitmapWidth;
+ height = Bitmap->BitmapHeight;
+ }
+ GSM_ResizeBitmap(&dest, Bitmap, width, height);
+ PHONE_ClearBitmap(Type, buffer, width, height);
+
+ for (x=0;x<width;x++) {
+ for (y=0;y<height;y++) {
+ if (GSM_IsPointBitmap(&dest,x,y)) PHONE_SetPointBitmap(Type, buffer, x, y, width, height);
+ }
+ }
+}
+
+void GSM_GetMaxBitmapWidthHeight(GSM_Bitmap_Types Type, unsigned char *width, unsigned char *height)
+{
+ switch (Type) {
+ case GSM_CallerGroupLogo: *width=72; *height=14; break;
+ case GSM_OperatorLogo : *width=101;*height=21; break;
+ case GSM_StartupLogo : *width=96; *height=65; break;
+ case GSM_PictureImage : *width=72; *height=28; break;
+ default : break;
+ }
+}
+
+void GSM_SetPointBitmap(GSM_Bitmap *bmp, int x, int y)
+{
+ SetBit(bmp->BitmapPoints,y*bmp->BitmapWidth+x);
+}
+
+void GSM_ClearPointBitmap(GSM_Bitmap *bmp, int x, int y)
+{
+ ClearBit(bmp->BitmapPoints,y*bmp->BitmapWidth+x);
+}
+
+bool GSM_IsPointBitmap(GSM_Bitmap *bmp, int x, int y)
+{
+ if (GetBit(bmp->BitmapPoints,y*bmp->BitmapWidth+x)) return true; else return false;
+}
+
+void GSM_ClearBitmap(GSM_Bitmap *bmp)
+{
+ memset(bmp->BitmapPoints,0,GSM_GetBitmapSize(bmp));
+}
+
+int GSM_GetBitmapSize(GSM_Bitmap *bmp)
+{
+ return bmp->BitmapWidth*bmp->BitmapHeight/8+1;
+}
+
+void GSM_PrintBitmap(FILE *file, GSM_Bitmap *bitmap)
+{
+ int x,y;
+
+ for (y=0;y<bitmap->BitmapHeight;y++) {
+ for (x=0;x<bitmap->BitmapWidth;x++) {
+ if (GSM_IsPointBitmap(bitmap,x,y)) {
+ fprintf(file,"#");
+ } else {
+ fprintf(file," ");
+ }
+ }
+ fprintf(file,"\n");
+ }
+}
+
+void GSM_ReverseBitmap(GSM_Bitmap *Bitmap)
+{
+ int x, y;
+
+ for (x=0;x<Bitmap->BitmapWidth;x++) {
+ for (y=0;y<Bitmap->BitmapHeight;y++) {
+ if (GSM_IsPointBitmap(Bitmap,x,y)) {
+ GSM_ClearPointBitmap(Bitmap, x, y);
+ } else {
+ GSM_SetPointBitmap(Bitmap, x, y);
+ }
+ }
+ }
+}
+
+void GSM_ResizeBitmap(GSM_Bitmap *dest, GSM_Bitmap *src, int width, int height)
+{
+ int startx=0,endx=0,setx=0, starty=0,endy=0,sety=0, x, y;
+
+ if (src->BitmapWidth<=width) {
+ startx = 0;
+ endx = src->BitmapWidth;
+ setx = (width-src->BitmapWidth)/2;
+ } else {
+ startx = (src->BitmapWidth-width)/2;
+ endx = startx + width;
+ setx = 0;
+ }
+ if (src->BitmapHeight<=height) {
+ starty = 0;
+ endy = src->BitmapHeight;
+ sety = (height-src->BitmapHeight)/2;
+ } else {
+ starty = (src->BitmapHeight-height)/2;
+ endy = starty + height;
+ sety = 0;
+ }
+ dest->BitmapHeight = height;
+ dest->BitmapWidth = width;
+ GSM_ClearBitmap(dest);
+ for (x=startx;x<endx;x++) {
+ for (y=starty;y<endy;y++) {
+ if (GSM_IsPointBitmap(src,x,y))
+ GSM_SetPointBitmap(dest,setx+x-startx,sety+y-starty);
+ }
+ }
+}
+
+GSM_Error Bitmap2BMP(unsigned char *buffer, FILE *file,GSM_Bitmap *bitmap)
+{
+ int x,y,pos,i,sizeimage,buffpos=0;
+ unsigned char buff[1];
+ div_t division;
+ bool isfile=false;
+
+ unsigned char header[]={
+/*1'st header*/ 'B','M', /* BMP file ID */
+ 0x00,0x00,0x00,0x00, /* Size of file */
+ 0x00,0x00, /* Reserved for future use */
+ 0x00,0x00, /* Reserved for future use */
+ 62,0x00,0x00,0x00, /* Offset for image data */
+
+/*2'nd header*/ 40,0x00,0x00,0x00, /* Length of this part of header */
+ 0x00,0x00,0x00,0x00, /* Width of image */
+ 0x00,0x00,0x00,0x00, /* Height of image */
+ 1,0x00, /* How many planes in target device */
+ 1,0x00, /* How many colors in image. 1 means 2^1=2 colors */
+ 0x00,0x00,0x00,0x00, /* Type of compression. 0 means no compression */
+/*Sometimes */ 0x00,0x00,0x00,0x00, /* Size of part with image data */
+/*ttttttt...*/ 0xE8,0x03,0x00,0x00, /* XPelsPerMeter */
+/*hhiiiiissss*/ 0xE8,0x03,0x00,0x00, /* YPelsPerMeter */
+/*part of header*/0x02,0x00,0x00,0x00, /* How many colors from palette is used */
+/*doesn't exist*/ 0x00,0x00,0x00,0x00, /* How many colors from palette is required to display image. 0 means all */
+
+/*Color palette*/ 0x00,0x00,0x00, /* First color in palette in Blue, Green, Red. Here white */
+ 0x00, /* Each color in palette is end by 4'th byte */
+ 102, 204, 102, /* Second color in palette in Blue, Green, Red. Here green */
+ 0x00}; /* Each color in palette is end by 4'th byte */
+
+ if (file!=NULL) isfile=true;
+
+ header[22]=bitmap->BitmapHeight;
+ header[18]=bitmap->BitmapWidth;
+
+ pos = 7;
+ sizeimage = 0;
+ /*lines are written from the last to the first*/
+ for (y=bitmap->BitmapHeight-1;y>=0;y--) {
+ i=1;
+ for (x=0;x<bitmap->BitmapWidth;x++) {
+ /*new byte !*/
+ if (pos==7) {
+ if (x!=0) sizeimage++;
+ i++;
+ /*each line is written in multiply of 4 bytes*/
+ if(i==5) i=1;
+ }
+ pos--;
+ /*going to new byte*/
+ if (pos<0) pos=7;
+ }
+ /*going to new byte*/
+ pos=7;
+ sizeimage++;
+ if (i!=1) {
+ /*each line is written in multiply of 4 bytes*/
+ while (i!=5) {
+ sizeimage++;
+ i++;
+ }
+ }
+ }
+ dbgprintf("Data size in BMP file: %i\n",sizeimage);
+ division=div(sizeimage,256);
+ header[35]=division.quot;
+ header[34]=sizeimage-(division.quot*256);
+ sizeimage=sizeimage+sizeof(header);
+ dbgprintf("Size of BMP file: %i\n",sizeimage);
+ division=div(sizeimage,256);
+ header[3]=division.quot;
+ header[2]=sizeimage-(division.quot*256);
+
+ if (isfile) {
+ fwrite(header,1,sizeof(header),file);
+ } else {
+ memcpy(buffer,header,sizeof(header));
+ buffpos += sizeof(header);
+ }
+
+ pos=7;
+ /*lines are written from the last to the first*/
+ for (y=bitmap->BitmapHeight-1;y>=0;y--) {
+ i=1;
+ for (x=0;x<bitmap->BitmapWidth;x++) {
+ /*new byte !*/
+ if (pos==7) {
+ if (x!=0) {
+ if (isfile) {
+ fwrite(buff, 1, sizeof(buff), file);
+ } else {
+ memcpy (buffer+buffpos,buff,1);
+ buffpos++;
+ }
+ }
+ i++;
+ /*each line is written in multiply of 4 bytes*/
+ if(i==5) i=1;
+ buff[0]=0;
+ }
+ if (!GSM_IsPointBitmap(bitmap,x,y)) buff[0]|=(1<<pos);
+ pos--;
+ /*going to new byte*/
+ if (pos<0) pos=7;
+ }
+ /*going to new byte*/
+ pos=7;
+ if (isfile) {
+ fwrite(buff, 1, sizeof(buff), file);
+ } else {
+ memcpy (buffer+buffpos,buff,1);
+ buffpos++;
+ }
+ if (i!=1) {
+ /*each line is written in multiply of 4 bytes*/
+ while (i!=5) {
+ buff[0]=0;
+ if (isfile) {
+ fwrite(buff, 1, sizeof(buff), file);
+ } else {
+ memcpy (buffer+buffpos,buff,1);
+ buffpos++;
+ }
+ i++;
+ }
+ }
+ }
+ return ERR_NONE;
+}
+
+static GSM_Error savebmp(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ GSM_Error error;
+
+ error=Bitmap2BMP(NULL,file,&bitmap->Bitmap[0]);
+ return error;
+}
+
+static void PrivSaveNLMWBMP(FILE *file, GSM_Bitmap *Bitmap)
+{
+ unsigned char buffer[1000];
+ int x,y,pos,pos2;
+ div_t division;
+
+ pos=0;pos2=7;
+ for (y=0;y<Bitmap->BitmapHeight;y++) {
+ for (x=0;x<Bitmap->BitmapWidth;x++) {
+ if (pos2==7) buffer[pos]=0;
+ if (GSM_IsPointBitmap(Bitmap,x,y)) buffer[pos]|=(1<<pos2);
+ pos2--;
+ /* going to new line */
+ if (pos2<0) {pos2=7;pos++;}
+ }
+ /* for startup logos - new line with new byte */
+ if (pos2!=7) {pos2=7;pos++;}
+ }
+
+ division=div(Bitmap->BitmapWidth,8);
+ /* For startup logos */
+ if (division.rem!=0) division.quot++;
+
+ fwrite(buffer,1,(division.quot*Bitmap->BitmapHeight),file);
+}
+
+static GSM_Error savenlm(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ int i;
+ char header[]={
+ 'N','L','M',' ', /* Nokia Logo Manager file ID. */
+ 0x01,
+ 0x00, /* 0x00 (OP), 0x01 (CLI), 0x02 (Startup), 0x03 (Picture)*/
+ 0x00, /* Number of images inside file - 1. 0x01==2 images, 0x03==4 images, etc. */
+ 0x00, /* Width. */
+ 0x00, /* Height. */
+ 0x01};
+
+ switch (bitmap->Bitmap[0].Type) {
+ case GSM_OperatorLogo : header[5]=0x00; break;
+ case GSM_CallerGroupLogo : header[5]=0x01; break;
+ case GSM_StartupLogo : header[5]=0x02; break;
+ case GSM_PictureImage : header[5]=0x03; break;
+ default : return ERR_UNKNOWN;
+ }
+ header[6] = bitmap->Number - 1;
+ header[7] = bitmap->Bitmap[0].BitmapWidth;
+ header[8] = bitmap->Bitmap[0].BitmapHeight;
+ fwrite(header,1,sizeof(header),file);
+
+ for (i=0;i<bitmap->Number;i++) {
+ PrivSaveNLMWBMP(file, &bitmap->Bitmap[i]);
+ }
+
+ return ERR_NONE;
+}
+
+static void PrivSaveNGGNOL(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ char buffer[GSM_BITMAP_SIZE];
+ int x,y,current=0;
+
+ for (y=0;y<bitmap->Bitmap[0].BitmapHeight;y++) {
+ for (x=0;x<bitmap->Bitmap[0].BitmapWidth;x++) {
+ if (GSM_IsPointBitmap(&bitmap->Bitmap[0],x,y)) {
+ buffer[current++] = '1';
+ } else {
+ buffer[current++] = '0';
+ }
+ }
+ }
+ fwrite(buffer,1,current,file);
+}
+
+static GSM_Error savengg(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ char header[]={
+ 'N','G','G',0x00,0x01,0x00,
+ 0x00,0x00, /* Width */
+ 0x00,0x00, /* Height */
+ 0x01,0x00,0x01,0x00,
+ 0x00, /* Unknown.Can't be checksum - for */
+ /* the same logo files can be different */
+ 0x00};
+
+ header[6] = bitmap->Bitmap[0].BitmapWidth;
+ header[8] = bitmap->Bitmap[0].BitmapHeight;
+ fwrite(header,1,sizeof(header),file);
+
+ PrivSaveNGGNOL(file,bitmap);
+
+ return ERR_NONE;
+}
+
+static GSM_Error savenol(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ int country,net;
+ char header[]={
+ 'N','O','L',0x00,0x01,0x00,
+ 0x00,0x00, /* MCC */
+ 0x00,0x00, /* MNC */
+ 0x00,0x00, /* Width */
+ 0x00,0x00, /* Height */
+ 0x01,0x00,0x01,0x00,
+ 0x00, /* Unknown.Can't be checksum - for */
+ /* the same logo files can be different */
+ 0x00};
+
+ if (bitmap->Bitmap[0].Type == GSM_OperatorLogo) sscanf(bitmap->Bitmap[0].NetworkCode, "%d %d", &country, &net);
+
+ header[6] = country%256;
+ header[7] = country/256;
+ header[8] = net%256;
+ header[9] = net/256;
+ header[10] = bitmap->Bitmap[0].BitmapWidth;
+ header[12] = bitmap->Bitmap[0].BitmapHeight;
+ fwrite(header,1,sizeof(header),file);
+
+ PrivSaveNGGNOL(file,bitmap);
+
+ return ERR_NONE;
+}
+
+static GSM_Error savexpm(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ int x,y;
+
+ fprintf(file,"/* XPM */\n");
+ fprintf(file,"static char * ala_xpm[] = {\n");
+ fprintf(file,"\"%i %i 2 1\",\n",bitmap->Bitmap[0].BitmapWidth,bitmap->Bitmap[0].BitmapHeight);
+ fprintf(file,"\". s c m #000000 g4 #000000 g #000000 c #000000\",\n");
+ fprintf(file,"\"# s c m #ffffff g4 #ffffff g #ffffff c #ffffff\",\n");
+
+ for (y=0;y<bitmap->Bitmap[0].BitmapHeight;y++) {
+ fprintf(file,"\"");
+ for (x=0;x<bitmap->Bitmap[0].BitmapWidth;x++)
+ if (GSM_IsPointBitmap(&bitmap->Bitmap[0],x,y)) {
+ fprintf(file,".");
+ } else {
+ fprintf(file,"#");
+ }
+ fprintf(file,"\"");
+ if (y==bitmap->Bitmap[0].BitmapHeight-1) {
+ fprintf(file,"};\n");
+ } else {
+ fprintf(file,",\n");
+ }
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error savensl(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ char buffer[GSM_BITMAP_SIZE];
+ unsigned char header[]={
+ 'F','O','R','M', 0x01,0xFE, /* File ID block, size 1*256+0xFE=510*/
+ 'N','S','L','D', 0x01,0xF8}; /* Startup Logo block, size 1*256+0xF8=504*/
+
+ fwrite(header,1,sizeof(header),file);
+ PHONE_EncodeBitmap(GSM_NokiaStartupLogo, buffer, &bitmap->Bitmap[0]);
+ fwrite(buffer,1,PHONE_GetBitmapSize(GSM_NokiaStartupLogo,0,0),file);
+
+ return ERR_NONE;
+}
+
+static GSM_Error savewbmp(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ unsigned char buffer[4];
+
+ buffer[0] = 0x00;
+ buffer[1] = 0x00;
+ buffer[2] = bitmap->Bitmap[0].BitmapWidth;
+ buffer[3] = bitmap->Bitmap[0].BitmapHeight;
+ fwrite(buffer,1,4,file);
+
+ PrivSaveNLMWBMP(file, &bitmap->Bitmap[0]);
+
+ return ERR_NONE;
+}
+
+GSM_Error GSM_SaveBitmapFile(char *FileName, GSM_MultiBitmap *bitmap)
+{
+ FILE *file;
+ GSM_Error error=ERR_NONE;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ /* Attempt to identify filetype */
+ if (strstr(FileName,".nlm")) {
+ error=savenlm(file,bitmap);
+ } else if (strstr(FileName,".ngg")) {
+ error=savengg(file,bitmap);
+ } else if (strstr(FileName,".nol")) {
+ error=savenol(file,bitmap);
+ } else if (strstr(FileName,".xpm")) {
+ error=savexpm(file,bitmap);
+ } else if (strstr(FileName,".nsl")) {
+ error=savensl(file,bitmap);
+ } else if (strstr(FileName,".wbmp")) {
+ error=savewbmp(file,bitmap);
+ } else {
+ error=savebmp(file,bitmap);
+ }
+ fclose(file);
+
+ return error;
+}
+
+GSM_Error BMP2Bitmap(unsigned char *buffer, FILE *file,GSM_Bitmap *bitmap)
+{
+ bool first_white,isfile=false;
+ unsigned char buff[34];
+ int w,h,pos,y,x,i,buffpos=0;
+#ifdef DEBUG
+ int sizeimage=0;
+#endif
+
+ if (bitmap->Type == GSM_None) bitmap->Type = GSM_StartupLogo;
+ if (file!=NULL) isfile=true;
+ if (isfile) {
+ fread(buff, 1, 34, file);
+ } else {
+ memcpy(buff,buffer,34);
+ }
+
+ /* height and width of image in the file */
+ h=buff[22]+256*buff[21];
+ w=buff[18]+256*buff[17];
+ dbgprintf("Image Size in BMP file: %dx%d\n",w,h);
+
+ GSM_GetMaxBitmapWidthHeight(bitmap->Type, &bitmap->BitmapWidth, &bitmap->BitmapHeight);
+ if (h<bitmap->BitmapHeight) bitmap->BitmapHeight=h;
+ if (w<bitmap->BitmapWidth) bitmap->BitmapWidth=w;
+ dbgprintf("Height %i %i, width %i %i\n",h,bitmap->BitmapHeight,w,bitmap->BitmapWidth);
+
+ GSM_ClearBitmap(bitmap);
+
+#ifdef DEBUG
+ dbgprintf("Number of colors in BMP file: ");
+ switch (buff[28]) {
+ case 1 : dbgprintf("2 (supported)\n"); break;
+ case 4 : dbgprintf("16 (NOT SUPPORTED)\n"); break;
+ case 8 : dbgprintf("256 (NOT SUPPORTED)\n"); break;
+ case 24 : dbgprintf("True Color (NOT SUPPORTED)\n"); break;
+ default : dbgprintf("unknown\n"); break;
+ }
+#endif
+ if (buff[28]!=1) {
+ dbgprintf("Wrong number of colors\n");
+ return ERR_FILENOTSUPPORTED;
+ }
+
+#ifdef DEBUG
+ dbgprintf("Compression in BMP file: ");
+ switch (buff[30]) {
+ case 0 :dbgprintf("no compression (supported)\n"); break;
+ case 1 :dbgprintf("RLE8 (NOT SUPPORTED)\n"); break;
+ case 2 :dbgprintf("RLE4 (NOT SUPPORTED)\n"); break;
+ default :dbgprintf("unknown\n"); break;
+ }
+#endif
+ if (buff[30]!=0) {
+ dbgprintf("Compression type not supported\n");
+ return ERR_FILENOTSUPPORTED;
+ }
+
+ /* read rest of header (if exists) and color palette */
+ if (isfile) {
+ pos=buff[10]-34;
+ fread(buff, 1, pos, file);
+ } else {
+ pos=buff[10]-34;
+ buffpos=buff[10];
+ memcpy (buff,buffer+34,pos);
+ }
+
+#ifdef DEBUG
+ dbgprintf("First color in BMP file: %i %i %i ",buff[pos-8], buff[pos-7], buff[pos-6]);
+ if (buff[pos-8]==0 && buff[pos-7]==0 && buff[pos-6]==0) dbgprintf("(white)");
+ if (buff[pos-8]==0xFF && buff[pos-7]==0xFF && buff[pos-6]==0xFF) dbgprintf("(black)");
+ if (buff[pos-8]==102 && buff[pos-7]==204 && buff[pos-6]==102) dbgprintf("(green)");
+ dbgprintf("\n");
+ dbgprintf("Second color in BMP file: %i %i %i ",buff[pos-38], buff[pos-37], buff[pos-36]);
+ if (buff[pos-4]==0 && buff[pos-3]==0 && buff[pos-2]==0) dbgprintf("(white)");
+ if (buff[pos-4]==0xFF && buff[pos-3]==0xFF && buff[pos-2]==0xFF) dbgprintf("(black)");
+ dbgprintf("\n");
+#endif
+ first_white=true;
+ if (buff[pos-8]!=0 || buff[pos-7]!=0 || buff[pos-6]!=0) first_white=false;
+
+ pos=7;
+ /* lines are written from the last to the first */
+ for (y=h-1;y>=0;y--) { i=1;
+ for (x=0;x<w;x++) { /* new byte ! */
+ if (pos==7) { if (isfile) {
+ fread(buff, 1, 1, file);
+ } else {
+ memcpy (buff,buffer+buffpos,1);
+ buffpos++;
+ }
+#ifdef DEBUG
+ sizeimage++;
+#endif
+ i++;
+ /* each line is written in multiply of 4 bytes */
+ if(i==5) i=1;
+ }
+ /* we have top left corner ! */
+ if (x<=bitmap->BitmapWidth && y<=bitmap->BitmapHeight) { if (first_white) {
+ if ((buff[0]&(1<<pos))<=0) GSM_SetPointBitmap(bitmap,x,y);
+ } else {
+ if ((buff[0]&(1<<pos))>0) GSM_SetPointBitmap(bitmap,x,y);
+ }
+ }
+ pos--;
+ /* going to new byte */
+ if (pos<0) pos=7;
+ }
+ /* going to new byte */
+ pos=7;
+ if (i!=1) {
+ /* each line is written in multiply of 4 bytes */
+ while (i!=5) {
+ if (isfile) {
+ fread(buff, 1, 1, file);
+ } else {
+ memcpy (buff,buffer+buffpos,1);
+ buffpos++;
+ }
+#ifdef DEBUG
+ sizeimage++;
+#endif
+ i++;
+ }
+ }
+ }
+#ifdef DEBUG
+ dbgprintf("Data size in BMP file: %i\n",sizeimage);
+#endif
+ return(ERR_NONE);
+}
+
+static GSM_Error loadbmp(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ GSM_Error error;
+
+ error=BMP2Bitmap(NULL,file,&bitmap->Bitmap[0]);
+ bitmap->Number = 1;
+ return error;
+}
+
+static GSM_Error loadnlm (FILE *file, GSM_MultiBitmap *bitmap)
+{
+ unsigned char buffer[1000];
+ int pos,pos2,x,y,h,w,i,number;
+ div_t division;
+
+ fread(buffer,1,5,file);
+
+ fread(buffer,1,1,file);
+ switch (buffer[0]) {
+ case 0x00:
+ dbgprintf("Operator logo\n");
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_OperatorLogo;
+ break;
+ case 0x01:
+ dbgprintf("Caller logo\n");
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_CallerGroupLogo;
+ break;
+ case 0x02:
+ dbgprintf("Startup logo\n");
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_StartupLogo;
+ break;
+ case 0x03:
+ dbgprintf("Picture Image logo\n");
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_PictureImage;
+ break;
+ }
+
+ bitmap->Number = 0;
+ fread(buffer,1,4,file);
+ number = buffer[0] + 1;
+ w = buffer[1];
+ h = buffer[2];
+ for (i=0;i<number;i++) {
+ bitmap->Bitmap[i].Type = bitmap->Bitmap[0].Type;
+ GSM_GetMaxBitmapWidthHeight(bitmap->Bitmap[i].Type, &bitmap->Bitmap[i].BitmapWidth, &bitmap->Bitmap[i].BitmapHeight);
+ if (h < bitmap->Bitmap[i].BitmapHeight) bitmap->Bitmap[i].BitmapHeight = h;
+ if (w < bitmap->Bitmap[i].BitmapWidth) bitmap->Bitmap[i].BitmapWidth = w;
+
+ division=div(w,8);
+ /* For startup logos */
+ if (division.rem!=0) division.quot++;
+ if (fread(buffer,1,(division.quot*h),file)!=(unsigned int)(division.quot*h)) return ERR_UNKNOWN;
+
+ GSM_ClearBitmap(&bitmap->Bitmap[i]);
+
+ pos=0;pos2=7;
+ for (y=0;y<h;y++) {
+ for (x=0;x<w;x++) {
+ if ((buffer[pos]&(1<<pos2))>0) {
+ if (y<bitmap->Bitmap[i].BitmapHeight && x<bitmap->Bitmap[i].BitmapWidth) GSM_SetPointBitmap(&bitmap->Bitmap[i],x,y);
+ }
+ pos2--;
+ /* going to new byte */
+ if (pos2<0) {pos2=7;pos++;}
+ }
+ /* for startup logos-new line means new byte */
+ if (pos2!=7) {pos2=7;pos++;}
+ }
+ bitmap->Number++;
+ if (bitmap->Number == MAX_MULTI_BITMAP) break;
+ }
+ return (ERR_NONE);
+}
+
+static GSM_Error loadnolngg(FILE *file, GSM_MultiBitmap *bitmap, bool nolformat)
+{
+ unsigned char buffer[2000];
+ int i,h,w,x,y;
+
+ fread(buffer, 1, 6, file);
+
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_CallerGroupLogo;
+ if (nolformat) {
+ fread(buffer, 1, 4, file);
+ sprintf(bitmap->Bitmap[0].NetworkCode, "%d %02d", buffer[0]+256*buffer[1], buffer[2]);
+ if (bitmap->Bitmap[0].Type == GSM_None) bitmap->Bitmap[0].Type = GSM_OperatorLogo;
+ }
+
+ fread(buffer, 1, 4, file);
+ w = buffer[0];
+ h = buffer[2];
+ GSM_GetMaxBitmapWidthHeight(bitmap->Bitmap[0].Type, &bitmap->Bitmap[0].BitmapWidth, &bitmap->Bitmap[0].BitmapHeight);
+ if (h < bitmap->Bitmap[0].BitmapHeight) bitmap->Bitmap[0].BitmapHeight = h;
+ if (w < bitmap->Bitmap[0].BitmapWidth) bitmap->Bitmap[0].BitmapWidth = w;
+
+ /* Unknown bytes. */
+ fread(buffer, 1, 6, file);
+
+ GSM_ClearBitmap(&bitmap->Bitmap[0]);
+
+ x=0; y=0;
+ for (i=0; i<w*h; i++) {
+ if (fread(buffer, 1, 1, file)!=1) return ERR_UNKNOWN;
+ if (buffer[0]=='1') GSM_SetPointBitmap(&bitmap->Bitmap[0],x,y);
+ x++;
+ if (x==w) {x=0; y++;}
+ }
+
+#ifdef DEBUG
+ /* Some programs writes here fileinfo */
+ if (fread(buffer, 1, 1, file)==1) {
+ dbgprintf("Fileinfo: %c",buffer[0]);
+ while (fread(buffer, 1, 1, file)==1) {
+ if (buffer[0]!=0x0A) dbgprintf("%c",buffer[0]);
+ }
+ dbgprintf("\n");
+ }
+#endif
+ bitmap->Number = 1;
+ return(ERR_NONE);
+}
+
+static GSM_Error loadnsl(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ unsigned char block[6],buffer[505];
+ int block_size;
+ GSM_Bitmap_Types OldType;
+
+ while (fread(block,1,6,file)==6) {
+ block_size = block[4]*256 + block[5];
+ dbgprintf("Block %c%c%c%c, size %i\n",block[0],block[1],block[2],block[3],block_size);
+ if (!strncmp(block, "FORM", 4)) {
+ dbgprintf("File ID\n");
+ } else {
+ if (block_size>504) return ERR_UNKNOWN;
+ if (block_size!=0) {
+ fread(buffer,1,block_size,file);
+ /* if it's string, we end it with 0 */
+ buffer[block_size]=0;
+#ifdef DEBUG
+ if (!strncmp(block, "VERS", 4)) dbgprintf("File saved by: %s\n",buffer);
+ if (!strncmp(block, "MODL", 4)) dbgprintf("Logo saved from: %s\n",buffer);
+ if (!strncmp(block, "COMM", 4)) dbgprintf("Phone was connected to COM port: %s\n",buffer);
+#endif
+ if (!strncmp(block, "NSLD", 4)) {
+ bitmap->Bitmap[0].BitmapHeight = 48;
+ bitmap->Bitmap[0].BitmapWidth = 84;
+ OldType = bitmap->Bitmap[0].Type;
+ PHONE_DecodeBitmap(GSM_NokiaStartupLogo, buffer, &bitmap->Bitmap[0]);
+ if (OldType != GSM_None) bitmap->Bitmap[0].Type = OldType;
+ dbgprintf("Startup logo (size %i)\n",block_size);
+ }
+ }
+ }
+ }
+ bitmap->Number = 1;
+ return(ERR_NONE);
+}
+
+static GSM_Error loadwbmp(FILE *file, GSM_MultiBitmap *bitmap)
+{
+ unsigned char buffer[10000];
+
+ fread(buffer,1,4,file);
+ bitmap->Bitmap[0].BitmapWidth = buffer[2];
+ bitmap->Bitmap[0].BitmapHeight = buffer[3];
+ bitmap->Number = 1;
+
+ fread(buffer,1,10000,file);
+ PHONE_DecodeBitmap(GSM_Nokia7110OperatorLogo, buffer, &bitmap->Bitmap[0]);
+ GSM_ReverseBitmap(&bitmap->Bitmap[0]);
+
+ return ERR_NONE;
+}
+
+GSM_Error GSM_ReadBitmapFile(char *FileName, GSM_MultiBitmap *bitmap)
+{
+ FILE *file;
+ unsigned char buffer[300];
+
+ file = fopen(FileName, "rb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ fread(buffer, 1, 9, file); /* Read the header of the file. */
+ rewind(file);
+
+ bitmap->Bitmap[0].DefaultBitmap = false;
+
+ /* Attempt to identify filetype */
+ if (memcmp(buffer, "BM",2)==0) {
+ return loadbmp(file,bitmap);
+ } else if (buffer[0] == 0x00 && buffer[1] == 0x00) {
+ return loadwbmp(file,bitmap);
+ } else if (memcmp(buffer, "NLM",3)==0) {
+ return loadnlm(file,bitmap);
+ } else if (memcmp(buffer, "NOL",3)==0) {
+ return loadnolngg(file,bitmap,true);
+ } else if (memcmp(buffer, "NGG",3)==0) {
+ return loadnolngg(file,bitmap,false);
+ } else if (memcmp(buffer, "FORM",4)==0) {
+ return loadnsl(file,bitmap);
+ }
+ return ERR_UNKNOWN;
+}
+
+void NOKIA_CopyBitmap(GSM_Phone_Bitmap_Types Type, GSM_Bitmap *Bitmap, char *Buffer, int *Length)
+{
+ int Width, Height;
+
+ Buffer[(*Length)++] = 0x00;
+ PHONE_GetBitmapWidthHeight(Type, &Width, &Height);
+ Buffer[(*Length)++] = Width;
+ Buffer[(*Length)++] = Height;
+ Buffer[(*Length)++] = 0x01;
+ PHONE_EncodeBitmap(Type, Buffer + (*Length), Bitmap);
+ (*Length) = (*Length) + PHONE_GetBitmapSize(Type,0,0);
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmlogo.h b/gammu/emb/common/service/gsmlogo.h
new file mode 100644
index 0000000..b1b579d
--- a/dev/null
+++ b/gammu/emb/common/service/gsmlogo.h
@@ -0,0 +1,180 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef __gsm_bitmaps_h
+#define __gsm_bitmaps_h
+
+#include "../gsmcomon.h"
+
+/**
+ * Enum to handle all possible bitmaps, which are not saved in various filesystems.
+ */
+typedef enum {
+ GSM_None = 1,
+ /**
+ * ID of static file in filesystem displayed during startup
+ */
+ GSM_ColourStartupLogo_ID,
+ /**
+ * Static mono bitmap/ID of animated mono bitmap displayed during startup
+ */
+ GSM_StartupLogo,
+ /**
+ * ID of static file in filesystem displayed instead of operator name
+ */
+ GSM_ColourOperatorLogo_ID,
+ /**
+ * Mono bitmap displayed instead of operator name
+ */
+ GSM_OperatorLogo,
+ /**
+ * ID of static file in filesystem displayed as wallpaper
+ */
+ GSM_ColourWallPaper_ID,
+ /**
+ * Mono bitmap assigned to caller group
+ */
+ GSM_CallerGroupLogo,
+ /**
+ * Text displayed during startup, which can't be removed from phone menu
+ */
+ GSM_DealerNote_Text,
+ /**
+ * Text displayed during startup
+ */
+ GSM_WelcomeNote_Text,
+ /**
+ * Image defined in Smart Messaging specification
+ */
+ GSM_PictureImage
+} GSM_Bitmap_Types;
+
+#define GSM_BITMAP_SIZE (65+7)/8*96
+#define GSM_BITMAP_TEXT_LENGTH 128
+
+/**
+ * Structure for all possible bitmaps, which are not saved in various filesystems
+ */
+typedef struct {
+ /**
+ * For all: bitmap type
+ */
+ GSM_Bitmap_Types Type;
+ /**
+ * For caller group logos: number of group
+ * For startup logos: number of animated bitmap
+ */
+ unsigned char Location;
+ /**
+ * For dealer/welcome note text: text
+ * For caller group logo: name of group
+ * For picture images: text assigned to it
+ */
+ unsigned char Text[2 * (GSM_BITMAP_TEXT_LENGTH + 1)];
+ /**
+ * For caller group logo: true, when logo is enabled in group
+ */
+ bool BitmapEnabled;
+ /**
+ * For caller group logo: true, when group has default name
+ */
+ bool DefaultName;
+ /**
+ * For caller group logo: true, when group has default bitmap
+ */
+ bool DefaultBitmap;
+ /**
+ * For caller group logo: true, when group has default ringtone
+ */
+ bool DefaultRingtone;
+ /**
+ * For caller group logo: ringtone ID. Phone model specific
+ */
+ unsigned char RingtoneID;
+ bool FileSystemRingtone;
+ /**
+ * For mono bitmaps: body of bitmap
+ */
+ unsigned char BitmapPoints[GSM_BITMAP_SIZE];
+ /**
+ * For mono bitmaps: height specified in pixels
+ */
+ unsigned char BitmapHeight;
+ /**
+ * For mono bitmaps: width specified in pixels
+ */
+ unsigned char BitmapWidth;
+ /**
+ * For operator logos: Network operator code
+ */
+ char NetworkCode[7];
+ /**
+ * For picture images: number of sender
+ */
+ unsigned char Sender[2 * (GSM_MAX_NUMBER_LENGTH + 1)];
+ /**
+ * For colour bitmaps: ID
+ */
+ unsigned char ID;
+} GSM_Bitmap;
+
+#define MAX_MULTI_BITMAP 6
+
+/**
+ * Structure to handle more than one bitmap
+ */
+typedef struct {
+ /**
+ * Number of bitmaps
+ */
+ unsigned char Number;
+ /**
+ * All bitmaps
+ */
+ GSM_Bitmap Bitmap[MAX_MULTI_BITMAP];
+} GSM_MultiBitmap;
+
+typedef enum {
+ GSM_NokiaStartupLogo = 1, /*size 84*48*/
+ GSM_NokiaOperatorLogo, /*size 72*14*/
+ GSM_Nokia7110OperatorLogo, /*size 78*21*/
+ GSM_Nokia6510OperatorLogo, /*size 78*21*/
+ GSM_NokiaCallerLogo, /*size 72*14*/
+ GSM_NokiaPictureImage, /*size 72*28*/
+ GSM_Nokia7110StartupLogo, /*size 96*65*/
+ GSM_Nokia6210StartupLogo, /*size 96*60*/
+ GSM_AlcatelBMMIPicture,
+ GSM_EMSSmallPicture, /*size 8* 8*/
+ GSM_EMSMediumPicture, /*size 16*16*/
+ GSM_EMSBigPicture, /*size 32*32*/
+ GSM_EMSVariablePicture
+} GSM_Phone_Bitmap_Types;
+
+bool GSM_IsPointBitmap (GSM_Bitmap *bmp, int x, int y);
+void GSM_SetPointBitmap (GSM_Bitmap *bmp, int x, int y);
+void GSM_ClearPointBitmap (GSM_Bitmap *bmp, int x, int y);
+void GSM_ClearBitmap (GSM_Bitmap *bmp);
+void GSM_ResizeBitmap (GSM_Bitmap *dest, GSM_Bitmap *src, int width, int height);
+void GSM_ReverseBitmap (GSM_Bitmap *Bitmap);
+void GSM_GetMaxBitmapWidthHeight(GSM_Bitmap_Types Type, unsigned char *width, unsigned char *height);
+int GSM_GetBitmapSize (GSM_Bitmap *bmp);
+void GSM_PrintBitmap (FILE *file, GSM_Bitmap *bitmap);
+
+GSM_Error GSM_SaveBitmapFile (char *FileName, GSM_MultiBitmap *bitmap);
+GSM_Error GSM_ReadBitmapFile (char *FileName, GSM_MultiBitmap *bitmap);
+
+GSM_Error BMP2Bitmap (unsigned char *buffer, FILE *file,GSM_Bitmap *bitmap);
+GSM_Error Bitmap2BMP (unsigned char *buffer, FILE *file,GSM_Bitmap *bitmap);
+
+void PHONE_GetBitmapWidthHeight (GSM_Phone_Bitmap_Types Type, int *width, int *height);
+int PHONE_GetBitmapSize (GSM_Phone_Bitmap_Types Type, int width, int height);
+void PHONE_ClearBitmap (GSM_Phone_Bitmap_Types Type, char *buffer, int width, int height);
+void PHONE_DecodeBitmap (GSM_Phone_Bitmap_Types Type, char *buffer, GSM_Bitmap *Bitmap);
+void PHONE_EncodeBitmap (GSM_Phone_Bitmap_Types Type, char *buffer, GSM_Bitmap *Bitmap);
+
+void NOKIA_CopyBitmap (GSM_Phone_Bitmap_Types Type, GSM_Bitmap *Bitmap, char *Buffer, int *Length);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmmisc.c b/gammu/emb/common/service/gsmmisc.c
new file mode 100644
index 0000000..6959a22
--- a/dev/null
+++ b/gammu/emb/common/service/gsmmisc.c
@@ -0,0 +1,262 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#include <string.h>
+#include <stdlib.h>
+#include <sys/stat.h>
+#ifdef WIN32
+# include <io.h>
+# include <fcntl.h>
+#endif
+
+#include "../misc/coding/coding.h"
+#include "../gsmcomon.h"
+#include "gsmmisc.h"
+
+struct keys_table_position {
+ char whatchar;
+ int whatcode;
+};
+
+static struct keys_table_position Keys[] = {
+ {'m',GSM_KEY_MENU}, {'M',GSM_KEY_MENU},
+ {'n',GSM_KEY_NAMES}, {'N',GSM_KEY_NAMES},
+ {'p',GSM_KEY_POWER}, {'P',GSM_KEY_POWER},
+ {'u',GSM_KEY_UP}, {'U',GSM_KEY_UP},
+ {'d',GSM_KEY_DOWN}, {'D',GSM_KEY_DOWN},
+ {'+',GSM_KEY_INCREASEVOLUME}, {'-',GSM_KEY_DECREASEVOLUME},
+ {'1',GSM_KEY_1}, {'2',GSM_KEY_2}, {'3',GSM_KEY_3},
+ {'4',GSM_KEY_4}, {'5',GSM_KEY_5}, {'6',GSM_KEY_6},
+ {'7',GSM_KEY_7}, {'8',GSM_KEY_8}, {'9',GSM_KEY_9},
+ {'*',GSM_KEY_ASTERISK}, {'0',GSM_KEY_0}, {'#',GSM_KEY_HASH},
+ {'g',GSM_KEY_GREEN}, {'G',GSM_KEY_GREEN},
+ {'r',GSM_KEY_RED}, {'R',GSM_KEY_RED},
+ {' ',0}
+};
+
+GSM_Error MakeKeySequence(char *text, GSM_KeyCode *KeyCode, int *Length)
+{
+ int i,j;
+ unsigned char key;
+
+ for (i=0;i<(int)(strlen(text));i++) {
+ key = text[i];
+ KeyCode[i] = GSM_KEY_NONE;
+ j = 0;
+ while (Keys[j].whatchar!=' ') {
+ if (Keys[j].whatchar==key) {
+ KeyCode[i]=Keys[j].whatcode;
+ break;
+ }
+ j++;
+ }
+ if (KeyCode[i] == GSM_KEY_NONE) {
+ *Length = i;
+ return ERR_NOTSUPPORTED;
+ }
+ }
+ *Length = i;
+ return ERR_NONE;
+}
+
+GSM_Error GSM_ReadFile(char *FileName, GSM_File *File)
+{
+ int i = 1000;
+ FILE *file;
+ struct stat fileinfo;
+
+ if (FileName[0] == 0x00) return ERR_UNKNOWN;
+ file = fopen(FileName,"rb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ free(File->Buffer);
+ File->Buffer = NULL;
+ File->Used = 0;
+ while (i == 1000) {
+ File->Buffer = realloc(File->Buffer,File->Used + 1000);
+ i = fread(File->Buffer+File->Used,1,1000,file);
+ File->Used = File->Used + i;
+ }
+ File->Buffer = realloc(File->Buffer,File->Used);
+ fclose(file);
+
+ File->ModifiedEmpty = true;
+ if (stat(FileName,&fileinfo) == 0) {
+ File->ModifiedEmpty = false;
+ dbgprintf("File info read correctly\n");
+ //st_mtime is time of last modification of file
+ Fill_GSM_DateTime(&File->Modified, fileinfo.st_mtime);
+ File->Modified.Year = File->Modified.Year + 1900;
+ dbgprintf("FileTime: %02i-%02i-%04i %02i:%02i:%02i\n",
+ File->Modified.Day,File->Modified.Month,File->Modified.Year,
+ File->Modified.Hour,File->Modified.Minute,File->Modified.Second);
+ }
+
+ return ERR_NONE;
+}
+
+static void GSM_JADFindLine(GSM_File File, char *Name, char *Value)
+{
+ unsigned char Line[2000];
+ int Pos = 0;
+
+ Value[0] = 0;
+
+ while (1) {
+ MyGetLine(File.Buffer, &Pos, Line, File.Used);
+ if (strlen(Line) == 0) break;
+ if (!strncmp(Line,Name,strlen(Name))) {
+ Pos = strlen(Name);
+ while (Line[Pos] == 0x20) Pos++;
+ strcpy(Value,Line+Pos);
+ return;
+ }
+ }
+}
+
+GSM_Error GSM_JADFindData(GSM_File File, char *Vendor, char *Name, char *JAR, char *Version, int *Size)
+{
+ char Size2[200];
+
+ GSM_JADFindLine(File, "MIDlet-Vendor:", Vendor);
+ if (Vendor[0] == 0x00) return ERR_FILENOTSUPPORTED;
+ dbgprintf("Vendor: \"%s\"\n",Vendor);
+
+ GSM_JADFindLine(File, "MIDlet-Name:", Name);
+ if (Name[0] == 0x00) return ERR_FILENOTSUPPORTED;
+ dbgprintf("Name: \"%s\"\n",Name);
+
+ GSM_JADFindLine(File, "MIDlet-Jar-URL:", JAR);
+ if (JAR[0] == 0x00) return ERR_FILENOTSUPPORTED;
+ dbgprintf("JAR file URL: \"%s\"\n",JAR);
+
+ GSM_JADFindLine(File, "MIDlet-Jar-Size:", Size2);
+ *Size = -1;
+ if (Size2[0] == 0x00) return ERR_FILENOTSUPPORTED;
+ dbgprintf("JAR size: \"%s\"\n",Size2);
+ (*Size) = atoi(Size2);
+
+ GSM_JADFindLine(File, "MIDlet-Version:", Version);
+ dbgprintf("Version: \"%s\"\n",Version);
+
+ return ERR_NONE;
+}
+
+void GSM_IdentifyFileFormat(GSM_File *File)
+{
+ File->Type = GSM_File_Other;
+ if (File->Used > 2) {
+ if (memcmp(File->Buffer, "BM",2)==0) {
+ File->Type = GSM_File_Image_BMP;
+ } else if (memcmp(File->Buffer, "GIF",3)==0) {
+ File->Type = GSM_File_Image_GIF;
+ } else if (File->Buffer[0] == 0x00 && File->Buffer[1] == 0x00) {
+ File->Type = GSM_File_Image_WBMP;
+ } else if (memcmp(File->Buffer+1, "PNG",3)==0) {
+ File->Type = GSM_File_Image_PNG;
+ } else if (File->Buffer[0] == 0xFF && File->Buffer[1] == 0xD8) {
+ File->Type = GSM_File_Image_JPG;
+ } else if (memcmp(File->Buffer, "MThd",4)==0) {
+ File->Type = GSM_File_Sound_MIDI;
+ } else if (File->Buffer[0] == 0x00 && File->Buffer[1] == 0x02) {
+ File->Type = GSM_File_Sound_NRT;
+ }
+ }
+}
+
+void SaveVCALDateTime(char *Buffer, int *Length, GSM_DateTime *Date, char *Start)
+{
+ if (Start != NULL) {
+ *Length+=sprintf(Buffer+(*Length), "%s:",Start);
+ }
+ *Length+=sprintf(Buffer+(*Length), "%04d%02d%02dT%02d%02d%02d%c%c",
+ Date->Year, Date->Month, Date->Day,
+ Date->Hour, Date->Minute, Date->Second,13,10);
+}
+
+void ReadVCALDateTime(char *Buffer, GSM_DateTime *dt)
+{
+ char year[5]="", month[3]="", day[3]="", hour[3]="", minute[3]="", second[3]="";
+
+ memset(dt,0,sizeof(dt));
+
+ strncpy(year, Buffer, 4);
+ strncpy(month, Buffer+4, 2);
+ strncpy(day, Buffer+6, 2);
+ strncpy(hour, Buffer+9, 2);
+ strncpy(minute, Buffer+11, 2);
+ strncpy(second, Buffer+13, 2);
+
+ /* FIXME: Should check ranges... */
+ dt->Year = atoi(year);
+ dt->Month = atoi(month);
+ dt->Day = atoi(day);
+ dt->Hour = atoi(hour);
+ dt->Minute = atoi(minute);
+ dt->Second = atoi(second);
+ /* FIXME */
+ dt->Timezone = 0;
+}
+
+void SaveVCALText(char *Buffer, int *Length, char *Text, char *Start)
+{
+ char buffer[1000];
+
+ if (UnicodeLength(Text) != 0) {
+ EncodeUTF8QuotedPrintable(buffer,Text);
+ if (UnicodeLength(Text)==strlen(buffer)) {
+ *Length+=sprintf(Buffer+(*Length), "%s:%s%c%c",Start,DecodeUnicodeString(Text),13,10);
+ } else {
+ *Length+=sprintf(Buffer+(*Length), "%s;CHARSET=UTF-8;ENCODING=QUOTED-PRINTABLE:%s%c%c",Start,buffer,13,10);
+ }
+ }
+}
+
+bool ReadVCALText(char *Buffer, char *Start, char *Value)
+{
+ unsigned char buff[200];
+
+ Value[0] = 0x00;
+ Value[1] = 0x00;
+
+ strcpy(buff,Start);
+ strcat(buff,":");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ EncodeUnicode(Value,Buffer+strlen(Start)+1,strlen(Buffer)-(strlen(Start)+1));
+ dbgprintf("ReadVCalText is \"%s\"\n",DecodeUnicodeConsole(Value));
+ return true;
+ }
+ /* SE T68i */
+ strcpy(buff,Start);
+ strcat(buff,";ENCODING=QUOTED-PRINTABLE:");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ DecodeUTF8QuotedPrintable(Value,Buffer+strlen(Start)+27,strlen(Buffer)-(strlen(Start)+27));
+ dbgprintf("ReadVCalText is \"%s\"\n",DecodeUnicodeConsole(Value));
+ return true;
+ }
+ strcpy(buff,Start);
+ strcat(buff,";CHARSET=UTF-8;ENCODING=QUOTED-PRINTABLE:");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ DecodeUTF8QuotedPrintable(Value,Buffer+strlen(Start)+41,strlen(Buffer)-(strlen(Start)+41));
+ dbgprintf("ReadVCalText is \"%s\"\n",DecodeUnicodeConsole(Value));
+ return true;
+ }
+ strcpy(buff,Start);
+ strcat(buff,";CHARSET=UTF-8:");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ DecodeUTF8(Value,Buffer+strlen(Start)+15,strlen(Buffer)-(strlen(Start)+15));
+ dbgprintf("ReadVCalText is \"%s\"\n",DecodeUnicodeConsole(Value));
+ return true;
+ }
+ strcpy(buff,Start);
+ strcat(buff,";CHARSET=UTF-7:");
+ if (!strncmp(Buffer,buff,strlen(buff))) {
+ DecodeUTF7(Value,Buffer+strlen(Start)+15,strlen(Buffer)-(strlen(Start)+15));
+ dbgprintf("ReadVCalText is \"%s\"\n",DecodeUnicodeConsole(Value));
+ return true;
+ }
+ return false;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmmisc.h b/gammu/emb/common/service/gsmmisc.h
new file mode 100644
index 0000000..37501ad
--- a/dev/null
+++ b/gammu/emb/common/service/gsmmisc.h
@@ -0,0 +1,316 @@
+/* (c) 2001-2004 by Marcin Wiacek, Walek and Michal Cihar */
+
+#ifndef __gsm_misc_h
+#define __gsm_misc_h
+
+#include "../gsmcomon.h"
+
+/**
+ * Enum defines ID for various phone and SIM memories.
+ * Phone modules can translate them to values specific for concrete models
+ * Two letter codes (excluding VM) are from GSM 07.07
+ */
+typedef enum {
+ /**
+ * Internal memory of the mobile equipment
+ */
+ MEM_ME=1,
+ /**
+ * SIM card memory
+ */
+ MEM_SM,
+ /**
+ * Own numbers
+ */
+ MEM_ON,
+ /**
+ * Dialled calls
+ */
+ MEM_DC,
+ /**
+ * Received calls
+ */
+ MEM_RC,
+ /**
+ * Missed calls
+ */
+ MEM_MC,
+ /**
+ * Combined ME and SIM phonebook
+ */
+ MEM_MT,
+ /**
+ * Fixed dial
+ */
+ MEM_FD,
+
+ /**
+ * Voice mailbox
+ */
+ MEM_VM
+} GSM_MemoryType;
+
+/* --------------------------- resetting phone settings ------------------- */
+
+typedef enum {
+ GSM_RESET_PHONESETTINGS = 1,
+ GSM_RESET_USERINTERFACE,
+ GSM_RESET_USERINTERFACE_PHONESETTINGS,
+ GSM_RESET_DEVICE,
+ GSM_RESET_FULLFACTORY
+} GSM_ResetSettingsType;
+
+/* --------------------------- security codes ------------------------------ */
+
+/**
+ * Definition of security codes.
+ */
+typedef enum {
+ /**
+ * Security code.
+ */
+ SEC_SecurityCode = 0x01,
+ /**
+ * PIN.
+ */
+ SEC_Pin,
+ /**
+ * PIN 2.
+ */
+ SEC_Pin2,
+ /**
+ * PUK.
+ */
+ SEC_Puk,
+ /**
+ * PUK 2.
+ */
+ SEC_Puk2,
+ /**
+ * Code not needed.
+ */
+ SEC_None
+} GSM_SecurityCodeType;
+
+#define GSM_SECURITY_CODE_LEN 15
+
+/**
+ * Security code definition.
+ */
+typedef struct {
+ /**
+ * Actual code.
+ */
+ char Code[GSM_SECURITY_CODE_LEN+1];
+ /**
+ * Type of the code.
+ */
+ GSM_SecurityCodeType Type;
+} GSM_SecurityCode;
+
+/* ---------------------------- keyboard ----------------------------------- */
+
+typedef enum {
+ GSM_KEY_NONE = 0x00,
+ GSM_KEY_1 = 0x01,
+ GSM_KEY_2,
+ GSM_KEY_3,
+ GSM_KEY_4,
+ GSM_KEY_5,
+ GSM_KEY_6,
+ GSM_KEY_7,
+ GSM_KEY_8,
+ GSM_KEY_9,
+ GSM_KEY_0,
+ /**
+ * #
+ */
+ GSM_KEY_HASH,
+ /**
+ * *
+ */
+ GSM_KEY_ASTERISK,
+ GSM_KEY_POWER,
+ /**
+ * in some phone ie. N5110 sometimes works identical to POWER
+ */
+ GSM_KEY_GREEN,
+ /**
+ * (c) key in some phone: ie. N5110
+ */
+ GSM_KEY_RED,
+ /**
+ * doesn't available in some phones as separate button: ie. N5110
+ */
+ GSM_KEY_INCREASEVOLUME,
+ /**
+ * doesn't available in some phones as separate button: ie. N5110
+ */
+ GSM_KEY_DECREASEVOLUME,
+ GSM_KEY_UP = 0x17,
+ GSM_KEY_DOWN,
+ GSM_KEY_MENU,
+ /**
+ * doesn't available in some phone: ie. N5110
+ */
+ GSM_KEY_NAMES
+} GSM_KeyCode;
+
+GSM_Error MakeKeySequence(char *text, GSM_KeyCode *KeyCode, int *Length);
+
+/* ------------------------------- display features ------------------------ */
+
+typedef enum {
+ GSM_CallActive = 1,
+ /**
+ * blinking envelope
+ */
+ GSM_SMSMemoryFull,
+ GSM_FaxCall,
+ GSM_UnreadSMS,
+ GSM_DataCall,
+ GSM_VoiceCall,
+ GSM_KeypadLocked
+} GSM_DisplayFeature;
+
+typedef struct {
+ int Number;
+ GSM_DisplayFeature Feature[7];
+} GSM_DisplayFeatures;
+
+/* ----------------------------- power source ------------------------------ */
+
+typedef enum {
+ GSM_BatteryPowered = 1,
+ GSM_BatteryConnected,
+ GSM_BatteryNotConnected,
+ GSM_PowerFault
+} GSM_ChargeState;
+
+typedef struct {
+ /**
+ * Signal strength in percent, -1 = unknown
+ */
+ int BatteryPercent;
+ /**
+ * Charge state
+ */
+ GSM_ChargeState ChargeState;
+} GSM_BatteryCharge;
+
+/* ------------------------------ categories ------------------------------- */
+
+#define GSM_MAX_CATEGORY_NAME_LENGTH 50
+
+typedef enum {
+ Category_ToDo = 1,
+ Category_Phonebook
+} GSM_CategoryType;
+
+typedef struct {
+ GSM_CategoryType Type;
+ int Location;
+ unsigned char Name[(GSM_MAX_CATEGORY_NAME_LENGTH + 1)*2];
+} GSM_Category;
+
+typedef struct {
+ GSM_CategoryType Type;
+ int Used;
+} GSM_CategoryStatus;
+
+/* ------------------- radio FM stations ---------------------------------- */
+
+#define GSM_MAX_FMSTATION_LENGTH 12
+#define GSM_MAX_FM_STATION 20
+
+typedef struct {
+ int Location;
+ char StationName [(GSM_MAX_FMSTATION_LENGTH+1)*2];
+ double Frequency;
+} GSM_FMStation;
+
+/* ----------------------- filesystem ------------------------------------- */
+
+typedef enum {
+ GSM_File_Java_JAR = 1,
+ GSM_File_Image_JPG,
+ GSM_File_Image_BMP,
+ GSM_File_Image_GIF,
+ GSM_File_Image_PNG,
+ GSM_File_Image_WBMP,
+ GSM_File_Video_3GP,
+ GSM_File_Sound_AMR,
+ GSM_File_Sound_NRT, /* DCT4 binary format */
+ GSM_File_Sound_MIDI,
+#ifdef DEVELOP
+ GSM_File_MMS,
+#endif
+ GSM_File_Other
+} GSM_FileType;
+
+typedef struct {
+ int Used; /* how many bytes used */
+ unsigned char Name[300]; /* Name */
+ bool Folder; /* true, when folder */
+ int Level;
+ GSM_FileType Type;
+ unsigned char ID_FullName[400];
+ unsigned char *Buffer;
+
+ GSM_DateTime Modified;
+ bool ModifiedEmpty;
+
+ /* File attributes */
+ bool Protected;
+ bool ReadOnly;
+ bool Hidden;
+ bool System;
+} GSM_File;
+
+GSM_Error GSM_ReadFile(char *FileName, GSM_File *File);
+
+GSM_Error GSM_JADFindData(GSM_File File, char *Vendor, char *Name, char *JAR, char *Version, int *Size);
+
+void GSM_IdentifyFileFormat(GSM_File *File);
+
+typedef struct {
+ int Free;
+ int Used;
+} GSM_FileSystemStatus;
+
+/* ----------------------------- GPRS access points ----------------------- */
+
+typedef struct {
+ int Location;
+ unsigned char Name[300];
+ unsigned char URL[500];
+ bool Active;
+} GSM_GPRSAccessPoint;
+
+/* ------------------------------------------------------------------------ */
+
+typedef enum {
+ GSM_Date_DDMMYYYY = 1,
+ GSM_Date_MMDDYYYY,
+ GSM_Date_YYYYMMDD
+} GSM_DateFormat;
+
+typedef struct {
+ unsigned char DateSeparator;
+ GSM_DateFormat DateFormat;
+ bool AMPMTime;
+} GSM_Locale;
+
+/* ------------------------------------------------------------------------ */
+
+void ReadVCALDateTime(char *Buffer, GSM_DateTime *dt);
+void SaveVCALDateTime(char *Buffer, int *Length, GSM_DateTime *Date, char *Start);
+
+void SaveVCALText(char *Buffer, int *Length, char *Text, char *Start);
+bool ReadVCALText(char *Buffer, char *Start, char *Value);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmnet.c b/gammu/emb/common/service/gsmnet.c
new file mode 100644
index 0000000..a0ddaa7
--- a/dev/null
+++ b/gammu/emb/common/service/gsmnet.c
@@ -0,0 +1,444 @@
+/* (c) 2001-2003 by Marcin Wiacek */
+
+#include <string.h>
+
+#include "gsmnet.h"
+#include "../misc/coding/coding.h"
+
+unsigned char *GSM_Countries[] = {
+ "202", "Greece",
+ "204", "Netherlands",
+ "206", "Belgium",
+ "208", "France",
+ "213", "Andorra" ,
+ "214", "Spain",
+ "216", "Hungary",
+ "218", "Bosnia Herzegovina",
+ "219", "Croatia",
+ "220", "Yugoslavia",
+ "222", "Italy",
+ "226", "Romania",
+ "228", "Switzerland",
+ "230", "Czech Republic",
+ "231", "Slovak Republic",
+ "232", "Austria",
+ "234", "United Kingdom",
+ "238", "Denmark",
+ "240", "Sweden",
+ "242", "Norway",
+ "244", "Finland",
+ "246", "Lithuania",
+ "247", "Latvia",
+ "248", "Estonia",
+ "250", "Russia",
+ "255", "Ukraine",
+ "259", "Moldova",
+ "260", "Poland",
+ "262", "Germany",
+ "266", "Gibraltar",
+ "268", "Portugal",
+ "270", "Luxembourg",
+ "272", "Ireland",
+ "274", "Iceland",
+ "276", "Albania",
+ "278", "Malta",
+ "280", "Cyprus",
+ "282", "Georgia",
+ "283", "Armenia",
+ "284", "Bulgaria",
+ "286", "Turkey",
+ "290", "Greenland",
+ "293", "Slovenia",
+ "294", "Macedonia",
+ "302", "Canada",
+ "310", "U.S.A.",
+ "340", "French West Indies",
+ "400", "Azerbaijan",
+ "404", "India",
+ "410", "Pakistan",
+ "413", "Sri Lanka",
+ "415", "Lebanon",
+ "416", "Jordan",
+ "417", "Syria",
+ "418", "Iraq",
+ "419", "Kuwait",
+ "420", "Saudi Arabia",
+ "422", "Oman",
+ "424", "United Arab Emirates",
+ "425", "Israel",
+ "426", "Bahrain",
+ "427", "Qatar",
+ "432", "Iran",
+ "434", "Uzbekistan",
+ "437", "Kyrgyz Republic",
+ "452", "Vietnam",
+ "454", "Hong Kong",
+ "455", "Macau",
+ "456", "Cambodia",
+ "457", "Lao",
+ "460", "China",
+ "466", "Taiwan",
+ "470", "Bangladesh",
+ "502", "Malaysia",
+ "505", "Australia",
+ "510", "Indonesia",
+ "515", "Philippines",
+ "520", "Thailand",
+ "525", "Singapore",
+ "528", "Brunei Darussalam",
+ "530", "New Zealand",
+ "542", "Fiji",
+ "546", "New Caledonia",
+ "547", "French Polynesia",
+ "602", "Egypt",
+ "603", "Algeria",
+ "604", "Morocco",
+ "605", "Tunisia",
+ "608", "Senegal",
+ "611", "Guinea",
+ "612", "Cote d'Ivoire",
+ "615", "Togo",
+ "617", "Mauritius",
+ "618", "Liberia",
+ "620", "Ghana",
+ "624", "Cameroon",
+ "625", "Cape Verde",
+ "633", "Seychelles",
+ "634", "Mozambique",
+ "634", "Sudan",
+ "635", "Rwanda",
+ "636", "Ethiopia",
+ "640", "Tanzania",
+ "641", "Uganda",
+ "645", "Zambia",
+ "646", "Madagascar",
+ "647", "Reunion",
+ "648", "Zimbabwe",
+ "649", "Namibia",
+ "650", "Malawi",
+ "651", "Lesotho",
+ "652", "Botswana",
+ "655", "South Africa",
+ "730", "Chile",
+ "734", "Venezuela",
+
+ NULL
+};
+
+unsigned char *GSM_Networks[] = {
+ "202 01", "Cosmote",
+ "202 05", "PANAFON",
+ "202 10", "TELESTET",
+ "204 04", "LIBERTEL",
+ "204 08", "KPN Telecom",
+ "204 12", "O2",
+ "204 16", "BEN",
+ "204 20", "Dutchtone NV",
+ "206 01", "PROXIMUS",
+ "206 10", "Mobistar",
+ "206 20", "Base",
+ "208 01", "ITINERIS",
+ "208 10", "SFR",
+ "208 20", "Bouygues Telecom",
+ "213 03", "MOBILAND",
+ "214 01", "Airtel GSM 900-Spain",
+ "214 03", "Retevision Movil",
+ "214 07", "MOVISTAR",
+ "216 01", "Pannon GSM",
+ "216 70", "Vodafone",
+ "216 30", "Westel 900",
+ "218 90", "GSMBIH",
+ "219 01", "CRONET",
+ "219 10", "VIP",
+ "220 01", "MOBTEL",
+ "220 02", "ProMonte GSM",
+ "220 03", "Telekom Srbije",
+ "222 01", "Telecom Italia Mobile",
+ "222 10", "OMNITEL",
+ "222 88", "Wind Telecomunicazioni SpA",
+ "226 01", "CONNEX GSM",
+ "226 10", "DIALOG",
+ "228 01", "NATEL International",
+ "228 02", "diAx Mobile AG",
+ "230 01", "T-Mobile CZ",
+ "230 02", "EuroTel",
+ "230 03", "Oskar",
+ "231 01", "Orange",
+ "231 02", "EuroTel GSM",
+ "232 01", "A1",
+ "232 03", "T-Mobile AT",
+ "232 05", "ONE",
+ "232 07", "tele.ring",
+ "234 10", "Cellnet",
+ "234 15", "Vodafone",
+ "234 30", "T-Mobile UK",
+ "234 33", "ORANGE",
+ "234 50", "Jersey Telecoms GSM",
+ "234 55", "Guernsey Telecoms GSM",
+ "234 58", "PRONTO GSM",
+ "238 01", "TDK-MOBIL",
+ "238 02", "SONOFON",
+ "238 20", "TELIA DK",
+ "238 30", "Mobilix",
+ "240 01", "Telia AB",
+ "240 07", "COMVIQ",
+ "240 08", "EUROPOLITAN",
+ "242 01", "Telenor Mobil",
+ "242 02", "NetCom GSM",
+ "244 03", "Telia City (Finland)",
+ "244 05", "Radiolinja",
+ "244 09", "Finnet",
+ "244 12", "DNA (FI2G)",
+ "244 14", "Alands Mobiltelefon",
+ "244 91", "Sonera",
+ "246 01", "OMNITEL",
+ "246 02", "Bite GSM",
+ "247 01", "LMT LV",
+ "247 02", "BALTCOM GSM",
+ "248 01", "EMT GSM",
+ "248 02", "Radiolinja Eesti AS",
+ "248 03", "Q GSM",
+ "250 01", "Mobile Telesystems",
+ "250 02", "North-West GSM",
+ "250 05", "Siberian Cellular Systems 900",
+ "250 07", "BM Telecom",
+ "250 10", "Don Telecom",
+ "250 12", "FECS-900",
+ "250 13", "Kuban GSM",
+ "250 39", "Uraltel",
+ "250 44", "North Caucasian GSM",
+ "250 99", "BeeLine",
+ "255 01", "UMC",
+ "255 02", "WellCOM",
+ "255 03", "Kyivstar",
+ "255 05", "Golden Telecom",
+ "259 01", "VOXTEL",
+ "260 01", "PLUS GSM",
+ "260 02", "ERA GSM",
+ "260 03", "IDEA Centertel",
+ "262 01", "T-Mobile D",
+ "262 02", "D2 PRIVAT",
+ "262 03", "E-Plus",
+ "262 07", "Interkom",
+ "266 01", "Gibtel GSM",
+ "268 01", "TELECEL",
+ "268 03", "OPTIMUS",
+ "268 06", "TMN",
+ "270 01", "LUXGSM",
+ "270 77", "TANGO",
+ "272 01", "EIRCELL-GSM",
+ "272 02", "Digifone",
+ "274 01", "Landssiminn GSM 900",
+ "274 02", "TAL hf",
+ "276 01", "AMC",
+ "278 01", "Vodafone Malta Limited",
+ "280 01", "CYTAGSM",
+ "282 01", "Geocell Limited",
+ "282 02", "Magti GSM",
+ "283 01", "ArmGSM",
+ "284 01", "M-TEL GSM BG",
+ "286 01", "Turkcell",
+ "286 02", "TELSIM GSM",
+ "288 01", "Faroese Telecom",
+ "290 01", "Tele Greenland",
+ "293 40", "SI.MOBIL d. d.",
+ "293 41", "MOBITEL",
+ "293 70", "SI VEGA 070",
+ "294 01", "MobiMak",
+ "302 37", "Microcell Connexions Inc",
+ "302 72", "Rogers AT&T",
+ "310 01", "Cellnet",
+ "310 02", "Sprint Spectrum",
+ "310 11", "Wireless 2000 Telephone Co.",
+ "310 15", "BellSouth Mobility DCS",
+ "310 16", "T-Mobile",
+ "310 17", "Pac Bell",
+ "310 20", "T-Mobile",
+ "310 21", "T-Mobile",
+ "310 22", "T-Mobile",
+ "310 23", "T-Mobile",
+ "310 24", "T-Mobile",
+ "310 25", "T-Mobile",
+ "310 26", "T-Mobile",
+ "310 27", "T-Mobile",
+ "310 31", "T-Mobile",
+ "310 38", "AT&T Wireless",
+ "310 58", "T-Mobile",
+ "310 66", "T-Mobile",
+ "310 77", "Iowa Wireless Services LP",
+ "310 80", "T-Mobile",
+ "340 01", "AMERIS",
+ "400 01", "AZERCELL GSM",
+ "400 02", "Bakcell GSM 2000",
+ "404 07", "TATA Cellular",
+ "404 10", "AirTel",
+ "404 11", "Essar Cellphone",
+ "404 12", "Escotel",
+ "404 14", "Modicom",
+ "404 15", "Essar Cellphone",
+ "404 20", "Max Touch",
+ "404 21", "BPL - Mobile",
+ "404 27", "BPL USWEST Cellular",
+ "404 30", "Command",
+ "404 40", "SkyCell",
+ "404 41", "RPG Cellular",
+ "404 42", "AIRCEL",
+ "410 01", "Mobilink",
+ "413 02", "DIALOG GSM",
+ "415 01", "CELLIS",
+ "415 03", "LIBANCELL",
+ "416 01", "Fastlink",
+ "417 09", "MOBILE SYRIA",
+ "419 02", "MTCNet",
+ "420 01", "Al Jawwal",
+ "420 07", "E.A.E",
+ "422 02", "GTO",
+ "424 02", "UAE-ETISALAT",
+ "425 01", "Partner Communications Company Ltd",
+ "425 02", "Cellcom Israel Ltd",
+ "426 01", "BHR MOBILE PLUS",
+ "427 01", "QATARNET",
+ "432 11", "TCI",
+ "434 04", "Daewoo Unitel",
+ "434 05", "Coscom",
+ "437 01", "Bitel",
+ "454 00", "TCSL GSM",
+ "454 04", "HKGHT",
+ "454 06", "SMARTONE GSM",
+ "454 10", "New World PCS",
+ "454 12", "PEOPLES",
+ "454 16", "SUNDAY",
+ "455 01", "TELEMOVEL+ GSM900-Macau",
+ "456 01", "MobiTel",
+ "456 02", "SAMART-GSM",
+ "457 01", "Lao Shinawatra Telecom",
+ "460 00", "China Telecom GSM",
+ "460 01", "CU-GSM",
+ "466 01", "Far EasTone Telecoms 900",
+ "466 06", "TUNTEX GSM 1800",
+ "466 88", "KG Telecom",
+ "466 92", "Chunghwa GSM",
+ "466 93", "MobiTai",
+ "466 97", "TWNGSM",
+ "466 99", "TransAsia",
+ "470 01", "GrameenPhone Ltd",
+ "470 19", "Mobile 2000",
+ "502 12", "Maxis Mobile",
+ "502 13", "TM Touch",
+ "502 16", "DiGi 1800",
+ "502 17", "ADAM",
+ "502 19", "CELCOM",
+ "505 01", "MobileNet",
+ "505 02", "OPTUS",
+ "505 03", "VODAFONE",
+ "505 08", "One.Tel",
+ "510 01", "SATELINDO",
+ "510 08", "LIPPO TELECOM",
+ "510 10", "TELKOMSEL",
+ "510 11", "Excelcom",
+ "510 21", "INDOSAT",
+ "515 01", "ISLACOM",
+ "515 02", "Globe Telecom",
+ "520 01", "AIS GSM",
+ "520 10", "WCS",
+ "520 18", "Worldphone 1800",
+ "520 23", "HELLO",
+ "525 01", "SingTel Mobile",
+ "525 02", "ST-PCN",
+ "525 03", "MOBILEONE",
+ "528 11", "DSTCom",
+ "530 01", "Vodafone New Zealand Limited",
+ "542 01", "Vodafone",
+ "546 01", "Mobilis",
+ "547 20", "VINI",
+ "602 01", "MobiNil",
+ "602 02", "Tunicell",
+ "603 01", "ALGERIAN MOBILE NETWORK",
+ "604 01", "I A M",
+ "608 01", "ALIZE",
+ "611 02", "Lagui",
+ "612 03", "IVOIRIS",
+ "612 05", "Telecel",
+ "615 01", "TOGO CELL",
+ "617 01", "Cellplus Mobile Comms",
+ "618 01", "Omega",
+ "620 01", "SPACEFON",
+ "625 01", "CVMOVEL",
+ "633 01", "Seychelles Cellular Services",
+ "633 10", "AIRTEL",
+ "634 01", "MobiTel",
+ "635 10", "Rwandacell",
+ "636 01", "ETMTN",
+ "640 01", "TRITEL",
+ "641 10", "MTN-Uganda",
+ "642 02", "ANTARIS",
+ "643 01", "T.D.M GSM 900",
+ "645 01", "ZAMCELL",
+ "646 01", "Madacom",
+ "646 03", "Sacel Madagascar S.A.",
+ "647 10", "SRR",
+ "648 01", "NET*ONE",
+ "648 03", "Telecel",
+ "649 01", "MTC",
+ "650 01", "Callpoint 900",
+ "651 01", "Vodacom Lesotho (Pty) Ltd",
+ "655 01", "Vodacom",
+ "655 10", "MTN",
+ "680 38", "NPI Wireless",
+ "730 01", "Entel Telefonia Movi",
+ "730 10", "Entel PCS",
+ "734 01", "Infonet",
+
+ NULL
+};
+
+char *GSM_GetNetworkName(char *NetworkCode)
+{
+ int i = 0;
+ static char retval[200];
+
+ EncodeUnicode(retval,"unknown",7);
+ while (GSM_Networks[i*2] != NULL) {
+ if (!strncmp(GSM_Networks[i*2],NetworkCode,6)) {
+ EncodeUnicode(retval, GSM_Networks[i*2+1], strlen(GSM_Networks[i*2+1]));
+ break;
+ }
+ i++;
+ }
+ return retval;
+}
+
+char *GSM_GetCountryName(char *CountryCode)
+{
+ int i = 0;
+ static char retval[200];
+
+ EncodeUnicode(retval,"unknown",7);
+ while (GSM_Countries[i*2] != NULL) {
+ if (!strncmp(GSM_Countries[i*2],CountryCode,3)) {
+ EncodeUnicode(retval, GSM_Countries[i*2+1], strlen(GSM_Countries[i*2+1]));
+ break;
+ }
+ i++;
+ }
+ return retval;
+}
+
+void NOKIA_EncodeNetworkCode(unsigned char* buffer, unsigned char* output)
+{
+ EncodeBCD(buffer, output, 6, false);
+ buffer[1] = buffer[1] | 0xf0;
+}
+
+void NOKIA_DecodeNetworkCode(unsigned char* buffer, unsigned char* output)
+{
+ DecodeBCD(output, buffer, 3);
+ output[6] = output[5];
+ output[5] = output[4];
+ output[4] = output[3];
+ output[3] = ' ';
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmnet.h b/gammu/emb/common/service/gsmnet.h
new file mode 100644
index 0000000..aaea22f
--- a/dev/null
+++ b/gammu/emb/common/service/gsmnet.h
@@ -0,0 +1,98 @@
+/* (c) 2002-2003 by Marcin Wiacek & Michal Cihar */
+
+#ifndef __gsm_net_h
+#define __gsm_net_h
+
+/**
+ * Find network name from given network code.
+ */
+char *GSM_GetNetworkName(char *NetworkCode);
+
+/**
+ * Find country name from given country code.
+ */
+char *GSM_GetCountryName(char *CountryCode);
+
+/**
+ * Status of network logging
+ */
+typedef enum {
+ /**
+ * Home network for used SIM card.
+ */
+ GSM_HomeNetwork = 1,
+ /**
+ * No network available for used SIM card.
+ */
+ GSM_NoNetwork,
+ /**
+ * SIM card uses roaming.
+ */
+ GSM_RoamingNetwork,
+ /**
+ * Network registration denied - card blocked or expired or disabled.
+ */
+ GSM_RegistrationDenied,
+ /**
+ * Unknown network status.
+ */
+ GSM_NetworkStatusUnknown,
+ /**
+ * Network explicitely requested by user.
+ */
+ GSM_RequestingNetwork
+} GSM_NetworkInfo_State;
+
+/**
+ * Structure for getting the current network info.
+ */
+typedef struct {
+ /**
+ * Cell ID (CID)
+ */
+ unsigned char CID[10];
+ /**
+ * GSM network code.
+ */
+ char NetworkCode[10];
+ /**
+ * Status of network logging. If phone is not logged into any network,
+ * some values are not filled
+ */
+ GSM_NetworkInfo_State State;
+ /**
+ * LAC (Local Area Code).
+ */
+ unsigned char LAC[10];
+ /**
+ * Name of current network like returned from phone (or empty).
+ */
+ unsigned char NetworkName[15*2];
+} GSM_NetworkInfo;
+
+void NOKIA_EncodeNetworkCode (unsigned char* buffer, unsigned char* output);
+void NOKIA_DecodeNetworkCode (unsigned char* buffer, unsigned char* output);
+
+/**
+ * Information about signal quality, all these should be -1 when unknown.
+ */
+typedef struct {
+ /*
+ * Signal strength in dBm
+ */
+ int SignalStrength;
+ /**
+ * Signal strength in percent.
+ */
+ int SignalPercent;
+ /**
+ * Bit error rate in percent.
+ */
+ int BitErrorRate;
+} GSM_SignalQuality;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmpbk.c b/gammu/emb/common/service/gsmpbk.c
new file mode 100644
index 0000000..05e5cb9
--- a/dev/null
+++ b/gammu/emb/common/service/gsmpbk.c
@@ -0,0 +1,370 @@
+/* (c) 2001-2003 by Marcin Wiacek,... */
+
+#include <string.h>
+
+#include "../misc/coding/coding.h"
+#include "gsmpbk.h"
+#include "gsmmisc.h"
+
+unsigned char *GSM_PhonebookGetEntryName (GSM_MemoryEntry *entry)
+{
+ /* We possibly store here "LastName, FirstName" so allocate enough memory */
+ static char dest[(GSM_PHONEBOOK_TEXT_LENGTH*2+2+1)*2];
+ static char split[] = { '\0', ',', '\0', ' ', '\0', '\0'};
+ int i;
+ int first = -1, last = -1, name = -1;
+ int len = 0;
+
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case PBK_Text_LastName:
+ last = i;
+ break;
+ case PBK_Text_FirstName:
+ first = i;
+ break;
+ case PBK_Text_Name:
+ name = i;
+ break;
+ default:
+ break;
+ }
+ }
+
+ if (name != -1) {
+ CopyUnicodeString(dest, entry->Entries[name].Text);
+ } else {
+ if (last != -1 && first != -1) {
+ len = UnicodeLength(entry->Entries[last].Text);
+ CopyUnicodeString(dest, entry->Entries[last].Text);
+ CopyUnicodeString(dest + 2*len, split);
+ CopyUnicodeString(dest + 2*len + 4, entry->Entries[first].Text);
+ } else if (last != -1) {
+ CopyUnicodeString(dest, entry->Entries[last].Text);
+ } else if (first != -1) {
+ CopyUnicodeString(dest, entry->Entries[first].Text);
+ } else {
+ return NULL;
+ }
+ }
+
+ return dest;
+}
+
+void GSM_PhonebookFindDefaultNameNumberGroup(GSM_MemoryEntry *entry, int *Name, int *Number, int *Group)
+{
+ int i;
+
+ *Name = -1;
+ *Number = -1;
+ *Group = -1;
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case PBK_Number_General : if (*Number == -1) *Number = i; break;
+ case PBK_Text_Name : if (*Name == -1) *Name = i; break;
+ case PBK_Caller_Group : if (*Group == -1) *Group = i; break;
+ default : break;
+ }
+ }
+ if ((*Number) == -1) {
+ for (i = 0; i < entry->EntriesNum; i++) {
+ switch (entry->Entries[i].EntryType) {
+ case PBK_Number_Mobile:
+ case PBK_Number_Work:
+ case PBK_Number_Fax:
+ case PBK_Number_Home:
+ case PBK_Number_Pager:
+ case PBK_Number_Other:
+ *Number = i;
+ break;
+ default:
+ break;
+ }
+ if (*Number != -1) break;
+ }
+ }
+ if ((*Name) == -1) {
+ for (i = 0; i < entry->EntriesNum; i++) {
+ if (entry->Entries[i].EntryType != PBK_Text_LastName) continue;
+ *Name = i;
+ break;
+ }
+ }
+ if ((*Name) == -1) {
+ for (i = 0; i < entry->EntriesNum; i++) {
+ if (entry->Entries[i].EntryType != PBK_Text_FirstName) continue;
+ *Name = i;
+ break;
+ }
+ }
+}
+
+void GSM_EncodeVCARD(char *Buffer, int *Length, GSM_MemoryEntry *pbk, bool header, GSM_VCardVersion Version)
+{
+ int Name, Number, Group, i;
+ bool ignore;
+
+ GSM_PhonebookFindDefaultNameNumberGroup(pbk, &Name, &Number, &Group);
+
+ if (Version == Nokia_VCard10) {
+ if (header) *Length+=sprintf(Buffer+(*Length),"BEGIN:VCARD%c%c",13,10);
+ if (Name != -1) {
+ *Length+=sprintf(Buffer+(*Length),"N:%s%c%c",DecodeUnicodeString(pbk->Entries[Name].Text),13,10);
+ }
+ if (Number != -1) {
+ *Length +=sprintf(Buffer+(*Length),"TEL:%s%c%c",DecodeUnicodeString(pbk->Entries[Number].Text),13,10);
+ }
+ if (header) *Length+=sprintf(Buffer+(*Length),"END:VCARD%c%c",13,10);
+ } else if (Version == Nokia_VCard21) {
+ if (header) *Length+=sprintf(Buffer+(*Length),"BEGIN:VCARD%c%cVERSION:2.1%c%c",13,10,13,10);
+ if (Name != -1) {
+ SaveVCALText(Buffer, Length, pbk->Entries[Name].Text, "N");
+ }
+ for (i=0; i < pbk->EntriesNum; i++) {
+ if (i != Name) {
+ ignore = false;
+ switch(pbk->Entries[i].EntryType) {
+ case PBK_Text_Name :
+ case PBK_Date :
+ case PBK_Caller_Group :
+ ignore = true;
+ break;
+ case PBK_Number_General :
+ *Length+=sprintf(Buffer+(*Length),"TEL");
+ if (Number == i) (*Length)+=sprintf(Buffer+(*Length),";PREF");
+ break;
+ case PBK_Number_Mobile :
+ *Length+=sprintf(Buffer+(*Length),"TEL");
+ if (Number == i) (*Length)+=sprintf(Buffer+(*Length),";PREF");
+ *Length+=sprintf(Buffer+(*Length),";CELL");
+ break;
+ case PBK_Number_Work :
+ *Length+=sprintf(Buffer+(*Length),"TEL");
+ if (Number == i) (*Length)+=sprintf(Buffer+(*Length),";PREF");
+ *Length+=sprintf(Buffer+(*Length),";WORK;VOICE");
+ break;
+ case PBK_Number_Fax :
+ *Length+=sprintf(Buffer+(*Length),"TEL");
+ if (Number == i) (*Length)+=sprintf(Buffer+(*Length),";PREF");
+ *Length+=sprintf(Buffer+(*Length),";FAX");
+ break;
+ case PBK_Number_Home :
+ *Length+=sprintf(Buffer+(*Length),"TEL");
+ if (Number == i) (*Length)+=sprintf(Buffer+(*Length),";PREF");
+ *Length+=sprintf(Buffer+(*Length),";HOME;VOICE");
+ break;
+ case PBK_Text_Note :
+ *Length+=sprintf(Buffer+(*Length),"NOTE");
+ break;
+ case PBK_Text_Postal :
+ /* Don't ask why. Nokia phones save postal address
+ * double - once like LABEL, second like ADR
+ */
+ SaveVCALText(Buffer, Length, pbk->Entries[i].Text, "LABEL");
+ *Length+=sprintf(Buffer+(*Length),"ADR");
+ break;
+ case PBK_Text_Email :
+ case PBK_Text_Email2 :
+ *Length+=sprintf(Buffer+(*Length),"EMAIL");
+ break;
+ case PBK_Text_URL :
+ *Length+=sprintf(Buffer+(*Length),"URL");
+ break;
+ default :
+ ignore = true;
+ break;
+ }
+ if (!ignore) {
+ SaveVCALText(Buffer, Length, pbk->Entries[i].Text, "");
+ }
+ }
+ }
+ if (header) *Length+=sprintf(Buffer+(*Length),"END:VCARD%c%c",13,10);
+ }
+}
+
+GSM_Error GSM_DecodeVCARD(unsigned char *Buffer, int *Pos, GSM_MemoryEntry *Pbk, GSM_VCardVersion Version)
+{
+ unsigned char Line[2000],Buff[2000];
+ int Level = 0;
+
+ Buff[0] = 0;
+ Pbk->EntriesNum = 0;
+
+ while (1) {
+ MyGetLine(Buffer, Pos, Line, strlen(Buffer));
+ if (strlen(Line) == 0) break;
+ switch (Level) {
+ case 0:
+ if (strstr(Line,"BEGIN:VCARD")) Level = 1;
+ break;
+ case 1:
+ if (strstr(Line,"END:VCARD")) {
+ if (Pbk->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+ }
+ if (ReadVCALText(Line, "N", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Name;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "TEL", Buff) ||
+ ReadVCALText(Line, "TEL;VOICE", Buff) ||
+ ReadVCALText(Line, "TEL;PREF", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;VOICE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_General;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "TEL;CELL", Buff) ||
+ ReadVCALText(Line, "TEL;CELL;VOICE", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;CELL", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;CELL;VOICE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Mobile;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "TEL;WORK", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;WORK", Buff) ||
+ ReadVCALText(Line, "TEL;WORK;VOICE", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;WORK;VOICE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Work;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "TEL;FAX", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;FAX", Buff) ||
+ ReadVCALText(Line, "TEL;FAX;VOICE", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;FAX;VOICE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Fax;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "TEL;HOME", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;HOME", Buff) ||
+ ReadVCALText(Line, "TEL;HOME;VOICE", Buff) ||
+ ReadVCALText(Line, "TEL;PREF;HOME;VOICE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Number_Home;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "NOTE", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Note;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "ADR", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Postal;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "EMAIL", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_Email;
+ Pbk->EntriesNum++;
+ }
+ if (ReadVCALText(Line, "URL", Buff)) {
+ CopyUnicodeString(Pbk->Entries[Pbk->EntriesNum].Text,Buff);
+ Pbk->Entries[Pbk->EntriesNum].EntryType = PBK_Text_URL;
+ Pbk->EntriesNum++;
+ }
+ break;
+ }
+ }
+
+ if (Pbk->EntriesNum == 0) return ERR_EMPTY;
+ return ERR_NONE;
+}
+
+/* -------------- OLD functions (c) by Timo Teras -------------------------- */
+
+#ifndef ENABLE_LGPL
+
+static void ParseVCardLine(char **pos, char *Name, char *Parameters, char *Value)
+{
+ int i;
+
+ Name[0] = Parameters[0] = Value[0] = 0;
+
+ if (**pos == 0) return;
+
+ for (i=0; **pos && **pos != ':' && **pos != ';'; i++, (*pos)++) Name[i] = **pos;
+ Name[i] = 0;
+
+ //dbgprintf("ParseVCardLine: name tag = '%s'\n", Name);
+ if (**pos == ';') {
+ (*pos)++;
+ for (i=0; **pos && **pos != ':'; i++, (*pos)++) Parameters[i] = **pos;
+ Parameters[i] = ';';
+ Parameters[i+1] = 0;
+ //dbgprintf("ParseVCardLine: parameter tag = '%s'\n", Parameters);
+ }
+
+ if (**pos != 0) (*pos)++;
+
+ i=0;
+ while (**pos) {
+ if ((*pos)[0] == '\x0d' && (*pos)[1] == '\x0a') {
+ (*pos) += 2;
+ if (**pos != '\t' && **pos != ' ') break;
+ while (**pos == '\t' || **pos == ' ') (*pos)++;
+ continue;
+ }
+ Value[i++] = **pos;
+ (*pos)++;
+ }
+ Value[i] = 0;
+
+ //dbgprintf("ParseVCardLine: value tag = '%s'\n", Value);
+}
+
+void DecodeVCARD21Text(char *VCard, GSM_MemoryEntry *pbk)
+{
+ char *pos = VCard;
+ char Name[32], Parameters[256], Value[1024];
+
+ dbgprintf("Parsing VCard:\n%s\n", VCard);
+
+ ParseVCardLine(&pos, Name, Parameters, Value);
+ if (!mystrncasecmp(Name, "BEGIN", 0) || !mystrncasecmp(Value, "VCARD", 0)) {
+ dbgprintf("No valid VCARD signature\n");
+ return;
+ }
+
+ while (1) {
+ GSM_SubMemoryEntry *pbe = &pbk->Entries[pbk->EntriesNum];
+
+ ParseVCardLine(&pos, Name, Parameters, Value);
+ if (Name[0] == 0x00 ||
+ (mystrncasecmp(Name, "END", 0) && mystrncasecmp(Value, "VCARD", 0)))
+ return;
+
+ if (mystrncasecmp(Name, "N", 0)) {
+ //FIXME: Name is tagged field which should be parsed
+ pbe->EntryType = PBK_Text_Name;
+ EncodeUnicode(pbe->Text, Value, strlen(Value));
+ pbk->EntriesNum++;
+ } else if (mystrncasecmp(Name, "EMAIL", 0)) {
+ pbe->EntryType = PBK_Text_Email;
+ EncodeUnicode(pbe->Text, Value, strlen(Value));
+ pbk->EntriesNum++;
+ } else if (mystrncasecmp(Name, "TEL", 0)) {
+ if (strstr(Parameters, "WORK;"))
+ pbe->EntryType = PBK_Number_Work;
+ else if (strstr(Name, "HOME;"))
+ pbe->EntryType = PBK_Number_Home;
+ else if (strstr(Name, "FAX;"))
+ pbe->EntryType = PBK_Number_Fax;
+ else pbe->EntryType = PBK_Number_General;
+
+ EncodeUnicode(pbe->Text, Value, strlen(Value));
+ pbk->EntriesNum++;
+ }
+ }
+}
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmpbk.h b/gammu/emb/common/service/gsmpbk.h
new file mode 100644
index 0000000..e556793
--- a/dev/null
+++ b/gammu/emb/common/service/gsmpbk.h
@@ -0,0 +1,270 @@
+/* (c) 2001-2004 by Marcin Wiacek and Michal Cihar */
+
+#ifndef __gsm_pbk_h
+#define __gsm_pbk_h
+
+#include <stdlib.h>
+
+#include "../gsmcomon.h"
+#include "gsmmisc.h"
+
+/**
+ * Structure contains info about number of used/free entries in phonebook
+ * memory
+ */
+typedef struct {
+ /**
+ * Number of used entries
+ */
+ int MemoryUsed;
+ /**
+ * Memory type
+ */
+ GSM_MemoryType MemoryType;
+ /**
+ * Number of free entries
+ */
+ int MemoryFree;
+} GSM_MemoryStatus;
+
+/**
+ * Type of specific phonebook entry. In parenthesis is specified in which
+ * member of @ref GSM_SubMemoryEntry value is stored.
+ */
+typedef enum {
+ /**
+ * General number. (Text)
+ */
+ PBK_Number_General = 1,
+ /**
+ * Mobile number. (Text)
+ */
+ PBK_Number_Mobile,
+ /**
+ * Work number. (Text)
+ */
+ PBK_Number_Work,
+ /**
+ * Fax number. (Text)
+ */
+ PBK_Number_Fax,
+ /**
+ * Home number. (Text)
+ */
+ PBK_Number_Home,
+ /**
+ * Pager number. (Text)
+ */
+ PBK_Number_Pager,
+ /**
+ * Other number. (Text)
+ */
+ PBK_Number_Other,
+ /**
+ * Note. (Text)
+ */
+ PBK_Text_Note,
+ /**
+ * Complete postal address. (Text)
+ */
+ PBK_Text_Postal,
+ /**
+ * Email. (Text)
+ */
+ PBK_Text_Email,
+ /**
+ * Second email. (Text)
+ */
+ PBK_Text_Email2,
+ /**
+ * URL (Text)
+ */
+ PBK_Text_URL,
+ /**
+ * Date and time. FIXME: describe better (Date)
+ */
+ PBK_Date,
+ /**
+ * Caller group. (Text)
+ */
+ PBK_Caller_Group,
+ /**
+ * Name (Text)
+ */
+ PBK_Text_Name,
+ /**
+ * Last name. (Text)
+ */
+ PBK_Text_LastName,
+ /**
+ * First name. (Text)
+ */
+ PBK_Text_FirstName,
+ /**
+ * Company. (Text)
+ */
+ PBK_Text_Company,
+ /**
+ * Job title. (Text)
+ */
+ PBK_Text_JobTitle,
+ /**
+ * Category. (Number)
+ */
+ PBK_Category,
+ /**
+ * Whether entry is private. (Number)
+ */
+ PBK_Private,
+ /**
+ * Street address. (Text)
+ */
+ PBK_Text_StreetAddress,
+ /**
+ * City. (Text)
+ */
+ PBK_Text_City,
+ /**
+ * State. (Text)
+ */
+ PBK_Text_State,
+ /**
+ * Zip code. (Text)
+ */
+ PBK_Text_Zip,
+ /**
+ * Country. (Text)
+ */
+ PBK_Text_Country,
+ /**
+ * Custom information 1. (Text)
+ */
+ PBK_Text_Custom1,
+ /**
+ * Custom information 2. (Text)
+ */
+ PBK_Text_Custom2,
+ /**
+ * Custom information 3. (Text)
+ */
+ PBK_Text_Custom3,
+ /**
+ * Custom information 4. (Text)
+ */
+ PBK_Text_Custom4,
+ /**
+ * Ringtone ID. (Number)
+ */
+ PBK_RingtoneID,
+ /**
+ * Ringtone ID in phone filesystem. (Number)
+ */
+ PBK_RingtoneFileSystemID,
+ /**
+ * Picture ID. (Number)
+ */
+ PBK_PictureID,
+ PBK_SMSListID,
+ /**
+ * User ID. (Text)
+ */
+ PBK_Text_UserID
+} GSM_EntryType;
+
+#define GSM_PHONEBOOK_TEXT_LENGTH 200
+#define GSM_PHONEBOOK_ENTRIES 26
+
+/**
+ * One value of phonebook memory entry.
+ */
+typedef struct {
+ /**
+ * Type of entry.
+ */
+ GSM_EntryType EntryType;
+ /**
+ * Text of entry (if applicable, see @ref GSM_EntryType).
+ */
+ unsigned char Text[(GSM_PHONEBOOK_TEXT_LENGTH+1)*2];
+ /**
+ * Text of entry (if applicable, see @ref GSM_EntryType).
+ */
+ GSM_DateTime Date;
+ /**
+ * Number of entry (if applicable, see @ref GSM_EntryType).
+ */
+ int Number;
+ /**
+ * Voice dialling tag.
+ */
+ int VoiceTag;
+ int SMSList[20];
+} GSM_SubMemoryEntry;
+
+/**
+ * Structure for saving phonebook entries
+ */
+typedef struct {
+ /**
+ * Used memory for phonebook entry
+ */
+ GSM_MemoryType MemoryType;
+ /**
+ * Used location for phonebook entry
+ */
+ int Location;
+ /**
+ * Number of SubEntries in Entries table.
+ */
+ int EntriesNum;
+ /**
+ * Values of SubEntries.
+ */
+ GSM_SubMemoryEntry Entries[GSM_PHONEBOOK_ENTRIES];
+} GSM_MemoryEntry;
+
+typedef enum {
+ Nokia_VCard10 = 1,
+ Nokia_VCard21,
+ SonyEricsson_VCard10,
+ SonyEricsson_VCard21
+} GSM_VCardVersion;
+
+void GSM_PhonebookFindDefaultNameNumberGroup(GSM_MemoryEntry *entry, int *Name, int *Number, int *Group);
+unsigned char *GSM_PhonebookGetEntryName (GSM_MemoryEntry *entry);
+
+void GSM_EncodeVCARD(char *Buffer, int *Length, GSM_MemoryEntry *pbk, bool header, GSM_VCardVersion Version);
+GSM_Error GSM_DecodeVCARD(unsigned char *Buffer, int *Pos, GSM_MemoryEntry *Pbk, GSM_VCardVersion Version);
+
+#ifndef ENABLE_LGPL
+/* (c) by Timo Teras */
+void DecodeVCARD21Text(char *VCard, GSM_MemoryEntry *pbk);
+#endif
+
+/**
+ * Structure for saving speed dials
+ */
+typedef struct {
+ /**
+ * Number of speed dial: 2,3..,8,9
+ */
+ int Location;
+ /**
+ * ID of phone number used in phonebook entry
+ */
+ int MemoryNumberID;
+ /**
+ * Memory, where is saved used phonebook entry
+ */
+ GSM_MemoryType MemoryType;
+ /**
+ * Location in memory, where is saved used phonebook entry
+ */
+ int MemoryLocation;
+} GSM_SpeedDial;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmprof.h b/gammu/emb/common/service/gsmprof.h
new file mode 100644
index 0000000..2e3d087
--- a/dev/null
+++ b/gammu/emb/common/service/gsmprof.h
@@ -0,0 +1,104 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef __gsm_prof_h
+#define __gsm_prof_h
+
+typedef enum {
+ PROFILE_KEYPAD_LEVEL1 = 1,
+ PROFILE_KEYPAD_LEVEL2,
+ PROFILE_KEYPAD_LEVEL3,
+ PROFILE_KEYPAD_OFF,
+ PROFILE_CALLALERT_RINGING,
+ PROFILE_CALLALERT_BEEPONCE,
+ PROFILE_CALLALERT_OFF,
+ PROFILE_CALLALERT_RINGONCE,
+ PROFILE_CALLALERT_ASCENDING,
+ PROFILE_CALLALERT_CALLERGROUPS,
+ PROFILE_VOLUME_LEVEL1,
+ PROFILE_VOLUME_LEVEL2,
+ PROFILE_VOLUME_LEVEL3,
+ PROFILE_VOLUME_LEVEL4,
+ PROFILE_VOLUME_LEVEL5,
+ PROFILE_MESSAGE_NOTONE,
+ PROFILE_MESSAGE_STANDARD,
+ PROFILE_MESSAGE_SPECIAL,
+ PROFILE_MESSAGE_BEEPONCE,
+ PROFILE_MESSAGE_ASCENDING,
+ PROFILE_MESSAGE_PERSONAL,
+ PROFILE_VIBRATION_OFF,
+ PROFILE_VIBRATION_ON,
+ PROFILE_VIBRATION_FIRST,
+ PROFILE_WARNING_ON,
+ PROFILE_WARNING_OFF,
+ PROFILE_AUTOANSWER_ON,
+ PROFILE_AUTOANSWER_OFF,
+ PROFILE_LIGHTS_OFF,
+ PROFILE_LIGHTS_AUTO,
+ PROFILE_SAVER_ON,
+ PROFILE_SAVER_OFF,
+ PROFILE_SAVER_TIMEOUT_5SEC,
+ PROFILE_SAVER_TIMEOUT_20SEC,
+ PROFILE_SAVER_TIMEOUT_1MIN,
+ PROFILE_SAVER_TIMEOUT_2MIN,
+ PROFILE_SAVER_TIMEOUT_5MIN,
+ PROFILE_SAVER_TIMEOUT_10MIN
+} GSM_Profile_Feat_Value;
+
+typedef enum {
+ Profile_KeypadTone = 1,
+ Profile_CallAlert,
+ Profile_RingtoneVolume,
+ Profile_MessageTone,
+ Profile_Vibration,
+ Profile_WarningTone,
+ Profile_AutoAnswer,
+ Profile_Lights,
+ Profile_ScreenSaverTime,
+ Profile_ScreenSaver,
+
+ Profile_ScreenSaverNumber,
+ Profile_RingtoneID,
+ Profile_MessageToneID,
+ Profile_CallerGroups
+} GSM_Profile_Feat_ID;
+
+/**
+ * It contains phone profiles
+ */
+typedef struct {
+ bool Active;
+
+ /**
+ * Profile number
+ */
+ int Location;
+ /**
+ * Profile name
+ */
+ char Name[40*2];
+ /**
+ * Is it default name for profile ?
+ */
+ bool DefaultName;
+ bool HeadSetProfile;
+ bool CarKitProfile;
+
+ int FeaturesNumber;
+ GSM_Profile_Feat_Value FeatureValue[15];
+ GSM_Profile_Feat_ID FeatureID[15];
+
+ bool CallerGroups[5];
+} GSM_Profile;
+
+typedef struct {
+ GSM_Profile_Feat_ID ID;
+ GSM_Profile_Feat_Value Value;
+ unsigned char PhoneID;
+ unsigned char PhoneValue;
+} GSM_Profile_PhoneTableValue;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmring.c b/gammu/emb/common/service/gsmring.c
new file mode 100644
index 0000000..5a1ff87
--- a/dev/null
+++ b/gammu/emb/common/service/gsmring.c
@@ -0,0 +1,1600 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* Based on some work from Ralf Thelen (7110 ringtones),
+ * Gnokii (RTTL and SM) and others
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include <math.h>
+#ifdef WIN32
+# include <windows.h>
+#endif
+
+#include "../gsmcomon.h"
+#include "../misc/coding/coding.h"
+#include "../gsmstate.h"
+#include "gsmring.h"
+#include "sms/gsmsms.h"
+
+int GSM_RingNoteGetFrequency(GSM_RingNote Note)
+{
+ double freq=0;
+
+ /* Values according to the software from http://iki.fi/too/sw/xring/
+ * generated with:
+ * perl -e 'print int(4400 * (2 **($_/12)) + .5)/10, "\n" for(3..14)'
+ */
+ switch (Note.Note) {
+ case Note_C : freq = 523.3; break;
+ case Note_Cis: freq = 554.4; break;
+ case Note_D : freq = 587.3; break;
+ case Note_Dis: freq = 622.3; break;
+ case Note_E : freq = 659.3; break;
+ case Note_F : freq = 698.5; break;
+ case Note_Fis: freq = 740; break;
+ case Note_G : freq = 784; break;
+ case Note_Gis: freq = 830.6; break;
+ case Note_A : freq = 880; break;
+ case Note_Ais: freq = 932.3; break;
+ case Note_H : freq = 987.8; break;
+ case Note_Pause: break;
+ }
+ switch (Note.Scale) {
+ case Scale_440 : freq = freq / 2; break;
+ case Scale_880 : break;
+ case Scale_1760: freq = freq * 2; break;
+ case Scale_3520: freq = freq * 4; break;
+ default : break;
+ }
+ return (int)freq;
+}
+
+int GSM_RingNoteGetFullDuration(GSM_RingNote Note)
+{
+ int duration = 1;
+
+ switch (Note.Duration) {
+ case Duration_Full : duration = 128; break;
+ case Duration_1_2 : duration = 64; break;
+ case Duration_1_4 : duration = 32; break;
+ case Duration_1_8 : duration = 16; break;
+ case Duration_1_16 : duration = 8; break;
+ case Duration_1_32 : duration = 4; break;
+ }
+ switch (Note.DurationSpec) {
+ case NoSpecialDuration : break;
+ case DottedNote : duration = duration * 3/2; break;
+ case DoubleDottedNote : duration = duration * 9/4; break;
+ case Length_2_3 : duration = duration * 2/3; break;
+ }
+ return duration;
+}
+
+#ifndef PI
+# define PI 3.141592654
+#endif
+
+#define WAV_SAMPLE_RATE 44100
+
+GSM_Error savewav(FILE *file, GSM_Ringtone *ringtone)
+{
+ unsigned char WAV_Header[] = {
+ 'R','I','F','F',
+ 0x00,0x00,0x00,0x00, /* Length */
+ 'W','A','V','E'};
+ unsigned char FMT_Header[] = {'f','m','t',' ',
+ 0x10,0x00,0x00,0x00,0x01,0x00,0x01,0x00,0x44,0xac,
+ 0x00,0x00,0x88,0x58,0x01,0x00,0x02,0x00,0x10,0x00};
+ unsigned char DATA_Header[] = {
+ 'd','a','t','a',
+ 0x00,0x00,0x00,0x00}; /* Length */
+ short DATA_Buffer[60000];
+ long wavfilesize;
+ GSM_RingNote *Note;
+ long i,j,length=0;
+ double phase=0,phase_step;
+
+ fwrite(&WAV_Header, 1, sizeof(WAV_Header), file);
+ fwrite(&FMT_Header, 1, sizeof(FMT_Header), file);
+ fwrite(&DATA_Header, 1, sizeof(DATA_Header), file);
+
+ for (i=0;i<ringtone->NoteTone.NrCommands;i++) {
+ if (ringtone->NoteTone.Commands[i].Type == RING_Note) {
+ Note = &ringtone->NoteTone.Commands[i].Note;
+ phase_step = GSM_RingNoteGetFrequency(*Note)*WAV_SAMPLE_RATE*1.5;
+ for (j=0;j<((long)(GSM_RingNoteGetFullDuration(*Note)*WAV_SAMPLE_RATE/70));j++) {
+ DATA_Buffer[j] = ((int)(sin(phase*PI)*50000));
+ phase = phase + phase_step;
+ length++;
+ }
+ fwrite(&DATA_Buffer,sizeof(short),j,file);
+ }
+ }
+
+ wavfilesize = sizeof(WAV_Header) + sizeof(FMT_Header) + sizeof(DATA_Header) + length*2;
+ WAV_Header[4] = ((unsigned char)wavfilesize % 256);
+ WAV_Header[5] = ((unsigned char)wavfilesize / 256);
+ WAV_Header[6] = ((unsigned char)wavfilesize / (256*256));
+ WAV_Header[7] = ((unsigned char)wavfilesize / (256*256*256));
+ wavfilesize = wavfilesize - 54;
+ DATA_Header[4] = ((unsigned char)wavfilesize % 256);
+ DATA_Header[5] = ((unsigned char)wavfilesize / 256);
+ DATA_Header[6] = ((unsigned char)wavfilesize / (256*256));
+ DATA_Header[7] = ((unsigned char)wavfilesize / (256*256*256));
+
+ fseek( file, 0, SEEK_SET);
+ fwrite(&WAV_Header, 1, sizeof(WAV_Header), file);
+ fwrite(&FMT_Header, 1, sizeof(FMT_Header), file);
+ fwrite(&DATA_Header, 1, sizeof(DATA_Header), file);
+
+ return ERR_NONE;
+}
+
+static GSM_Error savebin(FILE *file, GSM_Ringtone *ringtone)
+{
+ char nullchar=0x00;
+
+ fwrite(&nullchar,1,1,file);
+ fwrite(&nullchar,1,1,file);
+ fprintf(file,"\x0C\x01\x2C");
+ fprintf(file,"%s",DecodeUnicodeString(ringtone->Name));
+ fwrite(&nullchar,1,1,file);
+ fwrite(&nullchar,1,1,file);
+ fwrite(ringtone->NokiaBinary.Frame,1,ringtone->NokiaBinary.Length,file);
+ return ERR_NONE;
+}
+
+static GSM_Error savepuremidi(FILE *file, GSM_Ringtone *ringtone)
+{
+ fwrite(ringtone->NokiaBinary.Frame,1,ringtone->NokiaBinary.Length,file);
+ return ERR_NONE;
+}
+
+GSM_Error saverttl(FILE *file, GSM_Ringtone *ringtone)
+{
+ GSM_RingNoteScale DefNoteScale;
+ GSM_RingNoteDuration DefNoteDuration;
+
+ GSM_RingNoteStyle DefNoteStyle=0;
+ int DefNoteTempo=0;
+
+ bool started = false, firstcomma = true;
+ GSM_RingNote *Note;
+
+ unsigned char buffer[15];
+ int i,j,k=0;
+
+ /* Saves ringtone name */
+ fprintf(file,"%s:",DecodeUnicodeString(ringtone->Name));
+
+ /* Find the most frequently used duration */
+ for (i=0;i<6;i++) buffer[i]=0;
+ for (i=0;i<ringtone->NoteTone.NrCommands;i++) {
+ if (ringtone->NoteTone.Commands[i].Type == RING_Note) {
+ Note = &ringtone->NoteTone.Commands[i].Note;
+ /* some durations need 2 bytes in file, some 1 */
+ if (Note->Duration >= Duration_Full && Note->Duration <= Duration_1_8) {
+ buffer[Note->Duration/32]++;
+ }
+ if (Note->Duration >= Duration_1_16 && Note->Duration <= Duration_1_32) {
+ buffer[Note->Duration/32]+=2;
+ }
+ }
+ }
+ /* Now find the most frequently used */
+ j=0;
+ for (i=0;i<6;i++) {
+ if (buffer[i]>j) {
+ k=i;
+ j=buffer[i];
+ }
+ }
+ /* Finally convert the default duration */
+ DefNoteDuration = k * 32;
+ dbgprintf("DefNoteDuration=%d\n", DefNoteDuration);
+ switch (DefNoteDuration) {
+ case Duration_Full:fprintf(file,"d=1"); break;
+ case Duration_1_2 :fprintf(file,"d=2"); break;
+ case Duration_1_4 :fprintf(file,"d=4"); break;
+ case Duration_1_8 :fprintf(file,"d=8"); break;
+ case Duration_1_16:fprintf(file,"d=16");break;
+ case Duration_1_32:fprintf(file,"d=32");break;
+ }
+
+ /* Find the most frequently used scale */
+ for (i=0;i<9;i++) buffer[i]=0;
+ for (i=0;i<ringtone->NoteTone.NrCommands;i++) {
+ if (ringtone->NoteTone.Commands[i].Type == RING_Note) {
+ Note = &ringtone->NoteTone.Commands[i].Note;
+ if (Note->Note!=Note_Pause &&
+ Note->Scale >= Scale_55 && Note->Scale <= Scale_14080) {
+ buffer[Note->Scale - 1]++;
+ }
+ }
+ }
+ j=0;
+ for (i=0;i<9;i++) {
+ if (buffer[i]>j) {
+ k = i;
+ j=buffer[i];
+ }
+ }
+ DefNoteScale = k + 1;
+ /* Save the default scale */
+ fprintf(file,",o=%i,",DefNoteScale);
+ dbgprintf("DefNoteScale=%d\n", DefNoteScale);
+
+ for (i=0;i<ringtone->NoteTone.NrCommands;i++) {
+ if (ringtone->NoteTone.Commands[i].Type != RING_Note) continue;
+
+ Note = &ringtone->NoteTone.Commands[i].Note;
+
+ /* Trick from PPM Edit */
+ if (Note->DurationSpec == DoubleDottedNote) {
+ switch (Note->Duration) {
+ case Duration_Full:Note->Duration = Duration_Full;break;
+ case Duration_1_2 :Note->Duration = Duration_Full;break;
+ case Duration_1_4 :Note->Duration = Duration_1_2; break;
+ case Duration_1_8 :Note->Duration = Duration_1_4; break;
+ case Duration_1_16:Note->Duration = Duration_1_8; break;
+ case Duration_1_32:Note->Duration = Duration_1_16;break;
+ }
+ Note->DurationSpec = NoSpecialDuration;
+ }
+
+ if (!started) {
+ DefNoteTempo=Note->Tempo;
+ DefNoteStyle=Note->Style;
+ switch (Note->Style) {
+ case StaccatoStyle : fprintf(file,"s=S,"); break;
+ case NaturalStyle : fprintf(file,"s=N,"); break;
+ case ContinuousStyle : break;
+ }
+ /* Save the default tempo */
+ fprintf(file,"b=%i:",DefNoteTempo);
+ dbgprintf("DefNoteTempo=%d\n", DefNoteTempo);
+ started = true;
+ firstcomma = true;
+ }
+
+ if (!started) continue;
+
+ if (Note->Style!=DefNoteStyle) {
+ /* And a separator */
+ if (!firstcomma) fprintf(file,",");
+ firstcomma = false;
+ DefNoteStyle=Note->Style;
+ switch (Note->Style) {
+ case StaccatoStyle : fprintf(file,"s=S"); break;
+ case NaturalStyle : fprintf(file,"s=N"); break;
+ case ContinuousStyle: fprintf(file,"s=C"); break;
+ }
+ }
+ if (Note->Tempo!=DefNoteTempo) {
+ /* And a separator */
+ if (!firstcomma) fprintf(file,",");
+ firstcomma = false;
+ DefNoteTempo=Note->Tempo;
+ fprintf(file,"b=%i",DefNoteTempo);
+ }
+ /* This note has a duration different than the default. We must save it */
+ if (Note->Duration!=DefNoteDuration) {
+ /* And a separator */
+ if (!firstcomma) fprintf(file,",");
+ firstcomma = false;
+ switch (Note->Duration) {
+ case Duration_Full:fprintf(file,"1"); break;
+ case Duration_1_2 :fprintf(file,"2"); break;
+ case Duration_1_4 :fprintf(file,"4"); break;
+ case Duration_1_8 :fprintf(file,"8"); break;
+ case Duration_1_16:fprintf(file,"16");break;
+ case Duration_1_32:fprintf(file,"32");break;
+ }
+ } else {
+ /* And a separator */
+ if (!firstcomma) fprintf(file,",");
+ firstcomma = false;
+ }
+ /* Now save the actual note */
+ switch (Note->Note) {
+ case Note_C :fprintf(file,"c"); break;
+ case Note_Cis:fprintf(file,"c#"); break;
+ case Note_D :fprintf(file,"d"); break;
+ case Note_Dis:fprintf(file,"d#"); break;
+ case Note_E :fprintf(file,"e"); break;
+ case Note_F :fprintf(file,"f"); break;
+ case Note_Fis:fprintf(file,"f#"); break;
+ case Note_G :fprintf(file,"g"); break;
+ case Note_Gis:fprintf(file,"g#"); break;
+ case Note_A :fprintf(file,"a"); break;
+ case Note_Ais:fprintf(file,"a#"); break;
+ case Note_H :fprintf(file,"h"); break;
+ default :fprintf(file,"p"); break; /*Pause ?*/
+ }
+ switch (Note->DurationSpec) {
+ case DottedNote : fprintf(file,"."); break;
+ default : break;
+ }
+ if (Note->Note!=Note_Pause && Note->Scale != DefNoteScale) {
+ fprintf(file,"%i",Note->Scale);
+ }
+ }
+ return ERR_NONE;
+}
+
+void saveimelody(FILE *file, GSM_Ringtone *ringtone)
+{
+ char Buffer[2000];
+ int i=2000;
+
+ GSM_EncodeEMSSound(*ringtone, Buffer, &i, (float)1.2, true);
+
+ fwrite(Buffer, 1, i, file);
+}
+
+#ifndef ENABLE_LGPL
+
+static void WriteVarLen(unsigned char* midifile, int* current, long value)
+{
+ long buffer;
+
+ buffer = value & 0x7f;
+
+ while (value >>= 7) {
+ buffer <<= 8;
+ buffer |= 0x80;
+ buffer += (value & 0x7f);
+ }
+
+ while (1) {
+ midifile[(*current)++] = (unsigned char)buffer;
+ if (buffer & 0x80) {
+ buffer >>= 8;
+ } else {
+ break;
+ }
+ }
+}
+
+#define singlepauses
+
+/* FIXME: need adding tempo before each note and scale too ? */
+void savemid(FILE* file, GSM_Ringtone *ringtone)
+{
+ int pause = 0, current = 26, duration, i, note=0, length = 20;
+ bool started = false;
+ GSM_RingNote *Note;
+ unsigned char midifile[3000] = {
+ 0x4D, 0x54, 0x68, 0x64, // MThd
+ 0x00, 0x00, 0x00, 0x06, // chunk length
+ 0x00, 0x00, // format 0
+ 0x00, 0x01, // one track
+ 0x00, 0x20, // 32 per quarter note
+ 0x4D, 0x54, 0x72, 0x6B, // MTrk
+ 0x00, 0x00, 0x00, 0x00, // chunk length
+ 0x00, 0xFF, 0x51, 0x03, // tempo meta event
+ 0x00, 0x00, 0x00}; // 3 bytes for us for a quarter note
+
+ for (i = 0; i < ringtone->NoteTone.NrCommands; i++) {
+ if (ringtone->NoteTone.Commands[i].Type == RING_Note) {
+ Note = &ringtone->NoteTone.Commands[i].Note;
+ if (!started) {
+ /* readmid does not read pauses at the beginning */
+ if (Note->Note != Note_Pause) {
+ /* FIXME: we need add tempo before each note or so... */
+ long duration=60000000/Note->Tempo;
+
+ midifile[current++] = (unsigned char)(duration >> 16);
+ midifile[current++] = (unsigned char)(duration >> 8);
+ midifile[current++] = (unsigned char)duration;
+
+ started = true;
+ }
+ }
+ if (!started) continue;
+ duration = GSM_RingNoteGetFullDuration(*Note);
+ if (Note->Note == Note_Pause) {
+ pause += duration;
+#ifdef singlepauses
+ WriteVarLen(midifile,&current,pause);
+ pause=0;
+ midifile[current++]=0x00; // pause
+ midifile[current++]=0x00;
+#endif
+ } else {
+ if (Note->Note >= Note_C && Note->Note <= Note_H) {
+ note = Note->Note/16 + 12 * Note->Scale - 1;
+ }
+
+ WriteVarLen(midifile,&current,pause);
+ pause=0;
+ midifile[current++]=0x90; // note on
+ midifile[current++]=note;
+ midifile[current++]=0x64; // forte
+
+ WriteVarLen(midifile,&current,duration);
+ midifile[current++]=0x80; // note off
+ midifile[current++]=note;
+ midifile[current++]=0x64;
+ }
+ }
+ }
+ if (pause) {
+ WriteVarLen(midifile,&current,pause);
+ midifile[current++]=0x00; // pause
+ midifile[current++]=0x00; //
+ }
+ midifile[current++] = 0x00;
+ midifile[current++] = 0xFF; // track end
+ midifile[current++] = 0x2F;
+ midifile[current++] = 0x00;
+ midifile[length++] = (current-22) >> 8;
+ midifile[length++] = current-22;
+
+ fwrite(midifile,1,current,file);
+}
+
+#endif
+
+void saveott(FILE *file, GSM_Ringtone *ringtone)
+{
+ char Buffer[2000];
+ int i=2000;
+
+ GSM_EncodeNokiaRTTLRingtone(*ringtone, Buffer, &i);
+
+ fwrite(Buffer, 1, i, file);
+}
+
+GSM_Error GSM_SaveRingtoneFile(char *FileName, GSM_Ringtone *ringtone)
+{
+ FILE *file;
+
+ file = fopen(FileName, "wb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ switch (ringtone->Format) {
+ case RING_NOTETONE:
+ if (strstr(FileName,".ott")) {
+ saveott(file,ringtone);
+#ifndef ENABLE_LGPL
+ } else if (strstr(FileName,".mid")) {
+ savemid(file,ringtone);
+#endif
+ } else if (strstr(FileName,".rng")) {
+ saveott(file,ringtone);
+ } else if (strstr(FileName,".imy")) {
+ saveimelody(file,ringtone);
+ } else if (strstr(FileName,".ime")) {
+ saveimelody(file,ringtone);
+ } else if (strstr(FileName,".wav")) {
+ savewav(file,ringtone);
+ } else {
+ saverttl(file, ringtone);
+ }
+ break;
+ case RING_NOKIABINARY:
+ savebin(file, ringtone);
+ break;
+ case RING_MIDI:
+ savepuremidi(file, ringtone);
+ break;
+ }
+
+ fclose(file);
+
+ return ERR_NONE;
+}
+
+static GSM_Error loadrttl(FILE *file, GSM_Ringtone *ringtone)
+{
+ GSM_RingNoteScale DefNoteScale = Scale_880;
+ GSM_RingNoteDuration DefNoteDuration = Duration_1_4;
+ GSM_RingNoteStyle DefNoteStyle = NaturalStyle;
+ int DefNoteTempo = 63, i=0;
+
+ unsigned char buffer[2000],Name[100];
+ GSM_RingNote *Note;
+
+ fread(buffer, 2000, 1, file);
+
+ ringtone->NoteTone.NrCommands = 0;
+
+ /* -------------- name ---------------- */
+ while (buffer[i] != ':') {
+ if (buffer[i] == 0x00) return ERR_NONE;
+ i++;
+ }
+ if (i == 0) {
+ /* This is for RTTL ringtones without name. */
+ EncodeUnicode(ringtone->Name,"Gammu",5);
+ } else {
+ memcpy(Name,buffer,i);
+ Name[i] = 0x00;
+ EncodeUnicode(ringtone->Name,Name,strlen(Name));
+ }
+ i++;
+
+ /* --------- section with default ringtone settings ----------- */
+ while(1) {
+ switch (buffer[i]) {
+ case ':':
+ break;
+ case 0x00:
+ return ERR_NONE;
+ case 'd': case 'D':
+ switch (atoi(buffer+i+2)) {
+ case 1: DefNoteDuration = Duration_Full; break;
+ case 2: DefNoteDuration = Duration_1_2 ; break;
+ case 4: DefNoteDuration = Duration_1_4 ; break;
+ case 8: DefNoteDuration = Duration_1_8 ; break;
+ case 16: DefNoteDuration = Duration_1_16; break;
+ case 32: DefNoteDuration = Duration_1_32; break;
+ }
+ break;
+ case 'o': case 'O':
+ switch (atoi(buffer+i+2)) {
+ case 4: DefNoteScale = Scale_440 ; break;
+ case 5: DefNoteScale = Scale_880 ; break;
+ case 6: DefNoteScale = Scale_1760; break;
+ case 7: DefNoteScale = Scale_3520; break;
+ }
+ break;
+ case 'b': case 'B':
+ DefNoteTempo=atoi(buffer+i+2);
+ dbgprintf("Tempo = %i\n",DefNoteTempo);
+ break;
+ case 's': case 'S':
+ switch (buffer[i+1]) {
+ case 'C': case 'c': DefNoteStyle=ContinuousStyle; break;
+ case 'N': case 'n': DefNoteStyle=NaturalStyle; break;
+ case 'S': case 's': DefNoteStyle=StaccatoStyle; break;
+ }
+ switch (buffer[i+2]) {
+ case 'C': case 'c': DefNoteStyle=ContinuousStyle; break;
+ case 'N': case 'n': DefNoteStyle=NaturalStyle; break;
+ case 'S': case 's': DefNoteStyle=StaccatoStyle; break;
+ }
+ break;
+ }
+ while (buffer[i] != ':' && buffer[i] != ',') {
+ if (buffer[i] == 0x00) return ERR_NONE;
+ i++;
+ }
+ if (buffer[i] == ',') i++;
+ if (buffer[i] == ':') break;
+ }
+ dbgprintf("DefNoteDuration=%d\n", DefNoteDuration);
+ dbgprintf("DefNoteScale=%d\n", DefNoteScale);
+ i++;
+
+ /* ------------------------- notes ------------------------------ */
+ while (buffer[i] != 0x00 && ringtone->NoteTone.NrCommands != MAX_RINGTONE_NOTES) {
+ switch(buffer[i]) {
+ case 'z': case 'Z':
+ switch (buffer[i+1]) {
+ case 'd':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_DisableLED;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ case 'D':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_EnableLED;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ case 'v':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_DisableVibra;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ case 'V':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_EnableVibra;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ case 'l':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_DisableLight;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ case 'L':
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_EnableLight;
+ ringtone->NoteTone.NrCommands++;
+ }
+ break;
+ case 'o': case 'O':
+ switch (buffer[i+2]) {
+ case 4: DefNoteScale = Scale_440 ; break;
+ case 5: DefNoteScale = Scale_880 ; break;
+ case 6: DefNoteScale = Scale_1760; break;
+ case 7: DefNoteScale = Scale_3520; break;
+ }
+ break;
+ case 's': case 'S':
+ switch (buffer[i+1]) {
+ case 'C': case 'c': DefNoteStyle=ContinuousStyle; break;
+ case 'N': case 'n': DefNoteStyle=NaturalStyle; break;
+ case 'S': case 's': DefNoteStyle=StaccatoStyle; break;
+ }
+ switch (buffer[i+2]) {
+ case 'C': case 'c': DefNoteStyle=ContinuousStyle; break;
+ case 'N': case 'n': DefNoteStyle=NaturalStyle; break;
+ case 'S': case 's': DefNoteStyle=StaccatoStyle; break;
+ }
+ break;
+ default:
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_Note;
+ Note = &ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Note;
+ Note->Style = DefNoteStyle;
+ Note->Tempo = DefNoteTempo;
+ Note->Scale = DefNoteScale;
+ Note->Duration = DefNoteDuration;
+ Note->DurationSpec = NoSpecialDuration;
+ Note->Note = Note_Pause;
+
+ /* Duration */
+ switch (atoi(buffer+i)) {
+ case 1: Note->Duration = Duration_Full ; break;
+ case 2: Note->Duration = Duration_1_2 ; break;
+ case 4: Note->Duration = Duration_1_4 ; break;
+ case 8: Note->Duration = Duration_1_8 ; break;
+ case 16: Note->Duration = Duration_1_16 ; break;
+ case 32: Note->Duration = Duration_1_32 ; break;
+ }
+ /* We skip all numbers from duration specification */
+ while(isdigit(buffer[i])) i++;
+
+ /* Some files can have special duration here */
+ if (buffer[i]=='.') {
+ Note->DurationSpec = DottedNote;
+ i++;
+ }
+
+ /* Note */
+ /* B or b is not in specs, but I decided to put it, because
+ * it's in some RTTL files. It's the same to H note */
+ switch (buffer[i]) {
+ case 'A': case 'a': Note->Note = Note_A; break;
+ case 'B': case 'b': Note->Note = Note_H; break;
+ case 'C': case 'c': Note->Note = Note_C; break;
+ case 'D': case 'd': Note->Note = Note_D; break;
+ case 'E': case 'e': Note->Note = Note_E; break;
+ case 'F': case 'f': Note->Note = Note_F; break;
+ case 'G': case 'g': Note->Note = Note_G; break;
+ case 'H': case 'h': Note->Note = Note_H; break;
+ }
+ i++;
+
+ if (buffer[i]=='#') {
+ switch (Note->Note) {
+ case Note_A : Note->Note = Note_Ais; break;
+ case Note_C : Note->Note = Note_Cis; break;
+ case Note_D : Note->Note = Note_Dis; break;
+ case Note_F : Note->Note = Note_Fis; break;
+ case Note_G : Note->Note = Note_Gis; break;
+ default : break;
+ }
+ i++;
+ }
+
+ /* Some files can have special duration here */
+ if (buffer[i]=='.') {
+ Note->DurationSpec = DottedNote;
+ i++;
+ }
+
+ /* Scale */
+ if (Note->Note!=Note_Pause && isdigit(buffer[i])) {
+ switch (atoi(buffer+i)) {
+ case 4: Note->Scale = Scale_440 ; break;
+ case 5: Note->Scale = Scale_880 ; break;
+ case 6: Note->Scale = Scale_1760; break;
+ case 7: Note->Scale = Scale_3520; break;
+ }
+ i++;
+ }
+
+ ringtone->NoteTone.NrCommands++;
+ break;
+ }
+ while (buffer[i] != ',') {
+ if (buffer[i] == 0x00) return ERR_NONE;
+ i++;
+ }
+ if (buffer[i] == ',') i++;
+ }
+
+ return ERR_NONE;
+}
+
+static GSM_Error loadott(FILE *file, GSM_Ringtone *ringtone)
+{
+ char Buffer[2000];
+ int i;
+
+ i=fread(Buffer, 1, 2000, file);
+
+ return GSM_DecodeNokiaRTTLRingtone(ringtone, Buffer, i);
+}
+
+static GSM_Error loadcommunicator(FILE *file, GSM_Ringtone *ringtone)
+{
+ char Buffer[4000];
+ int i,j;
+
+ i=fread(Buffer, 1, 4000, file);
+
+ i=0;j=0;
+ while (true) {
+ if (Buffer[j] ==0x00 && Buffer[j+1]==0x02 &&
+ Buffer[j+2]==0x4a && Buffer[j+3]==0x3a) break;
+ if (j==i-4) return ERR_UNKNOWN;
+ j++;
+ }
+ j++;
+
+ return GSM_DecodeNokiaRTTLRingtone(ringtone, Buffer+j, i-j);
+}
+
+static GSM_Error loadbin(FILE *file, GSM_Ringtone *ringtone)
+{
+ int i;
+ unsigned char buffer[2000];
+
+ dbgprintf("loading binary\n");
+ ringtone->NokiaBinary.Length=fread(buffer, 1, 500, file);
+ i=5;
+ while (buffer[i]!=0x00) i++;
+ EncodeUnicode(ringtone->Name,buffer+5,i-5);
+ while (buffer[i]!=0x02 && buffer[i+1]!=0xFC && buffer[i+2]!=0x09) {
+ i++;
+ }
+ ringtone->NokiaBinary.Length=ringtone->NokiaBinary.Length-i;
+ memcpy(ringtone->NokiaBinary.Frame,buffer+i,ringtone->NokiaBinary.Length);
+ dbgprintf("Length %i name \"%s\"\n",ringtone->NokiaBinary.Length,DecodeUnicodeString(ringtone->Name));
+ return ERR_NONE;
+}
+
+static GSM_Error loadpuremidi(FILE *file, GSM_Ringtone *ringtone)
+{
+ unsigned char buffer[30000];
+
+ dbgprintf("loading midi\n");
+ EncodeUnicode(ringtone->Name,"MIDI",4);
+ ringtone->NokiaBinary.Length=fread(buffer, 1, 30000, file);
+ memcpy(ringtone->NokiaBinary.Frame,buffer,ringtone->NokiaBinary.Length);
+ dbgprintf("Length %i name \"%s\"\n",ringtone->NokiaBinary.Length,DecodeUnicodeString(ringtone->Name));
+ return ERR_NONE;
+}
+
+static GSM_Error loadre(FILE *file, GSM_Ringtone *ringtone)
+{
+ unsigned char buffer[2000];
+
+ ringtone->NokiaBinary.Length=fread(buffer, 1, 500, file);
+
+ if (buffer[18]==0x00 && buffer[21]!=0x02) {
+ /* DCT3, Unicode subformat, 62xx & 7110 */
+ CopyUnicodeString(ringtone->Name,buffer+18);
+ ringtone->NokiaBinary.Length = ringtone->NokiaBinary.Length - (21+UnicodeLength(ringtone->Name)*2);
+ memcpy(ringtone->NokiaBinary.Frame,buffer+21+UnicodeLength(ringtone->Name)*2,ringtone->NokiaBinary.Length);
+ } else {
+ /* DCT3, normal subformat, 32xx/33xx/51xx/5210/5510/61xx/8xxx */
+ EncodeUnicode(ringtone->Name,buffer+17,buffer[16]);
+ ringtone->NokiaBinary.Length = ringtone->NokiaBinary.Length - (19+UnicodeLength(ringtone->Name));
+ memcpy(ringtone->NokiaBinary.Frame,buffer+19+UnicodeLength(ringtone->Name),ringtone->NokiaBinary.Length);
+ }
+ dbgprintf("Name \"%s\"\n",DecodeUnicodeString(ringtone->Name));
+ return ERR_NONE;
+}
+
+GSM_Error GSM_ReadRingtoneFile(char *FileName, GSM_Ringtone *ringtone)
+{
+ FILE *file;
+ unsigned char buffer[300];
+ GSM_Error error = ERR_UNKNOWN;
+
+ dbgprintf("Loading ringtone %s\n",FileName);
+ file = fopen(FileName, "rb");
+ if (file == NULL) return ERR_CANTOPENFILE;
+
+ /* Read the header of the file. */
+ fread(buffer, 1, 4, file);
+ if (ringtone->Format == 0x00) {
+ ringtone->Format = RING_NOTETONE;
+ if (buffer[0]==0x00 && buffer[1]==0x00 &&
+ buffer[2]==0x0C && buffer[3]==0x01) {
+ ringtone->Format = RING_NOKIABINARY;
+ }
+ if (buffer[0]==0x00 && buffer[1]==0x00 &&
+ buffer[2]==0x00) {
+ ringtone->Format = RING_NOKIABINARY;
+ }
+ if (buffer[0]==0x4D && buffer[1]==0x54 &&
+ buffer[2]==0x68 && buffer[3]==0x64) {
+ ringtone->Format = RING_MIDI;
+ }
+ }
+ rewind(file);
+ switch (ringtone->Format) {
+ case RING_NOTETONE:
+ if (buffer[0]==0x02 && buffer[1]==0x4A) {
+ error=loadott(file,ringtone);
+ } else if (buffer[0]==0xC7 && buffer[1]==0x45) {
+ error=loadcommunicator(file,ringtone);
+ } else {
+ error=loadrttl(file,ringtone);
+ }
+ ringtone->NoteTone.AllNotesScale=false;
+ break;
+ case RING_NOKIABINARY:
+ if (buffer[0]==0x00 && buffer[1]==0x00 &&
+ buffer[2]==0x0C && buffer[3]==0x01) {
+ error=loadbin(file,ringtone);
+ }
+ if (buffer[0]==0x00 && buffer[1]==0x00 &&
+ buffer[2]==0x00) {
+ error=loadre(file,ringtone);
+ }
+ break;
+ case RING_MIDI:
+ EncodeUnicode(ringtone->Name,FileName,strlen(FileName));
+ error = loadpuremidi(file,ringtone);
+ }
+ fclose(file);
+ return(error);
+}
+
+/* -------------------------- required with Nokia & RTTL ------------------- */
+
+/* Beats per Minute like written in Smart Messaging */
+static int SM_BeatsPerMinute[] = {
+ 25, 28, 31, 35, 40, 45, 50, 56, 63, 70,
+ 80, 90, 100, 112, 125, 140, 160, 180, 200, 225,
+ 250, 285, 320, 355, 400, 450, 500, 565, 635, 715,
+ 800, 900
+};
+
+int GSM_RTTLGetTempo(int Beats)
+{
+ int i=0;
+
+ while (Beats > SM_BeatsPerMinute[i] && SM_BeatsPerMinute[i] != 900) i++;
+
+ return i<<3;
+}
+
+/* This function packs the ringtone from the structure "ringtone" to
+ "package", where maxlength means length of package.
+ Function returns number of packed notes and change maxlength to
+ number of used chars in "package" */
+unsigned char GSM_EncodeNokiaRTTLRingtone(GSM_Ringtone ringtone, unsigned char *package, int *maxlength)
+{
+ unsigned char CommandLength = 0x02;
+ unsigned char Loop = 0x15; /* Infinite */
+
+ unsigned char Buffer[200];
+ int StartBit=0, OldStartBit;
+ int StartBitHowManyCommands;
+ int HowManyCommands = 0; /* How many instructions packed */
+ int HowManyNotes = 0;
+ int i,j;
+ bool started;
+ GSM_RingNote *Note;
+
+ GSM_RingNoteScale DefScale = 255;
+ GSM_RingNoteStyle DefStyle = 255;
+ int DefTempo = 255;
+
+ AddBufferByte(package, &StartBit, CommandLength, 8);
+ AddBufferByte(package, &StartBit, SM_Command_RingingToneProgramming, 7);
+
+ /* According to specification we need have next part octet-aligned */
+ BufferAlign(package, &StartBit);
+
+ AddBufferByte(package, &StartBit, SM_Command_Sound, 7);
+ AddBufferByte(package, &StartBit, SM_Song_BasicSongType, 3);
+
+ /* Packing the name of the tune. */
+ EncodeUnicodeSpecialNOKIAChars(Buffer, ringtone.Name, UnicodeLength(ringtone.Name));
+ AddBufferByte(package, &StartBit, ((unsigned char)(UnicodeLength(Buffer)<<4)), 4);
+ AddBuffer(package, &StartBit, DecodeUnicodeString(Buffer), 8*UnicodeLength(Buffer));
+
+ /* Packing info about song pattern */
+ AddBufferByte(package, &StartBit, 0x01, 8); //one pattern
+ AddBufferByte(package, &StartBit, SM_InstructionID_PatternHeaderId, 3);
+ AddBufferByte(package, &StartBit, SM_PatternID_A_part, 2);
+ AddBufferByte(package, &StartBit, ((unsigned char)(Loop<<4)), 4);
+
+ /* Later here will be HowManyCommands */
+ StartBitHowManyCommands=StartBit;
+ StartBit = StartBit + 8;
+
+ started = false;
+ for (i=0; i<ringtone.NoteTone.NrCommands; i++) {
+ if (ringtone.NoteTone.Commands[i].Type != RING_Note) {
+ HowManyNotes++;
+ continue;
+ }
+ Note = &ringtone.NoteTone.Commands[i].Note;
+ if (!started) {
+ /* First note can't be Pause - it makes problems
+ * for example with PC Composer
+ */
+ if (Note->Note != Note_Pause) started = true;
+ }
+ if (!started) {
+ HowManyNotes++;
+ continue;
+ }
+ OldStartBit = StartBit;
+ /* we don't write Scale & Style info before "Pause" note - it saves place */
+ if (Note->Note!=Note_Pause) {
+ if (DefScale != Note->Scale || ringtone.NoteTone.AllNotesScale) {
+ j = StartBit+5+8;
+ BufferAlignNumber(&j);
+ if ((j/8)>(*maxlength)) {
+ StartBit = OldStartBit;
+ break;
+ }
+ DefScale = Note->Scale;
+ AddBufferByte(package, &StartBit, SM_InstructionID_ScaleInstructionId, 3);
+ AddBufferByte(package, &StartBit, ((unsigned char)((DefScale-4)<<6)), 2);
+ HowManyCommands++;
+ }
+ if (DefStyle != Note->Style) {
+ j = StartBit+5+8;
+ BufferAlignNumber(&j);
+ if ((j/8)>(*maxlength)) {
+ StartBit = OldStartBit;
+ break;
+ }
+ DefStyle = Note->Style;
+ AddBufferByte(package, &StartBit, SM_InstructionID_StyleInstructionId, 3);
+ AddBufferByte(package, &StartBit, ((unsigned char)DefStyle), 2);
+ HowManyCommands++;
+ }
+ }
+ /* Beats per minute/tempo of the tune */
+ if (DefTempo != GSM_RTTLGetTempo(Note->Tempo)) {
+ j = StartBit+8+8;
+ BufferAlignNumber(&j);
+ if ((j/8)>(*maxlength)) {
+ StartBit = OldStartBit;
+ break;
+ }
+ DefTempo=GSM_RTTLGetTempo(Note->Tempo);
+ /* Adding beats per minute (tempo) of the tune */
+ AddBufferByte(package, &StartBit, SM_InstructionID_TempoInstructionId, 3);
+ AddBufferByte(package, &StartBit, ((unsigned char)DefTempo), 5);
+ HowManyCommands++;
+ }
+ j = StartBit+12+8;
+ BufferAlignNumber(&j);
+ if ((j/8)>(*maxlength)) {
+ StartBit = OldStartBit;
+ break;
+ }
+ /* Note */
+ AddBufferByte(package, &StartBit, SM_InstructionID_NoteInstructionId, 3);
+ AddBufferByte(package, &StartBit, ((unsigned char)Note->Note), 4);
+ AddBufferByte(package, &StartBit, ((unsigned char)Note->Duration), 3);
+ AddBufferByte(package, &StartBit, ((unsigned char)Note->DurationSpec), 2);
+ HowManyCommands++;
+ /* We are sure, we pack it for SMS or setting to phone, not for OTT file */
+ if (*maxlength<1000) {
+ /* Like Pc Composer say - before of phone limitations...*/
+ if (HowManyNotes==130-1) break;
+ }
+ HowManyNotes++;
+ }
+
+ BufferAlign(package, &StartBit);
+ AddBufferByte(package, &StartBit, SM_CommandEnd_CommandEnd, 8);
+
+ OldStartBit = StartBit;
+ StartBit = StartBitHowManyCommands;
+ /* HowManyCommands */
+ AddBufferByte(package, &StartBit, ((unsigned char)HowManyCommands), 8);
+ StartBit = OldStartBit;
+
+ *maxlength=StartBit/8;
+
+ return(i);
+}
+
+GSM_Error GSM_DecodeNokiaRTTLRingtone(GSM_Ringtone *ringtone, unsigned char *package, int maxlength)
+{
+ int StartBit=0, HowMany, l, q, i, spec;
+ char Buffer[100];
+ GSM_RingNote *Note;
+
+ /* Default ringtone parameters */
+ GSM_RingNoteScale DefScale = Scale_880;
+ GSM_RingNoteStyle DefStyle = NaturalStyle;
+ int DefTempo = 63;
+
+ ringtone->Format = RING_NOTETONE;
+ ringtone->NoteTone.NrCommands = 0;
+
+ GetBufferInt(package,&StartBit,&l,8);
+ if (l!=0x02) {
+ dbgprintf("Not header\n");
+ return ERR_NOTSUPPORTED;
+ }
+
+ GetBufferInt(package,&StartBit,&l,7);
+ if (l!=SM_Command_RingingToneProgramming) {
+ dbgprintf("Not RingingToneProgramming\n");
+ return ERR_NOTSUPPORTED;
+ }
+
+ /* According to specification we need have next part octet-aligned */
+ BufferAlignNumber(&StartBit);
+
+ GetBufferInt(package,&StartBit,&l,7);
+ if (l!=SM_Command_Sound) {
+ dbgprintf("Not Sound\n");
+ return ERR_NOTSUPPORTED;
+ }
+
+ GetBufferInt(package,&StartBit,&l,3);
+ if (l!=SM_Song_BasicSongType) {
+ dbgprintf("Not BasicSongType\n");
+ return ERR_NOTSUPPORTED;
+ }
+
+ /* Getting length of the tune name */
+ GetBufferInt(package,&StartBit,&l,4);
+ l=l>>4;
+
+ /* Unpacking the name of the tune. */
+ GetBuffer(package, &StartBit, Buffer, 8*l);
+ Buffer[l]=0;
+ EncodeUnicode(ringtone->Name,Buffer,strlen(Buffer));
+ DecodeUnicodeSpecialNOKIAChars(Buffer, ringtone->Name, UnicodeLength(ringtone->Name));
+ CopyUnicodeString(ringtone->Name,Buffer);
+
+ GetBufferInt(package,&StartBit,&l,8);
+ dbgprintf("Number of song patterns: %i\n",l);
+ /* we support only one song pattern */
+ if (l!=1) return ERR_NOTSUPPORTED;
+
+ GetBufferInt(package,&StartBit,&l,3);
+ if (l!=SM_InstructionID_PatternHeaderId) {
+ dbgprintf("Not PatternHeaderId\n");
+ return ERR_NOTSUPPORTED;
+ }
+
+ /* Pattern ID - we ignore it */
+ StartBit+=2;
+
+ GetBufferInt(package,&StartBit,&l,4);
+ l=l>>4;
+ dbgprintf("Loop value: %i\n",l);
+
+ HowMany=0;
+ GetBufferInt(package, &StartBit, &HowMany, 8);
+
+ for (i=0;i<HowMany;i++) {
+ GetBufferInt(package,&StartBit,&q,3);
+ switch (q) {
+ case SM_InstructionID_VolumeInstructionId:
+ StartBit+=4;
+ break;
+ case SM_InstructionID_StyleInstructionId:
+ GetBufferInt(package,&StartBit,&l,2);
+ if (l>=NaturalStyle && l<=StaccatoStyle) DefStyle = l;
+ break;
+ case SM_InstructionID_TempoInstructionId:
+ GetBufferInt(package,&StartBit,&l,5);
+ DefTempo=SM_BeatsPerMinute[l>>3];
+ break;
+ case SM_InstructionID_ScaleInstructionId:
+ GetBufferInt(package,&StartBit,&l,2);
+ DefScale=(l>>6)+4;
+ break;
+ case SM_InstructionID_NoteInstructionId:
+ Note = &ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Note;
+ ringtone->NoteTone.Commands[ringtone->NoteTone.NrCommands].Type = RING_Note;
+
+ GetBufferInt(package,&StartBit,&l,4);
+ Note->Note=Note_Pause;
+ if (l >= Note_C && l <= Note_H) Note->Note = l;
+
+ GetBufferInt(package,&StartBit,&l,3);
+ if (l >= Duration_Full && l <= Duration_1_32) Note->Duration = l;
+
+ GetBufferInt(package,&StartBit,&spec,2);
+ if (spec >= NoSpecialDuration && spec <= Length_2_3) {
+ Note->DurationSpec = spec;
+ }
+
+ Note->Scale = DefScale;
+ Note->Style = DefStyle;
+ Note->Tempo = DefTempo;
+ if (ringtone->NoteTone.NrCommands==MAX_RINGTONE_NOTES) break;
+ ringtone->NoteTone.NrCommands++;
+ break;
+ default:
+ dbgprintf("Unsupported block %i %i\n",q,i);
+ return ERR_NOTSUPPORTED;
+ }
+ }
+ return ERR_NONE;
+}
+
+static void RTTL2Binary(GSM_Ringtone *dest, GSM_Ringtone *src)
+{
+ int current = 0, i, note, lastnote = 0, duration;
+ GSM_RingNote *Note;
+ unsigned char end[] = {0x40, 0x7D, 0x40, 0x5C, 0x0A, 0xFE, 0x40,
+ 0x20, 0x40, 0x7D, 0x40, 0x37, 0x0A, 0xFE,
+ 0x0A, 0x0A, 0x40, 0x32, 0x07, 0x0B};
+
+ strcpy(dest->NokiaBinary.Frame+current,"\x02\xFC\x09"); current=current+3;
+ dest->NokiaBinary.Frame[current++]=0x00;
+
+/* This command can be used to loop, where 0xLL = 0x01 - 0x10
+ * 0x01=loop once [...] 0x10=loop infinite
+ * Commented now
+
+ dest->NokiaBinary.Frame[current++]=0x05;
+ dest->NokiaBinary.Frame[current++]=0xLL;
+ */
+ strcpy(dest->NokiaBinary.Frame+current,"\x0A\x01"); current=current+2;
+
+ for (i=0; i<src->NoteTone.NrCommands; i++) {
+ if (src->NoteTone.Commands[i].Type != RING_Note) continue;
+
+ Note = &src->NoteTone.Commands[i].Note;
+ note = 64; /* Pause */
+ if (Note->Note!=Note_Pause) {
+ if (Note->Note >= Note_C && Note->Note <= Note_H) {
+ note = 113 + Note->Note/16;
+ }
+ switch (Note->Scale) {
+ case Scale_440 : break;
+ case Scale_880 : note = note + 12; break;
+ case Scale_1760: note = note + 24; break;
+ case Scale_3520: note = note + 36; break;
+ default : break;
+ }
+ }
+
+ /* In 7110 we have 8 ms long sounds */
+ duration = 60000 * GSM_RingNoteGetFullDuration(*Note) / Note->Tempo / 256;
+
+ switch (Note->Style) {
+ case StaccatoStyle:
+ if (duration) {
+ /* Note needs only one sound */
+ dest->NokiaBinary.Frame[current++] = note;
+ dest->NokiaBinary.Frame[current++] = 1;
+ duration--;
+ }
+ note = 0x40; /* The rest is pause */
+ case NaturalStyle:
+ if (note != 0x40 && duration) {
+ dest->NokiaBinary.Frame[current++] = 0x40;
+ /* There is small pause between notes */
+ dest->NokiaBinary.Frame[current++] = 1;
+ duration--;
+ }
+ default:
+ if (note != 0x40 && note == lastnote && duration) {
+ dest->NokiaBinary.Frame[current++] = 0x40;
+ /* There is small pause between same notes */
+ dest->NokiaBinary.Frame[current++] = 1;
+ duration--;
+ }
+ while (duration > 125) {
+ dest->NokiaBinary.Frame[current++] = note;
+ dest->NokiaBinary.Frame[current++] = 125;
+ duration -= 125;
+ }
+ dest->NokiaBinary.Frame[current++] = note;
+ dest->NokiaBinary.Frame[current++] = duration;
+ }
+ lastnote = note;
+ }
+ for (i = 0; i < (int)sizeof(end); i++) dest->NokiaBinary.Frame[current++] = end[i];
+ dest->NokiaBinary.Length=current;
+}
+
+static void Binary2RTTL(GSM_Ringtone *dest, GSM_Ringtone *src)
+{
+ int i = 3, j, z, NrNotes = 0, repeat = 0, accuracy;
+ int StartRepeat = 0, EndRepeat, Speed;
+ unsigned char command,length=0;
+ int NotesLen[500];
+ GSM_RingNoteScale NotesScale[500];
+ GSM_RingNoteNote Notes[500];
+ int Lengths[6*4];
+ GSM_RingNoteDurationSpec DurationSpec[6*4];
+ GSM_RingNoteDuration Duration[6*4];
+ bool foundlen;
+ GSM_RingNote *Note;
+
+ while (i<src->NokiaBinary.Length) {
+ command = src->NokiaBinary.Frame[i];
+ i++;
+ if (command != 0x06 && command != 0x00 && command != 0x09) {
+ length = src->NokiaBinary.Frame[i];
+ i++;
+ dbgprintf("Block %02x %02x - ",length,command);
+ } else dbgprintf("Block %02x - ",command);
+ if (command >= 114 && command <= 161) {
+ dbgprintf("note\n");
+ if (command >= 114 && command <= 124) {
+ NotesScale[NrNotes] = Scale_440; command -= 114;
+ } else if (command >= 125 && command <= 137) {
+ NotesScale[NrNotes] = Scale_880; command -= 126;
+ } else if (command >= 138 && command <= 149) {
+ NotesScale[NrNotes] = Scale_1760; command -= 138;
+ } else if (command >= 150 && command <= 161) {
+ NotesScale[NrNotes] = Scale_3520; command -= 150;
+ }
+ switch (command) {
+ case 0 : Notes[NrNotes] = Note_C; break;
+ case 1 : Notes[NrNotes] = Note_Cis; break;
+ case 2 : Notes[NrNotes] = Note_D; break;
+ case 3 : Notes[NrNotes] = Note_Dis; break;
+ case 4 : Notes[NrNotes] = Note_E; break;
+ case 5 : Notes[NrNotes] = Note_F; break;
+ case 6 : Notes[NrNotes] = Note_Fis; break;
+ case 7 : Notes[NrNotes] = Note_G; break;
+ case 8 : Notes[NrNotes] = Note_Gis; break;
+ case 9 : Notes[NrNotes] = Note_A; break;
+ case 10 : Notes[NrNotes] = Note_Ais; break;
+ case 11 : Notes[NrNotes] = Note_H; break;
+ }
+ if (NrNotes > 0) {
+ if (Notes[NrNotes-1] == Notes[NrNotes] &&
+ NotesScale[NrNotes-1] == NotesScale[NrNotes]) {
+ NotesLen[NrNotes-1]+=length;
+ } else {
+ NotesLen[NrNotes]=length;
+ NrNotes++;
+ }
+ } else {
+ NotesLen[NrNotes]=length;
+ NrNotes++;
+ }
+ } else switch (command) {
+ case 0x00:
+ dbgprintf("Unknown\n");
+ break;
+ case 0x05:
+ dbgprintf("repeat %i times\n",length);
+ repeat = length;
+ StartRepeat = NrNotes;
+ break;
+ case 0x06:
+ dbgprintf("end repeat\n");
+ EndRepeat = NrNotes;
+ for (z=0;z<repeat-1;z++) {
+ for (j=StartRepeat;j<EndRepeat;j++) {
+ Notes[NrNotes] = Notes[j];
+ NotesScale[NrNotes] = NotesScale[j];
+ NotesLen[NrNotes] = NotesLen[j];
+ NrNotes++;
+ dbgprintf("Adding repeat note %i %i\n",Notes[j],NotesLen[j]);
+ }
+ }
+ break;
+ case 0x07:
+ if (length == 0x0B) {
+ dbgprintf("Ringtone end\n");
+ i = src->NokiaBinary.Length + 1;
+ }
+ break;
+ case 0x09:
+ dbgprintf("Unknown\n");
+ break;
+ case 0x0A:
+ if (length == 0x01) {
+ dbgprintf("Let's start our song\n");
+ break;
+ }
+ if (length == 0x0A) {
+ dbgprintf("Ending joining note\n");
+ break;
+ }
+ if (length == 0xFE) {
+ dbgprintf("Starting joining note\n");
+ break;
+ }
+ break;
+ case 0x40:
+ dbgprintf("Pause\n");
+ Notes[NrNotes] = Note_Pause;
+ if (NrNotes > 0) {
+ if (Notes[NrNotes-1] == Notes[NrNotes] &&
+ NotesScale[NrNotes-1] == NotesScale[NrNotes]) {
+ NotesLen[NrNotes-1]+=length;
+ } else {
+ NotesLen[NrNotes]=length;
+ NrNotes++;
+ }
+ } else {
+ NotesLen[NrNotes]=length;
+ NrNotes++;
+ }
+ break;
+ default:
+ dbgprintf("Unknown\n");
+ }
+ }
+
+ while (NrNotes>0) {
+ if (Notes[NrNotes-1] == Note_Pause) {
+ NrNotes--;
+ } else break;
+ }
+
+ for (accuracy=1; accuracy<5; accuracy++) {
+ i = 1;
+ while (i < 1000) {
+ Lengths[0] = 30000/i;
+ for (j=0;j<5;j++) Lengths[j+1] = Lengths[j] / 2;
+ for (j=0;j<6;j++) Lengths[6+j] = Lengths[j] * 3/2;
+ for (j=0;j<6;j++) Lengths[12+j] = Lengths[j] * 9/4;
+ for (j=0;j<6;j++) Lengths[18+j] = Lengths[j] * 2/3;
+
+#ifdef DEBUG
+ dbgprintf("Length matrix (%i) : ",i);
+ for (j=0;j<6*4;j++) dbgprintf("%i ",Lengths[j]);
+ dbgprintf("\n");
+#endif
+ foundlen = false;
+
+ for (j=0;j<NrNotes;j++) {
+ dbgprintf("Comparing to %i\n",NotesLen[j]);
+ foundlen = false;
+ for (z=0;z<6*4;z++) {
+ if (NotesLen[j] - Lengths[z] > -accuracy &&
+ NotesLen[j] - Lengths[z] < accuracy) {
+ foundlen = true;
+ break;
+ }
+ }
+ if (!foundlen) break;
+ }
+ if (foundlen) break;
+ i++;
+ }
+
+ if (foundlen) {
+ Speed = i;
+ Duration[5] = Duration_1_32; Duration[4] = Duration_1_16;
+ Duration[3] = Duration_1_8; Duration[2] = Duration_1_4;
+ Duration[1] = Duration_1_2; Duration[0] = Duration_Full;
+ for (i=0;i<6;i++) Duration[i] = Duration[i];
+ for (i=0;i<6;i++) Duration[i+6] = Duration[i];
+ for (i=0;i<6;i++) Duration[i+12] = Duration[i];
+ for (i=0;i<6;i++) Duration[i+18] = Duration[i];
+ for (i=0;i<6;i++) DurationSpec[i] = NoSpecialDuration;
+ for (i=0;i<6;i++) DurationSpec[i+6] = DottedNote;
+ for (i=0;i<6;i++) DurationSpec[i+12] = DoubleDottedNote;
+ for (i=0;i<6;i++) DurationSpec[i+18] = Length_2_3;
+
+ for (i=0;i<NrNotes;i++) {
+ dest->NoteTone.Commands[i].Type = RING_Note;
+ Note = &dest->NoteTone.Commands[i].Note;
+ Note->Note = Notes[i];
+ Note->Tempo = Speed;
+ Note->Style = ContinuousStyle;
+ if (Notes[i] != Note_Pause) Note->Scale = NotesScale[i];
+ for (z=0;z<6*4;z++) {
+ if (NotesLen[i] - Lengths[z] > -accuracy &&
+ NotesLen[i] - Lengths[z] < accuracy) {
+ Note->Duration = Duration[z];
+ Note->DurationSpec = DurationSpec[z];
+ /* Trick from PPM Edit */
+ if (Note->DurationSpec == DoubleDottedNote) {
+ switch (Note->Duration) {
+ case Duration_Full:Note->Duration = Duration_Full;break;
+ case Duration_1_2 :Note->Duration = Duration_Full;break;
+ case Duration_1_4 :Note->Duration = Duration_1_2; break;
+ case Duration_1_8 :Note->Duration = Duration_1_4; break;
+ case Duration_1_16:Note->Duration = Duration_1_8; break;
+ case Duration_1_32:Note->Duration = Duration_1_16;break;
+ }
+ Note->DurationSpec = NoSpecialDuration;
+ }
+ /* Here happy creation */
+ if (Note->DurationSpec == Length_2_3) {
+ Note->DurationSpec = NoSpecialDuration;
+ }
+
+ break;
+ }
+ }
+ }
+ dest->NoteTone.NrCommands = NrNotes;
+ dbgprintf("speed = %i\n",Speed);
+ break;
+ }
+ }
+
+ if (!foundlen) dest->NoteTone.NrCommands = 0;
+}
+
+GSM_Error GSM_RingtoneConvert(GSM_Ringtone *dest, GSM_Ringtone *src, GSM_RingtoneFormat Format)
+{
+ dest->Format = Format;
+ CopyUnicodeString(dest->Name,src->Name);
+ if (src->Format==RING_NOTETONE && Format==RING_NOKIABINARY) {
+ RTTL2Binary(dest, src);
+ return ERR_NONE;
+ }
+ if (src->Format==RING_NOKIABINARY && Format==RING_NOTETONE) {
+ Binary2RTTL(dest, src);
+ return ERR_NONE;
+ }
+ /* The same source and target format */
+ if (src->Format==Format) {
+ memcpy(dest,src,sizeof(GSM_Ringtone));
+ return ERR_NONE;
+ }
+ return ERR_NOTIMPLEMENTED;
+}
+
+/* 0 = No header and footer, 0.5 = partial header and footer,
+ * 1.0 = IMelody 1.0, 1.2 = IMelody 1.2 */
+unsigned char GSM_EncodeEMSSound(GSM_Ringtone ringtone, unsigned char *package, int *maxlength, double version, bool start)
+{
+ int i, NrNotes = 0, Len, Max = *maxlength;
+
+ GSM_RingNote *Note;
+
+ GSM_RingNoteScale DefNoteScale;
+ GSM_RingNoteStyle DefNoteStyle=0;
+ int DefNoteTempo=0;
+
+ bool started = false, end;
+
+ *maxlength = 0;
+
+ if (start) {
+ if (version != 0) *maxlength+=sprintf(package,"BEGIN:IMELODY%c%c",13,10);
+ if (version == 1.0) *maxlength+=sprintf(package+(*maxlength),"VERSION:1.0%c%c",13,10);
+ if (version == 1.2) *maxlength+=sprintf(package+(*maxlength),"VERSION:1.2%c%c",13,10);
+ if (version >= 1.0) *maxlength+=sprintf(package+(*maxlength),"FORMAT:CLASS1.0%c%c",13,10);
+ if (version == 1.2) *maxlength+=sprintf(package+(*maxlength),"NAME:%s%c%c",DecodeUnicodeString(ringtone.Name),13,10);
+ }
+
+ DefNoteScale = Scale_880; /* by iMelody definition */
+
+ for (i=0;i<ringtone.NoteTone.NrCommands;i++) {
+ Len = *maxlength;
+ if (ringtone.NoteTone.Commands[i].Type != RING_Note) continue;
+
+ Note = &ringtone.NoteTone.Commands[i].Note;
+ if (Note->Note == Note_Pause) continue;
+
+ if (version == 1.2 && start) {
+ /* Save the default tempo */
+ DefNoteTempo = Note->Tempo;
+ Len+=sprintf(package+Len,"BEAT:%i%c%c",DefNoteTempo,13,10);
+ dbgprintf("DefNoteTempo=%d\n",DefNoteTempo);
+
+ /* Save default style */
+ DefNoteStyle = Note->Style;
+ switch (DefNoteStyle) {
+ case NaturalStyle :Len+=sprintf(package+Len,"STYLE:S0%c%c",13,10); break;
+ case ContinuousStyle:Len+=sprintf(package+Len,"STYLE:S1%c%c",13,10); break;
+ case StaccatoStyle :Len+=sprintf(package+Len,"STYLE:S2%c%c",13,10); break;
+ }
+ }
+ Len+=sprintf(package+Len,"MELODY:");
+ if (version != 0) {
+ /* 15 = Len of END:IMELODY... */
+ if ((Len+15) > Max) { end = true; break; }
+ } else {
+ if (Len > Max) { end = true; break; }
+ }
+ *maxlength = Len;
+ break;
+ }
+
+ for (i=0;i<ringtone.NoteTone.NrCommands;i++) {
+ end = false;
+ Len = *maxlength;
+ switch (ringtone.NoteTone.Commands[i].Type) {
+ case RING_Note:
+ Note = &ringtone.NoteTone.Commands[i].Note;
+ if (!started && Note->Note != Note_Pause) started = true;
+ if (!started) break;
+ if (Note->Note!=Note_Pause && Note->Scale != DefNoteScale) {
+ Len+=sprintf(package+Len,"*%i",Note->Scale-1);
+ }
+ switch (Note->Note) {
+ case Note_C :Len+=sprintf(package+Len,"c"); break;
+ case Note_Cis :Len+=sprintf(package+Len,"#c");break;
+ case Note_D :Len+=sprintf(package+Len,"d"); break;
+ case Note_Dis :Len+=sprintf(package+Len,"#d");break;
+ case Note_E :Len+=sprintf(package+Len,"e"); break;
+ case Note_F :Len+=sprintf(package+Len,"f"); break;
+ case Note_Fis :Len+=sprintf(package+Len,"#f");break;
+ case Note_G :Len+=sprintf(package+Len,"g"); break;
+ case Note_Gis :Len+=sprintf(package+Len,"#g");break;
+ case Note_A :Len+=sprintf(package+Len,"a"); break;
+ case Note_Ais :Len+=sprintf(package+Len,"#a");break;
+ case Note_H :Len+=sprintf(package+Len,"b"); break;
+ case Note_Pause :Len+=sprintf(package+Len,"r"); break;
+ }
+ switch (Note->Duration) {
+ case Duration_Full : package[Len++]='0'; break;
+ case Duration_1_2 : package[Len++]='1'; break;
+ case Duration_1_4 : package[Len++]='2'; break;
+ case Duration_1_8 : package[Len++]='3'; break;
+ case Duration_1_16 : package[Len++]='4'; break;
+ case Duration_1_32 : package[Len++]='5'; break;
+ default : break;
+ }
+ switch (Note->DurationSpec) {
+ case DottedNote : package[Len++] = '.'; break;
+ case DoubleDottedNote : package[Len++] = ':'; break;
+ case Length_2_3 : package[Len++] = ';'; break;
+ default : break;
+ }
+ if (version != 0) {
+ /* 15 = Len of END:IMELODY... */
+ if ((Len+15) > Max) { end = true; break; }
+ } else {
+ if (Len > Max) { end = true; break; }
+ }
+ *maxlength = Len;
+ break;
+ case RING_DisableLED:
+ if ((Len + 6) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"ledoff");
+ break;
+ case RING_EnableLED:
+ if ((Len + 5) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"ledon");
+ break;
+ case RING_DisableVibra:
+ if ((Len + 7) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"vibeoff");
+ break;
+ case RING_EnableVibra:
+ if ((Len + 6) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"vibeon");
+ break;
+ case RING_DisableLight:
+ if ((Len + 7) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"backoff");
+ break;
+ case RING_EnableLight:
+ if ((Len + 6) > Max) { end = true; break; }
+ (*maxlength)+=sprintf(package+Len,"backon");
+ break;
+ default:
+ break;
+ }
+ if (end) break;
+ NrNotes ++;
+ }
+
+ if (version != 0) *maxlength+=sprintf(package+(*maxlength),"%c%cEND:IMELODY%c%c",13,10,13,10);
+
+ return NrNotes;
+}
+
+char *GSM_GetRingtoneName(GSM_AllRingtonesInfo *Info, int ID)
+{
+ int i;
+ static char ala[2];
+
+ for (i=0;i<Info->Number;i++) {
+ if (Info->Ringtone[i].ID == ID) return Info->Ringtone[i].Name;
+ }
+
+ ala[0] = 0;
+ ala[1] = 0;
+ return ala;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/gsmring.h b/gammu/emb/common/service/gsmring.h
new file mode 100644
index 0000000..207cf31
--- a/dev/null
+++ b/gammu/emb/common/service/gsmring.h
@@ -0,0 +1,202 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef __gsm_ring_h
+#define __gsm_ring_h
+
+/* --------------- Smart Messaging Specification 2.0 & 3.0 ----------------- */
+
+#define SM_CommandEnd_CommandEnd 0x00
+
+/* specification gives also other */
+#define SM_Command_RingingToneProgramming 0x25<<1
+#define SM_Command_Sound 0x1d<<1
+/* specification gives also other */
+
+#define SM_Song_BasicSongType 0x01<<5
+/* specification gives also other */
+
+#define SM_PatternID_A_part 0x00<<6
+/* specification gives also other */
+
+#define SM_InstructionID_PatternHeaderId 0x00<<5
+#define SM_InstructionID_NoteInstructionId 0x01<<5
+#define SM_InstructionID_ScaleInstructionId 0x02<<5
+#define SM_InstructionID_StyleInstructionId 0x03<<5
+#define SM_InstructionID_TempoInstructionId 0x04<<5
+#define SM_InstructionID_VolumeInstructionId 0x05<<5
+
+/* ------ end of Smart Messaging Specification 2.0 & 3.0 definitions ------- */
+
+#define MAX_RINGTONE_NOTES 255
+
+typedef enum {
+ /**
+ * Natural style (rest between notes)
+ */
+ NaturalStyle = 0x00<<6,
+ /**
+ * Continuous style (no rest between notes)
+ */
+ ContinuousStyle = 0x01<<6,
+ /**
+ * Staccato style (shorter notes and longer rest period)
+ */
+ StaccatoStyle = 0x02<<6
+} GSM_RingNoteStyle;
+
+typedef enum {
+ Note_Pause = 0x00<<4,
+ Note_C = 0x01<<4,
+ Note_Cis = 0x02<<4,
+ Note_D = 0x03<<4,
+ Note_Dis = 0x04<<4,
+ Note_E = 0x05<<4,
+ Note_F = 0x06<<4,
+ Note_Fis = 0x07<<4,
+ Note_G = 0x08<<4,
+ Note_Gis = 0x09<<4,
+ Note_A = 0x0a<<4,
+ Note_Ais = 0x0b<<4,
+ Note_H = 0x0c<<4
+} GSM_RingNoteNote;
+
+typedef enum {
+ Duration_Full = 0x00<<5,
+ Duration_1_2 = 0x01<<5,
+ Duration_1_4 = 0x02<<5,
+ Duration_1_8 = 0x03<<5,
+ Duration_1_16 = 0x04<<5,
+ Duration_1_32 = 0x05<<5
+} GSM_RingNoteDuration;
+
+typedef enum {
+ NoSpecialDuration = 0x00<<6,
+ DottedNote = 0x01<<6,
+ DoubleDottedNote = 0x02<<6,
+ Length_2_3 = 0x03<<6
+} GSM_RingNoteDurationSpec;
+
+typedef enum {
+ Scale_55 = 1, /* 55 Hz for note A */
+ Scale_110, /* 110 Hz for note A */
+ Scale_220,
+ Scale_440, /* first scale for Nokia */
+ Scale_880,
+ Scale_1760,
+ Scale_3520, /* last scale for Nokia */
+ Scale_7040,
+ Scale_14080
+} GSM_RingNoteScale;
+
+typedef struct {
+ GSM_RingNoteDuration Duration;
+ GSM_RingNoteDurationSpec DurationSpec;
+ GSM_RingNoteNote Note;
+ GSM_RingNoteStyle Style;
+ GSM_RingNoteScale Scale;
+ int Tempo;
+} GSM_RingNote;
+
+typedef enum {
+ RING_Note = 1,
+ RING_EnableVibra,
+ RING_DisableVibra,
+ RING_EnableLight,
+ RING_DisableLight,
+ RING_EnableLED,
+ RING_DisableLED,
+ RING_Repeat
+} GSM_RingCommandType;
+
+typedef struct {
+ GSM_RingCommandType Type;
+ GSM_RingNote Note;
+ unsigned char Value;
+} GSM_RingCommand;
+
+typedef struct {
+ int NrCommands;
+ GSM_RingCommand Commands[MAX_RINGTONE_NOTES];
+ bool AllNotesScale;
+} GSM_NoteRingtone;
+
+/* Structure to hold Nokia binary ringtones. */
+typedef struct {
+ unsigned char Frame[30000];
+ int Length;
+} GSM_NokiaBinaryRingtone;
+
+typedef struct {
+ unsigned char *Frame;
+ int Length;
+} GSM_BinaryTone;
+
+typedef enum {
+ RING_NOTETONE = 1,
+ RING_NOKIABINARY,
+ RING_MIDI
+} GSM_RingtoneFormat;
+
+/**
+ * Structure for saving various ringtones formats
+ */
+typedef struct {
+ /**
+ * Ringtone saved in one of three formats
+ */
+ GSM_NokiaBinaryRingtone NokiaBinary;
+ GSM_BinaryTone BinaryTone;
+ GSM_NoteRingtone NoteTone;
+ /**
+ * Ringtone format
+ */
+ GSM_RingtoneFormat Format;
+ /**
+ * Ringtone name
+ */
+ char Name[20*2];
+ /**
+ * Ringtone location
+ */
+ int Location;
+} GSM_Ringtone;
+
+typedef struct {
+ int Group; //Nokia specific
+ int ID;
+ char Name[30*2];
+} GSM_RingtoneInfo;
+
+typedef struct {
+ int Number;
+ GSM_RingtoneInfo Ringtone[100];
+} GSM_AllRingtonesInfo;
+
+GSM_Error GSM_SaveRingtoneFile(char *FileName, GSM_Ringtone *ringtone);
+GSM_Error GSM_ReadRingtoneFile(char *FileName, GSM_Ringtone *ringtone);
+
+void saveott(FILE *file, GSM_Ringtone *ringtone);
+void savemid(FILE *file, GSM_Ringtone *ringtone);
+void saverng(FILE *file, GSM_Ringtone *ringtone);
+void saveimelody(FILE *file, GSM_Ringtone *ringtone);
+GSM_Error savewav(FILE *file, GSM_Ringtone *ringtone);
+GSM_Error saverttl(FILE *file, GSM_Ringtone *ringtone);
+
+unsigned char GSM_EncodeNokiaRTTLRingtone (GSM_Ringtone ringtone, unsigned char *package, int *maxlength);
+unsigned char GSM_EncodeEMSSound (GSM_Ringtone ringtone, unsigned char *package, int *maxlength, double version, bool start);
+
+GSM_Error GSM_DecodeNokiaRTTLRingtone (GSM_Ringtone *ringtone, unsigned char *package, int maxlength);
+
+GSM_Error GSM_RingtoneConvert(GSM_Ringtone *dest, GSM_Ringtone *src, GSM_RingtoneFormat Format);
+
+int GSM_RTTLGetTempo (int Beats);
+int GSM_RingNoteGetFrequency (GSM_RingNote Note);
+int GSM_RingNoteGetFullDuration (GSM_RingNote Note);
+
+char *GSM_GetRingtoneName(GSM_AllRingtonesInfo *Info, int ID);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmems.c b/gammu/emb/common/service/sms/gsmems.c
new file mode 100644
index 0000000..a7e20f4
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmems.c
@@ -0,0 +1,765 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#include <ctype.h>
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmcal.h"
+#include "../gsmpbk.h"
+#include "../gsmlogo.h"
+#include "../gsmring.h"
+#include "../gsmdata.h"
+#include "../gsmnet.h"
+#include "gsmsms.h"
+#include "gsmmulti.h"
+
+/* EMS Developers' Guidelines from www.sonyericsson.com
+ * docs from Alcatel
+ */
+GSM_Error GSM_EncodeEMSMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS,
+ GSM_UDH UDHType)
+{
+ unsigned char Buffer[GSM_MAX_SMS_LENGTH*2*MAX_MULTI_SMS];
+ int i,UsedText,j,Length,Width,Height,z,x,y;
+ unsigned int Len;
+ int Used,FreeText,FreeBytes,Width2,CopiedText,CopiedSMSText;
+ unsigned char UDHID;
+ GSM_Bitmap Bitmap,Bitmap2;
+ GSM_Ringtone Ring;
+ GSM_Coding_Type Coding = SMS_Coding_Default;
+ GSM_Phone_Bitmap_Types BitmapType;
+ MultiPartSMSEntry *Entry;
+ bool start;
+ GSM_DateTime Date;
+
+#ifdef DEBUG
+ if (UDHType != UDH_NoUDH) dbgprintf("linked EMS\n");
+#endif
+
+ if (Info->UnicodeCoding) Coding = SMS_Coding_Unicode;
+
+ /* Cleaning on the start */
+ for (i=0;i<MAX_MULTI_SMS;i++) {
+ GSM_SetDefaultSMSData(&SMS->SMS[i]);
+ SMS->SMS[i].UDH.Type = UDHType;
+ GSM_EncodeUDHHeader(&SMS->SMS[i].UDH);
+ SMS->SMS[i].Coding = Coding;
+ }
+
+ /* Packing */
+ for (i=0;i<Info->EntriesNum;i++) {
+ Entry = &Info->Entries[i];
+
+ switch (Entry->ID) {
+ case SMS_ConcatenatedTextLong:
+ case SMS_ConcatenatedTextLong16bit:
+ Len = 0;
+ while(1) {
+ if (Entry->Left || Entry->Right ||
+ Entry->Center || Entry->Large ||
+ Entry->Small || Entry->Bold ||
+ Entry->Italic || Entry->Underlined ||
+ Entry->Strikethrough) {
+ Buffer[0] = 0x0A; /* ID for text format */
+ Buffer[1] = 0x03; /* length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ Buffer[3] = 0x00; /* how many chars */
+ Buffer[4] = 0x00; /* formatting bits */
+ if (Entry->Left) {
+ } else if (Entry->Right) { Buffer[4] |= 1;
+ } else if (Entry->Center) { Buffer[4] |= 2;
+ } else Buffer[4] |= 3;
+ if (Entry->Large) { Buffer[4] |= 4;
+ } else if (Entry->Small) { Buffer[4] |= 8;}
+ if (Entry->Bold) Buffer[4] |= 16;
+ if (Entry->Italic) Buffer[4] |= 32;
+ if (Entry->Underlined) Buffer[4] |= 64;
+ if (Entry->Strikethrough) Buffer[4] |= 128;
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,5,true,&UsedText,&CopiedText,&CopiedSMSText);
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], &UsedText, &FreeText, &FreeBytes);
+ if (FreeText == 0) continue;
+ }
+ GSM_AddSMS_Text_UDH(SMS,Coding,Entry->Buffer+Len*2,UnicodeLength(Entry->Buffer) - Len,false,&UsedText,&CopiedText,&CopiedSMSText);
+ if (Entry->Left || Entry->Right ||
+ Entry->Center || Entry->Large ||
+ Entry->Small || Entry->Bold ||
+ Entry->Italic || Entry->Underlined ||
+ Entry->Strikethrough) {
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-3] = UsedText;
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-2] = CopiedSMSText;
+ }
+ Len += CopiedText;
+ if (Len == UnicodeLength(Entry->Buffer)) break;
+ dbgprintf("%i %i\n",Len,UnicodeLength(Entry->Buffer));
+ }
+ break;
+ case SMS_EMSPredefinedSound:
+ case SMS_EMSPredefinedAnimation:
+ if (Entry->ID == SMS_EMSPredefinedSound) {
+ Buffer[0] = 0x0B; /* ID for def.sound */
+ } else {
+ Buffer[0] = 0x0D; /* ID for def.animation */
+ }
+ Buffer[1] = 0x02; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ Buffer[3] = Entry->Number; /* Number of anim. */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-2] = UsedText;
+ break;
+ case SMS_EMSSonyEricssonSound:
+ case SMS_EMSSound10:
+ case SMS_EMSSound12:
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ Length = 128; /* 128 bytes is maximal length from specs */
+ switch (Entry->ID) {
+ case SMS_EMSSound10:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(*Entry->Ringtone, Buffer+3, &Length, 1.0, true);
+ break;
+ case SMS_EMSSound12:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(*Entry->Ringtone, Buffer+3, &Length, 1.2, true);
+ break;
+ case SMS_EMSSonyEricssonSound:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(*Entry->Ringtone, Buffer+3, &Length, 0, true);
+ break;
+ default:
+ break;
+ }
+
+ Buffer[0] = 0x0C; /* ID for EMS sound */
+ Buffer[1] = Length+1; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,Length+3,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-Length-1] = UsedText;
+ break;
+ case SMS_EMSSonyEricssonSoundLong:
+ case SMS_EMSSound10Long:
+ case SMS_EMSSound12Long:
+ Ring = *Entry->Ringtone;
+
+ /* First check if we can use classic format */
+ Length = 128; /* 128 bytes is maximal length from specs */
+ switch (Entry->ID) {
+ case SMS_EMSSound10Long:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.0, true);
+ break;
+ case SMS_EMSSound12Long:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.2, true);
+ break;
+ case SMS_EMSSonyEricssonSoundLong:
+ Entry->RingtoneNotes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 0, true);
+ break;
+ default:
+ break;
+ }
+ if (Entry->RingtoneNotes == Ring.NoteTone.NrCommands) {
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ Buffer[0] = 0x0C; /* ID for EMS sound */
+ Buffer[1] = Length+1; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,Length+3,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-Length-1] = UsedText;
+ break;
+ }
+
+ /* Find free place in first SMS */
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], &UsedText, &FreeText, &FreeBytes);
+ Length = FreeBytes - 3;
+ if (Entry->Protected) Length = Length - 4;
+ if (Length < 0) Length = 128;
+ if (Length > 128) Length = 128;
+
+ Ring = *Entry->Ringtone;
+
+ /* Checking number of SMS */
+ Used = 0;
+ FreeBytes = 0;
+ start = true;
+ while (1) {
+ if (FreeBytes != 0) {
+ z = 0;
+ for (j=FreeBytes;j<Ring.NoteTone.NrCommands;j++) {
+ Ring.NoteTone.Commands[z] = Ring.NoteTone.Commands[j];
+ z++;
+ }
+ Ring.NoteTone.NrCommands -= FreeBytes;
+ if (Ring.NoteTone.NrCommands == 0) break;
+ Length = 128; /* 128 bytes is maximal length from specs */
+ }
+ switch (Entry->ID) {
+ case SMS_EMSSound10Long:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.0, start);
+ break;
+ case SMS_EMSSound12Long:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.2, start);
+ break;
+ case SMS_EMSSonyEricssonSoundLong:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 0, start);
+ break;
+ default:
+ break;
+ }
+ start = false;
+ Used++;
+ }
+ dbgprintf("Used SMS: %i\n",Used);
+
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = Used+1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ /* Save UPI UDH */
+ Buffer[0] = 0x13; /* ID for UPI */
+ Buffer[1] = 1; /* Length of rest */
+ Buffer[2] = Used; /* Number of used parts */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,3,true,&UsedText,&CopiedText,&CopiedSMSText);
+
+ /* Find free place in first SMS */
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], &UsedText, &FreeText, &FreeBytes);
+ Length = FreeBytes - 3;
+ if (Length < 0) Length = 128;
+ if (Length > 128) Length = 128;
+
+ Ring = *Entry->Ringtone;
+
+ /* Saving */
+ FreeBytes = 0;
+ start = true;
+ while (1) {
+ if (FreeBytes != 0) {
+ z = 0;
+ for (j=FreeBytes;j<Ring.NoteTone.NrCommands;j++) {
+ Ring.NoteTone.Commands[z] = Ring.NoteTone.Commands[j];
+ z++;
+ }
+ Ring.NoteTone.NrCommands -= FreeBytes;
+ if (Ring.NoteTone.NrCommands == 0) break;
+ Length = 128; /* 128 bytes is maximal length from specs */
+ }
+ switch (Entry->ID) {
+ case SMS_EMSSound10Long:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.0, start);
+ break;
+ case SMS_EMSSound12Long:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 1.2, start);
+ break;
+ case SMS_EMSSonyEricssonSoundLong:
+ FreeBytes = GSM_EncodeEMSSound(Ring, Buffer+3, &Length, 0, start);
+ break;
+ default:
+ break;
+ }
+ Buffer[0] = 0x0C; /* ID for EMS sound */
+ Buffer[1] = Length+1; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,Length+3,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-Length-1] = UsedText;
+ start = false;
+ }
+
+ Entry->RingtoneNotes = Entry->Ringtone->NoteTone.NrCommands;
+
+ break;
+ case SMS_EMSAnimation:
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ if (Entry->Bitmap->Bitmap[0].BitmapWidth > 8 || Entry->Bitmap->Bitmap[0].BitmapHeight > 8) {
+ BitmapType = GSM_EMSMediumPicture; /* Bitmap 16x16 */
+ Buffer[0] = 0x0E; /* ID for 16x16 animation */
+ } else {
+ BitmapType = GSM_EMSSmallPicture; /* Bitmap 8x8 */
+ Buffer[0] = 0x0F; /* ID for 8x8 animation */
+ }
+ Length = PHONE_GetBitmapSize(BitmapType,0,0);
+
+ Buffer[1] = Length*Entry->Bitmap->Number + 1; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ for (j=0;j<Entry->Bitmap->Number;j++) {
+ PHONE_EncodeBitmap(BitmapType, Buffer+3+j*Length, &Entry->Bitmap->Bitmap[j]);
+ }
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,3+Length*Entry->Bitmap->Number,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-1-Length*Entry->Bitmap->Number] = UsedText;
+ break;
+ case SMS_EMSFixedBitmap:
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ if (Entry->Bitmap->Bitmap[0].BitmapWidth > 16 || Entry->Bitmap->Bitmap[0].BitmapHeight > 16) {
+ BitmapType = GSM_EMSBigPicture; /* Bitmap 32x32 */
+ Buffer[0] = 0x10; /* ID for EMS bitmap */
+ } else {
+ BitmapType = GSM_EMSMediumPicture; /* Bitmap 16x16 */
+ Buffer[0] = 0x11; /* ID for EMS bitmap */
+ }
+ Length = PHONE_GetBitmapSize(BitmapType,0,0);
+ PHONE_GetBitmapWidthHeight(BitmapType, &Width, &Height);
+
+ Buffer[1] = Length + 1; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ PHONE_EncodeBitmap(BitmapType,Buffer+3, &Entry->Bitmap->Bitmap[0]);
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,3+Length,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-1-Length] = UsedText;
+ break;
+ case SMS_EMSVariableBitmapLong:
+ BitmapType = GSM_EMSVariablePicture;
+ Width = Entry->Bitmap->Bitmap[0].BitmapWidth;
+ Height = Entry->Bitmap->Bitmap[0].BitmapHeight;
+ Bitmap = Entry->Bitmap->Bitmap[0];
+
+ /* First check if we can use classical format */
+ while (1) {
+ /* Width should be multiply of 8 */
+ while (Width % 8 != 0) Width--;
+
+ /* specs */
+ if (Width <= 96 && Height <= 128) break;
+
+ Height--;
+ }
+ Length = PHONE_GetBitmapSize(BitmapType,Width,Height);
+ if (Length <= 128) {
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ Buffer[0] = 0x12; /* ID for EMS bitmap */
+ Buffer[1] = Length + 3; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ Buffer[3] = Width/8; /* Bitmap width/8 */
+ Buffer[4] = Height; /* Bitmap height */
+
+ GSM_ResizeBitmap(&Bitmap, &Entry->Bitmap->Bitmap[0], Width, Height);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Bitmap);
+#endif
+ PHONE_EncodeBitmap(BitmapType,Buffer+5, &Bitmap);
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,5+Length,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-3-Length] = UsedText;
+ break;
+ }
+
+ /* Find free place in first SMS */
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], &UsedText, &FreeText, &FreeBytes);
+ Used = 0;
+ Length = FreeBytes - 3;
+ if (Entry->Protected) Length = Length - 4;
+ if (Length < 0) Length = 128;
+ if (Length > 128) Length = 128;
+
+ /* Checking number of SMS */
+ FreeBytes = 0;
+ while (FreeBytes != Width) {
+ Width2 = 8;
+ while (FreeBytes + Width2 != Width) {
+ if (PHONE_GetBitmapSize(BitmapType,Width2+8,Height) > Length) break;
+
+ Width2 = Width2 + 8;
+ }
+ FreeBytes = FreeBytes + Width2;
+ Length = 128;
+ Used ++;
+ }
+ dbgprintf("Used SMS: %i\n",Used);
+
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = Used+1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ /* Save UPI UDH */
+ Buffer[0] = 0x13; /* ID for UPI */
+ Buffer[1] = 1; /* Length of rest */
+ Buffer[2] = Used; /* Number of used parts */
+
+ /* Find free place in first SMS */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,3,true,&UsedText,&CopiedText,&CopiedSMSText);
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], &UsedText, &FreeText, &FreeBytes);
+ Length = FreeBytes - 3;
+ if (Length < 0) Length = 128;
+ if (Length > 128) Length = 128;
+
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Bitmap);
+#endif
+
+ /* Saving SMS */
+ FreeBytes = 0;
+ while (FreeBytes != Width) {
+ Width2 = 8;
+ while (FreeBytes + Width2 != Width) {
+ if (PHONE_GetBitmapSize(BitmapType,Width2+8,Height) > Length) break;
+
+ Width2 = Width2 + 8;
+ }
+
+ /* Copying part of bitmap to new structure */
+ Bitmap2.BitmapWidth = Width2;
+ Bitmap2.BitmapHeight = Height;
+ GSM_ClearBitmap(&Bitmap2);
+ for (x=0;x<Width2;x++) {
+ for (y=0;y<Height;y++) {
+ if (GSM_IsPointBitmap(&Bitmap,x+FreeBytes,y)) {
+ GSM_SetPointBitmap(&Bitmap2, x, y);
+ }
+ }
+ }
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Bitmap2);
+#endif
+
+ /* Adding new bitmap to SMS */
+ Length = PHONE_GetBitmapSize(BitmapType,Width2,Height);
+ Buffer[0] = 0x12; /* ID for EMS bitmap */
+ Buffer[1] = Length + 3; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ Buffer[3] = Width2/8; /* Bitmap width/8 */
+ Buffer[4] = Height; /* Bitmap height */
+ PHONE_EncodeBitmap(BitmapType,Buffer+5, &Bitmap2);
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,5+Length,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-3-Length] = UsedText;
+
+ FreeBytes = FreeBytes + Width2;
+ Length = 128;
+ }
+ break;
+ case SMS_EMSVariableBitmap:
+ if (Entry->Protected) {
+ Buffer[0] = 0x17; /* ID for ODI */
+ Buffer[1] = 2; /* Length of rest */
+ Buffer[2] = 1; /* Number of protected objects */
+ Buffer[3] = 1; /* 1=Protected,0=Not protected */
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,4,true,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+
+ BitmapType = GSM_EMSVariablePicture;
+ Width = Entry->Bitmap->Bitmap[0].BitmapWidth;
+ Height = Entry->Bitmap->Bitmap[0].BitmapHeight;
+
+ while (1) {
+ /* Width should be multiply of 8 */
+ while (Width % 8 != 0) Width--;
+
+ /* specs */
+ if (PHONE_GetBitmapSize(BitmapType,Width,Height) <= 128) break;
+
+ Height--;
+ }
+
+ Length = PHONE_GetBitmapSize(BitmapType,Width,Height);
+
+ Buffer[0] = 0x12; /* ID for EMS bitmap */
+ Buffer[1] = Length + 3; /* Length of rest */
+ Buffer[2] = 0x00; /* Position in EMS msg */
+ Buffer[3] = Width/8; /* Bitmap width/8 */
+ Buffer[4] = Height; /* Bitmap height */
+
+ GSM_ResizeBitmap(&Bitmap, &Entry->Bitmap->Bitmap[0], Width, Height);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Bitmap);
+#endif
+ PHONE_EncodeBitmap(BitmapType,Buffer+5, &Bitmap);
+ GSM_AddSMS_Text_UDH(SMS,Coding,Buffer,5+Length,true,&UsedText,&CopiedText,&CopiedSMSText);
+ SMS->SMS[SMS->Number].UDH.Text[SMS->SMS[SMS->Number].UDH.Length-3-Length] = UsedText;
+ break;
+ default:
+ break;
+ }
+ }
+
+ SMS->Number++;
+
+ if (UDHType == UDH_ConcatenatedMessages) {
+ UDHID = GSM_MakeSMSIDFromTime();
+ for (i=0;i<SMS->Number;i++) {
+ SMS->SMS[i].UDH.Text[2+1] = UDHID;
+ SMS->SMS[i].UDH.Text[3+1] = SMS->Number;
+ SMS->SMS[i].UDH.Text[4+1] = i+1;
+ }
+ }
+ if (UDHType == UDH_ConcatenatedMessages16bit) {
+ UDHID = GSM_MakeSMSIDFromTime();
+ GSM_GetCurrentDateTime (&Date);
+ for (i=0;i<SMS->Number;i++) {
+ SMS->SMS[i].UDH.Text[2+1] = Date.Hour;
+ SMS->SMS[i].UDH.Text[3+1] = UDHID;
+ SMS->SMS[i].UDH.Text[4+1] = SMS->Number;
+ SMS->SMS[i].UDH.Text[5+1] = i+1;
+ }
+ }
+
+#ifdef DEBUG
+ dbgprintf("SMS number is %i\n",SMS->Number);
+ for (i=0;i<SMS->Number;i++) {
+ dbgprintf("UDH length %i\n",SMS->SMS[i].UDH.Length);
+ DumpMessage(di.df, di.dl, SMS->SMS[i].UDH.Text, SMS->SMS[i].UDH.Length);
+ dbgprintf("SMS length %i\n",UnicodeLength(SMS->SMS[i].Text)*2);
+ DumpMessage(di.df, di.dl, SMS->SMS[i].Text, UnicodeLength(SMS->SMS[i].Text)*2);
+ }
+#endif
+ return ERR_NONE;
+}
+
+static bool AddEMSText(GSM_SMSMessage *SMS, GSM_MultiPartSMSInfo *Info, int *Pos, int Len)
+{
+ int BufferLen;
+
+ if (Len==0) return true;
+
+ if (Info->Entries[Info->EntriesNum].ID!=SMS_ConcatenatedTextLong &&
+ Info->Entries[Info->EntriesNum].ID!=0) {
+ (Info->EntriesNum)++;
+ }
+ BufferLen = UnicodeLength(Info->Entries[Info->EntriesNum].Buffer)*2;
+ switch (SMS->Coding) {
+ case SMS_Coding_8bit:
+// memcpy(Info->Entries[Info->EntriesNum].Buffer+BufferLen,SMS->Text+(*Pos),Len);
+// BufferLen+=Len;
+// (*Pos)+=Len;
+ break;
+ case SMS_Coding_Unicode:
+ case SMS_Coding_Default:
+ Info->Entries[Info->EntriesNum].Buffer = realloc(Info->Entries[Info->EntriesNum].Buffer, BufferLen + (Len * 2) + 2);
+ if (Info->Entries[Info->EntriesNum].Buffer == NULL) return false;
+ memcpy(Info->Entries[Info->EntriesNum].Buffer + BufferLen, SMS->Text + (*Pos) *2, Len * 2);
+ BufferLen += Len * 2;
+ break;
+ }
+ (*Pos)+=Len;
+ Info->Entries[Info->EntriesNum].Buffer[BufferLen] = 0;
+ Info->Entries[Info->EntriesNum].Buffer[BufferLen+1] = 0;
+ Info->Entries[Info->EntriesNum].ID = SMS_ConcatenatedTextLong;
+ return true;
+}
+
+bool GSM_DecodeEMSMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS)
+{
+ int i, w, Pos, z, UPI = 1, width, height;
+ bool RetVal = false, NewPicture = true;
+ GSM_Phone_Bitmap_Types BitmapType;
+ GSM_Bitmap Bitmap,Bitmap2;
+
+ for (i=0;i<MAX_MULTI_SMS;i++) {
+ Info->Entries[i].ID = 0;
+ }
+
+ for (i=0;i<SMS->Number;i++) {
+ Pos = 0;
+ w = 1;
+ while (w < SMS->SMS[i].UDH.Length) {
+ if (Info->EntriesNum + 1 == MAX_MULTI_SMS) {
+ dbgprintf("Couldn't parse SMS, contains too many EMS parts!\n");
+ return false;
+ }
+ switch(SMS->SMS[i].UDH.Text[w]) {
+ case 0x00:
+ dbgprintf("UDH part - linked SMS with 8 bit ID\n");
+ break;
+ case 0x08:
+ dbgprintf("UDH part - linked SMS with 16 bit ID\n");
+ break;
+// case 0x0A:
+// dbgprintf("UDH part - EMS text formatting\n");
+// break;
+ case 0x0B:
+ dbgprintf("UDH part - default EMS sound\n");
+ if (SMS->SMS[i].UDH.Text[w+2] > Pos) {
+ z = Pos;
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].UDH.Text[w+2]-z)) return false;
+ }
+ if (Info->Entries[Info->EntriesNum].ID != 0) (Info->EntriesNum)++;
+ Info->Entries[Info->EntriesNum].Number = SMS->SMS[i].UDH.Text[w+3];
+ Info->Entries[Info->EntriesNum].ID = SMS_EMSPredefinedSound;
+ RetVal = true;
+ break;
+// case 0x0C:
+// dbgprintf("UDH part - EMS sound\n");
+// break;
+ case 0x0D:
+ dbgprintf("UDH part - default EMS animation\n");
+ if (SMS->SMS[i].UDH.Text[w+2] > Pos) {
+ z = Pos;
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].UDH.Text[w+2]-z)) return false;
+ }
+ if (Info->Entries[Info->EntriesNum].ID != 0) (Info->EntriesNum)++;
+ Info->Entries[Info->EntriesNum].Number = SMS->SMS[i].UDH.Text[w+3];
+ Info->Entries[Info->EntriesNum].ID = SMS_EMSPredefinedAnimation;
+ RetVal = true;
+ break;
+ case 0x0E:
+ case 0x0F:
+ if (SMS->SMS[i].UDH.Text[w] == 0x0E) {
+ dbgprintf("UDH part - EMS 16x16 animation\n");
+ BitmapType = GSM_EMSMediumPicture;
+ } else {
+ dbgprintf("UDH part - EMS 8x8 animation\n");
+ BitmapType = GSM_EMSSmallPicture;
+ }
+ dbgprintf("Position - %i\n",SMS->SMS[i].UDH.Text[w+2]);
+ if (SMS->SMS[i].UDH.Text[w+2] > Pos) {
+ z = Pos;
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].UDH.Text[w+2]-z)) return false;
+ }
+ (Info->EntriesNum)++;
+ Info->Entries[Info->EntriesNum].ID = SMS_EMSAnimation;
+ Info->Entries[Info->EntriesNum].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[Info->EntriesNum].Bitmap == NULL) return false;
+ Info->Entries[Info->EntriesNum].Bitmap->Number = 0;
+ for (z=0;z<((SMS->SMS[i].UDH.Text[w+1]-1)/PHONE_GetBitmapSize(BitmapType,0,0));z++) {
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[z].Type = GSM_PictureImage;
+ PHONE_DecodeBitmap(BitmapType,
+ SMS->SMS[i].UDH.Text + w + 3 + PHONE_GetBitmapSize(BitmapType,0,0) * z,
+ &Info->Entries[Info->EntriesNum].Bitmap->Bitmap[z]);
+ Info->Entries[Info->EntriesNum].Bitmap->Number++;
+ }
+ RetVal = true;
+ break;
+ case 0x10:
+ case 0x11:
+ if (SMS->SMS[i].UDH.Text[w] == 0x10) {
+ dbgprintf("UDH part - EMS 32x32 picture\n");
+ BitmapType = GSM_EMSBigPicture;
+ } else {
+ dbgprintf("UDH part - EMS 16x16 picture\n");
+ BitmapType = GSM_EMSMediumPicture;
+ }
+ dbgprintf("Position - %i\n",SMS->SMS[i].UDH.Text[w+2]);
+ if (SMS->SMS[i].UDH.Text[w+2] > Pos) {
+ z = Pos;
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].UDH.Text[w+2]-z)) return false;
+ }
+ if (Info->Entries[Info->EntriesNum].ID != 0) (Info->EntriesNum)++;
+ Info->Entries[Info->EntriesNum].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[Info->EntriesNum].Bitmap == NULL) return false;
+ PHONE_DecodeBitmap(BitmapType,
+ SMS->SMS[i].UDH.Text + w + 3,
+ &Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0]);
+ Info->Entries[Info->EntriesNum].Bitmap->Number = 1;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].Text[0] = 0;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].Text[1] = 0;
+ Info->Entries[Info->EntriesNum].ID = SMS_EMSFixedBitmap;
+ RetVal = true;
+ break;
+ case 0x12:
+ dbgprintf("UDH part - EMS variable width bitmap\n");
+ if (SMS->SMS[i].UDH.Text[w+2] > Pos) {
+ z = Pos;
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].UDH.Text[w+2]-z)) return false;
+ }
+ if (NewPicture) {
+ (Info->EntriesNum)++;
+ Info->Entries[Info->EntriesNum].Bitmap->Number = 0;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapWidth = 0;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapHeight = 0;
+ }
+ Bitmap.BitmapWidth = SMS->SMS[i].UDH.Text[w+3]*8;
+ Bitmap.BitmapHeight = SMS->SMS[i].UDH.Text[w+4];
+ Info->Entries[Info->EntriesNum].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[Info->EntriesNum].Bitmap == NULL) return false;
+ if (NewPicture) {
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapWidth = Bitmap.BitmapWidth;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapHeight = Bitmap.BitmapHeight;
+ PHONE_DecodeBitmap(GSM_EMSVariablePicture,
+ SMS->SMS[i].UDH.Text + w + 5,
+ &Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0]);
+ } else {
+ PHONE_DecodeBitmap(GSM_EMSVariablePicture,
+ SMS->SMS[i].UDH.Text + w + 5,
+ &Bitmap);
+ Bitmap2 = Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0];
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapWidth = Bitmap.BitmapWidth+Bitmap2.BitmapWidth;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].BitmapHeight = Bitmap2.BitmapHeight;
+ for (width=0;width<Bitmap2.BitmapWidth;width++) {
+ for (height=0;height<Bitmap2.BitmapHeight;height++) {
+ if (GSM_IsPointBitmap(&Bitmap2, width, height)) {
+ GSM_SetPointBitmap(&Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0],width,height);
+ } else {
+ GSM_ClearPointBitmap(&Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0],width,height);
+ }
+ }
+ }
+ for (width=0;width<Bitmap.BitmapWidth;width++) {
+ for (height=0;height<Bitmap2.BitmapHeight;height++) {
+ if (GSM_IsPointBitmap(&Bitmap, width, height)) {
+ GSM_SetPointBitmap(&Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0],width+Bitmap2.BitmapWidth,height);
+ } else {
+ GSM_ClearPointBitmap(&Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0],width+Bitmap2.BitmapWidth,height);
+ }
+ }
+ }
+ }
+ if (UPI == 1) {
+ Info->Entries[Info->EntriesNum].Bitmap->Number = 1;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].Text[0] = 0;
+ Info->Entries[Info->EntriesNum].Bitmap->Bitmap[0].Text[1] = 0;
+ Info->Entries[Info->EntriesNum].ID = SMS_EMSVariableBitmap;
+ RetVal = true;
+ NewPicture = true;
+ dbgprintf("New variable picture\n");
+ } else {
+ NewPicture = false;
+ UPI--;
+ }
+ break;
+ case 0x13:
+ dbgprintf("UDH part - UPI\n");
+ dbgprintf("Value %i\n",SMS->SMS[i].UDH.Text[w+2]);
+ UPI = SMS->SMS[i].UDH.Text[w+2];
+ break;
+ case 0x17:
+ dbgprintf("UDH part - Object Distribution Indicator (Media Rights Protecting) ignored now\n");
+ break;
+ default:
+ dbgprintf("UDH part - block %02x\n",SMS->SMS[i].UDH.Text[w]);
+ Info->Unknown = true;
+ } /* switch */
+ w=w+SMS->SMS[i].UDH.Text[w+1]+2;
+ } /* while */
+ if (!AddEMSText(&SMS->SMS[i], Info, &Pos, SMS->SMS[i].Length-Pos)) return false;
+ RetVal = true;
+ }
+ if (RetVal) (Info->EntriesNum)++;
+ return RetVal;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmems.h b/gammu/emb/common/service/sms/gsmems.h
new file mode 100644
index 0000000..6701d28
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmems.h
@@ -0,0 +1,20 @@
+/* (c) 2002-2003 by Marcin Wiacek */
+
+#ifndef __gsm_ems_h
+#define __gsm_ems_h
+
+#include "../../gsmcomon.h"
+#include "gsmmulti.h"
+
+GSM_Error GSM_EncodeEMSMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS,
+ GSM_UDH UDHType);
+
+bool GSM_DecodeEMSMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmmulti.c b/gammu/emb/common/service/sms/gsmmulti.c
new file mode 100644
index 0000000..6c1cdcd
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmmulti.c
@@ -0,0 +1,1148 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#include <ctype.h>
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmcal.h"
+#include "../gsmpbk.h"
+#include "../gsmlogo.h"
+#include "../gsmring.h"
+#include "../gsmdata.h"
+#include "../gsmnet.h"
+#include "gsmsms.h"
+#include "gsmmulti.h"
+#include "gsmems.h"
+
+/* ----------------- Splitting SMS into parts ------------------------------ */
+
+unsigned char GSM_MakeSMSIDFromTime(void)
+{
+ GSM_DateTime Date;
+ unsigned char retval;
+
+ GSM_GetCurrentDateTime (&Date);
+ retval = Date.Second;
+ switch (Date.Minute/10) {
+ case 2: case 7: retval = retval + 60; break;
+ case 4: case 8: retval = retval + 120; break;
+ case 9: case 5: case 0: retval = retval + 180; break;
+ }
+ retval += Date.Minute/10;
+ return retval;
+}
+
+void GSM_Find_Free_Used_SMS2(GSM_Coding_Type Coding,GSM_SMSMessage SMS, int *UsedText, int *FreeText, int *FreeBytes)
+{
+ int UsedBytes;
+
+ switch (Coding) {
+ case SMS_Coding_Default:
+ FindDefaultAlphabetLen(SMS.Text,&UsedBytes,UsedText,500);
+ UsedBytes = *UsedText * 7 / 8;
+ if (UsedBytes * 8 / 7 != *UsedText) UsedBytes++;
+ *FreeBytes = GSM_MAX_8BIT_SMS_LENGTH - SMS.UDH.Length - UsedBytes;
+ *FreeText = (GSM_MAX_8BIT_SMS_LENGTH - SMS.UDH.Length) * 8 / 7 - *UsedText;
+ break;
+ case SMS_Coding_Unicode:
+ *UsedText = UnicodeLength(SMS.Text);
+ UsedBytes = *UsedText * 2;
+ *FreeBytes = GSM_MAX_8BIT_SMS_LENGTH - SMS.UDH.Length - UsedBytes;
+ *FreeText = *FreeBytes / 2;
+ break;
+ case SMS_Coding_8bit:
+ *UsedText = UsedBytes = SMS.Length;
+ *FreeBytes = GSM_MAX_8BIT_SMS_LENGTH - SMS.UDH.Length - UsedBytes;
+ *FreeText = *FreeBytes;
+ break;
+ }
+ dbgprintf("UDH len %i, UsedBytes %i, FreeText %i, UsedText %i, FreeBytes %i\n",SMS.UDH.Length,UsedBytes,*FreeText,*UsedText,*FreeBytes);
+}
+
+GSM_Error GSM_AddSMS_Text_UDH(GSM_MultiSMSMessage *SMS,
+ GSM_Coding_Type Coding,
+ char *Buffer,
+ int BufferLen,
+ bool UDH,
+ int *UsedText,
+ int *CopiedText,
+ int *CopiedSMSText)
+{
+ int FreeText,FreeBytes,Copy,i,j;
+
+ dbgprintf("Checking used\n");
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], UsedText, &FreeText, &FreeBytes);
+
+ if (UDH) {
+ dbgprintf("Adding UDH\n");
+ if (FreeBytes - BufferLen <= 0) {
+ dbgprintf("Going to the new SMS\n");
+ SMS->Number++;
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], UsedText, &FreeText, &FreeBytes);
+ }
+ if (SMS->SMS[SMS->Number].UDH.Length == 0) {
+ SMS->SMS[SMS->Number].UDH.Length = 1;
+ SMS->SMS[SMS->Number].UDH.Text[0] = 0x00;
+ }
+ memcpy(SMS->SMS[SMS->Number].UDH.Text+SMS->SMS[SMS->Number].UDH.Length,Buffer,BufferLen);
+ SMS->SMS[SMS->Number].UDH.Length += BufferLen;
+ SMS->SMS[SMS->Number].UDH.Text[0] += BufferLen;
+ SMS->SMS[SMS->Number].UDH.Type = UDH_UserUDH;
+ dbgprintf("UDH added %i\n",BufferLen);
+ } else {
+ dbgprintf("Adding text\n");
+ if (FreeText == 0) {
+ dbgprintf("Going to the new SMS\n");
+ SMS->Number++;
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], UsedText, &FreeText, &FreeBytes);
+ }
+
+ Copy = FreeText;
+ dbgprintf("copy %i\n",Copy);
+ if (BufferLen < Copy) Copy = BufferLen;
+ dbgprintf("copy %i\n",Copy);
+
+ switch (Coding) {
+ case SMS_Coding_Default:
+ FindDefaultAlphabetLen(Buffer,&i,&j,FreeText);
+ dbgprintf("def length %i %i\n",i,j);
+ SMS->SMS[SMS->Number].Text[UnicodeLength(SMS->SMS[SMS->Number].Text)*2+i*2] = 0;
+ SMS->SMS[SMS->Number].Text[UnicodeLength(SMS->SMS[SMS->Number].Text)*2+i*2+1] = 0;
+ memcpy(SMS->SMS[SMS->Number].Text+UnicodeLength(SMS->SMS[SMS->Number].Text)*2,Buffer,i*2);
+ *CopiedText = i;
+ *CopiedSMSText = j;
+ SMS->SMS[SMS->Number].Length += i;
+ break;
+ case SMS_Coding_Unicode:
+ SMS->SMS[SMS->Number].Text[UnicodeLength(SMS->SMS[SMS->Number].Text)*2+Copy*2] = 0;
+ SMS->SMS[SMS->Number].Text[UnicodeLength(SMS->SMS[SMS->Number].Text)*2+Copy*2+1] = 0;
+ memcpy(SMS->SMS[SMS->Number].Text+UnicodeLength(SMS->SMS[SMS->Number].Text)*2,Buffer,Copy*2);
+ *CopiedText = *CopiedSMSText = Copy;
+ SMS->SMS[SMS->Number].Length += Copy;
+ break;
+ case SMS_Coding_8bit:
+ memcpy(SMS->SMS[SMS->Number].Text+SMS->SMS[SMS->Number].Length,Buffer,Copy);
+ SMS->SMS[SMS->Number].Length += Copy;
+ *CopiedText = *CopiedSMSText = Copy;
+ break;
+ }
+ dbgprintf("Text added\n");
+ }
+
+ dbgprintf("Checking on the end\n");
+ GSM_Find_Free_Used_SMS2(Coding,SMS->SMS[SMS->Number], UsedText, &FreeText, &FreeBytes);
+
+ return ERR_NONE;
+}
+
+void GSM_MakeMultiPartSMS(GSM_MultiSMSMessage *SMS,
+ unsigned char *MessageBuffer,
+ int MessageLength,
+ GSM_UDH UDHType,
+ GSM_Coding_Type Coding,
+ int Class,
+ unsigned char ReplaceMessage)
+{
+ int j,Len,UsedText,CopiedText,CopiedSMSText;
+ unsigned char UDHID;
+ GSM_DateTime Date;
+
+ Len = 0;
+ while(1) {
+ GSM_SetDefaultSMSData(&SMS->SMS[SMS->Number]);
+ SMS->SMS[SMS->Number].Class = Class;
+ SMS->SMS[SMS->Number].Coding = Coding;
+
+ SMS->SMS[SMS->Number].UDH.Type = UDHType;
+ GSM_EncodeUDHHeader(&SMS->SMS[SMS->Number].UDH);
+
+ if (Coding == SMS_Coding_8bit) {
+ GSM_AddSMS_Text_UDH(SMS,Coding,MessageBuffer+Len,MessageLength - Len,false,&UsedText,&CopiedText,&CopiedSMSText);
+ } else {
+ GSM_AddSMS_Text_UDH(SMS,Coding,MessageBuffer+Len*2,MessageLength - Len,false,&UsedText,&CopiedText,&CopiedSMSText);
+ }
+ Len += CopiedText;
+ dbgprintf("%i %i\n",Len,MessageLength);
+ if (Len == MessageLength) break;
+ if (SMS->Number == MAX_MULTI_SMS) break;
+ SMS->Number++;
+ }
+
+ SMS->Number++;
+
+ UDHID = GSM_MakeSMSIDFromTime();
+ GSM_GetCurrentDateTime (&Date);
+ for (j=0;j<SMS->Number;j++) {
+ SMS->SMS[j].UDH.Type = UDHType;
+ SMS->SMS[j].UDH.ID8bit = UDHID;
+ SMS->SMS[j].UDH.ID16bit = UDHID + 256 * Date.Hour;
+ SMS->SMS[j].UDH.PartNumber = j+1;
+ SMS->SMS[j].UDH.AllParts = SMS->Number;
+ GSM_EncodeUDHHeader(&SMS->SMS[j].UDH);
+ }
+ if (SMS->Number == 1) SMS->SMS[0].ReplaceMessage = ReplaceMessage;
+}
+
+/* Calculates number of SMS and number of left chars in SMS */
+void GSM_SMSCounter(int MessageLength,
+ unsigned char *MessageBuffer,
+ GSM_UDH UDHType,
+ GSM_Coding_Type Coding,
+ int *SMSNum,
+ int *CharsLeft)
+{
+ int UsedText,FreeBytes;
+ GSM_MultiSMSMessage MultiSMS;
+
+ MultiSMS.Number = 0;
+ GSM_MakeMultiPartSMS(&MultiSMS,MessageBuffer,MessageLength,UDHType,Coding,-1,false);
+ GSM_Find_Free_Used_SMS2(Coding,MultiSMS.SMS[MultiSMS.Number-1], &UsedText, CharsLeft, &FreeBytes);
+ *SMSNum = MultiSMS.Number;
+}
+
+/* Nokia Smart Messaging 3.0 */
+static void GSM_EncodeSMS30MultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ char *Buffer, int *Length)
+{
+ int len;
+
+ /*SM version. Here 3.0*/
+ Buffer[(*Length)++] = 0x30;
+
+ if (Info->Entries[0].ID == SMS_NokiaProfileLong) {
+ if (Info->Entries[0].Buffer != NULL) {
+ if (Info->Entries[0].Buffer[0]!=0x00 || Info->Entries[0].Buffer[1]!=0x00) {
+ Buffer[(*Length)++] = SM30_PROFILENAME;
+ Buffer[(*Length)++] = 0x00;
+ Buffer[(*Length)++] = 2*UnicodeLength(Info->Entries[0].Buffer);
+ CopyUnicodeString(Buffer+(*Length),Info->Entries[0].Buffer);
+ *Length = *Length + 2*UnicodeLength(Info->Entries[0].Buffer);
+ }
+ }
+ if (Info->Entries[0].Ringtone != NULL) {
+ Buffer[(*Length)++] = SM30_RINGTONE;
+ /* Length for this part later will be changed */
+ Buffer[(*Length)++] = 0x01;
+ Buffer[(*Length)++] = 0x00;
+ /* Smart Messaging 3.0 says: 16*9=144 bytes,
+ * but on 3310 4.02 it was possible to save about 196 chars
+ * (without cutting) */
+ len = 196;
+ Info->Entries[0].RingtoneNotes=GSM_EncodeNokiaRTTLRingtone(*Info->Entries[0].Ringtone,Buffer+(*Length),&len);
+ Buffer[(*Length)-2] = len / 256;
+ Buffer[(*Length)-1] = len % 256;
+ *Length = *Length + len;
+ }
+ }
+ if (Info->Entries[0].Bitmap != NULL) {
+ if (Info->Entries[0].ID == SMS_NokiaPictureImageLong) {
+ Buffer[(*Length)++] = SM30_OTA;
+ } else {
+ Buffer[(*Length)++] = SM30_SCREENSAVER;
+ }
+ Buffer[(*Length)++] = 0x01;
+ Buffer[(*Length)++] = 0x00;
+ NOKIA_CopyBitmap(GSM_NokiaPictureImage, &Info->Entries[0].Bitmap->Bitmap[0], Buffer, Length);
+ if (Info->Entries[0].Bitmap->Bitmap[0].Text[0]!=0 || Info->Entries[0].Bitmap->Bitmap[0].Text[1]!=0) {
+ if (Info->UnicodeCoding) {
+ Buffer[(*Length)++] = SM30_UNICODETEXT;
+ /* Length for text part */
+ Buffer[(*Length)++] = 0x00;
+ Buffer[(*Length)++] = UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text)*2;
+ memcpy(Buffer+(*Length),Info->Entries[0].Bitmap->Bitmap[0].Text,UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text)*2);
+ *Length = *Length + UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text)*2;
+ } else {
+ /*ID for ISO-8859-1 text*/
+ Buffer[(*Length)++] = SM30_ISOTEXT;
+ Buffer[(*Length)++] = 0x00;
+ Buffer[(*Length)++] = UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text);
+ memcpy(Buffer+(*Length),DecodeUnicodeString(Info->Entries[0].Bitmap->Bitmap[0].Text),UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text));
+ *Length = *Length +UnicodeLength(Info->Entries[0].Bitmap->Bitmap[0].Text);
+ }
+ }
+ }
+}
+
+/* Alcatel docs from www.alcatel.com/wap/ahead */
+GSM_Error GSM_EncodeAlcatelMultiPartSMS(GSM_MultiSMSMessage *SMS,
+ unsigned char *Data,
+ int Len,
+ unsigned char *Name,
+ int Type)
+{
+ unsigned char buff[100],UDHID;
+ int p,i;
+ GSM_UDHHeader MyUDH;
+
+ for (i=0;i<MAX_MULTI_SMS;i++) {
+ GSM_SetDefaultSMSData(&SMS->SMS[i]);
+ SMS->SMS[i].UDH.Type = UDH_UserUDH;
+ SMS->SMS[i].UDH.Text[1] = 0x80; /* Alcatel */
+ p = UnicodeLength(Name);
+ EncodeDefault(buff, Name, &p, true, NULL);
+ SMS->SMS[i].UDH.Text[2] = GSM_PackSevenBitsToEight(0, buff, SMS->SMS[i].UDH.Text+3, p) + 4;
+ SMS->SMS[i].UDH.Text[3] = GSM_PackSevenBitsToEight(0, buff, SMS->SMS[i].UDH.Text+3, p);
+ SMS->SMS[i].UDH.Text[4] = Type;
+ SMS->SMS[i].UDH.Text[5] = Len / 256;
+ SMS->SMS[i].UDH.Text[6] = Len % 256;
+ SMS->SMS[i].UDH.Text[0] = 6 + SMS->SMS[i].UDH.Text[3];
+ SMS->SMS[i].UDH.Length = SMS->SMS[i].UDH.Text[0] + 1;
+
+ if (Len > 140 - SMS->SMS[i].UDH.Length) {
+ MyUDH.Type = UDH_ConcatenatedMessages;
+ GSM_EncodeUDHHeader(&MyUDH);
+
+ memcpy(SMS->SMS[i].UDH.Text+SMS->SMS[i].UDH.Length,MyUDH.Text+1,MyUDH.Length-1);
+ SMS->SMS[i].UDH.Text[0] += MyUDH.Length-1;
+ SMS->SMS[i].UDH.Length += MyUDH.Length-1;
+ }
+
+ SMS->SMS[i].Coding = SMS_Coding_8bit;
+ SMS->SMS[i].Class = 1;
+ }
+
+ p = 0;
+ while (p != Len) {
+ i = 140-SMS->SMS[SMS->Number].UDH.Length;
+ if (Len - p < i) i = Len - p;
+ memcpy(SMS->SMS[SMS->Number].Text,Data+p,i);
+ p += i;
+ SMS->SMS[SMS->Number].Length = i;
+ SMS->Number++;
+
+ }
+
+ /* Linked sms UDH */
+ if (SMS->Number != 1) {
+ UDHID = GSM_MakeSMSIDFromTime();
+ for (i=0;i<SMS->Number;i++) {
+ SMS->SMS[i].UDH.Text[SMS->SMS[i].UDH.Length-3] = UDHID;
+ SMS->SMS[i].UDH.Text[SMS->SMS[i].UDH.Length-2] = SMS->Number;
+ SMS->SMS[i].UDH.Text[SMS->SMS[i].UDH.Length-1] = i+1;
+ }
+ }
+
+ return ERR_NONE;
+}
+
+/* Alcatel docs from www.alcatel.com/wap/ahead and other */
+GSM_Error GSM_EncodeMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS)
+{
+ unsigned char Buffer[GSM_MAX_SMS_LENGTH*2*MAX_MULTI_SMS];
+ unsigned char Buffer2[GSM_MAX_SMS_LENGTH*2*MAX_MULTI_SMS];
+ int Length = 0,smslen,i, Class = -1, j,p;
+ GSM_Error error;
+ GSM_Coding_Type Coding = SMS_Coding_8bit;
+ GSM_UDH UDH = UDH_NoUDH;
+ GSM_UDHHeader UDHHeader;
+ bool EMS = false;
+ int textnum = 0;
+
+ SMS->Number = 0;
+
+ if (Info->Entries[0].ID == SMS_AlcatelSMSTemplateName) {
+ Buffer[Length++] = 0x00; //number of elements
+ for (i=1;i<Info->EntriesNum;i++) {
+ switch (Info->Entries[i].ID) {
+ case SMS_EMSSound10:
+ case SMS_EMSSound12:
+ case SMS_EMSSonyEricssonSound:
+ case SMS_EMSSound10Long:
+ case SMS_EMSSound12Long:
+ case SMS_EMSSonyEricssonSoundLong:
+ case SMS_EMSVariableBitmap:
+ case SMS_EMSAnimation:
+ case SMS_EMSVariableBitmapLong:
+ break;
+ case SMS_EMSPredefinedSound:
+ Buffer[0]++;
+ Buffer[Length++] = 0x01; //type of data
+ Buffer[Length++] = 1 % 256; //len
+ Buffer[Length++] = 1 / 256; //len
+ Buffer[Length++] = Info->Entries[i].Number;
+ break;
+ case SMS_EMSPredefinedAnimation:
+ Buffer[0]++;
+ Buffer[Length++] = 0x02; //type of data
+ Buffer[Length++] = 1 % 256; //len
+ Buffer[Length++] = 1 / 256; //len
+ Buffer[Length++] = Info->Entries[i].Number;
+ break;
+ case SMS_ConcatenatedTextLong:
+ Buffer[0]++;
+ p = UnicodeLength(Info->Entries[i].Buffer);
+ EncodeDefault(Buffer2, Info->Entries[i].Buffer, &p, true, NULL);
+ Buffer[Length++] = 0x00; //type of data
+ Length = Length + 2;
+ smslen = GSM_PackSevenBitsToEight(0, Buffer2, Buffer+Length, p);
+ Buffer[Length-2] = smslen % 256; //len
+ Buffer[Length-1] = smslen / 256; //len
+ Length = Length + smslen;
+ break;
+ default:
+ return ERR_UNKNOWN;
+ }
+ }
+ Buffer[0] = Buffer[0] * 2;
+ return GSM_EncodeAlcatelMultiPartSMS(SMS,Buffer,Length,Info->Entries[0].Buffer,ALCATELTDD_SMSTEMPLATE);
+ }
+
+ for (i=0;i<Info->EntriesNum;i++) {
+ switch (Info->Entries[i].ID) {
+ case SMS_EMSPredefinedAnimation:
+ case SMS_EMSPredefinedSound:
+ case SMS_EMSSound10:
+ case SMS_EMSSound12:
+ case SMS_EMSSonyEricssonSound:
+ case SMS_EMSSound10Long:
+ case SMS_EMSSound12Long:
+ case SMS_EMSSonyEricssonSoundLong:
+ case SMS_EMSFixedBitmap:
+ case SMS_EMSVariableBitmap:
+ case SMS_EMSAnimation:
+ case SMS_EMSVariableBitmapLong:
+ EMS = true;
+ break;
+ case SMS_ConcatenatedTextLong:
+ case SMS_ConcatenatedTextLong16bit:
+
+ /* This covers situation, when somebody will call function
+ * with two or more SMS_Concatenated.... entries only.
+ * It will be still only linked sms, but functions below
+ * will pack only first entry according to own limits.
+ * We redirect to EMS functions, because they are more generic
+ * here and will handle it correctly and produce linked sms
+ * from all entries
+ */
+ textnum ++;
+ if (textnum > 1) EMS = true;
+
+ if (Info->Entries[i].Left || Info->Entries[i].Right ||
+ Info->Entries[i].Center || Info->Entries[i].Large ||
+ Info->Entries[i].Small || Info->Entries[i].Bold ||
+ Info->Entries[i].Italic || Info->Entries[i].Underlined ||
+ Info->Entries[i].Strikethrough) {
+ EMS = true;
+ }
+ default:
+ break;
+ }
+ if (EMS) break;
+ }
+ if (EMS) {
+ error=GSM_EncodeEMSMultiPartSMS(Info,SMS,UDH_NoUDH);
+ if (error != ERR_NONE) return error;
+ if (SMS->Number != 1) {
+ SMS->Number = 0;
+ for (i=0;i<Info->EntriesNum;i++) {
+ if (Info->Entries[i].ID == SMS_ConcatenatedTextLong16bit) {
+ return GSM_EncodeEMSMultiPartSMS(Info,SMS,UDH_ConcatenatedMessages);
+ }
+ }
+ return GSM_EncodeEMSMultiPartSMS(Info,SMS,UDH_ConcatenatedMessages16bit);
+ }
+ return error;
+ }
+
+ if (Info->EntriesNum != 1) return ERR_UNKNOWN;
+
+ switch (Info->Entries[0].ID) {
+ case SMS_AlcatelMonoBitmapLong:
+ Buffer[0] = Info->Entries[0].Bitmap->Bitmap[0].BitmapWidth;
+ Buffer[1] = Info->Entries[0].Bitmap->Bitmap[0].BitmapHeight;
+ PHONE_EncodeBitmap(GSM_AlcatelBMMIPicture, Buffer+2, &Info->Entries[0].Bitmap->Bitmap[0]);
+ Length = PHONE_GetBitmapSize(GSM_AlcatelBMMIPicture,Info->Entries[0].Bitmap->Bitmap[0].BitmapWidth,Info->Entries[0].Bitmap->Bitmap[0].BitmapHeight)+2;
+ return GSM_EncodeAlcatelMultiPartSMS(SMS,Buffer,Length,Info->Entries[0].Bitmap->Bitmap[0].Text,ALCATELTDD_PICTURE);
+ case SMS_AlcatelMonoAnimationLong:
+ /* Number of sequence words */
+ Buffer[0] = (Info->Entries[0].Bitmap->Number+1) % 256;
+ Buffer[1] = (Info->Entries[0].Bitmap->Number+1) / 256;
+ /* Picture display time 1 second (1 = 100ms) */
+ Buffer[2] = 10 % 256;
+ Buffer[3] = 10 / 256 + 0xF0;
+
+ Length = 4;
+ j = 0;
+
+ /* Offsets to bitmaps */
+ for (i=0;i<Info->Entries[0].Bitmap->Number;i++) {
+ Buffer[Length++] = (4+j+Info->Entries[0].Bitmap->Number*2) % 256;
+ Buffer[Length++] = (4+j+Info->Entries[0].Bitmap->Number*2) / 256;
+ j += PHONE_GetBitmapSize(GSM_AlcatelBMMIPicture,Info->Entries[0].Bitmap->Bitmap[i].BitmapWidth,Info->Entries[0].Bitmap->Bitmap[i].BitmapHeight)+2;
+ }
+
+ /* Bitmaps */
+ for (i=0;i<Info->Entries[0].Bitmap->Number;i++) {
+ Buffer[Length++] = Info->Entries[0].Bitmap->Bitmap[i].BitmapWidth;
+ Buffer[Length++] = Info->Entries[0].Bitmap->Bitmap[i].BitmapHeight;
+ PHONE_EncodeBitmap(GSM_AlcatelBMMIPicture, Buffer+Length, &Info->Entries[0].Bitmap->Bitmap[i]);
+ Length += PHONE_GetBitmapSize(GSM_AlcatelBMMIPicture,Info->Entries[0].Bitmap->Bitmap[i].BitmapWidth,Info->Entries[0].Bitmap->Bitmap[i].BitmapHeight);
+ }
+ return GSM_EncodeAlcatelMultiPartSMS(SMS,Buffer,Length,Info->Entries[0].Bitmap->Bitmap[0].Text,ALCATELTDD_ANIMATION);
+ case SMS_MMSIndicatorLong:
+ Class = 1;
+ UDH = UDH_MMSIndicatorLong;
+ GSM_EncodeMMSIndicatorSMSText(Buffer,&Length,*Info->Entries[0].MMSIndicator);
+ break;
+ case SMS_NokiaRingtoneLong:
+ case SMS_NokiaRingtone:
+ UDH = UDH_NokiaRingtone;
+ Class = 1;
+ /* 7 = length of UDH_NokiaRingtone UDH header */
+ Length = GSM_MAX_8BIT_SMS_LENGTH-7;
+ Info->Entries[0].RingtoneNotes = GSM_EncodeNokiaRTTLRingtone(*Info->Entries[0].Ringtone,Buffer,&Length);
+ if (Info->Entries[0].ID == SMS_NokiaRingtone) break;
+ if (Info->Entries[0].RingtoneNotes != Info->Entries[0].Ringtone->NoteTone.NrCommands) {
+ UDH = UDH_NokiaRingtoneLong;
+ Length = (GSM_MAX_8BIT_SMS_LENGTH-12)*3;
+ Info->Entries[0].RingtoneNotes = GSM_EncodeNokiaRTTLRingtone(*Info->Entries[0].Ringtone,Buffer,&Length);
+ }
+ break;
+ case SMS_NokiaOperatorLogoLong:
+ if (Info->Entries[0].Bitmap->Bitmap[0].BitmapWidth > 72 || Info->Entries[0].Bitmap->Bitmap[0].BitmapHeight > 14) {
+ UDH = UDH_NokiaOperatorLogoLong;
+ Class = 1;
+ NOKIA_EncodeNetworkCode(Buffer, Info->Entries[0].Bitmap->Bitmap[0].NetworkCode);
+ Length = Length + 3;
+ NOKIA_CopyBitmap(GSM_Nokia7110OperatorLogo, &Info->Entries[0].Bitmap->Bitmap[0], Buffer, &Length);
+ break;
+ }
+ case SMS_NokiaOperatorLogo:
+ UDH = UDH_NokiaOperatorLogo;
+ Class = 1;
+ NOKIA_EncodeNetworkCode(Buffer, Info->Entries[0].Bitmap->Bitmap[0].NetworkCode);
+ Length = Length + 3;
+ NOKIA_CopyBitmap(GSM_NokiaOperatorLogo, &Info->Entries[0].Bitmap->Bitmap[0], Buffer, &Length);
+ break;
+ case SMS_NokiaCallerLogo:
+ UDH = UDH_NokiaCallerLogo;
+ Class = 1;
+ NOKIA_CopyBitmap(GSM_NokiaCallerLogo, &Info->Entries[0].Bitmap->Bitmap[0], Buffer, &Length);
+ break;
+ case SMS_NokiaProfileLong:
+ case SMS_NokiaPictureImageLong:
+ case SMS_NokiaScreenSaverLong:
+ Class = 1;
+ UDH = UDH_NokiaProfileLong;
+ GSM_EncodeSMS30MultiPartSMS(Info,Buffer,&Length);
+ break;
+ case SMS_NokiaWAPBookmarkLong:
+ Class = 1;
+ NOKIA_EncodeWAPBookmarkSMSText(Buffer,&Length,Info->Entries[0].Bookmark);
+ /* 7 = length of UDH_NokiaWAP UDH header */
+ if (Length>(GSM_MAX_8BIT_SMS_LENGTH-7)) {
+ UDH=UDH_NokiaWAPLong;
+ } else {
+ UDH=UDH_NokiaWAP;
+ }
+ break;
+ case SMS_NokiaWAPSettingsLong:
+ Class = 1;
+ UDH = UDH_NokiaWAPLong;
+ NOKIA_EncodeWAPMMSSettingsSMSText(Buffer,&Length,Info->Entries[0].Settings,false);
+ break;
+ case SMS_NokiaMMSSettingsLong:
+ Class = 1;
+ UDH = UDH_NokiaWAPLong;
+ NOKIA_EncodeWAPMMSSettingsSMSText(Buffer,&Length,Info->Entries[0].Settings,true);
+ break;
+ case SMS_NokiaVCARD10Long:
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard10);
+ /* is 1 SMS ? 8 = length of ..SCKE2 */
+ if (Length<=GSM_MAX_SMS_LENGTH-8) {
+ sprintf(Buffer,"//SCKE2 ");
+ Length = 8;
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard10);
+ } else {
+ /* FIXME: It wasn't checked */
+ UDH = UDH_NokiaPhonebookLong;
+ }
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_NokiaVCARD21Long:
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard21);
+ /* Is 1 SMS ? 12 = length of ..SCKL23F4 */
+ if (Length<=GSM_MAX_SMS_LENGTH-12) {
+ sprintf(Buffer,"//SCKL23F4%c%c",13,10);
+ Length = 12;
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard21);
+ } else {
+ UDH = UDH_NokiaPhonebookLong;
+ /* Here can be also 8 bit coding */
+ }
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_VCARD10Long:
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard10);
+ if (Length>GSM_MAX_SMS_LENGTH) UDH = UDH_ConcatenatedMessages;
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_VCARD21Long:
+ GSM_EncodeVCARD(Buffer,&Length,Info->Entries[0].Phonebook,true,Nokia_VCard21);
+ if (Length>GSM_MAX_SMS_LENGTH) UDH = UDH_ConcatenatedMessages;
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_NokiaVCALENDAR10Long:
+ error=GSM_EncodeVCALENDAR(Buffer,&Length,Info->Entries[0].Calendar,true,Nokia_VCalendar);
+ if (error != ERR_NONE) return error;
+ /* Is 1 SMS ? 8 = length of ..SCKE4 */
+ if (Length<=GSM_MAX_SMS_LENGTH-8) {
+ sprintf(Buffer,"//SCKE4 ");
+ Length = 8;
+ GSM_EncodeVCALENDAR(Buffer,&Length,Info->Entries[0].Calendar,true,Nokia_VCalendar);
+ } else {
+ UDH = UDH_NokiaCalendarLong;
+ /* can be here 8 bit coding ? */
+ }
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_NokiaVTODOLong:
+ error=GSM_EncodeVTODO(Buffer,&Length,Info->Entries[0].ToDo,true,Nokia_VToDo);
+ if (error != ERR_NONE) return error;
+ UDH = UDH_NokiaCalendarLong;
+ Coding = SMS_Coding_Default;
+ memcpy(Buffer2,Buffer,Length);
+ EncodeUnicode(Buffer,Buffer2,Length);
+ break;
+ case SMS_DisableVoice:
+ case SMS_DisableFax:
+ case SMS_DisableEmail:
+ case SMS_EnableVoice:
+ case SMS_EnableFax:
+ case SMS_EnableEmail:
+ case SMS_VoidSMS:
+ case SMS_Text:
+ Class = Info->Class;
+ switch (Info->Entries[0].ID) {
+ case SMS_DisableVoice : UDH = UDH_DisableVoice; break;
+ case SMS_DisableFax : UDH = UDH_DisableFax; break;
+ case SMS_DisableEmail : UDH = UDH_DisableEmail; break;
+ case SMS_EnableVoice : UDH = UDH_EnableVoice; break;
+ case SMS_EnableFax : UDH = UDH_EnableFax; break;
+ case SMS_EnableEmail : UDH = UDH_EnableEmail; break;
+ case SMS_VoidSMS : UDH = UDH_VoidSMS; break;
+ case SMS_Text : UDH = UDH_NoUDH; break;
+ default : break;
+ }
+ UDHHeader.Type = UDH;
+ GSM_EncodeUDHHeader(&UDHHeader);
+ memcpy(Buffer,Info->Entries[0].Buffer,UnicodeLength(Info->Entries[0].Buffer)*2+2);
+ if (Info->UnicodeCoding) {
+ Coding = SMS_Coding_Unicode;
+ Length = UnicodeLength(Info->Entries[0].Buffer);
+ if (Length>(140-UDHHeader.Length)/2) Length = (140-UDHHeader.Length)/2;
+ } else {
+ Coding = SMS_Coding_Default;
+ FindDefaultAlphabetLen(Info->Entries[0].Buffer,&Length,&smslen,(GSM_MAX_8BIT_SMS_LENGTH-UDHHeader.Length)*8/7);
+ }
+ break;
+ case SMS_ConcatenatedAutoTextLong:
+ case SMS_ConcatenatedAutoTextLong16bit:
+ smslen = UnicodeLength(Info->Entries[0].Buffer);
+ memcpy(Buffer,Info->Entries[0].Buffer,smslen*2);
+ EncodeDefault(Buffer2, Buffer, &smslen, true, NULL);
+ DecodeDefault(Buffer, Buffer2, smslen, true, NULL);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) {
+ dbgprintf("Info->Entries[0].Buffer:\n");
+ DumpMessage(di.df, di.dl, Info->Entries[0].Buffer, UnicodeLength(Info->Entries[0].Buffer)*2);
+ dbgprintf("Buffer:\n");
+ DumpMessage(di.df, di.dl, Buffer, UnicodeLength(Buffer)*2);
+ }
+#endif
+ Info->UnicodeCoding = false;
+ for (smslen=0;smslen<(int)(UnicodeLength(Info->Entries[0].Buffer)*2);smslen++) {
+ if (Info->Entries[0].Buffer[smslen] != Buffer[smslen]) {
+ Info->UnicodeCoding = true;
+ dbgprintf("Setting to Unicode %i\n",smslen);
+ break;
+ }
+ }
+ /* No break here - we go to the SMS_ConcatenatedTextLong */
+ case SMS_ConcatenatedTextLong:
+ case SMS_ConcatenatedTextLong16bit:
+ Class = Info->Class;
+ memcpy(Buffer,Info->Entries[0].Buffer,UnicodeLength(Info->Entries[0].Buffer)*2+2);
+ UDH = UDH_NoUDH;
+ if (Info->UnicodeCoding) {
+ Coding = SMS_Coding_Unicode;
+ Length = UnicodeLength(Info->Entries[0].Buffer);
+ if (Info->Entries[0].ID == SMS_ConcatenatedTextLong16bit ||
+ Info->Entries[0].ID == SMS_ConcatenatedAutoTextLong16bit) {
+ if (Length>70) UDH=UDH_ConcatenatedMessages16bit;
+ } else {
+ if (Length>70) UDH=UDH_ConcatenatedMessages;
+ }
+ } else {
+ Coding = SMS_Coding_Default;
+ FindDefaultAlphabetLen(Info->Entries[0].Buffer,&Length,&smslen,5000);
+ if (Info->Entries[0].ID == SMS_ConcatenatedTextLong16bit ||
+ Info->Entries[0].ID == SMS_ConcatenatedAutoTextLong16bit) {
+ if (smslen>GSM_MAX_SMS_LENGTH) UDH=UDH_ConcatenatedMessages16bit;
+ } else {
+ if (smslen>GSM_MAX_SMS_LENGTH) UDH=UDH_ConcatenatedMessages;
+ }
+ }
+ default:
+ break;
+ }
+ GSM_MakeMultiPartSMS(SMS,Buffer,Length,UDH,Coding,Class,Info->ReplaceMessage);
+ return ERR_NONE;
+}
+
+void GSM_ClearMultiPartSMSInfo(GSM_MultiPartSMSInfo *Info)
+{
+ int i;
+
+ for (i=0;i<MAX_MULTI_SMS;i++) {
+ Info->Entries[i].Number = 0;
+ Info->Entries[i].Ringtone = NULL;
+ Info->Entries[i].Bitmap = NULL;
+ Info->Entries[i].Bookmark = NULL;
+ Info->Entries[i].Settings = NULL;
+ Info->Entries[i].MMSIndicator = NULL;
+ Info->Entries[i].Phonebook = NULL;
+ Info->Entries[i].Calendar = NULL;
+ Info->Entries[i].ToDo = NULL;
+ Info->Entries[i].Protected = false;
+
+ Info->Entries[i].Buffer = NULL;
+ Info->Entries[i].Left = false;
+ Info->Entries[i].Right = false;
+ Info->Entries[i].Center = false;
+ Info->Entries[i].Large = false;
+ Info->Entries[i].Small = false;
+ Info->Entries[i].Bold = false;
+ Info->Entries[i].Italic = false;
+ Info->Entries[i].Underlined = false;
+ Info->Entries[i].Strikethrough = false;
+
+ Info->Entries[i].RingtoneNotes = 0;
+ }
+ Info->Unknown = false;
+ Info->EntriesNum = 0;
+ Info->Class = -1;
+ Info->ReplaceMessage = 0;
+ Info->UnicodeCoding = false;
+}
+
+void GSM_FreeMultiPartSMSInfo(GSM_MultiPartSMSInfo *Info)
+{
+ int i;
+
+ for (i=0;i<MAX_MULTI_SMS;i++) {
+ free(Info->Entries[i].Ringtone);
+ Info->Entries[i].Ringtone = NULL;
+ free(Info->Entries[i].Bitmap);
+ Info->Entries[i].Bitmap = NULL;
+ free(Info->Entries[i].Bookmark);
+ Info->Entries[i].Bookmark = NULL;
+ free(Info->Entries[i].Settings);
+ Info->Entries[i].Settings = NULL;
+ free(Info->Entries[i].MMSIndicator);
+ Info->Entries[i].MMSIndicator = NULL;
+ free(Info->Entries[i].Phonebook);
+ Info->Entries[i].Phonebook = NULL;
+ free(Info->Entries[i].Calendar);
+ Info->Entries[i].Calendar = NULL;
+ free(Info->Entries[i].ToDo);
+ Info->Entries[i].ToDo = NULL;
+ free(Info->Entries[i].Buffer);
+ Info->Entries[i].Buffer = NULL;
+ }
+}
+
+/* ----------------- Joining SMS from parts -------------------------------- */
+
+bool GSM_DecodeMultiPartSMS(GSM_MultiPartSMSInfo *Info,
+ GSM_MultiSMSMessage *SMS,
+ bool ems)
+{
+ int i, Length = 0;
+ char Buffer[GSM_MAX_SMS_LENGTH*2*MAX_MULTI_SMS];
+ bool emsexist = false;
+
+ GSM_ClearMultiPartSMSInfo(Info);
+ if (ems) {
+ emsexist = true;
+ for (i=0;i<SMS->Number;i++) {
+ if (SMS->SMS[i].UDH.Type != UDH_ConcatenatedMessages &&
+ SMS->SMS[i].UDH.Type != UDH_ConcatenatedMessages16bit &&
+ SMS->SMS[i].UDH.Type != UDH_UserUDH) {
+ emsexist = false;
+ break;
+ }
+ }
+ }
+
+ /* EMS decoding */
+ if (emsexist) return GSM_DecodeEMSMultiPartSMS(Info,SMS);
+
+ /* Smart Messaging decoding */
+ if (SMS->SMS[0].UDH.Type == UDH_NokiaRingtone && SMS->Number == 1) {
+ Info->Entries[0].Ringtone = (GSM_Ringtone *)malloc(sizeof(GSM_Ringtone));
+ if (Info->Entries[0].Ringtone == NULL) return false;
+ if (GSM_DecodeNokiaRTTLRingtone(Info->Entries[0].Ringtone, SMS->SMS[0].Text, SMS->SMS[0].Length)==ERR_NONE) {
+ Info->Entries[0].ID = SMS_NokiaRingtone;
+ Info->EntriesNum = 1;
+ return true;
+ }
+ }
+ if (SMS->SMS[0].UDH.Type == UDH_NokiaCallerLogo && SMS->Number == 1) {
+ Info->Entries[0].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[0].Bitmap == NULL) return false;
+ Info->Entries[0].Bitmap->Number = 1;
+ PHONE_DecodeBitmap(GSM_NokiaCallerLogo, SMS->SMS[0].Text+4, &Info->Entries[0].Bitmap->Bitmap[0]);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Info->Entries[0].Bitmap->Bitmap[0]);
+#endif
+ Info->Entries[0].ID = SMS_NokiaCallerLogo;
+ Info->EntriesNum = 1;
+ return true;
+ }
+ if (SMS->SMS[0].UDH.Type == UDH_NokiaOperatorLogo && SMS->Number == 1) {
+ Info->Entries[0].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[0].Bitmap == NULL) return false;
+ Info->Entries[0].Bitmap->Number = 1;
+ PHONE_DecodeBitmap(GSM_NokiaOperatorLogo, SMS->SMS[0].Text+7, &Info->Entries[0].Bitmap->Bitmap[0]);
+ NOKIA_DecodeNetworkCode(SMS->SMS[0].Text, Info->Entries[0].Bitmap->Bitmap[0].NetworkCode);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Info->Entries[0].Bitmap->Bitmap[0]);
+#endif
+ Info->Entries[0].ID = SMS_NokiaOperatorLogo;
+ Info->EntriesNum = 1;
+ return true;
+ }
+ if (SMS->SMS[0].UDH.Type == UDH_NokiaProfileLong) {
+ for (i=0;i<SMS->Number;i++) {
+ if (SMS->SMS[i].UDH.Type != UDH_NokiaProfileLong ||
+ SMS->SMS[i].UDH.Text[11] != i+1 ||
+ SMS->SMS[i].UDH.Text[10] != SMS->Number) {
+ return false;
+ }
+ memcpy(Buffer+Length,SMS->SMS[i].Text,SMS->SMS[i].Length);
+ Length = Length + SMS->SMS[i].Length;
+ }
+ Info->EntriesNum = 1;
+ Info->Entries[0].ID = SMS_NokiaPictureImageLong;
+ Info->Entries[0].Bitmap = (GSM_MultiBitmap *)malloc(sizeof(GSM_MultiBitmap));
+ if (Info->Entries[0].Bitmap == NULL) return false;
+ Info->Entries[0].Bitmap->Number = 1;
+ Info->Entries[0].Bitmap->Bitmap[0].Text[0] = 0;
+ Info->Entries[0].Bitmap->Bitmap[0].Text[1] = 0;
+ i=1;
+ while (i!=Length) {
+ switch (Buffer[i]) {
+ case SM30_ISOTEXT:
+ dbgprintf("ISO 8859-2 text\n");
+ Info->Unknown = true;
+ break;
+ case SM30_UNICODETEXT:
+ dbgprintf("Unicode text\n");
+ memcpy(Info->Entries[0].Bitmap->Bitmap[0].Text,Buffer+i+3,Buffer[i+1]*256+Buffer[i+2]);
+ Info->Entries[0].Bitmap->Bitmap[0].Text[Buffer[i+1]*256 + Buffer[i+2]] = 0;
+ Info->Entries[0].Bitmap->Bitmap[0].Text[Buffer[i+1]*256 + Buffer[i+2]+ 1] = 0;
+ dbgprintf("Unicode Text \"%s\"\n",DecodeUnicodeString(Info->Entries[0].Bitmap->Bitmap[0].Text));
+ break;
+ case SM30_OTA:
+ dbgprintf("OTA bitmap as Picture Image\n");
+ PHONE_DecodeBitmap(GSM_NokiaPictureImage, Buffer + i + 7, &Info->Entries[0].Bitmap->Bitmap[0]);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Info->Entries[0].Bitmap->Bitmap[0]);
+#endif
+ break;
+ case SM30_RINGTONE:
+ dbgprintf("RTTL ringtone\n");
+ Info->Unknown = true;
+ break;
+ case SM30_PROFILENAME:
+ dbgprintf("Profile Name\n");
+ Info->Entries[0].ID = SMS_NokiaProfileLong;
+ Info->Unknown = true;
+ break;
+ case SM30_SCREENSAVER:
+ dbgprintf("OTA bitmap as Screen Saver\n");
+ PHONE_DecodeBitmap(GSM_NokiaPictureImage, Buffer + i + 7, &Info->Entries[0].Bitmap->Bitmap[0]);
+#ifdef DEBUG
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) GSM_PrintBitmap(di.df,&Info->Entries[0].Bitmap->Bitmap[0]);
+#endif
+ Info->Entries[0].ID = SMS_NokiaScreenSaverLong;
+ break;
+ }
+ i = i + Buffer[i+1]*256 + Buffer[i+2] + 3;
+ dbgprintf("%i %i\n",i,Length);
+ }
+ return true;
+ }
+
+ /* Linked sms */
+ if (SMS->SMS[0].UDH.Type == UDH_ConcatenatedMessages ||
+ SMS->SMS[0].UDH.Type == UDH_ConcatenatedMessages16bit) {
+ Info->EntriesNum = 1;
+ Info->Entries[0].ID = SMS_ConcatenatedTextLong;
+ if (SMS->SMS[0].UDH.Type == UDH_ConcatenatedMessages16bit) {
+ Info->Entries[0].ID = SMS_ConcatenatedTextLong16bit;
+ }
+
+ for (i=0;i<SMS->Number;i++) {
+ switch (SMS->SMS[i].Coding) {
+ case SMS_Coding_8bit:
+ Info->Entries[0].Buffer = realloc(Info->Entries[0].Buffer, Length + SMS->SMS[i].Length + 2);
+ if (Info->Entries[0].Buffer == NULL) return false;
+
+ memcpy(Info->Entries[0].Buffer + Length, SMS->SMS[i].Text, SMS->SMS[i].Length);
+ Length=Length+SMS->SMS[i].Length;
+ break;
+ case SMS_Coding_Unicode:
+ if (Info->Entries[0].ID == SMS_ConcatenatedTextLong) {
+ Info->Entries[0].ID = SMS_ConcatenatedAutoTextLong;
+ }
+ if (Info->Entries[0].ID == SMS_ConcatenatedTextLong16bit) {
+ Info->Entries[0].ID = SMS_ConcatenatedAutoTextLong16bit;
+ }
+ case SMS_Coding_Default:
+ Info->Entries[0].Buffer = realloc(Info->Entries[0].Buffer, Length + UnicodeLength(SMS->SMS[i].Text)*2 + 2);
+ if (Info->Entries[0].Buffer == NULL) return false;
+
+ memcpy(Info->Entries[0].Buffer+Length,SMS->SMS[i].Text,UnicodeLength(SMS->SMS[i].Text)*2);
+ Length=Length+UnicodeLength(SMS->SMS[i].Text)*2;
+ break;
+ }
+ }
+ Info->Entries[0].Buffer[Length] = 0;
+ Info->Entries[0].Buffer[Length+1] = 0;
+ return true;
+ }
+
+ return false;
+}
+
+GSM_Error GSM_LinkSMS(GSM_MultiSMSMessage **INPUT, GSM_MultiSMSMessage **OUTPUT, bool ems)
+{
+ bool *INPUTSorted, copyit;
+ int i,OUTPUTNum,j,z,w;
+
+ i = 0;
+ while (INPUT[i] != NULL) i++;
+
+ INPUTSorted = calloc(i, sizeof(bool));
+ if (INPUTSorted == NULL) return ERR_MOREMEMORY;
+
+ OUTPUTNum = 0;
+ OUTPUT[0] = NULL;
+
+ if (ems) {
+ i=0;
+ while (INPUT[i] != NULL) {
+ if (INPUT[i]->SMS[0].UDH.Type == UDH_UserUDH) {
+ w=1;
+ while (w < INPUT[i]->SMS[0].UDH.Length) {
+ switch(INPUT[i]->SMS[0].UDH.Text[w]) {
+ case 0x00:
+ dbgprintf("Adding ID to user UDH - linked SMS with 8 bit ID\n");
+ INPUT[i]->SMS[0].UDH.ID8bit = INPUT[i]->SMS[0].UDH.Text[w+2];
+ INPUT[i]->SMS[0].UDH.AllParts = INPUT[i]->SMS[0].UDH.Text[w+3];
+ INPUT[i]->SMS[0].UDH.PartNumber = INPUT[i]->SMS[0].UDH.Text[w+4];
+ break;
+ case 0x08:
+ dbgprintf("Adding ID to user UDH - linked SMS with 16 bit ID\n");
+ INPUT[i]->SMS[0].UDH.ID16bit = INPUT[i]->SMS[0].UDH.Text[w+2]*256+INPUT[i]->SMS[0].UDH.Text[w+3];
+ INPUT[i]->SMS[0].UDH.AllParts = INPUT[i]->SMS[0].UDH.Text[w+4];
+ INPUT[i]->SMS[0].UDH.PartNumber = INPUT[i]->SMS[0].UDH.Text[w+5];
+ break;
+ default:
+ dbgprintf("Block %02x\n",INPUT[i]->SMS[0].UDH.Text[w]);
+ }
+ dbgprintf("%i %i %i %i\n",
+ INPUT[i]->SMS[0].UDH.ID8bit,
+ INPUT[i]->SMS[0].UDH.ID16bit,
+ INPUT[i]->SMS[0].UDH.PartNumber,
+ INPUT[i]->SMS[0].UDH.AllParts);
+ w=w+INPUT[i]->SMS[0].UDH.Text[w+1]+2;
+ }
+ }
+ i++;
+ }
+ }
+
+ i=0;
+ while (INPUT[i]!=NULL) {
+ /* If this one SMS was sorted earlier, do not touch */
+ if (INPUTSorted[i]) {
+ i++;
+ continue;
+ }
+ copyit = false;
+ /* If we have:
+ * - linked sms returned by phone driver
+ * - sms without linking
+ * we copy it to OUTPUT
+ */
+ if (INPUT[i]->Number != 1 ||
+ INPUT[i]->SMS[0].UDH.Type == UDH_NoUDH ||
+ INPUT[i]->SMS[0].UDH.PartNumber == -1) {
+ copyit = true;
+ }
+ /* If we have unknown UDH, we copy it to OUTPUT */
+ if (INPUT[i]->SMS[0].UDH.Type == UDH_UserUDH) {
+ if (!ems) copyit = true;
+ if (ems && INPUT[i]->SMS[0].UDH.PartNumber == -1) copyit = true;
+ }
+ if (copyit) {
+ OUTPUT[OUTPUTNum] = malloc(sizeof(GSM_MultiSMSMessage));
+ if (OUTPUT[OUTPUTNum] == NULL) {
+ free(INPUTSorted);
+ return ERR_MOREMEMORY;
+ }
+ OUTPUT[OUTPUTNum+1] = NULL;
+
+ memcpy(OUTPUT[OUTPUTNum],INPUT[i],sizeof(GSM_MultiSMSMessage));
+ INPUTSorted[i]=true;
+ OUTPUTNum++;
+ i = 0;
+ continue;
+ }
+ /* We have 1'st part of linked sms. It's single.
+ * We will try to find other parts
+ */
+ if (INPUT[i]->SMS[0].UDH.PartNumber == 1) {
+ OUTPUT[OUTPUTNum] = malloc(sizeof(GSM_MultiSMSMessage));
+ if (OUTPUT[OUTPUTNum] == NULL) {
+ free(INPUTSorted);
+ return ERR_MOREMEMORY;
+ }
+ OUTPUT[OUTPUTNum+1] = NULL;
+
+ memcpy(&OUTPUT[OUTPUTNum]->SMS[0],&INPUT[i]->SMS[0],sizeof(GSM_SMSMessage));
+ OUTPUT[OUTPUTNum]->Number = 1;
+ INPUTSorted[i] = true;
+ j = 1;
+ /* We're searching for other parts in sequence */
+ while (j!=INPUT[i]->SMS[0].UDH.AllParts) {
+ z=0;
+ while(INPUT[z]!=NULL) {
+ /* This was sorted earlier or is not single */
+ if (INPUTSorted[z] || INPUT[z]->Number != 1) {
+ z++;
+ continue;
+ }
+ if (ems && INPUT[i]->SMS[0].UDH.Type != UDH_ConcatenatedMessages &&
+ INPUT[i]->SMS[0].UDH.Type != UDH_ConcatenatedMessages16bit &&
+ INPUT[i]->SMS[0].UDH.Type != UDH_UserUDH &&
+ INPUT[z]->SMS[0].UDH.Type != UDH_ConcatenatedMessages &&
+ INPUT[z]->SMS[0].UDH.Type != UDH_ConcatenatedMessages16bit &&
+ INPUT[z]->SMS[0].UDH.Type != UDH_UserUDH) {
+ if (INPUT[z]->SMS[0].UDH.Type != INPUT[i]->SMS[0].UDH.Type) {
+ z++;
+ continue;
+ }
+ }
+ if (!ems && INPUT[z]->SMS[0].UDH.Type != INPUT[i]->SMS[0].UDH.Type) {
+ z++;
+ continue;
+ }
+ dbgprintf("compare %i %i %i %i %i",
+ j+1,
+ INPUT[i]->SMS[0].UDH.ID8bit,
+ INPUT[i]->SMS[0].UDH.ID16bit,
+ INPUT[i]->SMS[0].UDH.PartNumber,
+ INPUT[i]->SMS[0].UDH.AllParts);
+ dbgprintf(" %i %i %i %i\n",
+ INPUT[z]->SMS[0].UDH.ID8bit,
+ INPUT[z]->SMS[0].UDH.ID16bit,
+ INPUT[z]->SMS[0].UDH.PartNumber,
+ INPUT[z]->SMS[0].UDH.AllParts);
+ if (INPUT[z]->SMS[0].UDH.ID8bit != INPUT[i]->SMS[0].UDH.ID8bit ||
+ INPUT[z]->SMS[0].UDH.ID16bit != INPUT[i]->SMS[0].UDH.ID16bit ||
+ INPUT[z]->SMS[0].UDH.AllParts != INPUT[i]->SMS[0].UDH.AllParts ||
+ INPUT[z]->SMS[0].UDH.PartNumber != j+1) {
+ z++;
+ continue;
+ }
+ /* For SMS_Deliver compare also SMSC and Sender number */
+ if (INPUT[z]->SMS[0].PDU == SMS_Deliver &&
+ (strcmp(DecodeUnicodeString(INPUT[z]->SMS[0].SMSC.Number),DecodeUnicodeString(INPUT[i]->SMS[0].SMSC.Number)) ||
+ strcmp(DecodeUnicodeString(INPUT[z]->SMS[0].Number),DecodeUnicodeString(INPUT[i]->SMS[0].Number)))) {
+ z++;
+ continue;
+ }
+ /* DCT4 Outbox: SMS Deliver. Empty number and SMSC. We compare dates */
+ if (INPUT[z]->SMS[0].PDU == SMS_Deliver &&
+ UnicodeLength(INPUT[z]->SMS[0].SMSC.Number)==0 &&
+ UnicodeLength(INPUT[z]->SMS[0].Number)==0 &&
+ (INPUT[z]->SMS[0].DateTime.Day != INPUT[i]->SMS[0].DateTime.Day ||
+ INPUT[z]->SMS[0].DateTime.Month != INPUT[i]->SMS[0].DateTime.Month ||
+ INPUT[z]->SMS[0].DateTime.Year != INPUT[i]->SMS[0].DateTime.Year ||
+ INPUT[z]->SMS[0].DateTime.Hour != INPUT[i]->SMS[0].DateTime.Hour ||
+ INPUT[z]->SMS[0].DateTime.Minute != INPUT[i]->SMS[0].DateTime.Minute ||
+ INPUT[z]->SMS[0].DateTime.Second != INPUT[i]->SMS[0].DateTime.Second)) {
+ z++;
+ continue;
+ }
+ /* We found correct sms. Copy it */
+ memcpy(&OUTPUT[OUTPUTNum]->SMS[j],&INPUT[z]->SMS[0],sizeof(GSM_SMSMessage));
+ OUTPUT[OUTPUTNum]->Number++;
+ INPUTSorted[z]=true;
+ break;
+ }
+ /* Incomplete sequence */
+ if (OUTPUT[OUTPUTNum]->Number==j) {
+ dbgprintf("Incomplete sequence\n");
+ break;
+ }
+ j++;
+ }
+ OUTPUTNum++;
+ i = 0;
+ continue;
+ }
+ /* We have some next linked sms from sequence */
+ if (INPUT[i]->SMS[0].UDH.PartNumber > 1) {
+ j = 0;
+ while (INPUT[j]!=NULL) {
+ if (INPUTSorted[j]) {
+ j++;
+ continue;
+ }
+ /* We have some not unassigned first sms from sequence.
+ * We can't touch other sms from sequences
+ */
+ if (INPUT[j]->SMS[0].UDH.PartNumber == 1) break;
+ j++;
+ }
+ if (INPUT[j]==NULL) {
+ OUTPUT[OUTPUTNum] = malloc(sizeof(GSM_MultiSMSMessage));
+ if (OUTPUT[OUTPUTNum] == NULL) {
+ free(INPUTSorted);
+ return ERR_MOREMEMORY;
+ }
+ OUTPUT[OUTPUTNum+1] = NULL;
+
+ memcpy(OUTPUT[OUTPUTNum],INPUT[i],sizeof(GSM_MultiSMSMessage));
+ INPUTSorted[i]=true;
+ OUTPUTNum++;
+ i = 0;
+ continue;
+ } else i++;
+ }
+ }
+ free(INPUTSorted);
+ return ERR_NONE;
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmmulti.h b/gammu/emb/common/service/sms/gsmmulti.h
new file mode 100644
index 0000000..c672261
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmmulti.h
@@ -0,0 +1,271 @@
+/* (c) 2002-2004 by Marcin Wiacek */
+
+#ifndef __gsm_multi_h
+#define __gsm_multi_h
+
+#include "../../gsmcomon.h"
+#include "../gsmlogo.h"
+#include "../gsmcal.h"
+#include "../gsmpbk.h"
+#include "../gsmdata.h"
+#include "../gsmring.h"
+#include "gsmsms.h"
+
+/* ---------------------- multi SMS --------------------------------------- */
+
+/* Identifiers for Smart Messaging 3.0 multipart SMS */
+
+#define SM30_ISOTEXT 0 /* ISO 8859-1 text */
+#define SM30_UNICODETEXT 1
+#define SM30_OTA 2
+#define SM30_RINGTONE 3
+#define SM30_PROFILENAME 4
+/* ... */
+#define SM30_SCREENSAVER 6
+
+/* Identifiers for Alcatel Terminal Data Download */
+#define ALCATELTDD_PICTURE 4
+#define ALCATELTDD_ANIMATION 5
+#define ALCATELTDD_SMSTEMPLATE 6
+
+void GSM_SMSCounter(int MessageLength,
+ unsigned char *MessageBuffer,
+ GSM_UDH UDHType,
+ GSM_Coding_Type Coding,
+ int *SMSNum,
+ int *CharsLeft);
+
+#define MAX_MULTI_SMS 10
+
+/**
+ * Multiple SMS messages, used for Smart Messaging 3.0/EMS.
+ */
+typedef struct {
+ /**
+ * Sender or recipient number.
+ */
+ unsigned char Number;
+ /**
+ * Array of SMSes.
+ */
+ GSM_SMSMessage SMS[MAX_MULTI_SMS];
+} GSM_MultiSMSMessage;
+
+GSM_Error GSM_AddSMS_Text_UDH(GSM_MultiSMSMessage *SMS,
+ GSM_Coding_Type Coding,
+ char *Buffer,
+ int BufferLen,
+ bool UDH,
+ int *UsedText,
+ int *CopiedText,
+ int *CopiedSMSText);
+
+void GSM_MakeMultiPartSMS(GSM_MultiSMSMessage *SMS,
+ unsigned char *MessageBuffer,
+ int MessageLength,
+ GSM_UDH UDHType,
+ GSM_Coding_Type Coding,
+ int Class,
+ unsigned char RejectDuplicates);
+
+void GSM_Find_Free_Used_SMS2(GSM_Coding_Type Coding,GSM_SMSMessage SMS, int *UsedText, int *FreeText, int *FreeBytes);
+
+unsigned char GSM_MakeSMSIDFromTime(void);
+
+/**
+ * ID during packing SMS for Smart Messaging 3.0, EMS and other
+ */
+typedef enum {
+ /**
+ * 1 text SMS.
+ */
+ SMS_Text = 1,
+ /**
+ * Contacenated SMS, when longer than 1 SMS.
+ */
+ SMS_ConcatenatedTextLong,
+ /**
+ * Contacenated SMS, auto Default/Unicode coding.
+ */
+ SMS_ConcatenatedAutoTextLong,
+ SMS_ConcatenatedTextLong16bit,
+ SMS_ConcatenatedAutoTextLong16bit,
+ /**
+ * Nokia profile = Name, Ringtone, ScreenSaver
+ */
+ SMS_NokiaProfileLong,
+ /**
+ * Nokia Picture Image + (text)
+ */
+ SMS_NokiaPictureImageLong,
+ /**
+ * Nokia screen saver + (text)
+ */
+ SMS_NokiaScreenSaverLong,
+ /**
+ * Nokia ringtone - old SM2.0 format, 1 SMS
+ */
+ SMS_NokiaRingtone,
+ /**
+ * Nokia ringtone contacenated, when very long
+ */
+ SMS_NokiaRingtoneLong,
+ /**
+ * Nokia 72x14 operator logo, 1 SMS
+ */
+ SMS_NokiaOperatorLogo,
+ /**
+ * Nokia 72x14 op logo or 78x21 in 2 SMS
+ */
+ SMS_NokiaOperatorLogoLong,
+ /**
+ * Nokia 72x14 caller logo, 1 SMS
+ */
+ SMS_NokiaCallerLogo,
+ /**
+ * Nokia WAP bookmark in 1 or 2 SMS
+ */
+ SMS_NokiaWAPBookmarkLong,
+ /**
+ * Nokia WAP settings in 2 SMS
+ */
+ SMS_NokiaWAPSettingsLong,
+ /**
+ * Nokia MMS settings in 2 SMS
+ */
+ SMS_NokiaMMSSettingsLong,
+ /**
+ * Nokia VCARD 1.0 - only name and default number
+ */
+ SMS_NokiaVCARD10Long,
+ /**
+ * Nokia VCARD 2.1 - all numbers + text
+ */
+ SMS_NokiaVCARD21Long,
+ /**
+ * Nokia VCALENDAR 1.0 - can be in few sms
+ */
+ SMS_NokiaVCALENDAR10Long,
+ SMS_NokiaVTODOLong,
+ SMS_VCARD10Long,
+ SMS_VCARD21Long,
+ SMS_DisableVoice,
+ SMS_DisableFax,
+ SMS_DisableEmail,
+ SMS_EnableVoice,
+ SMS_EnableFax,
+ SMS_EnableEmail,
+ SMS_VoidSMS,
+ /**
+ * IMelody 1.0
+ */
+ SMS_EMSSound10,
+ /**
+ * IMelody 1.2
+ */
+ SMS_EMSSound12,
+ /**
+ * IMelody without header - SonyEricsson extension
+ */
+ SMS_EMSSonyEricssonSound,
+ /**
+ * IMelody 1.0 with UPI.
+ */
+ SMS_EMSSound10Long,
+ /***
+ * IMelody 1.2 with UPI.
+ */
+ SMS_EMSSound12Long,
+ /**
+ * IMelody without header with UPI.
+ */
+ SMS_EMSSonyEricssonSoundLong,
+ SMS_EMSPredefinedSound,
+ SMS_EMSPredefinedAnimation,
+ SMS_EMSAnimation,
+ /**
+ * Fixed bitmap of size 16x16 or 32x32.
+ */
+ SMS_EMSFixedBitmap,
+ SMS_EMSVariableBitmap,
+ SMS_EMSVariableBitmapLong,
+ SMS_MMSIndicatorLong,
+ /**
+ * Variable bitmap with black and white colors
+ */
+ SMS_AlcatelMonoBitmapLong,
+ /**
+ * Variable animation with black and white colors
+ */
+ SMS_AlcatelMonoAnimationLong,
+ SMS_AlcatelSMSTemplateName
+} EncodeMultiPartSMSID;
+
+typedef struct {
+ EncodeMultiPartSMSID ID;
+
+ int Number;
+ GSM_Ringtone *Ringtone;
+ GSM_MultiBitmap *Bitmap;
+ GSM_WAPBookmark *Bookmark;
+ GSM_WAPSettings *Settings;
+ GSM_MMSIndicator *MMSIndicator;
+ GSM_MemoryEntry *Phonebook;
+ GSM_CalendarEntry *Calendar;
+ GSM_ToDoEntry *ToDo;
+ bool Protected;
+
+ unsigned char *Buffer;
+ bool Left;
+ bool Right;
+ bool Center;
+ bool Large;
+ bool Small;
+ bool Bold;
+ bool Italic;
+ bool Underlined;
+ bool Strikethrough;
+
+ /* Return values */
+ int RingtoneNotes;
+} MultiPartSMSEntry;
+
+typedef struct {
+ MultiPartSMSEntry Entries[MAX_MULTI_SMS];
+ int EntriesNum;
+ bool UnicodeCoding;
+ int Class;
+ unsigned char ReplaceMessage;
+ bool Unknown;
+} GSM_MultiPartSMSInfo;
+
+/**
+ * Encodes multi part SMS from "readable" format.
+ */
+GSM_Error GSM_EncodeMultiPartSMS (GSM_MultiPartSMSInfo *Info, GSM_MultiSMSMessage *SMS);
+
+/**
+ * Decodes multi part SMS to "readable" format.
+ */
+bool GSM_DecodeMultiPartSMS (GSM_MultiPartSMSInfo *Info, GSM_MultiSMSMessage *SMS, bool ems);
+
+/**
+ * Clears @ref GSM_MultiPartSMSInfo to default values.
+ */
+void GSM_ClearMultiPartSMSInfo (GSM_MultiPartSMSInfo *Info);
+
+/**
+ * Frees any allocated structures inside @ref GSM_MultiPartSMSInfo.
+ */
+void GSM_FreeMultiPartSMSInfo (GSM_MultiPartSMSInfo *Info);
+
+/**
+ * Links SMS messages according to IDs.
+ */
+GSM_Error GSM_LinkSMS(GSM_MultiSMSMessage **INPUT, GSM_MultiSMSMessage **OUTPUT, bool ems);
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmsms.c b/gammu/emb/common/service/sms/gsmsms.c
new file mode 100644
index 0000000..9920835
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmsms.c
@@ -0,0 +1,663 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Pawel Kot, others and Gnokii */
+
+#include <ctype.h>
+#include <string.h>
+#include <time.h>
+
+#include "../../gsmcomon.h"
+#include "../../misc/coding/coding.h"
+#include "../gsmcal.h"
+#include "../gsmpbk.h"
+#include "../gsmlogo.h"
+#include "../gsmring.h"
+#include "../gsmdata.h"
+#include "../gsmnet.h"
+#include "gsmsms.h"
+
+/* User data headers */
+static GSM_UDHHeader UDHHeaders[] = {
+ /* See GSM 03.40 section 9.2.3.24.1
+ * 1 byte 0x00
+ * 1 byte 0x03
+ * 1 byte 0x01: unique ID for message series
+ * 1 byte 0x00: how many SMS in sequence
+ * 1 byte 0x00: number of current SMS in sequence */
+ { UDH_ConcatenatedMessages, 0x05, "\x00\x03\x01\x00\x00",2,-1,4,3},
+
+ /* See GSM 03.40 section 9.2.3.24.2 for voice, fax and email messages */
+ { UDH_DisableVoice, 0x04, "\x01\x02\x00\x00",-1,-1,-1,-1},
+ { UDH_DisableFax, 0x04, "\x01\x02\x01\x00",-1,-1,-1,-1},
+ { UDH_DisableEmail, 0x04, "\x01\x02\x02\x00",-1,-1,-1,-1},
+ { UDH_EnableVoice, 0x04, "\x01\x02\x00\x01",-1,-1,-1,-1},
+ { UDH_EnableFax, 0x04, "\x01\x02\x01\x01",-1,-1,-1,-1},
+ { UDH_EnableEmail, 0x04, "\x01\x02\x02\x01",-1,-1,-1,-1},
+
+ /* When send such SMS to some phones, they don't display anything,
+ * only beep and enable vibra/light
+ */
+ { UDH_VoidSMS, 0x08, "\x01\x02\x02\x01\x01\x02\x02\x00",-1,-1,-1,-1},
+
+ /* Nokia Smart Messaging (short version) UDH
+ * General format :
+ * 1 byte 0x05 : IEI application port addressing scheme, 16 bit address
+ * 1 byte 0x04 : IEI length
+ * 2 bytes : destination address : high & low byte
+ * 2 bytes 0x00 0x00 : originator address : high & low byte */
+ { UDH_NokiaRingtone, 0x06, "\x05\x04\x15\x81\x00\x00",-1,-1,-1,-1},
+ { UDH_NokiaOperatorLogo, 0x06, "\x05\x04\x15\x82\x00\x00",-1,-1,-1,-1},
+ { UDH_NokiaCallerLogo, 0x06, "\x05\x04\x15\x83\x00\x00",-1,-1,-1,-1},
+ { UDH_NokiaWAP, 0x06, "\x05\x04\xc3\x4f\x00\x00",-1,-1,-1,-1},
+
+ /* Nokia Smart Messaging (long version) UDH and other
+ * General format:
+ * 1 byte 0x05 : IEI application port addressing scheme, 16 bit address
+ * 1 byte 0x04 : IEI length
+ * 2 bytes 0x00 0x00 : destination address : high & low byte
+ * 2 bytes 0x00 0x00 : originator address : high & low byte
+ * 1 byte 0x00 : SAR
+ * 1 byte 0x03 : SAR length
+ * 1 byte : diagram reference number (unique ID for message series)
+ * 1 byte : number of all SMS
+ * 1 byte : number of current SMS */
+ { UDH_NokiaCalendarLong, 0x0b, "\x05\x04\x00\xe4\x00\x00\x00\x03\xc7\x00\x00",8,-1,10,9},
+ { UDH_MMSIndicatorLong, 0x0b, "\x05\x04\x0b\x84\x23\xf0\x00\x03\xe5\x00\x00",8,-1,10,9},
+ { UDH_NokiaRingtoneLong, 0x0b, "\x05\x04\x15\x81\x00\x00\x00\x03\x01\x00\x00",8,-1,10,9},
+ { UDH_NokiaOperatorLogoLong, 0x0b, "\x05\x04\x15\x82\x00\x00\x00\x03\x02\x00\x00",8,-1,10,9},
+ { UDH_NokiaProfileLong, 0x0b, "\x05\x04\x15\x8a\x00\x00\x00\x03\xce\x00\x00",8,-1,10,9},
+ { UDH_NokiaPhonebookLong, 0x0b, "\x05\x04\x23\xf4\x00\x00\x00\x03\x01\x00\x00",8,-1,10,9},
+ { UDH_NokiaWAPLong, 0x0b, "\x05\x04\xc3\x4f\x00\x00\x00\x03\x7f\x00\x00",8,-1,10,9},
+
+ { UDH_ConcatenatedMessages16bit,0x06, "\x08\x04\x00\x00\x00\x00",-1,2,5,4},
+
+ { UDH_NoUDH, 0x00, "",-1,-1,-1,-1}
+};
+
+/* --------------------------- Unpacking SMS ------------------------------- */
+
+/* See GSM 03.40 section 9.2.3.11 */
+static GSM_Error GSM_DecodeSMSDateTime(GSM_DateTime *DT, unsigned char *req)
+{
+ DT->Year = DecodeWithBCDAlphabet(req[0]);
+ if (DT->Year<90) DT->Year=DT->Year+2000; else DT->Year=DT->Year+1990;
+ DT->Month = DecodeWithBCDAlphabet(req[1]);
+ DT->Day = DecodeWithBCDAlphabet(req[2]);
+ DT->Hour = DecodeWithBCDAlphabet(req[3]);
+ DT->Minute = DecodeWithBCDAlphabet(req[4]);
+ DT->Second = DecodeWithBCDAlphabet(req[5]);
+
+ /* Base for timezone is GMT. It's in quarters */
+ DT->Timezone=(10*(req[6]&0x07)+(req[6]>>4))/4;
+
+ if (req[6]&0x08) DT->Timezone = -DT->Timezone;
+
+ dbgprintf("Decoding date & time: ");
+ dbgprintf("%s %4d/%02d/%02d ", DayOfWeek(DT->Year, DT->Month, DT->Day),
+ DT->Year, DT->Month, DT->Day);
+ dbgprintf("%02d:%02d:%02d %02d00\n", DT->Hour, DT->Minute, DT->Second, DT->Timezone);
+
+ return ERR_NONE;
+}
+
+void GSM_DecodeUDHHeader(GSM_UDHHeader *UDH)
+{
+ int i, tmp, w;
+ bool UDHOK;
+
+ UDH->Type = UDH_UserUDH;
+ UDH->ID8bit = -1;
+ UDH->ID16bit = -1;
+ UDH->PartNumber = -1;
+ UDH->AllParts = -1;
+
+ i=-1;
+ while (true) {
+ i++;
+ if (UDHHeaders[i].Type==UDH_NoUDH) break;
+
+ tmp=UDHHeaders[i].Length;
+ /* if length is the same */
+ if (tmp==UDH->Text[0]) {
+
+ if (tmp==0x05) tmp=tmp-3;/*three last bytes can be different for such UDH*/
+ if (tmp==0x0b) tmp=tmp-3;/*three last bytes can be different for such UDH*/
+ if (tmp==0x06 && UDH->Text[1] == 0x08) tmp=tmp-4;
+
+ UDHOK=true;
+ for (w=0;w<tmp;w++) {
+ if (UDHHeaders[i].Text[w]!=UDH->Text[w+1]) {
+ UDHOK=false;
+ break;
+ }
+ }
+ if (UDHOK) {
+ UDH->Type=UDHHeaders[i].Type;
+
+ if (UDHHeaders[i].ID8bit !=-1) UDH->ID8bit = UDH->Text[UDHHeaders[i].ID8bit+1];
+ if (UDHHeaders[i].ID16bit !=-1) UDH->ID16bit = UDH->Text[UDHHeaders[i].ID16bit+1]*256+UDH->Text[UDHHeaders[i].ID16bit+2];
+ if (UDHHeaders[i].PartNumber !=-1) UDH->PartNumber = UDH->Text[UDHHeaders[i].PartNumber+1];
+ if (UDHHeaders[i].AllParts !=-1) UDH->AllParts = UDH->Text[UDHHeaders[i].AllParts+1];
+ break;
+ }
+ }
+ }
+
+#ifdef DEBUG
+ dbgprintf("Type of UDH: ");
+ switch (UDH->Type) {
+ case UDH_ConcatenatedMessages : dbgprintf("Concatenated (linked) message"); break;
+ case UDH_ConcatenatedMessages16bit : dbgprintf("Concatenated (linked) message"); break;
+ case UDH_DisableVoice : dbgprintf("Disables voice indicator"); break;
+ case UDH_EnableVoice : dbgprintf("Enables voice indicator"); break;
+ case UDH_DisableFax : dbgprintf("Disables fax indicator"); break;
+ case UDH_EnableFax : dbgprintf("Enables fax indicator"); break;
+ case UDH_DisableEmail : dbgprintf("Disables email indicator"); break;
+ case UDH_EnableEmail : dbgprintf("Enables email indicator"); break;
+ case UDH_VoidSMS : dbgprintf("Void SMS"); break;
+ case UDH_NokiaWAP : dbgprintf("Nokia WAP Bookmark"); break;
+ case UDH_NokiaOperatorLogoLong : dbgprintf("Nokia operator logo"); break;
+ case UDH_NokiaWAPLong : dbgprintf("Nokia WAP Bookmark or WAP/MMS Settings"); break;
+ case UDH_NokiaRingtone : dbgprintf("Nokia ringtone"); break;
+ case UDH_NokiaRingtoneLong : dbgprintf("Nokia ringtone"); break;
+ case UDH_NokiaOperatorLogo : dbgprintf("Nokia GSM operator logo"); break;
+ case UDH_NokiaCallerLogo : dbgprintf("Nokia caller logo"); break;
+ case UDH_NokiaProfileLong : dbgprintf("Nokia profile"); break;
+ case UDH_NokiaCalendarLong : dbgprintf("Nokia calendar note"); break;
+ case UDH_NokiaPhonebookLong : dbgprintf("Nokia phonebook entry"); break;
+ case UDH_UserUDH : dbgprintf("User UDH"); break;
+ case UDH_MMSIndicatorLong : dbgprintf("MMS indicator"); break;
+ case UDH_NoUDH: break;
+ }
+ if (UDH->ID8bit != -1) dbgprintf(", ID 8 bit %i",UDH->ID8bit);
+ if (UDH->ID16bit != -1) dbgprintf(", ID 16 bit %i",UDH->ID16bit);
+ if (UDH->PartNumber != -1 && UDH->AllParts != -1) {
+ dbgprintf(", part %i of %i",UDH->PartNumber,UDH->AllParts);
+ }
+ dbgprintf("\n");
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, UDH->Text, UDH->Length);
+#endif
+}
+
+GSM_Error GSM_DecodeSMSFrameText(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout)
+{
+ int off=0; // length of the User Data Header
+ int w,i,tmp=0;
+ unsigned char output[161];
+
+ SMS->UDH.Length = 0;
+ /* UDH header available */
+ if (buffer[Layout.firstbyte] & 64) {
+ /* Length of UDH header */
+ off = (buffer[Layout.Text] + 1);
+ SMS->UDH.Length = off;
+ dbgprintf("UDH header available (length %i)\n",off);
+
+ /* Copy UDH header into SMS->UDH */
+ for (i = 0; i < off; i++) SMS->UDH.Text[i] = buffer[Layout.Text + i];
+
+ GSM_DecodeUDHHeader(&SMS->UDH);
+ }
+
+ /* GSM 03.40 section 9.2.3.10 (TP-Data-Coding-Scheme) and GSM 03.38 section 4 */
+ if ((buffer[Layout.TPDCS] & 0xf4) == 0xf4) SMS->Coding=SMS_Coding_8bit;
+ if ((buffer[Layout.TPDCS] & 0x08) == 0x08) SMS->Coding=SMS_Coding_Unicode;
+
+ switch (SMS->Coding) {
+ case SMS_Coding_Default:
+ i = 0;
+ do {
+ i+=7;
+ w=(i-off)%i;
+ } while (w<0);
+ SMS->Length=buffer[Layout.TPUDL] - (off*8 + w) / 7;
+ tmp=GSM_UnpackEightBitsToSeven(w, buffer[Layout.TPUDL]-off, SMS->Length, buffer+(Layout.Text+off), output);
+ dbgprintf("7 bit SMS, length %i\n",SMS->Length);
+ DecodeDefault (SMS->Text, output, SMS->Length, true, NULL);
+ dbgprintf("%s\n",DecodeUnicodeString(SMS->Text));
+ break;
+ case SMS_Coding_8bit:
+ SMS->Length=buffer[Layout.TPUDL] - off;
+ memcpy(SMS->Text,buffer+(Layout.Text+off),SMS->Length);
+#ifdef DEBUG
+ dbgprintf("8 bit SMS, length %i\n",SMS->Length);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, SMS->Text, SMS->Length);
+#endif
+ break;
+ case SMS_Coding_Unicode:
+ SMS->Length=(buffer[Layout.TPUDL] - off) / 2;
+ DecodeUnicodeSpecialNOKIAChars(SMS->Text,buffer+(Layout.Text+off), SMS->Length);
+#ifdef DEBUG
+ dbgprintf("Unicode SMS, length %i\n",SMS->Length);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, buffer+(Layout.Text+off), SMS->Length*2);
+ dbgprintf("%s\n",DecodeUnicodeString(SMS->Text));
+#endif
+ break;
+ }
+
+ return ERR_NONE;
+}
+
+GSM_Error GSM_DecodeSMSFrameStatusReportData(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout)
+{
+ SMS->DeliveryStatus = buffer[Layout.TPStatus];
+
+ if (buffer[Layout.TPStatus] < 0x03) {
+ EncodeUnicode(SMS->Text,"Delivered",9);
+ SMS->Length = 9;
+ } else if (buffer[Layout.TPStatus] & 0x40) {
+ EncodeUnicode(SMS->Text,"Failed",6);
+ SMS->Length = 6;
+ } else if (buffer[Layout.TPStatus] & 0x20) {
+ EncodeUnicode(SMS->Text,"Pending",7);
+ SMS->Length = 7;
+ } else {
+ EncodeUnicode(SMS->Text,"Unknown",7);
+ SMS->Length = 7;
+ }
+
+#ifdef DEBUG
+ /* See GSM 03.40 section 9.2.3.15 (TP-Status) */
+ if (buffer[Layout.TPStatus] & 0x40) {
+ if (buffer[Layout.TPStatus] & 0x20) {
+ /* 0x60, 0x61, ... */
+ dbgprintf("Temporary error, SC is not making any more transfer attempts\n");
+ } else {
+ /* 0x40, 0x41, ... */
+ dbgprintf("Permanent error, SC is not making any more transfer attempts\n");
+ }
+ } else if (buffer[Layout.TPStatus] & 0x20) {
+ /* 0x20, 0x21, ... */
+ dbgprintf("Temporary error, SC still trying to transfer SM\n");
+ }
+ switch (buffer[Layout.TPStatus]) {
+ case 0x00: dbgprintf("SM received by the SME"); break;
+ case 0x01: dbgprintf("SM forwarded by the SC to the SME but the SC is unable to confirm delivery");break;
+ case 0x02: dbgprintf("SM replaced by the SC"); break;
+ case 0x20: dbgprintf("Congestion"); break;
+ case 0x21: dbgprintf("SME busy"); break;
+ case 0x22: dbgprintf("No response from SME"); break;
+ case 0x23: dbgprintf("Service rejected"); break;
+ case 0x24: dbgprintf("Quality of service not available"); break;
+ case 0x25: dbgprintf("Error in SME"); break;
+ case 0x40: dbgprintf("Remote procedure error"); break;
+ case 0x41: dbgprintf("Incompatibile destination"); break;
+ case 0x42: dbgprintf("Connection rejected by SME"); break;
+ case 0x43: dbgprintf("Not obtainable"); break;
+ case 0x44: dbgprintf("Quality of service not available"); break;
+ case 0x45: dbgprintf("No internetworking available"); break;
+ case 0x46: dbgprintf("SM Validity Period Expired"); break;
+ case 0x47: dbgprintf("SM deleted by originating SME"); break;
+ case 0x48: dbgprintf("SM Deleted by SC Administration"); break;
+ case 0x49: dbgprintf("SM does not exist"); break;
+ case 0x60: dbgprintf("Congestion"); break;
+ case 0x61: dbgprintf("SME busy"); break;
+ case 0x62: dbgprintf("No response from SME"); break;
+ case 0x63: dbgprintf("Service rejected"); break;
+ case 0x64: dbgprintf("Quality of service not available"); break;
+ case 0x65: dbgprintf("Error in SME"); break;
+ default : dbgprintf("Reserved/Specific to SC: %x",buffer[Layout.TPStatus]); break;
+ }
+ dbgprintf("\n");
+#endif
+
+ return ERR_NONE;
+}
+
+GSM_Error GSM_DecodeSMSFrame(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout)
+{
+ GSM_DateTime zerodt = {0,0,0,0,0,0,0};
+#ifdef DEBUG
+ if (Layout.firstbyte == 255) {
+ dbgprintf("ERROR: firstbyte in SMS layout not set\n");
+ return ERR_UNKNOWN;
+ }
+ if (Layout.TPDCS != 255) dbgprintf("TPDCS : %02x %i\n",buffer[Layout.TPDCS] ,buffer[Layout.TPDCS]);
+ if (Layout.TPMR != 255) dbgprintf("TPMR : %02x %i\n",buffer[Layout.TPMR] ,buffer[Layout.TPMR]);
+ if (Layout.TPPID != 255) dbgprintf("TPPID : %02x %i\n",buffer[Layout.TPPID] ,buffer[Layout.TPPID]);
+ if (Layout.TPUDL != 255) dbgprintf("TPUDL : %02x %i\n",buffer[Layout.TPUDL] ,buffer[Layout.TPUDL]);
+ if (Layout.firstbyte != 255) dbgprintf("FirstByte : %02x %i\n",buffer[Layout.firstbyte],buffer[Layout.firstbyte]);
+ if (Layout.Text != 255) dbgprintf("Text : %02x %i\n",buffer[Layout.Text] ,buffer[Layout.Text]);
+#endif
+
+ SMS->UDH.Type = UDH_NoUDH;
+ SMS->Coding = SMS_Coding_Default;
+ SMS->Length = 0;
+ SMS->SMSC.Location = 0;
+ SMS->SMSC.DefaultNumber[0] = 0;
+ SMS->SMSC.DefaultNumber[1] = 0;
+ SMS->SMSC.Number[0] = 0;
+ SMS->SMSC.Number[1] = 0;
+ SMS->SMSC.Name[0] = 0;
+ SMS->SMSC.Name[1] = 0;
+ SMS->SMSC.Validity.Format = SMS_Validity_NotAvailable;
+ SMS->SMSC.Format = SMS_FORMAT_Text;
+ SMS->Number[0] = 0;
+ SMS->Number[1] = 0;
+ SMS->Name[0] = 0;
+ SMS->Name[1] = 0;
+ SMS->ReplyViaSameSMSC = false;
+ if (Layout.SMSCNumber!=255) {
+ GSM_UnpackSemiOctetNumber(SMS->SMSC.Number,buffer+Layout.SMSCNumber,false);
+ dbgprintf("SMS center number : \"%s\"\n",DecodeUnicodeString(SMS->SMSC.Number));
+ }
+ if ((buffer[Layout.firstbyte] & 0x80)!=0) SMS->ReplyViaSameSMSC=true;
+#ifdef DEBUG
+ if (SMS->ReplyViaSameSMSC) dbgprintf("SMS centre set for reply\n");
+#endif
+ if (Layout.Number!=255) {
+ GSM_UnpackSemiOctetNumber(SMS->Number,buffer+Layout.Number,true);
+ dbgprintf("Remote number : \"%s\"\n",DecodeUnicodeString(SMS->Number));
+ }
+ if (Layout.Text != 255 && Layout.TPDCS!=255 && Layout.TPUDL!=255) {
+ GSM_DecodeSMSFrameText(SMS, buffer, Layout);
+ }
+ if (Layout.DateTime != 255) {
+ GSM_DecodeSMSDateTime(&SMS->DateTime,buffer+(Layout.DateTime));
+ } else {
+ SMS->DateTime = zerodt;
+ }
+ if (Layout.SMSCTime != 255 && Layout.TPStatus != 255) {
+ /* GSM 03.40 section 9.2.3.11 (TP-Service-Centre-Time-Stamp) */
+ dbgprintf("SMSC response date: ");
+ GSM_DecodeSMSDateTime(&SMS->SMSCTime, buffer+(Layout.SMSCTime));
+ GSM_DecodeSMSFrameStatusReportData(SMS,buffer,Layout);
+ } else {
+ SMS->SMSCTime = zerodt;
+ }
+ SMS->Class = -1;
+ if (Layout.TPDCS != 255) {
+ if ((buffer[Layout.TPDCS] & 0xF3)==0xF0) SMS->Class = 0;
+ if ((buffer[Layout.TPDCS] & 0xF3)==0xF1) SMS->Class = 1;
+ if ((buffer[Layout.TPDCS] & 0xF3)==0xF2) SMS->Class = 2;
+ if ((buffer[Layout.TPDCS] & 0xF3)==0xF3) SMS->Class = 3;
+ }
+ dbgprintf("SMS class: %i\n",SMS->Class);
+
+ SMS->MessageReference = 0;
+ if (Layout.TPMR != 255) SMS->MessageReference = buffer[Layout.TPMR];
+
+ SMS->ReplaceMessage = 0;
+ if (Layout.TPPID != 255) {
+ if (buffer[Layout.TPPID] > 0x40 && buffer[Layout.TPPID] < 0x48) {
+ SMS->ReplaceMessage = buffer[Layout.TPPID] - 0x40;
+ }
+ }
+ SMS->RejectDuplicates = false;
+ if ((buffer[Layout.firstbyte] & 0x04)==0x04) SMS->RejectDuplicates = true;
+
+ return ERR_NONE;
+}
+
+/* ----------------------------- Packing SMS ------------------------------- */
+
+/* See GSM 03.40 section 9.2.3.11 */
+static GSM_Error GSM_EncodeSMSDateTime(GSM_DateTime *DT, unsigned char *req)
+{
+ int Year;
+
+ dbgprintf("Encoding SMS datetime: %02i/%02i/%04i %02i:%02i:%02i\n",
+ DT->Day,DT->Month,DT->Year,DT->Hour,DT->Minute,DT->Second);
+
+ /* We need to have only two last digits of year */
+ if (DT->Year>1900) {
+ if (DT->Year<2000) Year = DT->Year-1900;
+ else Year = DT->Year-2000;
+ } else Year = DT->Year;
+
+ req[0]=EncodeWithBCDAlphabet(Year);
+ req[1]=EncodeWithBCDAlphabet(DT->Month);
+ req[2]=EncodeWithBCDAlphabet(DT->Day);
+ req[3]=EncodeWithBCDAlphabet(DT->Hour);
+ req[4]=EncodeWithBCDAlphabet(DT->Minute);
+ req[5]=EncodeWithBCDAlphabet(DT->Second);
+
+ /* FIXME: do it */
+ req[6]=0; /* TimeZone = +-0 */
+
+ return ERR_NONE;
+}
+
+static int GSM_EncodeSMSFrameText(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout)
+{
+ int off = 0; // length of the User Data Header
+ int size = 0, size2 = 0, w,p;
+ char buff[200];
+
+ if (SMS->UDH.Type!=UDH_NoUDH) {
+ buffer[Layout.firstbyte] |= 0x40; /* GSM 03.40 section 9.2.3.23 (TP-User-Data-Header-Indicator) */
+ off = 1 + SMS->UDH.Text[0]; /* off - length of the User Data Header */
+ memcpy(buffer+Layout.Text, SMS->UDH.Text, off); /* we copy the udh */
+#ifdef DEBUG
+ dbgprintf("UDH, length %i\n",off);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, SMS->UDH.Text, off);
+#endif
+ }
+ switch (SMS->Coding) {
+ case SMS_Coding_8bit:
+ /* the mask for the 8-bit data */
+ /* GSM 03.40 section 9.2.3.10 (TP-Data-Coding-Scheme)
+ * and GSM 03.38 section 4 */
+ buffer[Layout.TPDCS] |= 0xf4;
+ memcpy(buffer+(Layout.Text+off), SMS->Text, SMS->Length);
+ size2 = size = SMS->Length+off;
+#ifdef DEBUG
+ dbgprintf("8 bit SMS, length %i\n",SMS->Length);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, SMS->Text, SMS->Length);
+#endif
+ break;
+ case SMS_Coding_Default:
+ p = 0;
+ do {
+ p+=7;
+ w=(p-off)%p;
+ } while (w<0);
+ p = UnicodeLength(SMS->Text);
+ EncodeDefault(buff, SMS->Text, &p, true, NULL);
+ size = GSM_PackSevenBitsToEight(w, buff, buffer+(Layout.Text+off), p);
+ size += off;
+ size2 = (off*8 + w) / 7 + p;
+ dbgprintf("7 bit SMS, length %i, %i\n",size,size2);
+ dbgprintf("%s\n",DecodeUnicodeString(SMS->Text));
+ if (size > GSM_MAX_8BIT_SMS_LENGTH) {
+ size = 0; size2 = 0;
+ }
+ break;
+ case SMS_Coding_Unicode:
+ /* GSM 03.40 section 9.2.3.10 (TP-Data-Coding-Scheme)
+ * and GSM 03.38 section 4 */
+ buffer[Layout.TPDCS] |= 0x08;
+ EncodeUnicodeSpecialNOKIAChars(buffer+(Layout.Text+off), SMS->Text, UnicodeLength(SMS->Text));
+ size=size2=UnicodeLength(buffer+(Layout.Text+off))*2+off;
+#ifdef DEBUG
+ dbgprintf("Unicode SMS, length %i\n",(size2-off)/2);
+ if (di.dl == DL_TEXTALL || di.dl == DL_TEXTALLDATE) DumpMessage(di.df, di.dl, buffer+(Layout.Text+off), size2-off);
+ dbgprintf("%s\n",DecodeUnicodeString(buffer+(Layout.Text+off)));
+#endif
+ break;
+ }
+
+ /* GSM 03.40 section 9.2.3.16 (TP-User-Data-Length)
+ * SMS->Length is:
+ - integer representation of the number od octets within the
+ user data when TP-User-Data is coded using 8 bit data
+ - the sum of the number of septets in UDH including any padding
+ and the number of septets in TP-User-Data in other case
+ */
+ buffer[Layout.TPUDL] = size2;
+ return size;
+}
+
+GSM_Error GSM_EncodeSMSFrame(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout, int *length, bool clear)
+{
+ int i;
+
+ if (clear) {
+ /* Cleaning up to the SMS text */
+ for (i=0;i<Layout.Text;i++) buffer[i] = 0;
+ }
+
+ /* GSM 03.40 section 9.2.3.1 (TP-Message-Type-Indicator) */
+ switch (SMS->PDU) {
+ case SMS_Submit:
+ buffer[Layout.firstbyte] |= 0x01;
+ break;
+ /* SMS_Status_Report when Submit sms should have delivery report */
+ /* We DON'T CREATE FRAME FOR REAL SMS_STATUS_REPORT */
+ case SMS_Status_Report:
+ buffer[Layout.firstbyte] |= 0x01;
+ /* GSM 03.40 section 9.2.3.5 (TP-Status-Raport-Request) */
+ /* Set when want delivery report from SMSC */
+ buffer[Layout.firstbyte] |= 0x20;
+ break;
+ case SMS_Deliver:
+ buffer[Layout.firstbyte] |= 0x00;
+ }
+
+ /* GSM 03.40 section 9.2.3.17 (TP-Reply-Path) */
+ if (SMS->ReplyViaSameSMSC) buffer[Layout.firstbyte] |= 0x80;
+
+ if (Layout.Number!=255) {
+ buffer[Layout.Number] = GSM_PackSemiOctetNumber(SMS->Number,buffer+(Layout.Number+1),true);
+ dbgprintf("Recipient number \"%s\"\n",DecodeUnicodeString(SMS->Number));
+ }
+ if (Layout.SMSCNumber!=255) {
+ buffer[Layout.SMSCNumber]=GSM_PackSemiOctetNumber(SMS->SMSC.Number,buffer+(Layout.SMSCNumber+1), false);
+ dbgprintf("SMSC number \"%s\"\n",DecodeUnicodeString(SMS->SMSC.Number));
+ }
+
+ /* Message Class*/
+ /* GSM 03.40 section 9.2.3.10 (TP-Data-Coding-Scheme) and GSM 03.38 section 4 */
+ if (Layout.TPDCS != 255) {
+ if (SMS->Class>=0 && SMS->Class<5) buffer[Layout.TPDCS] |= (240+SMS->Class);
+ dbgprintf("SMS class %i\n",SMS->Class);
+ }
+
+ if (Layout.TPVP != 255) {
+ /* GSM 03.40 section 9.2.3.3 (TP-Validity-Period-Format) */
+ /* Bits 4 and 3: 10. TP-VP field present and integer represent (relative) */
+ buffer[Layout.firstbyte] |= 0x10;
+ buffer[Layout.TPVP]=((unsigned char)SMS->SMSC.Validity.Relative);
+ dbgprintf("SMS validity %02x\n",SMS->SMSC.Validity.Relative);
+ }
+
+ if (Layout.DateTime != 255) {
+ GSM_EncodeSMSDateTime(&SMS->DateTime, buffer+Layout.DateTime);
+ }
+
+ if (Layout.TPMR != 255) {
+ dbgprintf("TPMR: %02x %i\n",SMS->MessageReference,SMS->MessageReference);
+ buffer[Layout.TPMR] = SMS->MessageReference;
+ }
+
+ if (SMS->RejectDuplicates) {
+ /* GSM 03.40 section 9.2.3.25 (TP Reject Duplicates) */
+ buffer[Layout.firstbyte] |= 0x04;
+ }
+
+ if (Layout.TPPID != 255) {
+ buffer[Layout.TPPID] = 0;
+ if (SMS->ReplaceMessage > 0 && SMS->ReplaceMessage < 8) {
+ buffer[Layout.TPPID] = 0x40 + SMS->ReplaceMessage;
+ }
+ }
+
+ /* size is the length of the data in octets including UDH */
+ *length=GSM_EncodeSMSFrameText(SMS,buffer,Layout);
+// if (*length == 0) return GE_UNKNOWN;
+ *length += Layout.Text;
+
+ return ERR_NONE;
+}
+
+/* ----------------- Some help functions ----------------------------------- */
+
+void GSM_SetDefaultSMSData(GSM_SMSMessage *SMS)
+{
+ SMS->Class = -1;
+ SMS->SMSC.Location = 1;
+ SMS->SMSC.Format = SMS_FORMAT_Text;
+ SMS->SMSC.Validity.Format = SMS_Validity_RelativeFormat;
+ SMS->SMSC.Validity.Relative = SMS_VALID_Max_Time;
+ SMS->ReplyViaSameSMSC = false;
+ SMS->UDH.Type = UDH_NoUDH;
+ SMS->UDH.Length = 0;
+ SMS->UDH.Text[0] = 0;
+ SMS->UDH.ID8bit = 0;
+ SMS->UDH.ID16bit = 0;
+ SMS->UDH.PartNumber = 0;
+ SMS->UDH.AllParts = 0;
+ SMS->Coding = SMS_Coding_Default;
+ SMS->Text[0] = 0;
+ SMS->Text[1] = 0;
+ SMS->PDU = SMS_Submit;
+ SMS->RejectDuplicates = false;
+ SMS->MessageReference = 0;
+ SMS->ReplaceMessage = 0;
+ SMS->Length = 0;
+
+ /* This part is required to save SMS */
+ SMS->State = SMS_UnSent;
+ SMS->Location = 0;
+ SMS->Folder = 0x02; /*Outbox*/
+ GSM_GetCurrentDateTime (&SMS->DateTime);
+ SMS->Name[0] = 0;
+ SMS->Name[1] = 0;
+}
+
+/**
+ * GSM 03.40 section 9.2.3.24
+ */
+void GSM_EncodeUDHHeader(GSM_UDHHeader *UDH)
+{
+ int i=0;
+
+ if (UDH->Type == UDH_NoUDH) {
+ UDH->Length = 0;
+ return;
+ }
+ if (UDH->Type == UDH_UserUDH) {
+ UDH->Length = UDH->Text[0] + 1;
+ return;
+ }
+ while (true) {
+ if (UDHHeaders[i].Type==UDH_NoUDH) {
+ dbgprintf("Not supported UDH type\n");
+ break;
+ }
+ if (UDHHeaders[i].Type!=UDH->Type) {
+ i++;
+ continue;
+ }
+ /* UDH Length */
+ UDH->Text[0] = UDHHeaders[i].Length;
+ memcpy(UDH->Text+1, UDHHeaders[i].Text, UDHHeaders[i].Length);
+ UDH->Length = UDH->Text[0] + 1;
+
+ if (UDHHeaders[i].ID8bit != -1) {
+ UDH->Text[UDHHeaders[i].ID8bit+1] = UDH->ID8bit % 256;
+ } else {
+ UDH->ID8bit = -1;
+ }
+ if (UDHHeaders[i].ID16bit != -1) {
+ UDH->Text[UDHHeaders[i].ID16bit+1] = UDH->ID16bit / 256;
+ UDH->Text[UDHHeaders[i].ID16bit+2] = UDH->ID16bit % 256;
+ } else {
+ UDH->ID16bit = -1;
+ }
+ if (UDHHeaders[i].PartNumber != -1) {
+ UDH->Text[UDHHeaders[i].PartNumber+1] = UDH->PartNumber;
+ } else {
+ UDH->PartNumber = -1;
+ }
+ if (UDHHeaders[i].AllParts != -1) {
+ UDH->Text[UDHHeaders[i].AllParts+1] = UDH->AllParts;
+ } else {
+ UDH->AllParts = -1;
+ }
+ break;
+ }
+}
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */
diff --git a/gammu/emb/common/service/sms/gsmsms.h b/gammu/emb/common/service/sms/gsmsms.h
new file mode 100644
index 0000000..d87ff60
--- a/dev/null
+++ b/gammu/emb/common/service/sms/gsmsms.h
@@ -0,0 +1,492 @@
+/* (c) 2001-2004 by Marcin Wiacek */
+/* based on some work from Pawel Kot, others and Gnokii */
+
+#ifndef __gsm_sms_h
+#define __gsm_sms_h
+
+#include "../../gsmcomon.h"
+#include "../gsmlogo.h"
+#include "../gsmcal.h"
+#include "../gsmpbk.h"
+#include "../gsmdata.h"
+#include "../gsmring.h"
+
+/* --------------------- Some general definitions ------------------------- */
+
+#define GSM_MAX_UDH_LENGTH 140
+#define GSM_MAX_SMS_LENGTH 160
+#define GSM_MAX_8BIT_SMS_LENGTH 140
+
+/* -------------------- Cell Broadcast ------------------------------------ */
+
+/**
+ * Structure for Cell Broadcast messages.
+ */
+typedef struct {
+ /**
+ * Message text.
+ */
+ char Text[300];
+ /**
+ * Channel number.
+ */
+ int Channel;
+} GSM_CBMessage;
+
+/* ------------------------ SMS status ------------------------------------ */
+
+/**
+ * Status of SMS memory.
+ */
+typedef struct {
+ /**
+ * Number of unread messages on SIM.
+ */
+ int SIMUnRead;
+ /**
+ * Number of all saved messages (including unread) on SIM.
+ */
+ int SIMUsed;
+ /**
+ * Number of all possible messages on SIM.
+ */
+ int SIMSize;
+ /**
+ * Number of used templates (62xx/63xx/7110/etc.).
+ */
+ int TemplatesUsed;
+ /**
+ * Number of unread messages in phone.
+ */
+ int PhoneUnRead;
+ /**
+ * Number of all saved messages in phone.
+ */
+ int PhoneUsed;
+ /**
+ * Number of all possible messages on phone.
+ */
+ int PhoneSize;
+} GSM_SMSMemoryStatus;
+
+/* --------------------- SMS Center --------------------------------------- */
+
+/**
+ * Enum defines format of SMS messages. See GSM 03.40 section 9.2.3.9
+ */
+typedef enum {
+ SMS_FORMAT_Pager = 1,
+ SMS_FORMAT_Fax,
+ SMS_FORMAT_Email,
+ SMS_FORMAT_Text
+ /* Some values not handled here */
+} GSM_SMSFormat;
+
+/**
+ * Enum defines some the most often used validity lengths for SMS messages
+ * for relative validity format. See GSM 03.40 section 9.2.3.12.1 - it gives
+ * more values
+ */
+typedef enum {
+ SMS_VALID_1_Hour = 0x0b,
+ SMS_VALID_6_Hours = 0x47,
+ SMS_VALID_1_Day = 0xa7,
+ SMS_VALID_3_Days = 0xa9,
+ SMS_VALID_1_Week = 0xad,
+ SMS_VALID_Max_Time = 0xff
+} GSM_ValidityPeriod;
+
+/**
+ * Enum defines format of validity period for SMS messages.
+ * See GSM 03.40 section 9.2.3.12
+ */
+typedef enum {
+ SMS_Validity_NotAvailable = 1,
+ SMS_Validity_RelativeFormat
+ /* Specification gives also other possibilities */
+} GSM_ValidityPeriodFormat;
+
+/**
+ * Structure for validity of SMS messages
+ */
+typedef struct {
+ GSM_ValidityPeriodFormat Format;
+ /**
+ * Value defines period for relative format
+ */
+ GSM_ValidityPeriod Relative;
+} GSM_SMSValidity;
+
+#define GSM_MAX_SMSC_NAME_LENGTH 30
+
+/**
+ * Structure for SMSC (SMS Center) information.
+ */
+typedef struct {
+ /**
+ * Number of the SMSC on SIM
+ */
+ int Location;
+ /**
+ * Name of the SMSC
+ */
+ unsigned char Name[(GSM_MAX_SMSC_NAME_LENGTH+1)*2];
+ /**
+ * SMSC phone number.
+ */
+ unsigned char Number[(GSM_MAX_NUMBER_LENGTH+1)*2];
+ /**
+ * Validity of SMS messages.
+ */
+ GSM_SMSValidity Validity;
+ /**
+ * Format of sent SMS messages.
+ */
+ GSM_SMSFormat Format;
+ /**
+ * Default recipient number. In old DCT3 ignored
+ */
+ unsigned char DefaultNumber[(GSM_MAX_NUMBER_LENGTH+1)*2];
+} GSM_SMSC;
+
+/* --------------------- single SMS --------------------------------------- */
+
+/**
+ * Status of SMS message.
+ */
+typedef enum {
+ SMS_Sent = 1,
+ SMS_UnSent,
+ SMS_Read,
+ SMS_UnRead
+} GSM_SMS_State;
+
+/**
+ * Coding type of SMS.
+ */
+typedef enum {
+ /**
+ * Unicode
+ */
+ SMS_Coding_Unicode = 1,
+ /**
+ * Default GSM aplhabet.
+ */
+ SMS_Coding_Default,
+ /**
+ * 8-bit.
+ */
+ SMS_Coding_8bit
+} GSM_Coding_Type;
+
+/**
+ * Types of UDH (User Data Header).
+ */
+typedef enum {
+ UDH_NoUDH = 1,
+ /**
+ * Linked SMS.
+ */
+ UDH_ConcatenatedMessages,
+ /**
+ * Linked SMS with 16 bit reference.
+ */
+ UDH_ConcatenatedMessages16bit,
+ UDH_DisableVoice,
+ UDH_DisableFax,
+ UDH_DisableEmail,
+ UDH_EnableVoice,
+ UDH_EnableFax,
+ UDH_EnableEmail,
+ UDH_VoidSMS,
+ UDH_NokiaRingtone,
+ UDH_NokiaRingtoneLong,
+ UDH_NokiaOperatorLogo,
+ UDH_NokiaOperatorLogoLong,
+ UDH_NokiaCallerLogo,
+ UDH_NokiaWAP,
+ UDH_NokiaWAPLong,
+ UDH_NokiaCalendarLong,
+ UDH_NokiaProfileLong,
+ UDH_NokiaPhonebookLong,
+ UDH_UserUDH,
+ UDH_MMSIndicatorLong
+} GSM_UDH;
+
+/**
+ * Structure for User Data Header.
+ */
+typedef struct {
+ /**
+ * UDH type.
+ */
+ GSM_UDH Type;
+ /**
+ * UDH length.
+ */
+ int Length;
+ /**
+ * UDH text.
+ */
+ unsigned char Text[GSM_MAX_UDH_LENGTH];
+ /**
+ * 8-bit ID, when required (-1 otherwise).
+ */
+ int ID8bit;
+ /**
+ * 16-bit ID, when required (-1 otherwise).
+ */
+ int ID16bit;
+ /**
+ * Number of current part.
+ */
+ int PartNumber;
+ /**
+ * Total number of parts.
+ */
+ int AllParts;
+} GSM_UDHHeader;
+
+/**
+ * TP-Message-Type-Indicator. See GSM 03.40 section 9.2.3.1.
+ */
+typedef enum {
+ /**
+ * SMS in Inbox.
+ */
+ SMS_Deliver = 1,
+ /**
+ * Delivery Report
+ */
+ SMS_Status_Report,
+ /**
+ * SMS for sending or in Outbox
+ */
+ SMS_Submit
+ /* specification gives more */
+} GSM_SMSMessageType;
+
+/**
+ * Maximal length of SMS name.
+ */
+#define GSM_MAX_SMS_NAME_LENGTH 40
+
+/**
+ * SMS message data.
+ */
+typedef struct {
+ /**
+ * Message to be replaced.
+ */
+ unsigned char ReplaceMessage;
+ /**
+ * Whether to reject duplicates.
+ */
+ bool RejectDuplicates;
+ /**
+ * UDH (User Data Header)
+ */
+ GSM_UDHHeader UDH;
+ /**
+ * Sender or recipient number.
+ */
+ unsigned char Number[(GSM_MAX_NUMBER_LENGTH+1)*2];
+ /**
+ * SMSC (SMS Center)
+ */
+ GSM_SMSC SMSC;
+ /**
+ * For saved SMS: where exactly it's saved (SIM/phone)
+ */
+ GSM_MemoryType Memory;
+ /**
+ * For saved SMS: location of SMS in memory.
+ */
+ int Location;
+ /**
+ * For saved SMS: number of folder, where SMS is saved
+ */
+ int Folder;
+ /**
+ * For saved SMS: whether SMS is really in Inbox.
+ */
+ bool InboxFolder;
+ /**
+ * Length of the SMS message.
+ */
+ int Length;
+ /**
+ * Status (read/unread/...) of SMS message.
+ */
+ GSM_SMS_State State;
+ /**
+ * Name in Nokia with SMS memory (6210/7110, etc.) Ignored in other.
+ */
+ unsigned char Name[(GSM_MAX_SMS_NAME_LENGTH+1)*2];
+ /**
+ * Text for SMS.
+ */
+ unsigned char Text[(GSM_MAX_SMS_LENGTH+1)*2];
+ /**
+ * Type of message.
+ */
+ GSM_SMSMessageType PDU;
+ /**
+ * Type of coding.
+ */
+ GSM_Coding_Type Coding;
+ /**
+ * Date and time, when SMS was saved or sent
+ */
+ GSM_DateTime DateTime;
+ /**
+ * Date of SMSC response in DeliveryReport messages.
+ */
+ GSM_DateTime SMSCTime;
+ /**
+ * In delivery reports: status.
+ */
+ unsigned char DeliveryStatus;
+ /**
+ * Indicates whether "Reply via same center" is set.
+ */
+ bool ReplyViaSameSMSC;
+ /**
+ * SMS class.
+ */
+ char Class;
+ /**
+ * Message reference.
+ */
+ unsigned char MessageReference;
+} GSM_SMSMessage;
+
+/* In layouts are saved locations for some SMS part. Below are listed
+ * specs, which describe them
+ */
+typedef struct {
+ /**
+ * TP-User-Data. GSM 03.40 section 9.2.3.24.
+ */
+ unsigned char Text;
+ /**
+ * - In SMS-Deliver: TP-Originating-Address. GSM 03.40 section 9.2.3.7.
+ * - In SMS-Submit: TP-Destination-Address. GSM 03.40 section 9.2.3.8.
+ * - In SMS-Status-Report: TP-Recipient-Address. GSM 03.40 section 9.2.3.14.
+ */
+ unsigned char Number;
+ /**
+ * SMSC number
+ */
+ unsigned char SMSCNumber;
+ /**
+ * TP-Data-Coding-Scheme. GSM 03.40 section 9.2.3.10
+ */
+ unsigned char TPDCS;
+ /**
+ * - For SMS-Submit: TP-Validity-Period. GSM 03.40 section 9.2.3.12.
+ * - For SMS-Status-Report: TP-Discharge Time. GSM 03.40 section 9.2.3.13.
+ */
+ unsigned char DateTime;
+ /**
+ * TP-Service-Centre-Time-Stamp in SMS-Status-Report. GSM 03.40 section 9.2.3.11.
+ */
+ unsigned char SMSCTime;
+ /**
+ * TP-Status in SMS-Status-Report. GSM 03.40 section 9.2.3.15.
+ */
+ unsigned char TPStatus;
+ /**
+ * TP-User-Data-Length. GSM 03.40 section 9.2.3.16.
+ */
+ unsigned char TPUDL;
+ /**
+ * TP-Validity Period in SMS-Submit. GSM 03.40 section 9.2.3.12.
+ */
+ unsigned char TPVP;
+ /**
+ * Byte contains in SMS-Deliver:
+ * - TP-Message-Type-Indicator (2 bits) GSM 03.40 section 9.2.3.1
+ * - TP-More-Messages-To-Send (1 bit). GSM 03.40 section 9.2.3.2
+ * - TP-Reply-Path (1 bit). GSM 03.40 section 9.2.3.17
+ * - TP-User-Data-Header-Indicator (1 bit). GSM 03.40 section 9.2.3.23
+ * - TP-Status-Report-Indicator (1 bit). GSM 03.40 section 9.2.3.4
+ *
+ * Byte contains in SMS-Submit:
+ * - TP-Message-Type-Indicator (2 bits) GSM 03.40 section 9.2.3.1
+ * - TP-Reject-Duplicates (1 bit). GSM 03.40 section
+ * - TP-Validity-Period-Format (2 bits).GSM 03.40 section 9.2.3.3
+ * - TP-Reply-Path (1 bit). GSM 03.40 section 9.2.3.17
+ * - TP-User-Data-Header-Indicator (1 bit). GSM 03.40 section 9.2.3.23
+ * - TP-Status-Report-Request (1 bit). GSM 03.40 section 9.2.3.5
+ */
+ unsigned char firstbyte;
+ /**
+ * TP-Message Reference in SMS-Submit. GSM 03.40 section 9.2.3.6
+ */
+ unsigned char TPMR;
+ /**
+ * TP-Protocol-Identifier. GSM 03.40 section 9.2.3.9
+ */
+ unsigned char TPPID;
+} GSM_SMSMessageLayout;
+
+GSM_Error GSM_DecodeSMSFrame(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout);
+GSM_Error GSM_EncodeSMSFrame(GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout, int *length, bool clear);
+
+GSM_Error GSM_DecodeSMSFrameStatusReportData (GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout);
+GSM_Error GSM_DecodeSMSFrameText (GSM_SMSMessage *SMS, unsigned char *buffer, GSM_SMSMessageLayout Layout);
+
+void GSM_DecodeUDHHeader(GSM_UDHHeader *UDH);
+void GSM_EncodeUDHHeader(GSM_UDHHeader *UDH);
+
+void GSM_SetDefaultSMSData(GSM_SMSMessage *SMS);
+
+/* ---------------------- SMS folders ------------------------------------- */
+
+/**
+ * Number of possible SMS folders.
+ */
+#define GSM_MAX_SMS_FOLDERS 24
+/**
+ * Maximal length of SMS folder name.
+ */
+#define GSM_MAX_SMS_FOLDER_NAME_LEN 20
+
+/**
+ * Information about SMS folder.
+ */
+typedef struct {
+ /**
+ * Whether it is really inbox.
+ */
+ bool InboxFolder;
+ /**
+ * Where exactly it's saved
+ */
+ GSM_MemoryType Memory;
+ /**
+ * Name for SMS folder.
+ */
+ char Name[(GSM_MAX_SMS_FOLDER_NAME_LEN+1)*2];
+} GSM_OneSMSFolder;
+
+/**
+ * List of SMS folders.
+ */
+typedef struct {
+ /**
+ * Array of structures holding information about each folder.
+ */
+ GSM_OneSMSFolder Folder[GSM_MAX_SMS_FOLDERS];
+ /**
+ * Number of SMS folders.
+ */
+ unsigned char Number;
+} GSM_SMSFolders;
+
+#endif
+
+/* How should editor hadle tabs in this file? Add editor commands here.
+ * vim: noexpandtab sw=8 ts=8 sts=8:
+ */