创建适用于Android的PDU,与SmsMessage.createFromPdu()(GSM 3gpp)配合使用

时间:2012-09-09 00:44:47

标签: java android pdu

目标:(注意:所选答案为CDMA (3gpp2) please refer here生成GSM(3gpp)PDU

创建可以传递到SmsMessage.createFromPdu(byte[] pdu)的PDU。 我正在向我的BroadcastReciever中的一个收听短信的“广播意图”。

一个BroadcastReciever

android.provider.Telephony.SMS_RECEIVED用于“真实”短信

对这些新的“应用程序短信”使用自定义intent-filter操作。

@Override
public void onReceive(Context context, Intent intent) {

    Bundle bundle = intent.getExtras();

    if (bundle != null) {
        Object[] pdusObj = (Object[]) bundle.get("pdus");
        SmsMessage[] messages = new SmsMessage[pdusObj.length];

        // getting SMS information from Pdu.
        for (int i = 0; i < pdusObj.length; i++) {
            messages[i] = SmsMessage.createFromPdu((byte[]) pdusObj[i]);
        }

        for (SmsMessage currentMessage : messages) {
            //the currentMessage.getDisplayOriginatingAddress() 
            //or .getDisplayMessageBody() is null if I Broadcast a fake sms
            Log.i("BB", "address:"+currentMessage.getDisplayOriginatingAddress()+" message:"+currentMessage.getDisplayMessageBody());
    ...

所以我希望我的BroadcastReciever能够在不添加额外代码的情况下处理这两种类型的消息

(是的,我知道我可以为不同的BroadcastReciever行动提供不同的intent-filter,但我想实际上取消它,因为我知道可以做到,我很固执)

研究

我整天都在做研究。虽然我对数学和转换以及创建合适的算法非常糟糕,但我已经尝试自己编写。我查看了Stack topics on PDUsCreate PDU Android,但答案中的链接已断开。我甚至看了com.google.android.mms.pdu源代码

到目前为止,我只能使用http://www.wrankl.de/JavaPC/SMSTools.html中的一些代码创建一个没有“始发地址”的PDU

PDU:

目的地:555消息:helloworld

"1100038155f50000aa0ae8329bfdbebfe56c32"

这显然无效......

旁注:

除了本地使用之外,我不打算对PDU做任何事情,我不想在我的代码中使用硬编码PDU,因为我没有重复使用PDU。

如果有什么我可以添加到我用来添加“始发地址”的代码中,那就可以了。或者有没有人知道我不知道的图书馆的信息?

由于

更新

尝试了

byte[] by =(byte[])(SmsMessage.getSubmitPdu("12345", "1234", "hello", false).encodedMessage);

给出了以下内容(以十六进制表示)

"0000100200000000000000000000000004010203040000000e000320ec400107102e8cbb366f00"

没有用

3 个答案:

答案 0 :(得分:27)

也许这个代码段没有您想要的许多详细信息字段,但对于我的简单用途,它可以像其他短信一样调用通知。

    private static void createFakeSms(Context context, String sender,
        String body) {
    byte[] pdu = null;
    byte[] scBytes = PhoneNumberUtils
            .networkPortionToCalledPartyBCD("0000000000");
    byte[] senderBytes = PhoneNumberUtils
            .networkPortionToCalledPartyBCD(sender);
    int lsmcs = scBytes.length;
    byte[] dateBytes = new byte[7];
    Calendar calendar = new GregorianCalendar();
    dateBytes[0] = reverseByte((byte) (calendar.get(Calendar.YEAR)));
    dateBytes[1] = reverseByte((byte) (calendar.get(Calendar.MONTH) + 1));
    dateBytes[2] = reverseByte((byte) (calendar.get(Calendar.DAY_OF_MONTH)));
    dateBytes[3] = reverseByte((byte) (calendar.get(Calendar.HOUR_OF_DAY)));
    dateBytes[4] = reverseByte((byte) (calendar.get(Calendar.MINUTE)));
    dateBytes[5] = reverseByte((byte) (calendar.get(Calendar.SECOND)));
    dateBytes[6] = reverseByte((byte) ((calendar.get(Calendar.ZONE_OFFSET) + calendar
            .get(Calendar.DST_OFFSET)) / (60 * 1000 * 15)));
    try {
        ByteArrayOutputStream bo = new ByteArrayOutputStream();
        bo.write(lsmcs);
        bo.write(scBytes);
        bo.write(0x04);
        bo.write((byte) sender.length());
        bo.write(senderBytes);
        bo.write(0x00);
        bo.write(0x00); // encoding: 0 for default 7bit
        bo.write(dateBytes);
        try {
            String sReflectedClassName = "com.android.internal.telephony.GsmAlphabet";
            Class cReflectedNFCExtras = Class.forName(sReflectedClassName);
            Method stringToGsm7BitPacked = cReflectedNFCExtras.getMethod(
                    "stringToGsm7BitPacked", new Class[] { String.class });
            stringToGsm7BitPacked.setAccessible(true);
            byte[] bodybytes = (byte[]) stringToGsm7BitPacked.invoke(null,
                    body);
            bo.write(bodybytes);
        } catch (Exception e) {
        }

        pdu = bo.toByteArray();
    } catch (IOException e) {
    }

    Intent intent = new Intent();
    intent.setClassName("com.android.mms",
            "com.android.mms.transaction.SmsReceiverService");
    intent.setAction("android.provider.Telephony.SMS_RECEIVED");
    intent.putExtra("pdus", new Object[] { pdu });
    intent.putExtra("format", "3gpp");
    context.startService(intent);
}

private static byte reverseByte(byte b) {
    return (byte) ((b & 0xF0) >> 4 | (b & 0x0F) << 4);
}

希望你能找到有用的东西

更新:

 public static final SmsMessage[] getMessagesFromIntent(
                Intent intent) {
            Object[] messages = (Object[]) intent.getSerializableExtra("pdus");
            byte[][] pduObjs = new byte[messages.length][];

            for (int i = 0; i < messages.length; i++) {
                pduObjs[i] = (byte[]) messages[i];
            }
            byte[][] pdus = new byte[pduObjs.length][];
            int pduCount = pdus.length;
            SmsMessage[] msgs = new SmsMessage[pduCount];
            for (int i = 0; i < pduCount; i++) {
                pdus[i] = pduObjs[i];
                msgs[i] = SmsMessage.createFromPdu(pdus[i]);
            }
            return msgs;
        }

答案 1 :(得分:1)

自从我完成任何直接的PDU争吵以来,这已经很长时间了,但是当我这么做时,我很快放弃并使用了SMSLib:它已经完美地用于通过诺基亚手机发送的PDU实用程序(通过一系列链接)。我的假设(可能是错误的)是它们也适用于Android,假设接口实际上符合规范。

答案 2 :(得分:0)

请检查此code in console.c。这是android模拟器创建pdu和RIL.java的地方,其中CMT消息被转换为SmsMessage。您可以使用SmsMessage.getPdu来获取pdu。但是SmsMessage.newFromCmt看起来像是一个内部的api。所以它可能不可靠。

此外,这仅适用于Gsm,cdma具有完全不同的代码流,并且由于RIL.java和调制解调器完全是制造商特定的,因此这可能仅适用于模拟器。

通常GSM代码在Android上更可靠,所以不妨在gsm手机上工作。试一试。