izapple2/cardBuilder.go

203 lines
5.0 KiB
Go
Raw Normal View History

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()
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
}