创建Android的PDU与SmsMessage.createFromPdu()(GSM 3GPP)

2019-06-18 02:19发布

目标:(注意:选择的答案产生一个GSM(3GPP)PDU)的CDMA(3GPP2)请参阅这里

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

一个BroadcastReciever

使用android.provider.Telephony.SMS_RECEIVED “真正的”短信的

使用自定义intent-filter ,这些新的“应用SMS的”行动。

@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作用,但我想把这事办成,因为我知道这是可以做到的,我固执)

研究:

我一直在做研究的所有昼/夜。 我试着写我自己,即使我很可怕与数学和转化创造一个合适的算法。 我看过了堆栈主题上的PDU ,并创建PDU的Android ,但该链接的答案打破。 我甚至在看着com.google.android.mms.pdu源代码

到目前为止,我只能够创建一个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"

我以前不工作

Answer 1:

也许这个片段没有很多的细节领域,如你想,但我的目的很简单就可以调用通知像另一个短信。

    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;
        }


Answer 2:

它因为我已经做了任何直接PDU争论了很长时间,但是当我做我很快放弃了,并使用SMSLib :在PDU公用事业得很完美了通过诺基亚手机发送(通过串行链路)。 我的假设(这可能是错误的)是,他们将针对Android以及工作,假设接口实际上是符合规范。



Answer 3:

请检查此console.c中的代码 。 这是机器人仿真器创建PDU和RIL.java其中CMT消息被转换为一个SmsMessage。 您可以使用SmsMessage.getPdu获得PDU。 但SmsMessage.newFromCmt看起来是一个内部API。 因此,它可能不可靠。

而且这还只是针对GSM,CDMA有着完全不同的codeflow以来RIL.java和调制解调器完全特定制造商,这可能只在模拟器上工作。

通常GSM代码是在Android上更可靠,所以可能的GSM手机上使用工作。 不要试试看。



文章来源: Create PDU for Android that works with SmsMessage.createFromPdu() (GSM 3gpp)
标签: java android pdu