/*
* @Author: hungeazy
* @Date:   2025-10-21 10:00:32
* @Last Modified by:   hungeazy
* @Last Modified time: 2025-10-21 15:55:28
*/
#include <bits/stdc++.h>
#include <ext/pb_ds/assoc_container.hpp> 
#include <ext/pb_ds/tree_policy.hpp> 
// #pragma GCC optimize("O3")  
// #pragma GCC optimize("unroll-loops")  
// #pragma GCC target("avx2,bmi,bmi2,popcnt,lzcnt")  
using namespace std;
using namespace __gnu_pbds; 
bool M1;
#define fast ios_base::sync_with_stdio(false); cin.tie(NULL); cout.tie(NULL);
#define int long long
#define ll long long 
#define ull unsigned long long
#define sz(x) x.size()
#define sqr(x) (1LL * (x) * (x))
#define all(x) x.begin(), x.end()
#define fill(f,x) memset(f,x,sizeof(f))
#define FOR(i,l,r) for(int i=l;i<=r;i++)
#define FOD(i,r,l) for(int i=r;i>=l;i--)
#define debug(x) cout << #x << " = " << x << '\n'
#define ii pair<int,int>
#define iii pair<int,ii>
#define di pair<ii,ii>
#define vi vector<int>
#define vii vector<ii>
#define mii map<int,int>
#define fi first
#define se second
#define pb push_back
#define MOD 1000000007
#define __lcm(a,b) (1ll * ((a) / __gcd((a), (b))) * (b))
#define YES cout << "YES\n"
#define NO cout << "NO\n"
#define MASK(i) (1LL << (i))
#define c_bit(i) __builtin_popcountll(i)
#define BIT(x,i) ((x) & MASK(i))
#define SET_ON(x,i) ((x) | MASK(i))
#define SET_OFF(x,i) ((x) & ~MASK(i))
#define oo 1e18
#define name ""
#define endl '\n'
#define memory() cerr << abs(&M2-&M1)/1024.0/1024 << " MB" << endl
#define time() cerr << endl << "-------------Time:" << 1000.0 * clock() / CLOCKS_PER_SEC << "ms." << endl
template<typename T> bool maximize(T &res, const T &val) { if (res < val){ res = val; return true; }; return false; }
template<typename T> bool minimize(T &res, const T &val) { if (res > val){ res = val; return true; }; return false; }
template <class T> using ordered_set = tree <T, null_type, less_equal <T>, rb_tree_tag,tree_order_statistics_node_update>;
const int N = (int)5e5+10;
int n,W;
 
struct Data {
	int v,w;
	bool operator<(const Data &other) {
		return w > other.w;
	}
} a[N];
 
namespace sub3 {
 
	bool approved() {
		return n <= 1e3 and W <= 1e4;
	}
 
	void solve(void)
	{
		sort(a+1,a+n+1);
		mii dp;
		dp[W] = 0;
		FOR(i,1,n)
			for (auto &x : dp)
				if (x.fi >= a[i].w)
					maximize(dp[x.fi-a[i].w],x.se+a[i].v);
		int ans = 0;
		for (auto &x : dp) maximize(ans,x.se);
		cout << ans;
	}
 
}
 
namespace sub5 {
 
	bool approved() {
		int x = a[1].w;
		FOR(i,1,n)
			if (a[i].w != x) return false;
		return W <= 1e8;
	}
 
	void solve(void)
	{
		sort(a+1,a+n+1,[&](Data &x, Data &y) {
			return x.v > y.v;
		});
		int w0 = a[1].w, k = min(W/w0,n);
		int ans = 0;
		FOR(i,1,k) ans += a[i].v;
		cout << ans;
	}
 
}
 
namespace sub6 {
 
	map<int,vi> val;
	int pre1[N],pre2[N];
 
	bool approved() {
		set<int> s;
		FOR(i,1,n) s.insert(a[i].w);
		return W <= 1e8 and sz(s) <= 2;
	}
 
