Skip to content
Snippets Groups Projects
  1. Oct 06, 2017
  2. Oct 05, 2017
  3. Oct 04, 2017
  4. Oct 03, 2017
    • Vladislav Shpilevoy's avatar
      vinyl: introduce a function to validate vy_index formats · 69da8060
      Vladislav Shpilevoy authored
      vy_index_validate_formats() check consistency of vy_index formats.
      For example, all formats must have the same value of index_field_count
      and mem_format must be equal to disk_format for the primary index.
      
      Update disk_format for the primary key in the case when ALTER adds
      a new space:format() field.
      69da8060
  5. Oct 02, 2017
    • Vladislav Shpilevoy's avatar
      Refactor field_def · dcc5963a
      Vladislav Shpilevoy authored
      Extract struct tuple_field from field_def. Store field_def in space_def
      and tuple_field in tuple_format.
      It allows to do not store field names in tuple_format and do not
      store binary offsets and internal flags in field definition
      (specified by an user).
      dcc5963a
    • Vladislav Shpilevoy's avatar
      vinyl: reference iterator keys · be8ee29a
      Vladislav Shpilevoy authored
      Make all iterators keys not const and reference them. Actually
      they were not const before, because they had been casted to
      (struct tuple *) explicitly by vy_tx_track().
      
      Also it is needed for #1557, where iterators must be able to store
      key, not stored in a read iterator.
      be8ee29a
    • Vladislav Shpilevoy's avatar
      tuple: optimize mp_compare_integer · 9e273a99
      Vladislav Shpilevoy authored
      Mp_compare_integer can be called from mp_compare_number. Both of
      them calls mp_typeof() for their arguments. But mp_typeof() makes
      lookup in a big table in msgpuck.h. Lets pass already decoded types
      from mp_compare_number to mp_compare_integer.
      
      Introduce similar functions with hint for other comparators.
      9e273a99
  6. Sep 30, 2017
  7. Sep 28, 2017
    • Vladislav Shpilevoy's avatar
      vinyl: do not check types of not indexed surrogate fields · 223c6265
      Vladislav Shpilevoy authored
      Vinyl surrogate tuple can be constructed using a key and filling
      not indexed fields with NULL values. But not indexed field can be
      formatted using space:format. Such fields can not store NULLs and
      tuple_init_field_map raises an error on it.
      
      But vinyl actually does not use not indexed fields in surrogate
      tuples so we can store NULLs in surrogate not indexed formatted
      fields.
      
      Besides, we can not use field->type != FIELD_TYPE_ANY as a flag
      that a field is not indexed and is not formatted. Now ANY field
      type is available for users. To check if a field can be nullified
      in a surrogate tuple we must use key_def.
      
      The same problem is actual for construction a surrogate tuple from
      a normal tuple - we can not anymore use type != FIELD_TYPE_ANY to
      define a field is not indexed. Moreover in such a case we also can
      skip fields not used by an index and placed after the last field
      used by an index.
      For this optimization introduce tuple_format.index_field_count and
      field_def.is_key_part.
      
      Closes #2784
      223c6265
  8. Sep 26, 2017
    • Vladimir Davydov's avatar
      box: implement auto increment · d121e7c4
      Vladimir Davydov authored
      The value returned by space.auto_increment() is not guaranteed to be
      monotonically growing, because this method uses max+1 for the next
      generated value, which can be less than a value generated before in case
      the max key is deleted. Besides, calling max() may be expensive, e.g. it
      requires a disk access in case of Vinyl. So this patch implements auto
      increment based on persistent sequences.
      
      To be used for auto increment, a sequence needs to be attached to a
      space. To attach an existing sequence to a space, pass 'sequence=ID' in
      index options when creating or modifying the primary key. Here ID is the
      name or the numeric identifier of the sequence to attach. Note,
      sequences can be attached only to spaces with integer primary key.
      After a sequence is attached, it will be used for auto increment when
      the user passes 'nil' for the primary key value. If the user passes a
      value different from 'nil', the value will be used to update the
      attached sequence.
      
      To detach a sequence from the space it was attached to, either drop the
      primary key of the space or alter the primary key with 'sequence=false'.
      
      It is not necessary to pre-create a sequence to use it for auto
      increment - a sequence can be automatically generated. To generate a
      sequence automatically, pass 'sequence=true' in index options.
      Automatically generated sequences are named '_auto_ID', where ID is the
      space id, and removed when the space is dropped or the sequence is
      detached.
      
      A sequence cannot be dropped as long as it is attached to any spaces.
      
      Closes #389
      d121e7c4
    • Konstantin Osipov's avatar
      sequence: speed up allocation of sequence data tuple during checkpoint · c0a30d17
      Konstantin Osipov authored
      No need to allocate a tuple buffer for every sequence value.
      c0a30d17
    • Vladimir Davydov's avatar
      box: write up-to-date sequence values to snapshot · 775c02f7
      Vladimir Davydov authored
      Currently, _sequence_data is written to the snapshot the same way as any
      other memtx space - the snapshot iterator simply walks over the space
      tuples and dumps them one-by-one. This is OK now, because _sequence_data
      is always up-to-date. However, after the next patch, its contents may
      become stale - due to technical reasons we won't update _sequence_data
      if a sequence is used for auto increment in a space. To make sure we
      still snapshot sequence values properly, let's introduce a special
      iterator for this space that walks over the sequence cache instead of
      the space contents.
      
      Needed for #389
      775c02f7
    • Vladimir Davydov's avatar
      index: use snapshot_iterator for writing snapshot · 7675c28b
      Vladimir Davydov authored
      Currently, Index::createSnapshotIterator() returns a pointer to struct
      iterator, so the iterator implementation has to deal with struct tuple,
      which might not always be an option. Since the iterator is only used for
      writing a snapshot, struct tuple isn't really necessary, raw msgpack
      would be enough. So let's introduce snapshot_iterator which returns raw
      msgpack instead of struct tuple. It is required by the next patch to
      make a snapshot of _sequence_data space.
      
      Needed for #389
      7675c28b
    • Vladimir Davydov's avatar
      test: vinyl: check select consistency · 738c4a18
      Vladimir Davydov authored
      738c4a18
    • Vladimir Davydov's avatar
      vinyl: track DELETEs skipped by read iterator in conflict manager · bcb1bbec
      Vladimir Davydov authored
      Currently, read iterator adds only final statements (i.e. statements
      returned to the user) to the conflict manager while intermediate DELETEs
      are silently ignored. This can result to the following select()
      inconsistency.
      
      Suppose there are REPLACE{10}, DELETE{20} and REPLACE{30} (it does not
      matter which source owns which statement in this example) and select({})
      is called. The read iterator returns REPLACE{10} and is called again. It
      scans the sources, finds DELETE{20} and silently skips to the next
      statement. If its fiber yields now, and REPLACE{15} is inserted, the
      read iterator will ignore it, because its curr_stmt is {20} and so it
      will return {30}, i.e. the select() will return [{10}, {30}]. The same
      select() called from the same transaction afterwards will, however,
      return [{10}, {20}, {30}], which is inconsistent with the previous
      result. This happens, because interval [{10}, {20}] was not added to the
      conflict manager before the yield and so the transaction was not sent to
      read view.
      
      To avoid this, we must track not only final statements, but also
      intermediate DELETEs.
      
      There's a nuance to this patch regarding REQ case that needs to be noted
      explicitly. Since source iterators can't handle REQ, in contrast to EQ,
      we change vy_read_iterator->iterator_type from REQ to LE and set the
      need_check_eq flag when the iterator is opened. The need_check_eq flag
      makes read iterator check if the resulting statement equals the search
      key before returning it to the user. This check must be performed before
      vy_tx_track() is called, otherwise we might add a larger interval to the
      conflict manager than necessary, resulting in unnecessary conflicts
      (vinyl/tx_gap_lock test will surely catch that). So this patch moves
      call to vy_tx_track() along with the need_check_eq check immediately
      after vy_read_iterator_next_key() irrespective of the statement type.
      The problem is vy_cache_add() still has to be called on the final
      statement:
      
       * WAS:
         vy_cache_add(result)
         need_check_eq check; may set result to NULL
         vy_tx_track(result)
      
       * NOW:
         need_check_eq check; may set result to NULL
         vy_tx_track(result)
         vy_cache_add(result)
      
      Beside the result, vy_cache_add() is passed the iterator type so it
      should handle NULL properly and it sure does in case of EQ, but not in
      case of REQ, because REQ was changed to LE. That is, when vy_cache_add()
      is passed NULL (search end marker) in case of REQ, it thinks it saw the
      leftmost statement across all keys, not the leftmost statement equal to
      the search key.
      
      I decided that the best way to overcome this is to make read iterator
      handle REQ separately, just like EQ, and only replace it with LE right
      before passing it down to a source iterator. This gives vy_cache_add()
      enough info to handle the REQ case properly and makes the code look more
      symmetrical IMO.
      bcb1bbec
    • Vladimir Davydov's avatar
      vinyl: fix read source restore after statement removal · 330d53f0
      Vladimir Davydov authored
      Suppose, a source iterator was positioned at the minimal statement
      across all sources before yielding to read a disk source. Then it has
      front_id equal to the front_id of the read iterator, which indicates
      that the source iterator needs to be advanced on the next iteration.
      However, if the statement was removed during the yield, the iterator
      gets repositioned on the next statement by the restoration procedure
      (see vy_read_iterator_next_key()). In this case it isn't positioned on
      the minimal statement any more and hence must not be advanced on the
      next iteration, but its front_id remains the same, which means it will
      be advanced, possibly skipping a statement.
      
      The situation described above can happen only in case of the tuple
      cache, because other sources are append-only. Obviously, to avoid this
      kind of trouble, we must reset the front_id if a source iterator is
      restored to a statement that is greater than the minimal statement.
      330d53f0
    • Vladimir Davydov's avatar
      vinyl: fix double free in cache iterator restore · a2a82803
      Vladimir Davydov authored
      We drop a reference to itr->curr_stmt before restoring the iterator
      position so we must not drop it while iterating over the cache tree.
      This mistake results in funky crashes in various places, e.g. in
      vy_stmt_compare(). Shame on me: I missed that during self review.
      
      Fixes: 073c8d07 ("vinyl: rework cache iterator restore")
      a2a82803
    • Vladimir Davydov's avatar
      vinyl: fold vy_merge_iterator in vy_read_iterator · 4fed8854
      Vladimir Davydov authored
      Initially merge sort implementation was supposed to be shared between
      read and write iterators hence we got vy_merge_iterator. However, over
      time such a design decision proved to be inadequate, because it turned
      out that the two iterators can barely share a few lines - so different
      are the tasks they solve. The write iterator has already been rewritten
      without the use of the vy_merge_iterator so we can now fold the latter
      in vy_read_iterator. While we are at it, spruce up some comments a bit.
      4fed8854
    • Vladimir Davydov's avatar
      vinyl: rename vy_read_iterator->curr_stmt to last_stmt · 6fc49730
      Vladimir Davydov authored
      So as not to confuse it with vy_merge_iterator->curr_stmt.
      6fc49730
Loading