1
0
mirror of https://github.com/golang/go synced 2024-11-22 04:34:39 -07:00

chan and map of [] and struct

R=r
DELTA=192  (145 added, 8 deleted, 39 changed)
OCL=21609
CL=21614
This commit is contained in:
Russ Cox 2008-12-19 12:05:22 -08:00
parent cbff09d666
commit eee50ae1ac
8 changed files with 186 additions and 45 deletions

View File

@ -41,6 +41,8 @@ enum
ASTRING, ASTRING,
APTR, APTR,
AINTER, AINTER,
AARRAY,
ASTRUCT,
BADWIDTH = -1000000000 BADWIDTH = -1000000000
}; };

View File

@ -301,6 +301,12 @@ algtype(Type *t)
if(isptr[t->etype]) if(isptr[t->etype])
a = APTR; // pointer a = APTR; // pointer
else else
if(t->etype == TARRAY)
a = AARRAY;
else
if(t->etype == TSTRUCT)
a = ASTRUCT;
else
if(isinter(t)) if(isinter(t))
a = AINTER; // interface a = AINTER; // interface
// else // else

View File

@ -86,14 +86,17 @@ sys·newchan(uint32 elemsize, uint32 elemalg, uint32 hint,
Hchan *c; Hchan *c;
int32 i; int32 i;
if(elemalg >= nelem(algarray)) { switch(elemalg){
prints("0<="); case ASIMP:
sys·printint(elemalg); case ASTRING:
prints("<"); case APTR:
sys·printint(nelem(algarray)); case AINTER:
prints("\n"); case AARRAY:
case ASTRUCT:
throw("sys·newchan: elem algorithm out of range"); break;
default:
printf("chan(alg=%d)\n", elemalg);
throw("sys·newchan: unsupported channel element type");
} }
c = mal(sizeof(*c)); c = mal(sizeof(*c));

View File

@ -663,19 +663,30 @@ sys·newmap(uint32 keysize, uint32 valsize,
{ {
Hmap *h; Hmap *h;
if(keyalg >= 4 || switch(keyalg) {
valalg >= 4) { case ASIMP:
prints("0<="); case ASTRING:
sys·printint(keyalg); case APTR:
prints("<"); case AINTER:
sys·printint(nelem(algarray)); case AARRAY:
prints("\n0<="); case ASTRUCT:
sys·printint(valalg); break;
prints("<"); default:
sys·printint(nelem(algarray)); printf("map(keyalg=%d)\n", keyalg);
prints("\n"); throw("sys·newmap: unsupported map key type");
}
throw("sys·newmap: key/val algorithm out of range"); switch(valalg) {
case ASIMP:
case ASTRING:
case APTR:
case AINTER:
case AARRAY:
case ASTRUCT:
break;
default:
printf("map(valalg=%d)\n", valalg);
throw("sys·newmap: unsupported map value type");
} }
h = mal(sizeof(*h)); h = mal(sizeof(*h));

View File

@ -6,18 +6,13 @@
static int32 debug = 0; static int32 debug = 0;
enum
{
ASIMP = 0,
ASTRING,
APTR,
AINTER,
};
typedef struct Sigt Sigt; typedef struct Sigt Sigt;
typedef struct Sigi Sigi; typedef struct Sigi Sigi;
typedef struct Map Map; typedef struct Map Map;
/*
* the layout of Sigt and Sigi are known to the compiler
*/
struct Sigt struct Sigt
{ {
byte* name; byte* name;

View File

@ -580,7 +580,7 @@ memcopy(uint32 s, void *a, void *b)
} }
static uint64 static uint64
stringhash(uint32 s, string *a) strhash(uint32 s, string *a)
{ {
USED(s); USED(s);
if(*a == nil) if(*a == nil)
@ -589,21 +589,21 @@ stringhash(uint32 s, string *a)
} }
static uint32 static uint32
stringequal(uint32 s, string *a, string *b) strequal(uint32 s, string *a, string *b)
{ {
USED(s); USED(s);
return cmpstring(*a, *b) == 0; return cmpstring(*a, *b) == 0;
} }
static void static void
stringprint(uint32 s, string *a) strprint(uint32 s, string *a)
{ {
USED(s); USED(s);
sys·printstring(*a); sys·printstring(*a);
} }
static void static void
stringcopy(uint32 s, string *a, string *b) strcopy(uint32 s, string *a, string *b)
{ {
USED(s); USED(s);
if(b == nil) { if(b == nil) {
@ -614,28 +614,28 @@ stringcopy(uint32 s, string *a, string *b)
} }
static uint64 static uint64
pointerhash(uint32 s, void **a) ptrhash(uint32 s, void **a)
{ {
return memhash(s, *a); return memhash(s, *a);
} }
static uint32 static uint32
pointerequal(uint32 s, void **a, void **b) ptrequal(uint32 s, void **a, void **b)
{ {
USED(s, a, b); USED(s, a, b);
prints("pointerequal\n"); prints("ptrequal\n");
return 0; return 0;
} }
static void static void
pointerprint(uint32 s, void **a) ptrprint(uint32 s, void **a)
{ {
USED(s, a); USED(s, a);
prints("pointerprint\n"); prints("ptrprint\n");
} }
static void static void
pointercopy(uint32 s, void **a, void **b) ptrcopy(uint32 s, void **a, void **b)
{ {
USED(s); USED(s);
if(b == nil) { if(b == nil) {
@ -646,12 +646,13 @@ pointercopy(uint32 s, void **a, void **b)
} }
Alg Alg
algarray[4] = algarray[] =
{ {
{ memhash, memequal, memprint, memcopy }, // 0 [ASIMP] { memhash, memequal, memprint, memcopy },
{ stringhash, stringequal, stringprint, stringcopy }, // 1 [ASTRING] { strhash, strequal, strprint, strcopy },
// { pointerhash, pointerequal, pointerprint, pointercopy }, // 2 [APTR] { memhash, memequal, memprint, memcopy }, // TODO: ptr routines
{ memhash, memequal, memprint, memcopy }, // 2 - treat pointers as ints [AINTER] { memhash, memequal, memprint, memcopy }, // TODO: interface routines
{ memhash, memequal, memprint, memcopy }, // 3 - treat interfaces as memory [ASTRUCT] { memhash, memequal, memprint, memcopy }, // TODO: what goes here?
[AARRAY] { memhash, memequal, memprint, memcopy }, // TODO: what goes here?
}; };

View File

@ -214,10 +214,23 @@ struct Func
#define nelem(x) (sizeof(x)/sizeof((x)[0])) #define nelem(x) (sizeof(x)/sizeof((x)[0]))
#define nil ((void*)0) #define nil ((void*)0)
/*
* known to compiler
*/
enum
{
ASIMP = 0,
ASTRING,
APTR,
AINTER,
AARRAY,
ASTRUCT,
};
/* /*
* external data * external data
*/ */
extern Alg algarray[4]; extern Alg algarray[];
extern string emptystring; extern string emptystring;
G* allg; G* allg;
int32 goidgen; int32 goidgen;

110
test/bigalg.go Normal file
View File

@ -0,0 +1,110 @@
// $G $D/$F.go && $L $F.$A && ./$A.out
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package main
import (
"os";
"fmt";
)
type T struct {
a float64;
b int64;
c string;
d byte;
}
var a = []int{ 1, 2, 3 }
var NIL []int;
func arraycmptest() {
a1 := a;
if NIL != nil {
println("fail1:", NIL, "!= nil");
}
if nil != NIL {
println("fail2: nil !=", NIL);
}
if a == nil || nil == a {
println("fail3:", a, "== nil");
}
if a == NIL || NIL == a {
println("fail4:", a, "==", NIL);
}
if a != a {
println("fail5:", a, "!=", a);
}
if a1 != a {
println("fail6:", a1, "!=", a);
}
}
var t = T{1.5, 123, "hello", 255}
var mt = new(map[int]T)
var ma = new(map[int][]int)
func maptest() {
mt[0] = t;
t1 := mt[0];
if t1.a != t.a || t1.b != t.b || t1.c != t.c || t1.d != t.d {
println("fail: map val struct", t1.a, t1.b, t1.c, t1.d);
}
ma[1] = a;
a1 := ma[1];
if a1 != a {
println("fail: map val array", a, a1);
}
}
var mt1 = new(map[T]int)
var ma1 = new(map[[]int] int)
func maptest2() {
mt1[t] = 123;
t1 := t;
val, ok := mt1[t1];
if val != 123 || !ok {
println("fail: map key struct", val, ok);
}
ma1[a] = 345;
a1 := a;
val, ok = ma1[a1];
if val != 345 || !ok {
panic("map key array", val, ok);
}
}
var ct = new(chan T)
var ca = new(chan []int)
func send() {
ct <- t;
ca <- a;
}
func chantest() {
go send();
t1 := <-ct;
if t1.a != t.a || t1.b != t.b || t1.c != t.c || t1.d != t.d {
println("fail: chan struct", t1.a, t1.b, t1.c, t1.d);
}
a1 := <-ca;
if a1 != a {
println("fail: chan array", a, a1);
}
}
func main() {
arraycmptest();
maptest();
maptest2();
chantest();
}