	void solve(void)
	{
		FOR(i,1,n) val[a[i].w].pb(a[i].v);
		int val1 = 0, len1 = 0, val2 = 0, len2 = 0;
		for (auto x : val)
			if (!val1)
			{
				len1 = sz(x.se);
				val1 = x.fi;
				sort(all(x.se),greater<int>());
				FOR(i,0,len1-1)
					pre1[i+1] = pre1[i]+x.se[i];
			}
			else
			{	
				len2 = sz(x.se);
				val2 = x.fi;
				sort(all(x.se),greater<int>());
				FOR(i,0,len2-1)
					pre2[i+1] = pre2[i]+x.se[i];
			}
		int ans = 0;
		FOR(i,0,len1-1)
			if (val1*i >= W) break;
			else 
			{
				int cur = (W-(i*val1))/val2;
				maximize(ans,pre1[i]+pre2[cur]);
			}
		cout << ans;
	}
 
}
 
namespace sub7 {
 
	vi val[N];
 
	vi mergeSort(vi &a, vi &b)
	{
		int n = sz(a)-1, m = sz(b)-1;
		vi ans(n+m+1);
		int pos = 0;
		FOR(i,0,n+m)
		{
			int l = max(0LL,i-m), r = min(i,n);
			if (pos < l) pos = l;
			if (pos > r) pos = r;
			while (pos < r and a[pos+1]+b[i-(pos+1)] >= a[pos]+b[i-pos]) 
				++pos;
			ans[i] = a[pos]+b[i-pos];
		}
		return ans;
	}
 
	vi calc(vi &v, int tmp, int sz)
	{
		vi ans;
		int pos = 1, sum = v[0];
		while (pos < sz(v) and tmp > 0)
		{
			tmp--;
			sum = v[pos++];
		}
		ans.pb(sum);
		while (pos < sz(v))
		{
			int cur = pos+sz;
			while (pos < sz(v) and pos < cur)
				sum = v[pos++];
			ans.pb(sum);
		}
		return ans;
	}
 
	void solve(void)
	{
		sort(a+1,a+n+1,[&](Data &x, Data &y) {
			if (x.w == y.w) return x.v > y.v;
			return x.w < y.w;
		});	
		while (n and a[n].w > W) n--;
		vi vec;
		FOR(i,1,n) vec.pb(a[i].w);
		vec.erase(unique(all(vec)),vec.end());
		int len = sz(vec);
		if (vec.empty())
		{
			cout << 0 << endl;
			return;
		}
		FOR(i,0,len-1) val[i].pb(0);
		int pos = 0;
		FOR(i,1,n)
		{
			if (vec[pos] != a[i].w) pos++;
			val[pos].pb(a[i].v);
		}
		FOR(i,0,pos)
			FOR(j,1,sz(val[i])-1) val[i][j] += val[i][j-1];
		FOR(i,0,pos-1)
		{
			int cur = (W-(W/vec[i+1])*vec[i+1])/vec[i], sz = vec[i+1]/vec[i];
			val[i] = calc(val[i],cur,sz);
			val[i+1] = mergeSort(val[i],val[i+1]);
		}
		int tmp = W/vec[pos];
		cout << val[pos][min(tmp,(int)sz(val[pos])-1)];
	}
 
}
 
