// 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.
#include "runtime.h"
#include "type.h"
#include "malloc.h"
static int32 debug = 0;
// see also unsafe·NewArray
// makeslice(typ *Type, nel int, cap int) (ary []any);
void
runtime·makeslice(SliceType *t, uint32 nel, uint32 cap, Slice ret)
{
uint64 size;
if(cap < nel)
cap = nel;
size = cap*t->elem->size;
ret.len = nel;
ret.cap = cap;
if((t->elem->kind&KindNoPointers))
ret.array = mallocgc(size, RefNoPointers, 1);
else
ret.array = mal(size);
FLUSH(&ret);
if(debug) {
printf("makeslice(%S, %d, %d); ret=",
*t->string, nel, cap);
runtime·printslice(ret);
}
}
static void
throwslice(uint32 lb, uint32 hb, uint32 n)
{
prints("slice[");
runtime·printint(lb);
prints(":");
runtime·printint(hb);
prints("] of [");
runtime·printint(n);
prints("] array\n");
throw("array slice");
}
// sliceslice(old []any, lb int, hb int, width int) (ary []any);
void
runtime·sliceslice(Slice old, uint32 lb, uint32 hb, uint32 width, Slice ret)
{
if(hb > old.cap || lb > hb) {
if(debug) {
prints("runtime·sliceslice: old=");
runtime·printslice(old);
prints("; lb=");
runtime·printint(lb);
prints("; hb=");
runtime·printint(hb);
prints("; width=");
runtime·printint(width);
prints("\n");
prints("oldarray: nel=");
runtime·printint(old.len);
prints("; cap=");
runtime·printint(old.cap);
prints("\n");
}
throwslice(lb, hb, old.cap);
}
// new array is inside old array
ret.len = hb - lb;
ret.cap = old.cap - lb;
ret.array = old.array + lb*width;
FLUSH(&ret);
if(debug) {
prints("runtime·sliceslice: old=");
runtime·printslice(old);
prints("; lb=");
runtime·printint(lb);
prints("; hb=");
runtime·printint(hb);
prints("; width=");
runtime·printint(width);
prints("; ret=");
runtime·printslice(ret);
prints("\n");
}
}
// sliceslice1(old []any, lb int, width int) (ary []any);
void
runtime·sliceslice1(Slice old, uint32 lb, uint32 width, Slice ret)
{
if(lb > old.len) {
if(debug) {
prints("runtime·sliceslice: old=");
runtime·printslice(old);
prints("; lb=");
runtime·printint(lb);
prints("; width=");
runtime·printint(width);
prints("\n");
prints("oldarray: nel=");
runtime·printint(old.len);
prints("; cap=");
runtime·printint(old.cap);
prints("\n");
}
throwslice(lb, old.len, old.cap);
}
// new array is inside old array
ret.len = old.len - lb;
ret.cap = old.cap - lb;
ret.array = old.array + lb*width;
FLUSH(&ret);
if(debug) {
prints("runtime·sliceslice: old=");
runtime·printslice(old);
prints("; lb=");
runtime·printint(lb);
prints("; width=");
runtime·printint(width);
prints("; ret=");
runtime·printslice(ret);
prints("\n");
}
}
// slicearray(old *any, nel int, lb int, hb int, width int) (ary []any);
void
runtime·slicearray(byte* old, uint32 nel, uint32 lb, uint32 hb, uint32 width, Slice ret)
{
if(nel > 0 && old == nil) {
// crash if old == nil.
// could give a better message
// but this is consistent with all the in-line checks
// that the compiler inserts for other uses.
*old = 0;
}
if(hb > nel || lb > hb) {
if(debug) {
prints("runtime·slicearray: old=");
runtime·printpointer(old);
prints("; nel=");
runtime·printint(nel);
prints("; lb=");
runtime·printint(lb);
prints("; hb=");
runtime·printint(hb);
prints("; width=");
runtime·printint(width);
prints("\n");
}
throwslice(lb, hb, nel);
}
// new array is inside old array
ret.len = hb-lb;
ret.cap = nel-lb;
ret.array = old + lb*width;
FLUSH(&ret);
if(debug) {
prints("runtime·slicearray: old=");
runtime·printpointer(old);
prints("; nel=");
runtime·printint(nel);
prints("; lb=");
runtime·printint(lb);
prints("; hb=");
runtime·printint(hb);
prints("; width=");
runtime·printint(width);
prints("; ret=");
runtime·printslice(ret);
prints("\n");
}
}
// slicecopy(to any, fr any, wid uint32) int
void
runtime·slicecopy(Slice to, Slice fm, uintptr width, int32 ret)
{
if(fm.array == nil || fm.len == 0 ||
to.array == nil || to.len == 0 ||
width == 0) {
ret = 0;
goto out;
}
ret = fm.len;
if(to.len < ret)
ret = to.len;
if(ret == 1 && width == 1) { // common case worth about 2x to do here
*to.array = *fm.array; // known to be a byte pointer
} else {
memmove(to.array, fm.array, ret*width);
}
out:
FLUSH(&ret);
if(debug) {
prints("main·copy: to=");
runtime·printslice(to);
prints("; fm=");
runtime·printslice(fm);
prints("; width=");
runtime·printint(width);
prints("; ret=");
runtime·printint(ret);
prints("\n");
}
}
void
runtime·printslice(Slice a)
{
prints("[");
runtime·printint(a.len);
prints("/");
runtime·printint(a.cap);
prints("]");
runtime·printpointer(a.array);
}
|