github.com/whtcorpsinc/milevadb-prod@v0.0.0-20211104133533-f57f4be3b597/dbs/memristed/memex/builtin_miscellaneous_vec.go (about)

     1  // Copyright 2020 WHTCORPS INC, Inc.
     2  //
     3  // Licensed under the Apache License, Version 2.0 (the "License");
     4  // you may not use this file except in compliance with the License.
     5  // You may obtain a copy of the License at
     6  //
     7  //     http://www.apache.org/licenses/LICENSE-2.0
     8  //
     9  // Unless required by applicable law or agreed to in writing, software
    10  // distributed under the License is distributed on an "AS IS" BASIS,
    11  // See the License for the specific language governing permissions and
    12  // limitations under the License.
    13  
    14  package memex
    15  
    16  import (
    17  	"bytes"
    18  	"encoding/binary"
    19  	"fmt"
    20  	"math"
    21  	"net"
    22  	"strings"
    23  	"sync/atomic"
    24  	"time"
    25  
    26  	"github.com/google/uuid"
    27  	"github.com/whtcorpsinc/milevadb/stochastikctx/variable"
    28  	"github.com/whtcorpsinc/milevadb/types"
    29  	"github.com/whtcorpsinc/milevadb/soliton/chunk"
    30  )
    31  
    32  func (b *builtinInetNtoaSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
    33  	n := input.NumEvents()
    34  	buf, err := b.bufSlabPredictor.get(types.ETInt, n)
    35  	if err != nil {
    36  		return err
    37  	}
    38  	defer b.bufSlabPredictor.put(buf)
    39  	if err := b.args[0].VecEvalInt(b.ctx, input, buf); err != nil {
    40  		return err
    41  	}
    42  
    43  	result.ReserveString(n)
    44  	i64s := buf.Int64s()
    45  	ip := make(net.IP, net.IPv4len)
    46  	for i := 0; i < n; i++ {
    47  		val := i64s[i]
    48  		if buf.IsNull(i) || val < 0 || uint64(val) > math.MaxUint32 {
    49  			result.AppendNull()
    50  			continue
    51  		}
    52  		binary.BigEndian.PutUint32(ip, uint32(val))
    53  		ipv4 := ip.To4()
    54  		if ipv4 == nil {
    55  			//Not a vaild ipv4 address.
    56  			result.AppendNull()
    57  			continue
    58  		}
    59  		result.AppendString(ipv4.String())
    60  	}
    61  	return nil
    62  }
    63  
    64  func (b *builtinInetNtoaSig) vectorized() bool {
    65  	return true
    66  }
    67  
    68  func (b *builtinIsIPv4Sig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
    69  	n := input.NumEvents()
    70  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
    71  	if err != nil {
    72  		return err
    73  	}
    74  	defer b.bufSlabPredictor.put(buf)
    75  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
    76  		return err
    77  	}
    78  	result.ResizeInt64(n, false)
    79  	i64s := result.Int64s()
    80  	for i := 0; i < n; i++ {
    81  		// Note that even when the i-th input string is null, the output is
    82  		// 0 instead of null, therefore we do not set the null bit mask in
    83  		// result's corresponding event.
    84  		// See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_is-ipv4
    85  		if isIPv4(buf.GetString(i)) {
    86  			i64s[i] = 1
    87  		} else {
    88  			i64s[i] = 0
    89  		}
    90  	}
    91  	return nil
    92  }
    93  
    94  func (b *builtinIsIPv4Sig) vectorized() bool {
    95  	return true
    96  }
    97  func (b *builtinJSONAnyValueSig) vectorized() bool {
    98  	return true
    99  }
   100  
   101  func (b *builtinJSONAnyValueSig) vecEvalJSON(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   102  	return b.args[0].VecEvalJSON(b.ctx, input, result)
   103  }
   104  
   105  func (b *builtinRealAnyValueSig) vectorized() bool {
   106  	return true
   107  }
   108  
   109  func (b *builtinRealAnyValueSig) vecEvalReal(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   110  	return b.args[0].VecEvalReal(b.ctx, input, result)
   111  }
   112  
   113  func (b *builtinStringAnyValueSig) vectorized() bool {
   114  	return true
   115  }
   116  
   117  func (b *builtinStringAnyValueSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   118  	return b.args[0].VecEvalString(b.ctx, input, result)
   119  }
   120  
   121  func (b *builtinIsIPv6Sig) vectorized() bool {
   122  	return true
   123  }
   124  
   125  func (b *builtinIsIPv6Sig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   126  	n := input.NumEvents()
   127  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
   128  	if err != nil {
   129  		return err
   130  	}
   131  	defer b.bufSlabPredictor.put(buf)
   132  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
   133  		return err
   134  	}
   135  	result.ResizeInt64(n, false)
   136  	i64s := result.Int64s()
   137  	for i := 0; i < n; i++ {
   138  		// Note that even when the i-th input string is null, the output is
   139  		// 0 instead of null, therefore we do not set the null bit mask in
   140  		// result's corresponding event.
   141  		// See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_is-ipv6
   142  		if buf.IsNull(i) {
   143  			i64s[i] = 0
   144  		} else {
   145  			ipStr := buf.GetString(i)
   146  			if ip := net.ParseIP(ipStr); ip != nil && !isIPv4(ipStr) {
   147  				i64s[i] = 1
   148  			} else {
   149  				i64s[i] = 0
   150  			}
   151  		}
   152  	}
   153  	return nil
   154  }
   155  
   156  func (b *builtinNameConstStringSig) vectorized() bool {
   157  	return true
   158  }
   159  
   160  func (b *builtinNameConstStringSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   161  	return b.args[1].VecEvalString(b.ctx, input, result)
   162  }
   163  
   164  func (b *builtinDecimalAnyValueSig) vectorized() bool {
   165  	return true
   166  }
   167  
   168  func (b *builtinDecimalAnyValueSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   169  	return b.args[0].VecEvalDecimal(b.ctx, input, result)
   170  }
   171  
   172  func (b *builtinUUIDSig) vectorized() bool {
   173  	return true
   174  }
   175  
   176  func (b *builtinUUIDSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   177  	n := input.NumEvents()
   178  	result.ReserveString(n)
   179  	var id uuid.UUID
   180  	var err error
   181  	for i := 0; i < n; i++ {
   182  		id, err = uuid.NewUUID()
   183  		if err != nil {
   184  			return err
   185  		}
   186  		result.AppendString(id.String())
   187  	}
   188  	return nil
   189  }
   190  
   191  func (b *builtinNameConstDurationSig) vectorized() bool {
   192  	return true
   193  }
   194  
   195  func (b *builtinNameConstDurationSig) vecEvalDuration(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   196  	return b.args[1].VecEvalDuration(b.ctx, input, result)
   197  }
   198  
   199  func (b *builtinLockSig) vectorized() bool {
   200  	return true
   201  }
   202  
   203  // See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_get-dagger
   204  // The dagger function will do nothing.
   205  // Warning: get_lock() function is parsed but ignored.
   206  func (b *builtinLockSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   207  	n := input.NumEvents()
   208  	result.ResizeInt64(n, false)
   209  	i64s := result.Int64s()
   210  	for i := range i64s {
   211  		i64s[i] = 1
   212  	}
   213  	return nil
   214  }
   215  
   216  func (b *builtinDurationAnyValueSig) vectorized() bool {
   217  	return true
   218  }
   219  
   220  func (b *builtinDurationAnyValueSig) vecEvalDuration(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   221  	return b.args[0].VecEvalDuration(b.ctx, input, result)
   222  }
   223  
   224  func (b *builtinIntAnyValueSig) vectorized() bool {
   225  	return true
   226  }
   227  
   228  func (b *builtinIntAnyValueSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   229  	return b.args[0].VecEvalInt(b.ctx, input, result)
   230  }
   231  
   232  func (b *builtinIsIPv4CompatSig) vectorized() bool {
   233  	return true
   234  }
   235  
   236  func (b *builtinIsIPv4CompatSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   237  	n := input.NumEvents()
   238  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
   239  	if err != nil {
   240  		return err
   241  	}
   242  	defer b.bufSlabPredictor.put(buf)
   243  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
   244  		return err
   245  	}
   246  	result.ResizeInt64(n, false)
   247  	i64s := result.Int64s()
   248  	prefixCompat := []byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
   249  	for i := 0; i < n; i++ {
   250  		if buf.IsNull(i) {
   251  			i64s[i] = 0
   252  		} else {
   253  			// Note that the input should be IP address in byte format.
   254  			// For IPv4, it should be byte slice with 4 bytes.
   255  			// For IPv6, it should be byte slice with 16 bytes.
   256  			// See example https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_is-ipv4-compat
   257  			ipAddress := buf.GetBytes(i)
   258  			if len(ipAddress) != net.IPv6len || !bytes.HasPrefix(ipAddress, prefixCompat) {
   259  				//Not an IPv6 address, return false
   260  				i64s[i] = 0
   261  			} else {
   262  				i64s[i] = 1
   263  			}
   264  		}
   265  	}
   266  	return nil
   267  }
   268  
   269  func (b *builtinNameConstIntSig) vectorized() bool {
   270  	return true
   271  }
   272  
   273  func (b *builtinNameConstIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   274  	return b.args[1].VecEvalInt(b.ctx, input, result)
   275  }
   276  
   277  func (b *builtinNameConstTimeSig) vectorized() bool {
   278  	return true
   279  }
   280  
   281  func (b *builtinNameConstTimeSig) vecEvalTime(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   282  	return b.args[1].VecEvalTime(b.ctx, input, result)
   283  }
   284  
   285  func (b *builtinSleepSig) vectorized() bool {
   286  	return true
   287  }
   288  
   289  // vecEvalInt evals a builtinSleepSig in a vectorized manner.
   290  // See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_sleep
   291  func (b *builtinSleepSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   292  	n := input.NumEvents()
   293  	buf, err := b.bufSlabPredictor.get(types.ETReal, n)
   294  	if err != nil {
   295  		return err
   296  	}
   297  	defer b.bufSlabPredictor.put(buf)
   298  
   299  	err = b.args[0].VecEvalReal(b.ctx, input, buf)
   300  	if err != nil {
   301  		return err
   302  	}
   303  
   304  	result.ResizeInt64(n, false)
   305  	i64s := result.Int64s()
   306  
   307  	for i := 0; i < n; i++ {
   308  		isNull := buf.IsNull(i)
   309  		val := buf.GetFloat64(i)
   310  
   311  		sessVars := b.ctx.GetStochastikVars()
   312  		if isNull || val < 0 {
   313  			if sessVars.StrictALLEGROSQLMode {
   314  				return errIncorrectArgs.GenWithStackByArgs("sleep")
   315  			}
   316  			err := errIncorrectArgs.GenWithStackByArgs("sleep")
   317  			sessVars.StmtCtx.AppendWarning(err)
   318  			continue
   319  		}
   320  
   321  		if val > math.MaxFloat64/float64(time.Second.Nanoseconds()) {
   322  			return errIncorrectArgs.GenWithStackByArgs("sleep")
   323  		}
   324  
   325  		if isKilled := doSleep(val, sessVars); isKilled {
   326  			for j := i; j < n; j++ {
   327  				i64s[j] = 1
   328  			}
   329  			return nil
   330  		}
   331  	}
   332  
   333  	return nil
   334  }
   335  
   336  func doSleep(secs float64, sessVars *variable.StochastikVars) (isKilled bool) {
   337  	if secs <= 0.0 {
   338  		return false
   339  	}
   340  	dur := time.Duration(secs * float64(time.Second.Nanoseconds()))
   341  	ticker := time.NewTicker(10 * time.Millisecond)
   342  	defer ticker.Stop()
   343  	timer := time.NewTimer(dur)
   344  	for {
   345  		select {
   346  		case <-ticker.C:
   347  			if atomic.CompareAndSwapUint32(&sessVars.Killed, 1, 0) {
   348  				timer.Stop()
   349  				return true
   350  			}
   351  		case <-timer.C:
   352  			return false
   353  		}
   354  	}
   355  }
   356  
   357  func (b *builtinIsIPv4MappedSig) vectorized() bool {
   358  	return true
   359  }
   360  
   361  func (b *builtinIsIPv4MappedSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   362  	n := input.NumEvents()
   363  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
   364  	if err != nil {
   365  		return err
   366  	}
   367  	defer b.bufSlabPredictor.put(buf)
   368  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
   369  		return err
   370  	}
   371  	result.ResizeInt64(n, false)
   372  	i64s := result.Int64s()
   373  	prefixMapped := []byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff}
   374  	for i := 0; i < n; i++ {
   375  		if buf.IsNull(i) {
   376  			i64s[i] = 0
   377  		} else {
   378  			// Note that the input should be IP address in byte format.
   379  			// For IPv4, it should be byte slice with 4 bytes.
   380  			// For IPv6, it should be byte slice with 16 bytes.
   381  			// See example https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_is-ipv4-mapped
   382  			ipAddress := buf.GetBytes(i)
   383  			if len(ipAddress) != net.IPv6len || !bytes.HasPrefix(ipAddress, prefixMapped) {
   384  				//Not an IPv6 address, return false
   385  				i64s[i] = 0
   386  			} else {
   387  				i64s[i] = 1
   388  			}
   389  		}
   390  	}
   391  	return nil
   392  }
   393  
   394  func (b *builtinNameConstDecimalSig) vectorized() bool {
   395  	return true
   396  }
   397  
   398  func (b *builtinNameConstDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   399  	return b.args[1].VecEvalDecimal(b.ctx, input, result)
   400  }
   401  
   402  func (b *builtinNameConstJSONSig) vectorized() bool {
   403  	return true
   404  }
   405  
   406  func (b *builtinNameConstJSONSig) vecEvalJSON(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   407  	return b.args[1].VecEvalJSON(b.ctx, input, result)
   408  }
   409  
   410  func (b *builtinInet6AtonSig) vectorized() bool {
   411  	return true
   412  }
   413  
   414  // vecEvalString evals a builtinInet6AtonSig.
   415  // See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_inet6-aton
   416  func (b *builtinInet6AtonSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   417  	n := input.NumEvents()
   418  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
   419  	if err != nil {
   420  		return err
   421  	}
   422  	defer b.bufSlabPredictor.put(buf)
   423  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
   424  		return err
   425  	}
   426  
   427  	var (
   428  		resv4 []byte
   429  		resv6 []byte
   430  		res   []byte
   431  	)
   432  	result.ReserveString(n)
   433  	for i := 0; i < n; i++ {
   434  		if buf.IsNull(i) {
   435  			result.AppendNull()
   436  			continue
   437  		}
   438  		val := buf.GetString(i)
   439  		if len(val) == 0 {
   440  			result.AppendNull()
   441  			continue
   442  		}
   443  		ip := net.ParseIP(val)
   444  		if ip == nil {
   445  			result.AppendNull()
   446  			continue
   447  		}
   448  		var isMappedIpv6 bool
   449  		ipTo4 := ip.To4()
   450  		if ipTo4 != nil && strings.Contains(val, ":") {
   451  			//mapped ipv6 address.
   452  			isMappedIpv6 = true
   453  		}
   454  
   455  		if isMappedIpv6 || ipTo4 == nil {
   456  			if resv6 == nil {
   457  				resv6 = make([]byte, net.IPv6len)
   458  			}
   459  			res = resv6
   460  		} else {
   461  			if resv4 == nil {
   462  				resv4 = make([]byte, net.IPv4len)
   463  			}
   464  			res = resv4
   465  		}
   466  
   467  		if isMappedIpv6 {
   468  			copy(res[12:], ipTo4)
   469  			res[11] = 0xff
   470  			res[10] = 0xff
   471  		} else if ipTo4 == nil {
   472  			copy(res, ip.To16())
   473  		} else {
   474  			copy(res, ipTo4)
   475  		}
   476  		result.AppendBytes(res)
   477  	}
   478  	return nil
   479  }
   480  
   481  func (b *builtinTimeAnyValueSig) vectorized() bool {
   482  	return true
   483  }
   484  
   485  func (b *builtinTimeAnyValueSig) vecEvalTime(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   486  	return b.args[0].VecEvalTime(b.ctx, input, result)
   487  }
   488  
   489  func (b *builtinInetAtonSig) vectorized() bool {
   490  	return true
   491  }
   492  
   493  func (b *builtinInetAtonSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   494  	n := input.NumEvents()
   495  	buf, err := b.bufSlabPredictor.get(types.ETString, n)
   496  	if err != nil {
   497  		return err
   498  	}
   499  	defer b.bufSlabPredictor.put(buf)
   500  	if err := b.args[0].VecEvalString(b.ctx, input, buf); err != nil {
   501  		return err
   502  	}
   503  	var (
   504  		byteResult, res uint64
   505  		dotCount        int
   506  	)
   507  	result.ResizeInt64(n, false)
   508  	i64s := result.Int64s()
   509  	result.MergeNulls(buf)
   510  	for i := 0; i < n; i++ {
   511  		if result.IsNull(i) {
   512  			continue
   513  		}
   514  		ipAddr := buf.GetString(i)
   515  		if len(ipAddr) == 0 || ipAddr[len(ipAddr)-1] == '.' {
   516  			// ip address should not end with '.'.
   517  			result.SetNull(i, true)
   518  			continue
   519  		}
   520  		//reset
   521  		byteResult = 0
   522  		res = 0
   523  		dotCount = 0
   524  		for _, c := range ipAddr {
   525  			if c >= '0' && c <= '9' {
   526  				digit := uint64(c - '0')
   527  				byteResult = byteResult*10 + digit
   528  				if byteResult > 255 {
   529  					result.SetNull(i, true)
   530  					break
   531  				}
   532  			} else if c == '.' {
   533  				dotCount++
   534  				if dotCount > 3 {
   535  					result.SetNull(i, true)
   536  					break
   537  				}
   538  				res = (res << 8) + byteResult
   539  				byteResult = 0
   540  			} else {
   541  				result.SetNull(i, true)
   542  				break // illegal char (not number or .)
   543  			}
   544  		}
   545  		// 127 		-> 0.0.0.127
   546  		// 127.255 	-> 127.0.0.255
   547  		// 127.256	-> NULL
   548  		// 127.2.1	-> 127.2.0.1
   549  		if !result.IsNull(i) {
   550  			if dotCount == 1 {
   551  				res <<= 16
   552  			}
   553  			if dotCount == 2 {
   554  				res <<= 8
   555  			}
   556  			i64s[i] = int64((res << 8) + byteResult)
   557  		}
   558  	}
   559  	return nil
   560  }
   561  
   562  func (b *builtinInet6NtoaSig) vectorized() bool {
   563  	return true
   564  }
   565  
   566  func (b *builtinInet6NtoaSig) vecEvalString(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   567  	n := input.NumEvents()
   568  	val, err := b.bufSlabPredictor.get(types.ETString, n)
   569  	if err != nil {
   570  		return err
   571  	}
   572  	defer b.bufSlabPredictor.put(val)
   573  	if err := b.args[0].VecEvalString(b.ctx, input, val); err != nil {
   574  		return err
   575  	}
   576  	result.ReserveString(n)
   577  	for i := 0; i < n; i++ {
   578  		if val.IsNull(i) {
   579  			result.AppendNull()
   580  			continue
   581  		}
   582  		valI := val.GetString(i)
   583  		ip := net.IP(valI).String()
   584  		if len(valI) == net.IPv6len && !strings.Contains(ip, ":") {
   585  			ip = fmt.Sprintf("::ffff:%s", ip)
   586  		}
   587  		if net.ParseIP(ip) == nil {
   588  			result.AppendNull()
   589  			continue
   590  		}
   591  		result.AppendString(ip)
   592  	}
   593  	return nil
   594  }
   595  
   596  func (b *builtinNameConstRealSig) vectorized() bool {
   597  	return true
   598  }
   599  
   600  func (b *builtinNameConstRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   601  	return b.args[1].VecEvalReal(b.ctx, input, result)
   602  }
   603  
   604  func (b *builtinReleaseLockSig) vectorized() bool {
   605  	return true
   606  }
   607  
   608  // See https://dev.allegrosql.com/doc/refman/5.7/en/miscellaneous-functions.html#function_release-dagger
   609  // The release dagger function will do nothing.
   610  // Warning: release_lock() function is parsed but ignored.
   611  func (b *builtinReleaseLockSig) vecEvalInt(input *chunk.Chunk, result *chunk.DeferredCauset) error {
   612  	n := input.NumEvents()
   613  	result.ResizeInt64(n, false)
   614  	i64s := result.Int64s()
   615  	for i := range i64s {
   616  		i64s[i] = 1
   617  	}
   618  	return nil
   619  }