| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534 | /* <!-- copyright *//* * aria2 - The high speed download utility * * Copyright (C) 2006 Tatsuhiro Tsujikawa * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA * * In addition, as a special exception, the copyright holders give * permission to link the code of portions of this program with the * OpenSSL library under certain conditions as described in each * individual source file, and distribute linked combinations * including the two. * You must obey the GNU General Public License in all respects * for all of the code used other than OpenSSL.  If you modify * file(s) with this exception, you may extend this exception to your * version of the file(s), but you are not obligated to do so.  If you * do not wish to do so, delete this exception statement from your * version.  If you delete this exception statement from all source * files in the program, then also delete it here. *//* copyright --> */#include "BitfieldMan.h"#include "Util.h"#include <string.h>BitfieldMan::BitfieldMan(int blockLength, long long int totalLength)  :blockLength(blockLength), totalLength(totalLength), filterBitfield(0),   filterEnabled(false) {  if(blockLength > 0 && totalLength > 0) {    blocks = totalLength/blockLength+(totalLength%blockLength ? 1 : 0);    bitfieldLength = blocks/8+(blocks%8 ? 1 : 0);    bitfield = new unsigned char[bitfieldLength];    useBitfield = new unsigned char[bitfieldLength];    memset(bitfield, 0, bitfieldLength);    memset(useBitfield, 0, bitfieldLength);  }}BitfieldMan::BitfieldMan(const BitfieldMan& bitfieldMan) {  blockLength = bitfieldMan.blockLength;  totalLength = bitfieldMan.totalLength;  blocks = bitfieldMan.blocks;  bitfieldLength = bitfieldMan.bitfieldLength;  bitfield = new unsigned char[bitfieldLength];  useBitfield = new unsigned char[bitfieldLength];  memcpy(bitfield, bitfieldMan.bitfield, bitfieldLength);  memcpy(useBitfield, bitfieldMan.useBitfield, bitfieldLength);  filterEnabled = bitfieldMan.filterEnabled;  if(bitfieldMan.filterBitfield) {    filterBitfield = new unsigned char[bitfieldLength];    memcpy(filterBitfield, bitfieldMan.filterBitfield, bitfieldLength);  } else {    filterBitfield = 0;  }}BitfieldMan::~BitfieldMan() {  delete [] bitfield;  delete [] useBitfield;  if(filterBitfield) {    delete [] filterBitfield;  }}int BitfieldMan::countSetBit(const unsigned char* bitfield, int len) const {  int count = 0;  int size = sizeof(unsigned int);  for(int i = 0; i < len/size; i++) {    count += Util::countBit(*(unsigned int*)&bitfield[i*size]);  }  for(int i = len-len%size; i < len; i++) {    count += Util::countBit((unsigned int)bitfield[i]);  }  return count;}int BitfieldMan::getNthBitIndex(const unsigned char bitfield, int nth) const {  int index = -1;  for(int bs = 7; bs >= 0; bs--) {    unsigned char mask = 1 << bs;    if(bitfield & mask) {      nth--;      if(nth == 0) {	index = 7-bs;	break;      }    }  }  return index;}intBitfieldMan::getMissingIndexRandomly(const unsigned char* bitfield,				     int bitfieldLength) const{  int byte = (int)(((double)bitfieldLength)*random()/(RAND_MAX+1.0));  unsigned char lastMask = 0;  int lastByteLength = totalLength%(blockLength*8);  int lastBlockCount = DIV_FLOOR(lastByteLength, blockLength);  for(int i = 0; i < lastBlockCount; i++) {    lastMask >>= 1;    lastMask |= 0x80;  }  for(int i = 0; i < bitfieldLength; i++) {    unsigned char mask;    if(byte == bitfieldLength-1) {      mask = lastMask;    } else {      mask = 0xff;    }    if(bitfield[byte]&mask) {      int index = byte*8+getNthBitIndex(bitfield[byte], 1);      return index;    }    byte++;    if(byte == bitfieldLength) {      byte = 0;    }  }  return -1;}bool BitfieldMan::hasMissingPiece(const unsigned char* peerBitfield, int length) const {  if(bitfieldLength != length) {    return false;  }  bool retval = false;  for(int i = 0; i < bitfieldLength; i++) {    unsigned char temp = peerBitfield[i] & ~bitfield[i];    if(filterEnabled) {      temp &= filterBitfield[i];    }    if(temp&0xff) {      retval = true;      break;    }  }  return retval;}int BitfieldMan::getMissingIndex(const unsigned char* peerBitfield, int length) const {  if(bitfieldLength != length) {    return -1;  }  unsigned char* tempBitfield = new unsigned char[bitfieldLength];  for(int i = 0; i < bitfieldLength; i++) {    tempBitfield[i] = peerBitfield[i] & ~bitfield[i];    if(filterEnabled) {      tempBitfield[i] &= filterBitfield[i];    }  }  int index = getMissingIndexRandomly(tempBitfield, bitfieldLength);  delete [] tempBitfield;  return index;}int BitfieldMan::getMissingUnusedIndex(const unsigned char* peerBitfield, int length) const {  if(bitfieldLength != length) {    return -1;  }  unsigned char* tempBitfield = new unsigned char[bitfieldLength];  for(int i = 0; i < bitfieldLength; i++) {    tempBitfield[i] = peerBitfield[i] & ~bitfield[i] & ~useBitfield[i];    if(filterEnabled) {      tempBitfield[i] &= filterBitfield[i];    }  }  int index = getMissingIndexRandomly(tempBitfield, bitfieldLength);  delete [] tempBitfield;  return index;}int BitfieldMan::getFirstMissingUnusedIndex(const unsigned char* peerBitfield, int length) const {  if(bitfieldLength != length) {    return -1;  }  for(int i = 0; i < bitfieldLength; i++) {    unsigned char bit = peerBitfield[i] & ~bitfield[i] & ~useBitfield[i];    if(filterEnabled) {      bit &= filterBitfield[i];    }    for(int bs = 7; bs >= 0 && i*8+7-bs < blocks; bs--) {      unsigned char mask = 1 << bs;      if(bit & mask) {	return i*8+7-bs;      }    }  }  return -1;}int BitfieldMan::getFirstMissingUnusedIndex() const {  for(int i = 0; i < bitfieldLength; i++) {    unsigned char bit = ~bitfield[i] & ~useBitfield[i];    if(filterEnabled) {      bit &= filterBitfield[i];    }    for(int bs = 7; bs >= 0 && i*8+7-bs < blocks; bs--) {      unsigned char mask = 1 << bs;      if(bit & mask) {	return i*8+7-bs;      }    }  }  return -1;}int BitfieldMan::getMissingIndex() const {  unsigned char* tempBitfield = new unsigned char[bitfieldLength];  for(int i = 0; i < bitfieldLength; i++) {    tempBitfield[i] = ~bitfield[i];    if(filterEnabled) {      tempBitfield[i] &= filterBitfield[i];    }  }  int index = getMissingIndexRandomly(tempBitfield, bitfieldLength);  delete [] tempBitfield;  return index;}int BitfieldMan::getMissingUnusedIndex() const {  unsigned char* tempBitfield = new unsigned char[bitfieldLength];  memset(tempBitfield, 0xff, bitfieldLength);  int index = getMissingUnusedIndex(tempBitfield, bitfieldLength);  delete [] tempBitfield;  return index;}// [startIndex, endIndex)class Range {public:  int startIndex;  int endIndex;  Range(int startIndex = 0, int endIndex = 0):startIndex(startIndex),					      endIndex(endIndex) {}    int getSize() const {    return endIndex-startIndex;  }  int getMidIndex() const {    return (endIndex-startIndex)/2+startIndex;  }  bool operator<(const Range& range) const {    return getSize() < range.getSize();  }};int BitfieldMan::getStartIndex(int index) const {  while(index < blocks &&	(isUseBitSet(index) || isBitSet(index))) {    index++;  }  if(blocks <= index) {    return -1;  } else {    return index;  }}int BitfieldMan::getEndIndex(int index) const {  while(index < blocks &&	(!isUseBitSet(index) && !isBitSet(index))) {    index++;  }  return index;}int BitfieldMan::getSparseMissingUnusedIndex() const {  Range maxRange;  int index = 0;  int blocks = countBlock();  Range currentRange;  while(index < blocks) {    currentRange.startIndex = getStartIndex(index);    if(currentRange.startIndex == -1) {      break;    }    currentRange.endIndex = getEndIndex(currentRange.startIndex);    if(maxRange < currentRange) {      maxRange = currentRange;    }    index = currentRange.endIndex;  }  if(maxRange.getSize()) {    if(maxRange.startIndex == 0) {      return 0;    } else {      return maxRange.getMidIndex();    }  } else {    return -1;  }}BlockIndexes BitfieldMan::getAllMissingIndexes() const {  BlockIndexes missingIndexes;  for(int i = 0; i < bitfieldLength; i++) {    unsigned char bit = ~bitfield[i];    if(filterEnabled) {      bit &= filterBitfield[i];    }    for(int bs = 7; bs >= 0 && i*8+7-bs < blocks; bs--) {      unsigned char mask = 1 << bs;      if(bit & mask) {	missingIndexes.push_back(i*8+7-bs);      }    }  }  return missingIndexes;}BlockIndexes BitfieldMan::getAllMissingIndexes(const unsigned char* peerBitfield, int peerBitfieldLength) const {  BlockIndexes missingIndexes;  if(bitfieldLength != peerBitfieldLength) {    return missingIndexes;  }  for(int i = 0; i < bitfieldLength; i++) {    unsigned char bit = peerBitfield[i] & ~bitfield[i];    if(filterEnabled) {      bit &= filterBitfield[i];    }    for(int bs = 7; bs >= 0 && i*8+7-bs < blocks; bs--) {      unsigned char mask = 1 << bs;      if(bit & mask) {	missingIndexes.push_back(i*8+7-bs);      }    }  }  return missingIndexes;}int BitfieldMan::countMissingBlock() const {  if(filterEnabled) {    unsigned char* temp = new unsigned char[bitfieldLength];    for(int i = 0; i < bitfieldLength; i++) {      temp[i] = bitfield[i]&filterBitfield[i];    }    int count =  countSetBit(filterBitfield, bitfieldLength)-      countSetBit(temp, bitfieldLength);    delete [] temp;    return count;  } else {    return blocks-countSetBit(bitfield, bitfieldLength);  }}int BitfieldMan::countBlock() const {  if(filterEnabled) {    return countSetBit(filterBitfield, bitfieldLength);  } else {    return blocks;  }}bool BitfieldMan::setBitInternal(unsigned char* bitfield, int index, bool on) {  if(blocks <= index) { return false; }  unsigned char mask = 128 >> index%8;  if(on) {    bitfield[index/8] |= mask;  } else {    bitfield[index/8] &= ~mask;  }  return true;}bool BitfieldMan::setUseBit(int index) {  return setBitInternal(useBitfield, index, true);}bool BitfieldMan::unsetUseBit(int index) {  return setBitInternal(useBitfield, index, false);}bool BitfieldMan::setBit(int index) {  return setBitInternal(bitfield, index, true);}bool BitfieldMan::unsetBit(int index) {  return setBitInternal(bitfield, index, false);}bool BitfieldMan::isAllBitSet() const {  if(filterEnabled) {    for(int i = 0; i < bitfieldLength; i++) {      if((bitfield[i]&filterBitfield[i]) != filterBitfield[i]) {	return false;      }    }    return true;  } else {    for(int i = 0; i < bitfieldLength-1; i++) {      if(bitfield[i] != 0xff) {	return false;      }    }    unsigned char b = ~((128 >> (blocks-1)%8)-1);    if(bitfield[bitfieldLength-1] != b) {      return false;    }    return true;  }}bool BitfieldMan::isBitSetInternal(const unsigned char* bitfield, int index) const {  if(index < 0 || blocks <= index) { return false; }  unsigned char mask = 128 >> index%8;  return (bitfield[index/8] & mask) != 0;}bool BitfieldMan::isBitSet(int index) const {  return isBitSetInternal(bitfield, index);}bool BitfieldMan::isUseBitSet(int index) const {  return isBitSetInternal(useBitfield, index);}void BitfieldMan::setBitfield(const unsigned char* bitfield, int bitfieldLength) {  if(this->bitfieldLength != bitfieldLength) {    return;  }  memcpy(this->bitfield, bitfield, this->bitfieldLength);  memset(this->useBitfield, 0, this->bitfieldLength);}void BitfieldMan::clearAllBit() {  memset(this->bitfield, 0, this->bitfieldLength);}void BitfieldMan::setAllBit() {  for(int i = 0; i < blocks; i++) {    setBit(i);  }}void BitfieldMan::clearAllUseBit() {  memset(this->useBitfield, 0, this->bitfieldLength);}void BitfieldMan::setAllUseBit() {  for(int i = 0; i < blocks; i++) {    setUseBit(i);  }}bool BitfieldMan::setFilterBit(int index) {  return setBitInternal(filterBitfield, index, true);}void BitfieldMan::addFilter(long long int offset, long long int length) {  if(!filterBitfield) {    filterBitfield = new unsigned char[bitfieldLength];    memset(filterBitfield, 0, bitfieldLength);  }  int startBlock = offset/blockLength;  int endBlock = (offset+length-1)/blockLength;  for(int i = startBlock; i <= endBlock && i < blocks; i++) {    setFilterBit(i);  }}void BitfieldMan::enableFilter() {  filterEnabled = true;}void BitfieldMan::disableFilter() {  filterEnabled = false;}void BitfieldMan::clearFilter() {  if(filterBitfield) {    delete [] filterBitfield;    filterBitfield = 0;  }  filterEnabled = false;}bool BitfieldMan::isFilterEnabled() const {  return filterEnabled;}long long int BitfieldMan::getFilteredTotalLength() const {  if(!filterBitfield) {    return 0;  }  int filteredBlocks = countSetBit(filterBitfield, bitfieldLength);  if(filteredBlocks == 0) {    return 0;  }  if(isBitSetInternal(filterBitfield, blocks-1)) {    return ((long long int)filteredBlocks-1)*blockLength+getLastBlockLength();  } else {    return ((long long int)filteredBlocks)*blockLength;  }}long long int BitfieldMan::getCompletedLength() const {  unsigned char* temp = new unsigned char[bitfieldLength];  for(int i = 0; i < bitfieldLength; i++) {    temp[i] = bitfield[i];    if(filterEnabled) {      temp[i] &= filterBitfield[i];    }  }  int completedBlocks = countSetBit(temp, bitfieldLength);  long long int completedLength = 0;  if(completedBlocks == 0) {    completedLength = 0;  } else {    if(isBitSetInternal(temp, blocks-1)) {      completedLength = ((long long int)completedBlocks-1)*blockLength+getLastBlockLength();    } else {      completedLength = ((long long int)completedBlocks)*blockLength;    }  }  delete [] temp;  return completedLength;}
 |