12

目标:(注意:选择的答案为 CDMA(3gpp2)生成一个 GSM(3gpp)PDU )请参考这里

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

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动作采取不同的方式,但我想真正做到这一点,因为我知道这是可以做到的,我很固执)

研究:

我整天/晚上都在做研究。即使我对数学和转换以及创建合适的算法非常糟糕,我也尝试过编写自己的代码。我查看了有关 PDUs和Create PDU Android的Stack 主题,但答案中的链接已损坏。我什至看了源代码com.google.android.mms.pdu

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

配电单元:

目的地:555 消息:helloworld

"1100038155f50000aa0ae8329bfdbebfe56c32"

这显然是无效的......

旁注:

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

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

谢谢

更新:

试过了

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

这给了我以下(以十六进制表示)

"0000100200000000000000000000000004010203040000000e000320ec400107102e8cbb366f00"

没用

4

3 回答 3

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;
        }
于 2012-09-09T11:05:20.443 回答
1

自从我进行任何直接的 PDU 争吵以来已经很久了,但是当我这样做时,我很快放弃并使用了SMSLib:PDU 实用程序非常适合通过诺基亚手机(通过串行链接)发送。我的假设(可能是错误的)是它们也适用于 Android,假设接口实际上符合规范。

于 2012-09-09T01:01:58.300 回答
0

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

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

通常 GSM 代码在 android 上更可靠,所以也可以在 gsm 手机上工作。一定要试一试。

于 2012-09-09T16:02:28.270 回答