#include <algorithm>
#include <cassert>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <functional>
#include <iterator>
#include <list>
#include <map>
#include <numeric>
#include <queue>
#include <set>
#include <stack>
#include <string>
#include <vector>
#pragma warning(disable : 4996)
typedef long long ll;
typedef unsigned long long ull;
#define MIN(a, b) ((a) > (b) ? (b) : (a))
#define MAX(a, b) ((a) < (b) ? (b) : (a))
#define LINF 9223300000000000000
#define LINF2 1223300000000000000
#define INF 2140000000
const long long MOD = 1000000007;
using namespace std;
void solve() {
  int n, W;
  scanf("%d%d", &n, &W);
  vector<int> vv[4];
  vector<ll> svv[4];
  int w0;
  int i;
  for (i = 0; i < n; i++) {
    int w, v;
    scanf("%d%d", &w, &v);
    if (i == 0)
      w0 = w;
    int tmp = w - w0;
    vv[tmp].push_back(v);
  }
  int k;
  for (k = 0; k < 4; k++) {
    sort(vv[k].rbegin(), vv[k].rend());
    svv[k].resize(vv[k].size() + 1);
    for (i = 0; i < (int)vv[k].size(); i++) {
      svv[k][i + 1] = svv[k][i] + vv[k][i];
    }
  }
  ll ans = 0;
  int j;
  for (i = 0; i <= (int)vv[0].size(); i++) {
    for (j = 0; j <= (int)vv[1].size(); j++) {
      for (k = 0; k <= (int)vv[2].size(); k++) {
        ll sum = (ll)i * (w0) + (ll)j * (w0 + 1) + (ll)k * (w0 + 2);
        ll sumval = svv[0][i] + svv[1][j] + svv[2][k];
        if (sum <= W) {
          int k0 = (W - sum) / (w0 + 3);
          k0 = MIN(k0, (int)vv[3].size());
          sumval += svv[3][k0];
          ans = MAX(ans, sumval);
        }
      }
    }
  }
  printf("%lld\n", ans);
  return;
}
int main(int argc, char *argv[]) {
#if 1
  solve();
#else
  int T;
  scanf("%d", &T);
  int t;
  for (t = 0; t < T; t++) {
    solve();
  }
#endif
  return 0;
}
