author | zautrix <zautrix> | 2004-08-07 17:24:40 (UTC) |
---|---|---|
committer | zautrix <zautrix> | 2004-08-07 17:24:40 (UTC) |
commit | 88b0d33b8b0b1f6ae320cfc863ca6a47fa8fec22 (patch) (side-by-side diff) | |
tree | 6331418973714243beb674abc87692277b83869d /gammu/emb/common/phone/nokia | |
parent | ef8a09ce74ad3f0a51484d03fdf009bd5b3677bf (diff) | |
download | kdepimpi-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)
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: + */ |