mirror of
https://github.com/Xpl0itU/WiiUDownloader.git
synced 2025-05-13 06:34:44 -04:00
Add support for Wii's certificate generation
This commit is contained in:
parent
a6ccbb2473
commit
e93a238be9
5 changed files with 29 additions and 46 deletions
|
@ -10,23 +10,6 @@ import (
|
||||||
|
|
||||||
var cetkData []byte
|
var cetkData []byte
|
||||||
|
|
||||||
func getCert(tmdData []byte, id int, numContents uint16) ([]byte, error) { // TODO: Add support for Wii's TMD
|
|
||||||
var certSlice []byte
|
|
||||||
if len(tmdData) == int((0x0B04+0x30*numContents+0xA00)-0x300) {
|
|
||||||
certSlice = tmdData[0x0B04+0x30*numContents : 0x0B04+0x30*numContents+0xA00-0x300]
|
|
||||||
} else {
|
|
||||||
certSlice = tmdData[0x0B04+0x30*numContents : 0x0B04+0x30*numContents+0xA00]
|
|
||||||
}
|
|
||||||
switch id {
|
|
||||||
case 0:
|
|
||||||
return certSlice[:0x400], nil
|
|
||||||
case 1:
|
|
||||||
return certSlice[0x400 : 0x400+0x300], nil
|
|
||||||
default:
|
|
||||||
return nil, fmt.Errorf("invalid id: %d", id)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func getDefaultCert(progressReporter ProgressReporter, client *http.Client) ([]byte, error) {
|
func getDefaultCert(progressReporter ProgressReporter, client *http.Client) ([]byte, error) {
|
||||||
if len(cetkData) >= 0x350+0x300 {
|
if len(cetkData) >= 0x350+0x300 {
|
||||||
return cetkData[0x350 : 0x350+0x300], nil
|
return cetkData[0x350 : 0x350+0x300], nil
|
||||||
|
@ -50,20 +33,16 @@ func getDefaultCert(progressReporter ProgressReporter, client *http.Client) ([]b
|
||||||
return nil, fmt.Errorf("failed to download OSv10 cetk, length: %d", len(cetkData))
|
return nil, fmt.Errorf("failed to download OSv10 cetk, length: %d", len(cetkData))
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateCert(tmdData []byte, contentCount uint16, progressReporter ProgressReporter, client *http.Client) (bytes.Buffer, error) {
|
func GenerateCert(tmd *TMD, progressReporter ProgressReporter, client *http.Client) (bytes.Buffer, error) {
|
||||||
cert := bytes.Buffer{}
|
cert := bytes.Buffer{}
|
||||||
|
|
||||||
cert0, err := getCert(tmdData, 0, contentCount)
|
if _, err := cert.Write(tmd.Certificate1); err != nil {
|
||||||
if err != nil {
|
|
||||||
return bytes.Buffer{}, err
|
return bytes.Buffer{}, err
|
||||||
}
|
}
|
||||||
cert.Write(cert0)
|
|
||||||
|
|
||||||
cert1, err := getCert(tmdData, 1, contentCount)
|
if _, err := cert.Write(tmd.Certificate2); err != nil {
|
||||||
if err != nil {
|
|
||||||
return bytes.Buffer{}, err
|
return bytes.Buffer{}, err
|
||||||
}
|
}
|
||||||
cert.Write(cert1)
|
|
||||||
|
|
||||||
defaultCert, err := getDefaultCert(progressReporter, client)
|
defaultCert, err := getDefaultCert(progressReporter, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"context"
|
"context"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
@ -242,29 +241,16 @@ func (mw *MainWindow) ShowAll() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
var contentCount uint16
|
tmd, err := wiiudownloader.ParseTMD(tmdData)
|
||||||
if err := binary.Read(bytes.NewReader(tmdData[478:480]), binary.BigEndian, &contentCount); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var titleVersion uint16
|
titleKey, err := wiiudownloader.GenerateKey(fmt.Sprintf("%016x", tmd.TitleID))
|
||||||
if err := binary.Read(bytes.NewReader(tmdData[476:478]), binary.BigEndian, &titleVersion); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var titleID uint64
|
|
||||||
if err := binary.Read(bytes.NewReader(tmdData[0x018C:0x0194]), binary.BigEndian, &titleID); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
titleKey, err := wiiudownloader.GenerateKey(fmt.Sprintf("%016x", titleID))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
wiiudownloader.GenerateTicket(filepath.Join(parentDir, "title.tik"), titleID, titleKey, titleVersion)
|
wiiudownloader.GenerateTicket(filepath.Join(parentDir, "title.tik"), tmd.TitleID, titleKey, tmd.TitleVersion)
|
||||||
|
|
||||||
cert, err := wiiudownloader.GenerateCert(tmdData, contentCount, mw.progressWindow, http.DefaultClient)
|
cert, err := wiiudownloader.GenerateCert(tmd, mw.progressWindow, http.DefaultClient)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -416,7 +416,7 @@ func DecryptContents(path string, progressReporter ProgressReporter, deleteEncry
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
tmd, err := parseTMD(tmdData)
|
tmd, err := ParseTMD(tmdData)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -184,7 +184,7 @@ func DownloadTitle(titleID, outputDirectory string, doDecryption bool, progressR
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
tmd, err := parseTMD(tmdData)
|
tmd, err := ParseTMD(tmdData)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -211,7 +211,7 @@ func DownloadTitle(titleID, outputDirectory string, doDecryption bool, progressR
|
||||||
|
|
||||||
progressReporter.SetDownloadSize(int64(titleSize))
|
progressReporter.SetDownloadSize(int64(titleSize))
|
||||||
|
|
||||||
cert, err := GenerateCert(tmdData, tmd.ContentCount, progressReporter, client)
|
cert, err := GenerateCert(tmd, progressReporter, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if progressReporter.Cancelled() {
|
if progressReporter.Cancelled() {
|
||||||
return nil
|
return nil
|
||||||
|
|
20
tmd.go
20
tmd.go
|
@ -18,9 +18,11 @@ type TMD struct {
|
||||||
TitleVersion uint16
|
TitleVersion uint16
|
||||||
ContentCount uint16
|
ContentCount uint16
|
||||||
Contents []Content
|
Contents []Content
|
||||||
|
Certificate1 []byte
|
||||||
|
Certificate2 []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseTMD(data []byte) (*TMD, error) {
|
func ParseTMD(data []byte) (*TMD, error) {
|
||||||
tmd := &TMD{}
|
tmd := &TMD{}
|
||||||
reader := bytes.NewReader(data)
|
reader := bytes.NewReader(data)
|
||||||
|
|
||||||
|
@ -75,6 +77,14 @@ func parseTMD(data []byte) (*TMD, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
tmd.Certificate1 = make([]byte, 0x400)
|
||||||
|
if _, err := io.ReadFull(reader, tmd.Certificate1); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
tmd.Certificate2 = make([]byte, 0x300)
|
||||||
|
if _, err := io.ReadFull(reader, tmd.Certificate2); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
case TMD_VERSION_WIIU:
|
case TMD_VERSION_WIIU:
|
||||||
reader.Seek(0x18C, io.SeekStart)
|
reader.Seek(0x18C, io.SeekStart)
|
||||||
if err := binary.Read(reader, binary.BigEndian, &tmd.TitleID); err != nil {
|
if err := binary.Read(reader, binary.BigEndian, &tmd.TitleID); err != nil {
|
||||||
|
@ -121,6 +131,14 @@ func parseTMD(data []byte) (*TMD, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
tmd.Certificate1 = make([]byte, 0x400)
|
||||||
|
if _, err := io.ReadFull(reader, tmd.Certificate1); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
tmd.Certificate2 = make([]byte, 0x300)
|
||||||
|
if _, err := io.ReadFull(reader, tmd.Certificate2); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("unknown TMD version: %d", tmd.Version)
|
return nil, fmt.Errorf("unknown TMD version: %d", tmd.Version)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue