KeyCrypterScrypt.smali
.class public Lorg/bitcoinj/crypto/KeyCrypterScrypt;
.super Ljava/lang/Object;
.source "KeyCrypterScrypt.java"
# interfaces
.implements Lorg/bitcoinj/crypto/KeyCrypter;
# static fields
.field public static final BLOCK_LENGTH:I = 0x10
.field public static final KEY_LENGTH:I = 0x20
.field public static final SALT_LENGTH:I = 0x8
.field private static final log:Lorg/slf4j/Logger;
.field private static final secureRandom:Ljava/security/SecureRandom;
# instance fields
.field private final scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
# direct methods
.method static constructor <clinit>()V
.registers 1
.prologue
.line 58
const-class v0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;
invoke-static {v0}, Lorg/slf4j/LoggerFactory;->getLogger(Ljava/lang/Class;)Lorg/slf4j/Logger;
move-result-object v0
sput-object v0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->log:Lorg/slf4j/Logger;
.line 78
invoke-static {}, Lorg/bitcoinj/core/Utils;->isAndroidRuntime()Z
move-result v0
if-eqz v0, :cond_13
.line 79
new-instance v0, Lorg/bitcoinj/crypto/LinuxSecureRandom;
invoke-direct {v0}, Lorg/bitcoinj/crypto/LinuxSecureRandom;-><init>()V
.line 81
:cond_13
new-instance v0, Ljava/security/SecureRandom;
invoke-direct {v0}, Ljava/security/SecureRandom;-><init>()V
sput-object v0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->secureRandom:Ljava/security/SecureRandom;
.line 82
return-void
.end method
.method public constructor <init>()V
.registers 3
.prologue
.line 99
invoke-direct {p0}, Ljava/lang/Object;-><init>()V
.line 100
invoke-static {}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->newBuilder()Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;
move-result-object v0
.line 101
invoke-static {}, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->randomSalt()[B
move-result-object v1
invoke-static {v1}, Lcom/google/protobuf/ByteString;->copyFrom([B)Lcom/google/protobuf/ByteString;
move-result-object v1
.line 100
invoke-virtual {v0, v1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;->setSalt(Lcom/google/protobuf/ByteString;)Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;
move-result-object v0
.line 102
invoke-virtual {v0}, Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;->build()Lorg/bitcoinj/wallet/Protos$ScryptParameters;
move-result-object v0
iput-object v0, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
.line 103
return-void
.end method
.method public constructor <init>(I)V
.registers 6
.prologue
.line 112
invoke-direct {p0}, Ljava/lang/Object;-><init>()V
.line 113
invoke-static {}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->newBuilder()Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;
move-result-object v0
.line 114
invoke-static {}, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->randomSalt()[B
move-result-object v1
invoke-static {v1}, Lcom/google/protobuf/ByteString;->copyFrom([B)Lcom/google/protobuf/ByteString;
move-result-object v1
invoke-virtual {v0, v1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;->setSalt(Lcom/google/protobuf/ByteString;)Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;
move-result-object v0
int-to-long v2, p1
invoke-virtual {v0, v2, v3}, Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;->setN(J)Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;
move-result-object v0
.line 115
invoke-virtual {v0}, Lorg/bitcoinj/wallet/Protos$ScryptParameters$Builder;->build()Lorg/bitcoinj/wallet/Protos$ScryptParameters;
move-result-object v0
iput-object v0, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
.line 116
return-void
.end method
.method public constructor <init>(Lorg/bitcoinj/wallet/Protos$ScryptParameters;)V
.registers 4
.prologue
.line 124
invoke-direct {p0}, Ljava/lang/Object;-><init>()V
.line 125
invoke-static {p1}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
move-result-object v0
check-cast v0, Lorg/bitcoinj/wallet/Protos$ScryptParameters;
iput-object v0, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
.line 128
invoke-virtual {p1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getSalt()Lcom/google/protobuf/ByteString;
move-result-object v0
if-eqz v0, :cond_26
.line 129
invoke-virtual {p1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getSalt()Lcom/google/protobuf/ByteString;
move-result-object v0
invoke-virtual {v0}, Lcom/google/protobuf/ByteString;->toByteArray()[B
move-result-object v0
if-eqz v0, :cond_26
.line 130
invoke-virtual {p1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getSalt()Lcom/google/protobuf/ByteString;
move-result-object v0
invoke-virtual {v0}, Lcom/google/protobuf/ByteString;->toByteArray()[B
move-result-object v0
array-length v0, v0
if-nez v0, :cond_2d
.line 131
:cond_26
sget-object v0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->log:Lorg/slf4j/Logger;
const-string v1, "You are using a ScryptParameters with no salt. Your encryption may be vulnerable to a dictionary attack."
invoke-interface {v0, v1}, Lorg/slf4j/Logger;->warn(Ljava/lang/String;)V
.line 133
:cond_2d
return-void
.end method
.method private static convertToByteArray(Ljava/lang/CharSequence;)[B
.registers 5
.prologue
.line 238
invoke-static {p0}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
.line 240
invoke-interface {p0}, Ljava/lang/CharSequence;->length()I
move-result v0
shl-int/lit8 v0, v0, 0x1
new-array v1, v0, [B
.line 241
const/4 v0, 0x0
:goto_c
invoke-interface {p0}, Ljava/lang/CharSequence;->length()I
move-result v2
if-ge v0, v2, :cond_29
.line 242
shl-int/lit8 v2, v0, 0x1
.line 243
invoke-interface {p0, v0}, Ljava/lang/CharSequence;->charAt(I)C
move-result v3
shr-int/lit8 v3, v3, 0x8
int-to-byte v3, v3
aput-byte v3, v1, v2
.line 244
add-int/lit8 v2, v2, 0x1
invoke-interface {p0, v0}, Ljava/lang/CharSequence;->charAt(I)C
move-result v3
int-to-byte v3, v3
aput-byte v3, v1, v2
.line 241
add-int/lit8 v0, v0, 0x1
goto :goto_c
.line 246
:cond_29
return-object v1
.end method
.method public static randomSalt()[B
.registers 2
.prologue
.line 88
const/16 v0, 0x8
new-array v0, v0, [B
.line 89
sget-object v1, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->secureRandom:Ljava/security/SecureRandom;
invoke-virtual {v1, v0}, Ljava/security/SecureRandom;->nextBytes([B)V
.line 90
return-object v0
.end method
# virtual methods
.method public decrypt(Lorg/bitcoinj/crypto/EncryptedData;Lorg/spongycastle/crypto/params/KeyParameter;)[B
.registers 9
.prologue
.line 211
invoke-static {p1}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
.line 212
invoke-static {p2}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
.line 215
:try_start_6
new-instance v1, Lorg/spongycastle/crypto/params/ParametersWithIV;
new-instance v0, Lorg/spongycastle/crypto/params/KeyParameter;
invoke-virtual {p2}, Lorg/spongycastle/crypto/params/KeyParameter;->getKey()[B
move-result-object v2
invoke-direct {v0, v2}, Lorg/spongycastle/crypto/params/KeyParameter;-><init>([B)V
iget-object v2, p1, Lorg/bitcoinj/crypto/EncryptedData;->initialisationVector:[B
invoke-direct {v1, v0, v2}, Lorg/spongycastle/crypto/params/ParametersWithIV;-><init>(Lorg/spongycastle/crypto/CipherParameters;[B)V
.line 218
new-instance v0, Lorg/spongycastle/crypto/paddings/PaddedBufferedBlockCipher;
new-instance v2, Lorg/spongycastle/crypto/modes/CBCBlockCipher;
new-instance v3, Lorg/spongycastle/crypto/engines/AESFastEngine;
invoke-direct {v3}, Lorg/spongycastle/crypto/engines/AESFastEngine;-><init>()V
invoke-direct {v2, v3}, Lorg/spongycastle/crypto/modes/CBCBlockCipher;-><init>(Lorg/spongycastle/crypto/BlockCipher;)V
invoke-direct {v0, v2}, Lorg/spongycastle/crypto/paddings/PaddedBufferedBlockCipher;-><init>(Lorg/spongycastle/crypto/BlockCipher;)V
.line 219
const/4 v2, 0x0
invoke-virtual {v0, v2, v1}, Lorg/spongycastle/crypto/BufferedBlockCipher;->init(ZLorg/spongycastle/crypto/CipherParameters;)V
.line 221
iget-object v1, p1, Lorg/bitcoinj/crypto/EncryptedData;->encryptedBytes:[B
.line 222
array-length v2, v1
invoke-virtual {v0, v2}, Lorg/spongycastle/crypto/BufferedBlockCipher;->getOutputSize(I)I
move-result v2
new-array v4, v2, [B
.line 223
const/4 v2, 0x0
array-length v3, v1
const/4 v5, 0x0
invoke-virtual/range {v0 .. v5}, Lorg/spongycastle/crypto/BufferedBlockCipher;->processBytes([BII[BI)I
move-result v1
.line 224
invoke-virtual {v0, v4, v1}, Lorg/spongycastle/crypto/BufferedBlockCipher;->doFinal([BI)I
move-result v0
.line 226
add-int/2addr v0, v1
invoke-static {v4, v0}, Ljava/util/Arrays;->copyOf([BI)[B
:try_end_41
.catch Ljava/lang/Exception; {:try_start_6 .. :try_end_41} :catch_43
move-result-object v0
return-object v0
.line 227
:catch_43
move-exception v0
.line 228
new-instance v1, Lorg/bitcoinj/crypto/KeyCrypterException;
const-string v2, "Could not decrypt bytes"
invoke-direct {v1, v2, v0}, Lorg/bitcoinj/crypto/KeyCrypterException;-><init>(Ljava/lang/String;Ljava/lang/Throwable;)V
throw v1
.end method
.method public deriveKey(Ljava/lang/CharSequence;)Lorg/spongycastle/crypto/params/KeyParameter;
.registers 11
.prologue
const/4 v7, 0x0
.line 146
const/4 v1, 0x0
.line 148
:try_start_2
invoke-static {p1}, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->convertToByteArray(Ljava/lang/CharSequence;)[B
:try_end_5
.catch Ljava/lang/Exception; {:try_start_2 .. :try_end_5} :catch_77
.catchall {:try_start_2 .. :try_end_5} :catchall_6b
move-result-object v0
.line 149
const/4 v1, 0x0
:try_start_7
new-array v1, v1, [B
.line 150
iget-object v2, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v2}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getSalt()Lcom/google/protobuf/ByteString;
move-result-object v2
if-eqz v2, :cond_57
.line 151
iget-object v1, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getSalt()Lcom/google/protobuf/ByteString;
move-result-object v1
invoke-virtual {v1}, Lcom/google/protobuf/ByteString;->toByteArray()[B
move-result-object v1
.line 158
:goto_1b
invoke-static {}, Lcom/google/common/base/s;->wT()Lcom/google/common/base/s;
move-result-object v6
.line 159
iget-object v2, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v2}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getN()J
move-result-wide v2
long-to-int v2, v2
iget-object v3, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v3}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getR()I
move-result v3
iget-object v4, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v4}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getP()I
move-result v4
const/16 v5, 0x20
invoke-static/range {v0 .. v5}, Lcom/lambdaworks/crypto/SCrypt;->a([B[BIIII)[B
move-result-object v1
.line 160
invoke-virtual {v6}, Lcom/google/common/base/s;->wU()Lcom/google/common/base/s;
.line 161
sget-object v2, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->log:Lorg/slf4j/Logger;
const-string v3, "Deriving key took {} for {} scrypt iterations."
iget-object v4, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v4}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getN()J
move-result-wide v4
invoke-static {v4, v5}, Ljava/lang/Long;->valueOf(J)Ljava/lang/Long;
move-result-object v4
invoke-interface {v2, v3, v6, v4}, Lorg/slf4j/Logger;->info(Ljava/lang/String;Ljava/lang/Object;Ljava/lang/Object;)V
.line 162
new-instance v2, Lorg/spongycastle/crypto/params/KeyParameter;
invoke-direct {v2, v1}, Lorg/spongycastle/crypto/params/KeyParameter;-><init>([B)V
:try_end_51
.catch Ljava/lang/Exception; {:try_start_7 .. :try_end_51} :catch_5f
.catchall {:try_start_7 .. :try_end_51} :catchall_72
.line 167
if-eqz v0, :cond_56
.line 168
invoke-static {v0, v7}, Ljava/util/Arrays;->fill([BB)V
.line 162
:cond_56
return-object v2
.line 155
:cond_57
:try_start_57
sget-object v2, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->log:Lorg/slf4j/Logger;
const-string v3, "You are using a ScryptParameters with no salt. Your encryption may be vulnerable to a dictionary attack."
invoke-interface {v2, v3}, Lorg/slf4j/Logger;->warn(Ljava/lang/String;)V
:try_end_5e
.catch Ljava/lang/Exception; {:try_start_57 .. :try_end_5e} :catch_5f
.catchall {:try_start_57 .. :try_end_5e} :catchall_72
goto :goto_1b
.line 163
:catch_5f
move-exception v1
move-object v8, v1
move-object v1, v0
move-object v0, v8
.line 164
:goto_63
:try_start_63
new-instance v2, Lorg/bitcoinj/crypto/KeyCrypterException;
const-string v3, "Could not generate key from password and salt."
invoke-direct {v2, v3, v0}, Lorg/bitcoinj/crypto/KeyCrypterException;-><init>(Ljava/lang/String;Ljava/lang/Throwable;)V
throw v2
:try_end_6b
.catchall {:try_start_63 .. :try_end_6b} :catchall_6b
.line 167
:catchall_6b
move-exception v0
:goto_6c
if-eqz v1, :cond_71
.line 168
invoke-static {v1, v7}, Ljava/util/Arrays;->fill([BB)V
:cond_71
throw v0
.line 167
:catchall_72
move-exception v1
move-object v8, v1
move-object v1, v0
move-object v0, v8
goto :goto_6c
.line 163
:catch_77
move-exception v0
goto :goto_63
.end method
.method public encrypt([BLorg/spongycastle/crypto/params/KeyParameter;)Lorg/bitcoinj/crypto/EncryptedData;
.registers 10
.prologue
.line 178
invoke-static {p1}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
.line 179
invoke-static {p2}, Lcom/google/common/base/n;->checkNotNull(Ljava/lang/Object;)Ljava/lang/Object;
.line 183
const/16 v0, 0x10
:try_start_8
new-array v6, v0, [B
.line 184
sget-object v0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->secureRandom:Ljava/security/SecureRandom;
invoke-virtual {v0, v6}, Ljava/security/SecureRandom;->nextBytes([B)V
.line 186
new-instance v1, Lorg/spongycastle/crypto/params/ParametersWithIV;
invoke-direct {v1, p2, v6}, Lorg/spongycastle/crypto/params/ParametersWithIV;-><init>(Lorg/spongycastle/crypto/CipherParameters;[B)V
.line 189
new-instance v0, Lorg/spongycastle/crypto/paddings/PaddedBufferedBlockCipher;
new-instance v2, Lorg/spongycastle/crypto/modes/CBCBlockCipher;
new-instance v3, Lorg/spongycastle/crypto/engines/AESFastEngine;
invoke-direct {v3}, Lorg/spongycastle/crypto/engines/AESFastEngine;-><init>()V
invoke-direct {v2, v3}, Lorg/spongycastle/crypto/modes/CBCBlockCipher;-><init>(Lorg/spongycastle/crypto/BlockCipher;)V
invoke-direct {v0, v2}, Lorg/spongycastle/crypto/paddings/PaddedBufferedBlockCipher;-><init>(Lorg/spongycastle/crypto/BlockCipher;)V
.line 190
const/4 v2, 0x1
invoke-virtual {v0, v2, v1}, Lorg/spongycastle/crypto/BufferedBlockCipher;->init(ZLorg/spongycastle/crypto/CipherParameters;)V
.line 191
array-length v1, p1
invoke-virtual {v0, v1}, Lorg/spongycastle/crypto/BufferedBlockCipher;->getOutputSize(I)I
move-result v1
new-array v4, v1, [B
.line 192
const/4 v2, 0x0
array-length v3, p1
const/4 v5, 0x0
move-object v1, p1
invoke-virtual/range {v0 .. v5}, Lorg/spongycastle/crypto/BufferedBlockCipher;->processBytes([BII[BI)I
move-result v1
.line 193
invoke-virtual {v0, v4, v1}, Lorg/spongycastle/crypto/BufferedBlockCipher;->doFinal([BI)I
move-result v0
.line 195
new-instance v2, Lorg/bitcoinj/crypto/EncryptedData;
add-int/2addr v0, v1
invoke-static {v4, v0}, Ljava/util/Arrays;->copyOf([BI)[B
move-result-object v0
invoke-direct {v2, v6, v0}, Lorg/bitcoinj/crypto/EncryptedData;-><init>([B[B)V
:try_end_44
.catch Ljava/lang/Exception; {:try_start_8 .. :try_end_44} :catch_45
return-object v2
.line 196
:catch_45
move-exception v0
.line 197
new-instance v1, Lorg/bitcoinj/crypto/KeyCrypterException;
const-string v2, "Could not encrypt bytes."
invoke-direct {v1, v2, v0}, Lorg/bitcoinj/crypto/KeyCrypterException;-><init>(Ljava/lang/String;Ljava/lang/Throwable;)V
throw v1
.end method
.method public equals(Ljava/lang/Object;)Z
.registers 4
.prologue
.line 274
if-ne p0, p1, :cond_4
const/4 v0, 0x1
.line 276
:goto_3
return v0
.line 275
:cond_4
if-eqz p1, :cond_10
invoke-virtual {p0}, Ljava/lang/Object;->getClass()Ljava/lang/Class;
move-result-object v0
invoke-virtual {p1}, Ljava/lang/Object;->getClass()Ljava/lang/Class;
move-result-object v1
if-eq v0, v1, :cond_12
:cond_10
const/4 v0, 0x0
goto :goto_3
.line 276
:cond_12
iget-object v0, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
check-cast p1, Lorg/bitcoinj/crypto/KeyCrypterScrypt;
iget-object v1, p1, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-static {v0, v1}, Lcom/google/common/base/m;->equal(Ljava/lang/Object;Ljava/lang/Object;)Z
move-result v0
goto :goto_3
.end method
.method public getScryptParameters()Lorg/bitcoinj/wallet/Protos$ScryptParameters;
.registers 2
.prologue
.line 250
iget-object v0, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
return-object v0
.end method
.method public getUnderstoodEncryptionType()Lorg/bitcoinj/wallet/Protos$Wallet$EncryptionType;
.registers 2
.prologue
.line 259
sget-object v0, Lorg/bitcoinj/wallet/Protos$Wallet$EncryptionType;->ENCRYPTED_SCRYPT_AES:Lorg/bitcoinj/wallet/Protos$Wallet$EncryptionType;
return-object v0
.end method
.method public hashCode()I
.registers 4
.prologue
.line 269
const/4 v0, 0x1
new-array v0, v0, [Ljava/lang/Object;
const/4 v1, 0x0
iget-object v2, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
aput-object v2, v0, v1
.line 1084
invoke-static {v0}, Ljava/util/Arrays;->hashCode([Ljava/lang/Object;)I
move-result v0
.line 269
return v0
.end method
.method public toString()Ljava/lang/String;
.registers 5
.prologue
.line 264
new-instance v0, Ljava/lang/StringBuilder;
const-string v1, "AES-256-CBC, Scrypt (N: "
invoke-direct {v0, v1}, Ljava/lang/StringBuilder;-><init>(Ljava/lang/String;)V
iget-object v1, p0, Lorg/bitcoinj/crypto/KeyCrypterScrypt;->scryptParameters:Lorg/bitcoinj/wallet/Protos$ScryptParameters;
invoke-virtual {v1}, Lorg/bitcoinj/wallet/Protos$ScryptParameters;->getN()J
move-result-wide v2
invoke-virtual {v0, v2, v3}, Ljava/lang/StringBuilder;->append(J)Ljava/lang/StringBuilder;
move-result-object v0
const-string v1, ")"
invoke-virtual {v0, v1}, Ljava/lang/StringBuilder;->append(Ljava/lang/String;)Ljava/lang/StringBuilder;
move-result-object v0
invoke-virtual {v0}, Ljava/lang/StringBuilder;->toString()Ljava/lang/String;
move-result-object v0
return-object v0
.end method