summaryrefslogtreecommitdiffabout
path: root/gammu/emb/common/phone/nokia
authorzautrix <zautrix>2004-08-07 17:24:40 (UTC)
committer zautrix <zautrix>2004-08-07 17:24:40 (UTC)
commit88b0d33b8b0b1f6ae320cfc863ca6a47fa8fec22 (patch) (side-by-side diff)
tree6331418973714243beb674abc87692277b83869d /gammu/emb/common/phone/nokia
parentef8a09ce74ad3f0a51484d03fdf009bd5b3677bf (diff)
downloadkdepimpi-88b0d33b8b0b1f6ae320cfc863ca6a47fa8fec22.zip
kdepimpi-88b0d33b8b0b1f6ae320cfc863ca6a47fa8fec22.tar.gz
kdepimpi-88b0d33b8b0b1f6ae320cfc863ca6a47fa8fec22.tar.bz2
Initial revision
Diffstat (limited to 'gammu/emb/common/phone/nokia') (more/less context) (ignore whitespace changes)
-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
23 files changed, 16177 insertions, 0 deletions
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:
+ */