bool M2;
signed main()
{
    fast;
    if (fopen(name".inp","r"))
    {
    	freopen(name".inp","r",stdin);
    	freopen(name".out","w",stdout);
    }
    cin >> n >> W;
    FOR(i,1,n) cin >> a[i].v >> a[i].w;
    if (sub3::approved()) return sub3::solve(), time(), memory(), 0;
    if (sub5::approved()) return sub5::solve(), time(), memory(), 0;
    if (sub6::approved()) return sub6::solve(), time(), memory(), 0;
    sub7::solve();
    time();
    memory();
    return 0;
}
// ██░ ██  █    ██  ███▄    █   ▄████
//▓██░ ██▒ ██  ▓██▒ ██ ▀█   █  ██▒ ▀█▒
//▒██▀▀██░▓██  ▒██░▓██  ▀█ ██▒▒██░▄▄▄░
//░▓█ ░██ ▓▓█  ░██░▓██▒  ▐▌██▒░▓█  ██▓
//░▓█▒░██▓▒▒█████▓ ▒██░   ▓██░░▒▓███▀▒
// ▒ ░░▒░▒░▒▓▒ ▒ ▒ ░ ▒░   ▒ ▒  ░▒   ▒
// ▒ ░▒░ ░░░▒░ ░ ░ ░ ░░   ░ ▒░  ░   ░
// ░  ░░ ░ ░░░ ░ ░    ░   ░ ░ ░ ░   ░
// ░  ░  ░   ░              ░       ░
				/*
* @Author: hungeazy
* @Date:   2025-10-21 10:00:32
* @Last Modified by:   hungeazy
* @Last Modified time: 2025-10-21 15:55:28
*/
#include <bits/stdc++.h>
#include <ext/pb_ds/assoc_container.hpp> 
#include <ext/pb_ds/tree_policy.hpp> 
// #pragma GCC optimize("O3")  
// #pragma GCC optimize("unroll-loops")  
// #pragma GCC target("avx2,bmi,bmi2,popcnt,lzcnt")  
using namespace std;
using namespace __gnu_pbds; 
bool M1;
#define fast ios_base::sync_with_stdio(false); cin.tie(NULL); cout.tie(NULL);
#define int long long
#define ll long long 
#define ull unsigned long long
#define sz(x) x.size()
#define sqr(x) (1LL * (x) * (x))
#define all(x) x.begin(), x.end()
#define fill(f,x) memset(f,x,sizeof(f))
#define FOR(i,l,r) for(int i=l;i<=r;i++)
#define FOD(i,r,l) for(int i=r;i>=l;i--)
#define debug(x) cout << #x << " = " << x << '\n'
#define ii pair<int,int>
#define iii pair<int,ii>
#define di pair<ii,ii>
#define vi vector<int>
#define vii vector<ii>
#define mii map<int,int>
#define fi first
#define se second
#define pb push_back
#define MOD 1000000007
#define __lcm(a,b) (1ll * ((a) / __gcd((a), (b))) * (b))
#define YES cout << "YES\n"
#define NO cout << "NO\n"
#define MASK(i) (1LL << (i))
#define c_bit(i) __builtin_popcountll(i)
#define BIT(x,i) ((x) & MASK(i))
#define SET_ON(x,i) ((x) | MASK(i))
#define SET_OFF(x,i) ((x) & ~MASK(i))
#define oo 1e18
#define name ""
#define endl '\n'
#define memory() cerr << abs(&M2-&M1)/1024.0/1024 << " MB" << endl
#define time() cerr << endl << "-------------Time:" << 1000.0 * clock() / CLOCKS_PER_SEC << "ms." << endl
template<typename T> bool maximize(T &res, const T &val) { if (res < val){ res = val; return true; }; return false; }
template<typename T> bool minimize(T &res, const T &val) { if (res > val){ res = val; return true; }; return false; }
template <class T> using ordered_set = tree <T, null_type, less_equal <T>, rb_tree_tag,tree_order_statistics_node_update>;
const int N = (int)5e5+10;
int n,W;

struct Data {
	int v,w;
	bool operator<(const Data &other) {
		return w > other.w;
	}
} a[N];

namespace sub3 {

	bool approved() {
		return n <= 1e3 and W <= 1e4;
	}

	void solve(void)
	{
		sort(a+1,a+n+1);
		mii dp;
		dp[W] = 0;
		FOR(i,1,n)
			for (auto &x : dp)
				if (x.fi >= a[i].w)
					maximize(dp[x.fi-a[i].w],x.se+a[i].v);
		int ans = 0;
		for (auto &x : dp) maximize(ans,x.se);
		cout << ans;
	}
	
}

namespace sub5 {

	bool approved() {
		int x = a[1].w;
		FOR(i,1,n)
			if (a[i].w != x) return false;
		return W <= 1e8;
	}

	void solve(void)
	{
		sort(a+1,a+n+1,[&](Data &x, Data &y) {
			return x.v > y.v;
		});
		int w0 = a[1].w, k = min(W/w0,n);
		int ans = 0;
		FOR(i,1,k) ans += a[i].v;
		cout << ans;
	}

}

namespace sub6 {

	map<int,vi> val;
	int pre1[N],pre2[N];

	bool approved() {
		set<int> s;
		FOR(i,1,n) s.insert(a[i].w);
		return W <= 1e8 and sz(s) <= 2;
	}

