// Copyright 2015 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.

package types

import (
	"math"

	"github.com/hanchuanchuan/goInception/util/testleak"
	. "github.com/pingcap/check"
)

var _ = Suite(&testOverflowSuite{})

type testOverflowSuite struct {
}

func (s *testOverflowSuite) TestAdd(c *C) {
	defer testleak.AfterTest(c)()
	tblUint64 := []struct {
		lsh      uint64
		rsh      uint64
		ret      uint64
		overflow bool
	}{
		{math.MaxUint64, 1, 0, true},
		{math.MaxUint64, 0, math.MaxUint64, false},
		{1, 1, 2, false},
	}

	for _, t := range tblUint64 {
		ret, err := AddUint64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt64 := []struct {
		lsh      int64
		rsh      int64
		ret      int64
		overflow bool
	}{
		{math.MaxInt64, 1, 0, true},
		{math.MaxInt64, 0, math.MaxInt64, false},
		{0, math.MinInt64, math.MinInt64, false},
		{-1, math.MinInt64, 0, true},
		{math.MaxInt64, math.MinInt64, -1, false},
		{1, 1, 2, false},
		{1, -1, 0, false},
	}

	for _, t := range tblInt64 {
		ret, err := AddInt64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt := []struct {
		lsh      uint64
		rsh      int64
		ret      uint64
		overflow bool
	}{
		{math.MaxUint64, math.MinInt64, math.MaxUint64 + math.MinInt64, false},
		{math.MaxInt64, math.MinInt64, 0, true},
		{0, -1, 0, true},
		{1, -1, 0, false},
		{0, 1, 1, false},
		{1, 1, 2, false},
	}

	for _, t := range tblInt {
		ret, err := AddInteger(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}
}

func (s *testOverflowSuite) TestSub(c *C) {
	defer testleak.AfterTest(c)()
	tblUint64 := []struct {
		lsh      uint64
		rsh      uint64
		ret      uint64
		overflow bool
	}{
		{math.MaxUint64, 1, math.MaxUint64 - 1, false},
		{math.MaxUint64, 0, math.MaxUint64, false},
		{0, math.MaxUint64, 0, true},
		{0, 1, 0, true},
		{1, math.MaxUint64, 0, true},
		{1, 1, 0, false},
	}

	for _, t := range tblUint64 {
		ret, err := SubUint64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt64 := []struct {
		lsh      int64
		rsh      int64
		ret      int64
		overflow bool
	}{
		{math.MinInt64, 0, math.MinInt64, false},
		{math.MinInt64, 1, 0, true},
		{math.MaxInt64, -1, 0, true},
		{0, math.MinInt64, 0, true},
		{-1, math.MinInt64, math.MaxInt64, false},
		{math.MinInt64, math.MaxInt64, 0, true},
		{math.MinInt64, math.MinInt64, 0, false},
		{math.MinInt64, -math.MaxInt64, -1, false},
		{1, 1, 0, false},
	}

	for _, t := range tblInt64 {
		ret, err := SubInt64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt := []struct {
		lsh      uint64
		rsh      int64
		ret      uint64
		overflow bool
	}{
		{0, math.MinInt64, -math.MinInt64, false},
		{0, 1, 0, true},
		{math.MaxUint64, math.MinInt64, 0, true},
		{math.MaxInt64, math.MinInt64, 2*math.MaxInt64 + 1, false},
		{math.MaxUint64, -1, 0, true},
		{0, -1, 1, false},
		{1, 1, 0, false},
	}

	for _, t := range tblInt {
		ret, err := SubUintWithInt(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt2 := []struct {
		lsh      int64
		rsh      uint64
		ret      uint64
		overflow bool
	}{
		{math.MinInt64, 0, 0, true},
		{math.MaxInt64, 0, math.MaxInt64, false},
		{math.MaxInt64, math.MaxUint64, 0, true},
		{math.MaxInt64, -math.MinInt64, 0, true},
		{-1, 0, 0, true},
		{1, 1, 0, false},
	}

	for _, t := range tblInt2 {
		ret, err := SubIntWithUint(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}
}

func (s *testOverflowSuite) TestMul(c *C) {
	defer testleak.AfterTest(c)()
	tblUint64 := []struct {
		lsh      uint64
		rsh      uint64
		ret      uint64
		overflow bool
	}{
		{math.MaxUint64, 1, math.MaxUint64, false},
		{math.MaxUint64, 0, 0, false},
		{math.MaxUint64, 2, 0, true},
		{1, 1, 1, false},
	}

	for _, t := range tblUint64 {
		ret, err := MulUint64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt64 := []struct {
		lsh      int64
		rsh      int64
		ret      int64
		overflow bool
	}{
		{math.MaxInt64, 1, math.MaxInt64, false},
		{math.MinInt64, 1, math.MinInt64, false},
		{math.MaxInt64, -1, -math.MaxInt64, false},
		{math.MinInt64, -1, 0, true},
		{math.MinInt64, 0, 0, false},
		{math.MaxInt64, 0, 0, false},
		{math.MaxInt64, math.MaxInt64, 0, true},
		{math.MaxInt64, math.MinInt64, 0, true},
		{math.MinInt64 / 10, 11, 0, true},
		{1, 1, 1, false},
	}

	for _, t := range tblInt64 {
		ret, err := MulInt64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt := []struct {
		lsh      uint64
		rsh      int64
		ret      uint64
		overflow bool
	}{
		{math.MaxUint64, 0, 0, false},
		{0, -1, 0, false},
		{1, -1, 0, true},
		{math.MaxUint64, -1, 0, true},
		{math.MaxUint64, 10, 0, true},
		{1, 1, 1, false},
	}

	for _, t := range tblInt {
		ret, err := MulInteger(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}
}

func (s *testOverflowSuite) TestDiv(c *C) {
	defer testleak.AfterTest(c)()
	tblInt64 := []struct {
		lsh      int64
		rsh      int64
		ret      int64
		overflow bool
	}{
		{math.MaxInt64, 1, math.MaxInt64, false},
		{math.MinInt64, 1, math.MinInt64, false},
		{math.MinInt64, -1, 0, true},
		{math.MaxInt64, -1, -math.MaxInt64, false},
		{1, -1, -1, false},
		{-1, 1, -1, false},
		{-1, 2, 0, false},
		{math.MinInt64, 2, math.MinInt64 / 2, false},
	}

	for _, t := range tblInt64 {
		ret, err := DivInt64(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt := []struct {
		lsh      uint64
		rsh      int64
		ret      uint64
		overflow bool
	}{
		{0, -1, 0, false},
		{1, -1, 0, true},
		{math.MaxInt64, math.MinInt64, 0, false},
		{math.MaxInt64, -1, 0, true},
	}

	for _, t := range tblInt {
		ret, err := DivUintWithInt(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}

	tblInt2 := []struct {
		lsh      int64
		rsh      uint64
		ret      uint64
		overflow bool
	}{
		{math.MinInt64, math.MaxInt64, 0, true},
		{0, 1, 0, false},
		{-1, math.MaxInt64, 0, false},
	}

	for _, t := range tblInt2 {
		ret, err := DivIntWithUint(t.lsh, t.rsh)
		if t.overflow {
			c.Assert(err, NotNil)
		} else {
			c.Assert(ret, Equals, t.ret)
		}
	}
}