2024-01-06 21:48:23 +01:00
|
|
|
package izapple2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"golang.org/x/exp/maps"
|
2024-02-08 22:17:14 +01:00
|
|
|
"golang.org/x/exp/slices"
|
2024-01-06 21:48:23 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type paramSpec struct {
|
|
|
|
name string
|
|
|
|
description string
|
|
|
|
defaultValue string
|
|
|
|
}
|
|
|
|
|
|
|
|
type cardBuilder struct {
|
|
|
|
name string
|
|
|
|
description string
|
|
|
|
defaultParams *[]paramSpec
|
|
|
|
requiresIIe bool
|
|
|
|
buildFunc func(params map[string]string) (Card, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
const noCardName = "empty"
|
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
var commonParams = []paramSpec{
|
2024-01-31 23:17:47 +01:00
|
|
|
{"trace", "Enable debug messages", "false"},
|
2024-01-27 17:19:44 +01:00
|
|
|
{"tracess", "Trace softswitches", "false"},
|
|
|
|
}
|
|
|
|
|
2024-01-06 21:48:23 +01:00
|
|
|
var cardFactory map[string]*cardBuilder
|
|
|
|
|
|
|
|
func getCardFactory() map[string]*cardBuilder {
|
|
|
|
if cardFactory != nil {
|
|
|
|
return cardFactory
|
|
|
|
}
|
|
|
|
cardFactory = make(map[string]*cardBuilder)
|
2024-01-27 17:21:40 +01:00
|
|
|
cardFactory["brainboard"] = newCardBrainBoardBuilder()
|
2024-01-27 17:19:44 +01:00
|
|
|
cardFactory["brainboard2"] = newCardBrainBoardIIBuilder()
|
2024-01-30 00:33:53 +01:00
|
|
|
cardFactory["dan2sd"] = newCardDan2ControllerBuilder()
|
2024-01-06 21:48:23 +01:00
|
|
|
cardFactory["diskii"] = newCardDisk2Builder()
|
|
|
|
cardFactory["diskiiseq"] = newCardDisk2SequencerBuilder()
|
|
|
|
cardFactory["fastchip"] = newCardFastChipBuilder()
|
|
|
|
cardFactory["fujinet"] = newCardSmartPortFujinetBuilder()
|
|
|
|
cardFactory["inout"] = newCardInOutBuilder()
|
|
|
|
cardFactory["language"] = newCardLanguageBuilder()
|
|
|
|
cardFactory["softswitchlogger"] = newCardLoggerBuilder()
|
|
|
|
cardFactory["memexp"] = newCardMemoryExpansionBuilder()
|
|
|
|
cardFactory["mouse"] = newCardMouseBuilder()
|
2024-01-15 21:09:02 +01:00
|
|
|
cardFactory["multirom"] = newMultiRomCardBuilder()
|
2024-01-06 21:48:23 +01:00
|
|
|
cardFactory["parallel"] = newCardParallelPrinterBuilder()
|
|
|
|
cardFactory["saturn"] = newCardSaturnBuilder()
|
|
|
|
cardFactory["smartport"] = newCardSmartPortStorageBuilder()
|
|
|
|
cardFactory["swyftcard"] = newCardSwyftBuilder()
|
|
|
|
cardFactory["thunderclock"] = newCardThunderClockPlusBuilder()
|
|
|
|
cardFactory["videx"] = newCardVidexBuilder()
|
|
|
|
cardFactory["vidhd"] = newCardVidHDBuilder()
|
|
|
|
return cardFactory
|
|
|
|
}
|
|
|
|
|
|
|
|
func availableCards() []string {
|
2024-02-08 22:17:14 +01:00
|
|
|
names := maps.Keys(getCardFactory())
|
|
|
|
slices.Sort(names)
|
|
|
|
return names
|
2024-01-06 21:48:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func setupCard(a *Apple2, slot int, paramString string) (Card, error) {
|
2024-01-27 17:19:44 +01:00
|
|
|
actualArgs := splitConfigurationString(paramString, ',')
|
2024-01-06 21:48:23 +01:00
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
cardName := actualArgs[0]
|
2024-01-06 21:48:23 +01:00
|
|
|
if cardName == "" || cardName == noCardName {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
builder, ok := getCardFactory()[cardName]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("unknown card %s", cardName)
|
|
|
|
}
|
|
|
|
|
|
|
|
if builder.requiresIIe && !a.isApple2e {
|
|
|
|
return nil, fmt.Errorf("card %s requires an Apple IIe", builder.name)
|
|
|
|
}
|
|
|
|
|
|
|
|
finalParams := make(map[string]string)
|
2024-01-27 17:19:44 +01:00
|
|
|
for _, commonParam := range commonParams {
|
|
|
|
finalParams[commonParam.name] = commonParam.defaultValue
|
|
|
|
}
|
2024-01-06 21:48:23 +01:00
|
|
|
if builder.defaultParams != nil {
|
|
|
|
for _, defaultParam := range *builder.defaultParams {
|
|
|
|
finalParams[defaultParam.name] = defaultParam.defaultValue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
for i := 1; i < len(actualArgs); i++ {
|
|
|
|
actualArgSides := splitConfigurationString(actualArgs[i], '=')
|
|
|
|
actualArgName := strings.ToLower(actualArgSides[0])
|
2024-01-06 21:48:23 +01:00
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
if _, ok := finalParams[actualArgName]; !ok {
|
|
|
|
return nil, fmt.Errorf("unknown parameter %s", actualArgSides[0])
|
2024-01-06 21:48:23 +01:00
|
|
|
}
|
2024-01-27 17:19:44 +01:00
|
|
|
if len(actualArgSides) > 2 {
|
|
|
|
return nil, fmt.Errorf("invalid parameter value for %s", actualArgSides[0])
|
2024-01-06 21:48:23 +01:00
|
|
|
}
|
2024-01-27 17:19:44 +01:00
|
|
|
if len(actualArgSides) == 1 {
|
|
|
|
finalParams[actualArgName] = "true"
|
2024-01-06 21:48:23 +01:00
|
|
|
} else {
|
2024-01-27 17:19:44 +01:00
|
|
|
finalParams[actualArgName] = actualArgSides[1]
|
2024-01-06 21:48:23 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
card, err := builder.buildFunc(finalParams)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
// Common parameters
|
|
|
|
traceSS := paramsGetBool(finalParams, "tracess")
|
|
|
|
if traceSS {
|
|
|
|
a.io.traceSlot(slot)
|
|
|
|
}
|
|
|
|
|
2024-01-31 23:17:47 +01:00
|
|
|
debug := paramsGetBool(finalParams, "trace")
|
2024-01-06 21:48:23 +01:00
|
|
|
|
2024-01-30 00:33:53 +01:00
|
|
|
card.setName(builder.name)
|
|
|
|
card.setDebug(debug)
|
2024-01-06 21:48:23 +01:00
|
|
|
card.assign(a, slot)
|
|
|
|
a.cards[slot] = card
|
|
|
|
return card, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func paramsGetBool(params map[string]string, name string) bool {
|
|
|
|
value, ok := params[name]
|
|
|
|
if !ok {
|
|
|
|
value = "false"
|
|
|
|
}
|
|
|
|
return value == "true"
|
|
|
|
}
|
|
|
|
|
|
|
|
func paramsGetString(params map[string]string, name string) string {
|
|
|
|
value, ok := params[name]
|
|
|
|
if !ok {
|
|
|
|
value = ""
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
|
|
|
func paramsGetPath(params map[string]string, name string) string {
|
|
|
|
value := paramsGetString(params, name)
|
|
|
|
if strings.HasPrefix(value, "\"") && strings.HasSuffix(value, "\"") {
|
|
|
|
value = value[1 : len(value)-1]
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
|
|
|
func paramsGetInt(params map[string]string, name string) (int, error) {
|
|
|
|
value, ok := params[name]
|
|
|
|
if !ok {
|
2024-01-27 17:19:44 +01:00
|
|
|
return 0, fmt.Errorf("missing parameter %s", name)
|
2024-01-06 21:48:23 +01:00
|
|
|
}
|
|
|
|
return strconv.Atoi(value)
|
|
|
|
}
|
|
|
|
|
2024-01-27 17:19:44 +01:00
|
|
|
// Returns a 1 based array of bools
|
|
|
|
func paramsGetDIPs(params map[string]string, name string, size int) ([]bool, error) {
|
|
|
|
value, ok := params[name]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("missing parameter %s", name)
|
|
|
|
}
|
|
|
|
if len(value) != 8 {
|
|
|
|
return nil, fmt.Errorf("DIP switches must be 8 characters long")
|
|
|
|
}
|
|
|
|
result := make([]bool, size+1)
|
|
|
|
for i := 0; i < 8; i++ {
|
|
|
|
result[i+1] = value[i] == '1'
|
|
|
|
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2024-01-06 21:48:23 +01:00
|
|
|
func splitConfigurationString(s string, separator rune) []string {
|
|
|
|
// Split by comma, but not inside quotes
|
|
|
|
var result []string
|
|
|
|
var current string
|
|
|
|
inQuote := false
|
|
|
|
for _, c := range s {
|
|
|
|
if c == '"' {
|
|
|
|
inQuote = !inQuote
|
|
|
|
}
|
|
|
|
if c == separator && !inQuote {
|
|
|
|
result = append(result, current)
|
|
|
|
current = ""
|
|
|
|
} else {
|
|
|
|
current += string(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if current != "" {
|
|
|
|
result = append(result, current)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|