	void solve(void)
	{
		FOR(i,1,n) val[a[i].w].pb(a[i].v);
		int val1 = 0, len1 = 0, val2 = 0, len2 = 0;
		for (auto x : val)
			if (!val1)
			{
				len1 = sz(x.se);
				val1 = x.fi;
				sort(all(x.se),greater<int>());
				FOR(i,0,len1-1)
					pre1[i+1] = pre1[i]+x.se[i];
			}
			else
			{	
				len2 = sz(x.se);
				val2 = x.fi;
				sort(all(x.se),greater<int>());
				FOR(i,0,len2-1)
					pre2[i+1] = pre2[i]+x.se[i];
			}
		int ans = 0;
		FOR(i,0,len1-1)
			if (val1*i >= W) break;
			else 
			{
				int cur = (W-(i*val1))/val2;
				maximize(ans,pre1[i]+pre2[cur]);
			}
		cout << ans;
	}

}

namespace sub7 {

	vi val[N];

	vi mergeSort(vi &a, vi &b)
	{
		int n = sz(a)-1, m = sz(b)-1;
		vi ans(n+m+1);
		int pos = 0;
		FOR(i,0,n+m)
		{
			int l = max(0LL,i-m), r = min(i,n);
			if (pos < l) pos = l;
			if (pos > r) pos = r;
			while (pos < r and a[pos+1]+b[i-(pos+1)] >= a[pos]+b[i-pos]) 
				++pos;
			ans[i] = a[pos]+b[i-pos];
		}
		return ans;
	}

	vi calc(vi &v, int tmp, int sz)
	{
		vi ans;
		int pos = 1, sum = v[0];
		while (pos < sz(v) and tmp > 0)
		{
			tmp--;
			sum = v[pos++];
		}
		ans.pb(sum);
		while (pos < sz(v))
		{
			int cur = pos+sz;
			while (pos < sz(v) and pos < cur)
				sum = v[pos++];
			ans.pb(sum);
		}
		return ans;
	}

	void solve(void)
	{
		sort(a+1,a+n+1,[&](Data &x, Data &y) {
			if (x.w == y.w) return x.v > y.v;
			return x.w < y.w;
		});	
		while (n and a[n].w > W) n--;
		vi vec;
		FOR(i,1,n) vec.pb(a[i].w);
		vec.erase(unique(all(vec)),vec.end());
		int len = sz(vec);
		if (vec.empty())
		{
			cout << 0 << endl;
			return;
		}
		FOR(i,0,len-1) val[i].pb(0);
		int pos = 0;
		FOR(i,1,n)
		{
			if (vec[pos] != a[i].w) pos++;
			val[pos].pb(a[i].v);
		}
		FOR(i,0,pos)
			FOR(j,1,sz(val[i])-1) val[i][j] += val[i][j-1];
		FOR(i,0,pos-1)
		{
			int cur = (W-(W/vec[i+1])*vec[i+1])/vec[i], sz = vec[i+1]/vec[i];
			val[i] = calc(val[i],cur,sz);
			val[i+1] = mergeSort(val[i],val[i+1]);
		}
		int tmp = W/vec[pos];
		cout << val[pos][min(tmp,(int)sz(val[pos])-1)];
	}

}

bool M2;
signed main()
{
    fast;
    if (fopen(name".inp","r"))
    {
    	freopen(name".inp","r",stdin);
    	freopen(name".out","w",stdout);
    }
    cin >> n >> W;
    FOR(i,1,n) cin >> a[i].v >> a[i].w;
    if (sub3::approved()) return sub3::solve(), time(), memory(), 0;
    if (sub5::approved()) return sub5::solve(), time(), memory(), 0;
    if (sub6::approved()) return sub6::solve(), time(), memory(), 0;
    sub7::solve();
    time();
    memory();
    return 0;
}
// ██░ ██  █    ██  ███▄    █   ▄████
//▓██░ ██▒ ██  ▓██▒ ██ ▀█   █  ██▒ ▀█▒
//▒██▀▀██░▓██  ▒██░▓██  ▀█ ██▒▒██░▄▄▄░
//░▓█ ░██ ▓▓█  ░██░▓██▒  ▐▌██▒░▓█  ██▓
//░▓█▒░██▓▒▒█████▓ ▒██░   ▓██░░▒▓███▀▒
// ▒ ░░▒░▒░▒▓▒ ▒ ▒ ░ ▒░   ▒ ▒  ░▒   ▒
// ▒ ░▒░ ░░░▒░ ░ ░ ░ ░░   ░ ▒░  ░   ░
// ░  ░░ ░ ░░░ ░ ░    ░   ░ ░ ░ ░   ░
// ░  ░  ░   ░              ░       ░